diff options
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 <> #{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"); |