aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadPeriodsStep.java16
-rw-r--r--server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadPeriodsStepTest.java48
-rw-r--r--server/sonar-server/src/test/resources/org/sonar/server/computation/step/LoadPeriodsStepTest/previous_version_is_last_one.xml16
-rw-r--r--sonar-core/src/main/resources/org/sonar/l10n/core.properties1
-rw-r--r--sonar-db/src/main/java/org/sonar/core/timemachine/Periods.java14
-rw-r--r--sonar-db/src/main/java/org/sonar/db/component/SnapshotDao.java19
-rw-r--r--sonar-db/src/main/java/org/sonar/db/component/SnapshotMapper.java3
-rw-r--r--sonar-db/src/main/resources/org/sonar/db/component/SnapshotMapper.xml14
-rw-r--r--sonar-db/src/test/java/org/sonar/core/timemachine/PeriodsTest.java7
-rw-r--r--sonar-db/src/test/java/org/sonar/db/component/SnapshotDaoTest.java23
10 files changed, 140 insertions, 21 deletions
diff --git a/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadPeriodsStep.java b/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadPeriodsStep.java
index cd2fe44638b..4fd1cfe3d72 100644
--- a/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadPeriodsStep.java
+++ b/server/sonar-server/src/main/java/org/sonar/server/computation/step/LoadPeriodsStep.java
@@ -50,6 +50,7 @@ import org.sonar.server.computation.component.TypeAwareVisitorAdapter;
import org.sonar.server.computation.period.Period;
import org.sonar.server.computation.period.PeriodsHolderImpl;
+import static org.sonar.db.component.SnapshotDto.STATUS_PROCESSED;
import static org.sonar.db.component.SnapshotQuery.SORT_FIELD.BY_DATE;
import static org.sonar.db.component.SnapshotQuery.SORT_ORDER.ASC;
import static org.sonar.db.component.SnapshotQuery.SORT_ORDER.DESC;
@@ -223,7 +224,8 @@ public class LoadPeriodsStep implements ComputationStep {
}
List<SnapshotDto> snapshotDtos = dbClient.snapshotDao().selectPreviousVersionSnapshots(session, projectId, currentVersion);
if (snapshotDtos.isEmpty()) {
- return null;
+ // If no previous version is found, the first analysis is returned
+ return findByFirstAnalysis(index);
}
SnapshotDto snapshotDto = snapshotDtos.get(0);
LOG.debug("Compare to previous version ({})", formatDate(snapshotDto.getCreatedAt()));
@@ -231,6 +233,16 @@ public class LoadPeriodsStep implements ComputationStep {
}
@CheckForNull
+ private Period findByFirstAnalysis(int index) {
+ SnapshotDto snapshotDto = dbClient.snapshotDao().selectOldestSnapshot(session, projectId);
+ if (snapshotDto == null) {
+ return null;
+ }
+ LOG.debug("Compare to first analysis ({})", formatDate(snapshotDto.getCreatedAt()));
+ return new Period(index, CoreProperties.TIMEMACHINE_MODE_PREVIOUS_VERSION, null, snapshotDto.getCreatedAt(), snapshotDto.getId());
+ }
+
+ @CheckForNull
private Period findByVersion(int index, String version) {
SnapshotDto snapshot = findFirstSnapshot(session, createCommonQuery(projectId).setVersion(version).setSort(BY_DATE, DESC));
if (snapshot == null) {
@@ -275,7 +287,7 @@ public class LoadPeriodsStep implements ComputationStep {
}
private static SnapshotQuery createCommonQuery(Long projectId) {
- return new SnapshotQuery().setComponentId(projectId).setStatus(SnapshotDto.STATUS_PROCESSED);
+ return new SnapshotQuery().setComponentId(projectId).setStatus(STATUS_PROCESSED);
}
private static String formatDate(long date) {
diff --git a/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadPeriodsStepTest.java b/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadPeriodsStepTest.java
index 154bd7b3cc2..29430c796b7 100644
--- a/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadPeriodsStepTest.java
+++ b/server/sonar-server/src/test/java/org/sonar/server/computation/step/LoadPeriodsStepTest.java
@@ -343,7 +343,7 @@ public class LoadPeriodsStepTest extends BaseStepTest {
}
@Test
- public void feed_period_by_previous_version_wit_previous_version_deleted() {
+ public void feed_period_by_previous_version_with_previous_version_deleted() {
setupRoot(PROJECT_ROOT);
dbTester.prepareDbUnit(getClass(), "previous_version_deleted.xml");
@@ -376,9 +376,46 @@ public class LoadPeriodsStepTest extends BaseStepTest {
}
@Test
- @UseDataProvider("projectAndViewRoots")
- public void no_period_by_previous_version_when_no_event_version(Component root) {
- setupRoot(root);
+ public void feed_period_by_previous_version_with_first_analysis_when_no_previous_version_found() {
+ setupRoot(PROJECT_ROOT);
+
+ dbTester.prepareDbUnit(getClass(), "no_previous_version.xml");
+
+ settings.setProperty("sonar.timemachine.period1", "previous_version");
+
+ underTest.execute();
+ List<Period> periods = periodsHolder.getPeriods();
+ assertThat(periods).hasSize(1);
+
+ Period period = periods.get(0);
+ assertThat(period.getMode()).isEqualTo(CoreProperties.TIMEMACHINE_MODE_PREVIOUS_VERSION);
+ assertThat(period.getModeParameter()).isNull();
+ assertThat(period.getSnapshotDate()).isEqualTo(1226379600000L);
+ assertThat(period.getSnapshotId()).isEqualTo(1000L);
+ }
+
+ @Test
+ public void feed_period_by_previous_version_with_first_analysis_when_previous_snapshot_is_the_last_one() {
+ setupRoot(PROJECT_ROOT);
+
+ dbTester.prepareDbUnit(getClass(), "previous_version_is_last_one.xml");
+
+ settings.setProperty("sonar.timemachine.period1", "previous_version");
+
+ underTest.execute();
+ List<Period> periods = periodsHolder.getPeriods();
+ assertThat(periods).hasSize(1);
+
+ Period period = periods.get(0);
+ assertThat(period.getMode()).isEqualTo(CoreProperties.TIMEMACHINE_MODE_PREVIOUS_VERSION);
+ assertThat(period.getModeParameter()).isNull();
+ assertThat(period.getSnapshotDate()).isEqualTo(1226379600000L);
+ assertThat(period.getSnapshotId()).isEqualTo(1000L);
+ }
+
+ @Test
+ public void no_period_by_previous_version_when_no_event_version_for_views() {
+ setupRoot(VIEW_ROOT);
dbTester.prepareDbUnit(getClass(), "no_previous_version.xml");
@@ -480,7 +517,8 @@ public class LoadPeriodsStepTest extends BaseStepTest {
underTest.execute();
List<Period> periods = periodsHolder.getPeriods();
- assertThat(periods).extracting("mode").containsExactly(CoreProperties.TIMEMACHINE_MODE_DATE, CoreProperties.TIMEMACHINE_MODE_DAYS,
+ assertThat(periods).extracting("mode").containsExactly(
+ CoreProperties.TIMEMACHINE_MODE_DATE, CoreProperties.TIMEMACHINE_MODE_DAYS,
CoreProperties.TIMEMACHINE_MODE_PREVIOUS_ANALYSIS, CoreProperties.TIMEMACHINE_MODE_VERSION);
assertThat(periods.get(0).getMode()).isEqualTo(CoreProperties.TIMEMACHINE_MODE_DATE);
diff --git a/server/sonar-server/src/test/resources/org/sonar/server/computation/step/LoadPeriodsStepTest/previous_version_is_last_one.xml b/server/sonar-server/src/test/resources/org/sonar/server/computation/step/LoadPeriodsStepTest/previous_version_is_last_one.xml
new file mode 100644
index 00000000000..f61194367cf
--- /dev/null
+++ b/server/sonar-server/src/test/resources/org/sonar/server/computation/step/LoadPeriodsStepTest/previous_version_is_last_one.xml
@@ -0,0 +1,16 @@
+<dataset>
+
+ <projects id="1" kee="ROOT_KEY" name="project" uuid="ABCD"/>
+
+ <!-- 2008-11-11 -->
+ <!-- Version 0.9 -->
+ <snapshots id="1000" purge_status="[null]" period1_mode="[null]" period1_param="[null]" period1_date="[null]"
+ period2_mode="[null]" period2_param="[null]" period2_date="[null]"
+ period3_mode="[null]" period3_param="[null]" period3_date="[null]" period4_mode="[null]"
+ period4_param="[null]" period4_date="[null]" period5_mode="[null]"
+ period5_param="[null]" period5_date="[null]"
+ project_id="1" parent_snapshot_id="[null]" root_project_id="1" root_snapshot_id="[null]"
+ scope="PRJ" qualifier="TRK" created_at="1226379600000" build_date="1226379600000" version="0.9" path=""
+ status="P" islast="[true]" depth="0"/>
+
+</dataset>
diff --git a/sonar-core/src/main/resources/org/sonar/l10n/core.properties b/sonar-core/src/main/resources/org/sonar/l10n/core.properties
index 67cacb16ffe..d414fc20436 100644
--- a/sonar-core/src/main/resources/org/sonar/l10n/core.properties
+++ b/sonar-core/src/main/resources/org/sonar/l10n/core.properties
@@ -268,6 +268,7 @@ since_version_detailed.short={0} ({1})
since_previous_version=since previous version
since_previous_version.short=\u0394 version
since_previous_version_detailed=since previous version ({0} - {1})
+since_previous_version_with_only_date=since previous version ({0})
since_previous_version_detailed.short=\u0394 version ({0})
time_changes=Time changes
work_duration.x_days={0}d
diff --git a/sonar-db/src/main/java/org/sonar/core/timemachine/Periods.java b/sonar-db/src/main/java/org/sonar/core/timemachine/Periods.java
index 2765f3d8764..fbb06254223 100644
--- a/sonar-db/src/main/java/org/sonar/core/timemachine/Periods.java
+++ b/sonar-db/src/main/java/org/sonar/core/timemachine/Periods.java
@@ -22,13 +22,13 @@ package org.sonar.core.timemachine;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
-import java.util.Locale;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import org.sonar.api.config.Settings;
import org.sonar.api.i18n.I18n;
import static com.google.common.base.Preconditions.checkArgument;
+import static java.util.Locale.ENGLISH;
import static org.apache.commons.lang.StringUtils.isNotBlank;
import static org.sonar.api.CoreProperties.TIMEMACHINE_MODE_DATE;
import static org.sonar.api.CoreProperties.TIMEMACHINE_MODE_DAYS;
@@ -47,10 +47,6 @@ public class Periods {
this.i18n = i18n;
}
- private static Locale getLocale() {
- return Locale.ENGLISH;
- }
-
@CheckForNull
private static String convertDate(@Nullable Date date) {
if (date != null) {
@@ -129,9 +125,13 @@ public class Periods {
}
private String labelForPreviousVersion(@Nullable String param, @Nullable String date, boolean shortLabel) {
- if (param == null) {
+ if (param == null && date == null) {
return label("since_previous_version", shortLabel);
}
+ if (param == null) {
+ // Special case when no snapshot for previous version is found. The first analysis is then returned -> Display only the date.
+ return label("since_previous_version_with_only_date", shortLabel, date);
+ }
if (date == null) {
return label("since_previous_version_detailed", shortLabel, param);
}
@@ -143,7 +143,7 @@ public class Periods {
if (shortLabel) {
msgKey += ".short";
}
- return i18n.message(getLocale(), msgKey, null, parameters);
+ return i18n.message(ENGLISH, msgKey, null, parameters);
}
private static class PeriodParameters {
diff --git a/sonar-db/src/main/java/org/sonar/db/component/SnapshotDao.java b/sonar-db/src/main/java/org/sonar/db/component/SnapshotDao.java
index 55d3fbbb630..575d5ac26a0 100644
--- a/sonar-db/src/main/java/org/sonar/db/component/SnapshotDao.java
+++ b/sonar-db/src/main/java/org/sonar/db/component/SnapshotDao.java
@@ -26,6 +26,7 @@ import java.util.Collection;
import java.util.List;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
+import org.apache.ibatis.session.RowBounds;
import org.sonar.api.resources.Scopes;
import org.sonar.db.Dao;
import org.sonar.db.DbSession;
@@ -35,6 +36,10 @@ import static com.google.common.collect.FluentIterable.from;
public class SnapshotDao implements Dao {
+ public static boolean isLast(SnapshotDto snapshotTested, @Nullable SnapshotDto previousLastSnapshot) {
+ return previousLastSnapshot == null || previousLastSnapshot.getCreatedAt() < snapshotTested.getCreatedAt();
+ }
+
@CheckForNull
public SnapshotDto selectById(DbSession session, long id) {
return mapper(session).selectByKey(id);
@@ -69,6 +74,12 @@ public class SnapshotDao implements Dao {
return mapper(session).selectPreviousVersionSnapshots(componentId, lastVersion);
}
+ @CheckForNull
+ public SnapshotDto selectOldestSnapshot(DbSession session, long componentId) {
+ List<SnapshotDto> snapshotDtos = mapper(session).selectOldestSnapshots(componentId, new RowBounds(0, 1));
+ return snapshotDtos.isEmpty() ? null : snapshotDtos.get(0);
+ }
+
public List<SnapshotDto> selectSnapshotAndChildrenOfProjectScope(DbSession session, long snapshotId) {
return mapper(session).selectSnapshotAndChildrenOfScope(snapshotId, Scopes.PROJECT);
}
@@ -89,10 +100,6 @@ public class SnapshotDao implements Dao {
return mapper(session).updateSnapshotAndChildrenLastFlag(rootId, pathRootId, path, isLast);
}
- public static boolean isLast(SnapshotDto snapshotTested, @Nullable SnapshotDto previousLastSnapshot) {
- return previousLastSnapshot == null || previousLastSnapshot.getCreatedAt() < snapshotTested.getCreatedAt();
- }
-
public SnapshotDto insert(DbSession session, SnapshotDto item) {
mapper(session).insert(item);
return item;
@@ -111,8 +118,8 @@ public class SnapshotDao implements Dao {
@CheckForNull
public ViewsSnapshotDto selectSnapshotBefore(long componentId, long date, DbSession dbSession) {
return from(mapper(dbSession).selectSnapshotBefore(componentId, date))
- .first()
- .orNull();
+ .first()
+ .orNull();
}
@CheckForNull
diff --git a/sonar-db/src/main/java/org/sonar/db/component/SnapshotMapper.java b/sonar-db/src/main/java/org/sonar/db/component/SnapshotMapper.java
index a77633f7d1a..5cc7e177d5b 100644
--- a/sonar-db/src/main/java/org/sonar/db/component/SnapshotMapper.java
+++ b/sonar-db/src/main/java/org/sonar/db/component/SnapshotMapper.java
@@ -23,6 +23,7 @@ package org.sonar.db.component;
import java.util.List;
import javax.annotation.CheckForNull;
import org.apache.ibatis.annotations.Param;
+import org.apache.ibatis.session.RowBounds;
public interface SnapshotMapper {
@@ -40,6 +41,8 @@ public interface SnapshotMapper {
List<SnapshotDto> selectPreviousVersionSnapshots(@Param(value = "componentId") Long componentId, @Param(value = "lastVersion") String lastVersion);
+ List<SnapshotDto> selectOldestSnapshots(@Param(value = "componentId") Long componentId, RowBounds rowBounds);
+
List<SnapshotDto> selectSnapshotAndChildrenOfScope(@Param(value = "snapshot") Long resourceId, @Param(value = "scope") String scope);
int updateSnapshotAndChildrenLastFlagAndStatus(@Param(value = "root") Long rootId, @Param(value = "pathRootId") Long pathRootId,
diff --git a/sonar-db/src/main/resources/org/sonar/db/component/SnapshotMapper.xml b/sonar-db/src/main/resources/org/sonar/db/component/SnapshotMapper.xml
index 01b89c7e1aa..597f033cc50 100644
--- a/sonar-db/src/main/resources/org/sonar/db/component/SnapshotMapper.xml
+++ b/sonar-db/src/main/resources/org/sonar/db/component/SnapshotMapper.xml
@@ -114,10 +114,22 @@
<include refid="snapshotColumns"/>
FROM snapshots s
INNER JOIN events e ON s.id = e.snapshot_id AND e.name &lt;&gt; #{lastVersion} AND e.category='Version'
- INNER JOIN projects p ON p.uuid=e.component_uuid AND p.id=#{componentId}
+ <where>
+ s.project_id=#{componentId}
+ </where>
ORDER BY e.event_date DESC
</select>
+ <select id="selectOldestSnapshots" parameterType="map" resultType="Snapshot">
+ SELECT
+ <include refid="snapshotColumns"/>
+ FROM snapshots s
+ <where>
+ s.project_id=#{componentId}
+ </where>
+ ORDER BY s.created_at ASC
+ </select>
+
<select id="selectSnapshotAndChildrenOfScope" parameterType="map" resultType="Snapshot">
select
<include refid="snapshotColumns"/>
diff --git a/sonar-db/src/test/java/org/sonar/core/timemachine/PeriodsTest.java b/sonar-db/src/test/java/org/sonar/core/timemachine/PeriodsTest.java
index fc1e34b98fe..f124e96d8ff 100644
--- a/sonar-db/src/test/java/org/sonar/core/timemachine/PeriodsTest.java
+++ b/sonar-db/src/test/java/org/sonar/core/timemachine/PeriodsTest.java
@@ -202,6 +202,13 @@ public class PeriodsTest {
}
@Test
+ public void return_since_previous_version_with_only_date_label_when_no_param_and_date_is_set() {
+ periods.label(TIMEMACHINE_MODE_PREVIOUS_VERSION, null, STRING_DATE);
+
+ verify(i18n).message(any(Locale.class), eq("since_previous_version_with_only_date"), isNull(String.class), eq(STRING_DATE));
+ }
+
+ @Test
public void return_since_previous_version_detailed_abbreviation_when_param_and_date_are_set() {
periods.abbreviation(TIMEMACHINE_MODE_PREVIOUS_VERSION, VERSION, DATE);
diff --git a/sonar-db/src/test/java/org/sonar/db/component/SnapshotDaoTest.java b/sonar-db/src/test/java/org/sonar/db/component/SnapshotDaoTest.java
index 9768a52e1c6..37675670851 100644
--- a/sonar-db/src/test/java/org/sonar/db/component/SnapshotDaoTest.java
+++ b/sonar-db/src/test/java/org/sonar/db/component/SnapshotDaoTest.java
@@ -29,6 +29,7 @@ import org.sonar.api.resources.Qualifiers;
import org.sonar.api.resources.Scopes;
import org.sonar.api.utils.DateUtils;
import org.sonar.api.utils.System2;
+import org.sonar.db.DbSession;
import org.sonar.db.DbTester;
import org.sonar.test.DbTests;
@@ -36,6 +37,7 @@ import static org.assertj.core.api.Assertions.assertThat;
import static org.sonar.db.component.SnapshotQuery.SORT_FIELD.BY_DATE;
import static org.sonar.db.component.SnapshotQuery.SORT_ORDER.ASC;
import static org.sonar.db.component.SnapshotQuery.SORT_ORDER.DESC;
+import static org.sonar.db.component.SnapshotTesting.newSnapshotForProject;
@Category(DbTests.class)
public class SnapshotDaoTest {
@@ -43,6 +45,8 @@ public class SnapshotDaoTest {
@Rule
public DbTester db = DbTester.create(System2.INSTANCE);
+ DbSession dbSession = db.getSession();
+
SnapshotDao underTest = db.getDbClient().snapshotDao();
@Test
@@ -173,6 +177,25 @@ public class SnapshotDaoTest {
}
@Test
+ public void select_first_snapshots() throws Exception {
+ ComponentDto project = ComponentTesting.newProjectDto();
+ db.getDbClient().componentDao().insert(dbSession, project);
+
+ db.getDbClient().snapshotDao().insert(dbSession,
+ newSnapshotForProject(project).setCreatedAt(5L),
+ newSnapshotForProject(project).setCreatedAt(2L),
+ newSnapshotForProject(project).setCreatedAt(1L)
+ );
+ dbSession.commit();
+
+ SnapshotDto dto = underTest.selectOldestSnapshot(dbSession, project.getId());
+ assertThat(dto).isNotNull();
+ assertThat(dto.getCreatedAt()).isEqualTo(1L);
+
+ assertThat(underTest.selectOldestSnapshot(dbSession, 123456789)).isNull();
+ }
+
+ @Test
public void insert() {
db.prepareDbUnit(getClass(), "empty.xml");