diff options
Diffstat (limited to 'plugins')
14 files changed, 421 insertions, 275 deletions
diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java index f1e767eec96..27da9438c3b 100644 --- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/CorePlugin.java @@ -19,7 +19,7 @@ */ package org.sonar.plugins.core; -import org.sonar.plugins.core.timemachine.NewViolationsDecorator; +import org.sonar.plugins.core.timemachine.*; import com.google.common.collect.Lists; import org.sonar.api.CoreProperties; @@ -42,10 +42,6 @@ import org.sonar.plugins.core.metrics.UserManagedMetrics; import org.sonar.plugins.core.security.ApplyProjectRolesDecorator; import org.sonar.plugins.core.sensors.*; import org.sonar.plugins.core.testdetailsviewer.TestsViewerDefinition; -import org.sonar.plugins.core.timemachine.VariationDecorator; -import org.sonar.plugins.core.timemachine.PeriodLocator; -import org.sonar.plugins.core.timemachine.TendencyDecorator; -import org.sonar.plugins.core.timemachine.TimeMachineConfiguration; import org.sonar.plugins.core.ui.pageselector.GwtPageSelector; import org.sonar.plugins.core.violationsviewer.ViolationsViewerDefinition; import org.sonar.plugins.core.widgets.*; @@ -117,6 +113,27 @@ import java.util.List; description = "Any new users will automatically join this group.", project = false, global = true + ), + @Property( + key = "sonar.timemachine.variation1", + name = "Variation 1", + description = "To be defined. For the moment the number of days", + project = false, + global = true + ), + @Property( + key = "sonar.timemachine.variation2", + name = "Variation 2", + description = "To be defined. For the moment the number of days", + project = false, + global = true + ), + @Property( + key = "sonar.timemachine.variation3", + name = "Variation 3", + description = "To be defined. For the moment the number of days", + project = false, + global = true ) }) public class CorePlugin implements Plugin { @@ -197,9 +214,10 @@ public class CorePlugin implements Plugin { // time machine extensions.add(TendencyDecorator.class); extensions.add(PeriodLocator.class); + extensions.add(PastMeasuresLoader.class); + extensions.add(TimeMachineConfiguration.class); extensions.add(VariationDecorator.class); extensions.add(NewViolationsDecorator.class); - extensions.add(TimeMachineConfiguration.class); return extensions; } diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/NewViolationsDecorator.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/NewViolationsDecorator.java index d64935cf3a3..1c1eab8638c 100644 --- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/NewViolationsDecorator.java +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/NewViolationsDecorator.java @@ -19,22 +19,18 @@ */ package org.sonar.plugins.core.timemachine; -import org.apache.commons.lang.time.DateUtils; -import org.sonar.api.batch.Decorator; -import org.sonar.api.batch.DecoratorBarriers; -import org.sonar.api.batch.DecoratorContext; -import org.sonar.api.batch.DependedUpon; +import org.sonar.api.batch.*; import org.sonar.api.measures.CoreMetrics; import org.sonar.api.measures.Measure; import org.sonar.api.measures.Metric; import org.sonar.api.resources.Project; import org.sonar.api.resources.Resource; import org.sonar.api.rules.Violation; -import org.sonar.api.utils.SonarException; import java.util.Date; +import java.util.List; -@DependedUpon(DecoratorBarriers.END_OF_VIOLATIONS_GENERATION) +@DependsUpon(DecoratorBarriers.END_OF_VIOLATIONS_GENERATION) public class NewViolationsDecorator implements Decorator { private TimeMachineConfiguration timeMachineConfiguration; @@ -44,7 +40,7 @@ public class NewViolationsDecorator implements Decorator { } public boolean shouldExecuteOnProject(Project project) { - return true; + return project.isLatestAnalysis(); } @DependedUpon @@ -54,20 +50,17 @@ public class NewViolationsDecorator implements Decorator { public void decorate(Resource resource, DecoratorContext context) { Measure measure = new Measure(CoreMetrics.NEW_VIOLATIONS); - for (int index = 0; index < 3; index++) { - Integer days = timeMachineConfiguration.getDiffPeriodInDays(index); - if (days != null) { - double value = calculate(context, days) + sumChildren(context, index); - setDiffValue(measure, index, value); - } + for (VariationTarget variationTarget : timeMachineConfiguration.getVariationTargets()) { + Date date = variationTarget.getDate(); + double value = countViolationsAfterDate(context.getViolations(), date) + sumChildren(context, variationTarget.getIndex()); + measure.setVariation(variationTarget.getIndex(), value); } context.saveMeasure(measure); } - int calculate(DecoratorContext context, int days) { - Date targetDate = getTargetDate(context.getProject(), days); + int countViolationsAfterDate(List<Violation> violations, Date targetDate) { int newViolations = 0; - for (Violation violation : context.getViolations()) { + for (Violation violation : violations) { if (!violation.getCreatedAt().before(targetDate)) { newViolations++; } @@ -75,46 +68,17 @@ public class NewViolationsDecorator implements Decorator { return newViolations; } - double sumChildren(DecoratorContext context, int index) { - double sum = 0; + int sumChildren(DecoratorContext context, int index) { + int sum = 0; for (Measure measure : context.getChildrenMeasures(CoreMetrics.NEW_VIOLATIONS)) { - sum = sum + getDiffValue(measure, index); + Double var = measure.getVariation(index); + if (var != null) { + sum = sum + var.intValue(); + } } return sum; } - private Date getTargetDate(Project project, int distanceInDays) { - return DateUtils.addDays(project.getAnalysisDate(), -distanceInDays); - } - - private void setDiffValue(Measure measure, int index, double value) { - switch (index) { - case 0: - measure.setDiffValue1(value); - break; - case 1: - measure.setDiffValue2(value); - break; - case 2: - measure.setDiffValue3(value); - break; - default: - throw new SonarException("Should never happen"); - } - } - - private double getDiffValue(Measure measure, int index) { - switch (index) { - case 0: - return measure.getDiffValue1(); - case 1: - return measure.getDiffValue2(); - case 2: - return measure.getDiffValue3(); - default: - throw new SonarException("Should never happen"); - } - } @Override public String toString() { diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PastMeasuresLoader.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PastMeasuresLoader.java new file mode 100644 index 00000000000..be7fcfe30b1 --- /dev/null +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PastMeasuresLoader.java @@ -0,0 +1,77 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2009 SonarSource SA + * mailto:contact AT sonarsource DOT com + * + * Sonar is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * Sonar is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ +package org.sonar.plugins.core.timemachine; + +import com.google.common.collect.Maps; +import org.apache.commons.lang.ObjectUtils; +import org.sonar.api.BatchExtension; +import org.sonar.api.database.DatabaseSession; +import org.sonar.api.database.model.MeasureModel; +import org.sonar.api.database.model.Snapshot; +import org.sonar.api.measures.Metric; +import org.sonar.api.measures.MetricFinder; +import org.sonar.api.resources.Resource; + +import java.util.Collection; +import java.util.List; +import java.util.Map; + +public class PastMeasuresLoader implements BatchExtension { + + private Map<Integer, Metric> metricByIds; + private DatabaseSession session; + + public PastMeasuresLoader(DatabaseSession session, MetricFinder metricFinder) { + this(session, metricFinder.findAll()); + } + + PastMeasuresLoader(DatabaseSession session, Collection<Metric> metrics) { + this.session = session; + this.metricByIds = Maps.newHashMap(); + for (Metric metric : metrics) { + if (metric.isNumericType()) { + metricByIds.put(metric.getId(), metric); + } + } + } + + public Collection<Metric> getMetrics() { + return metricByIds.values(); + } + + public List<MeasureModel> getPastMeasures(Resource resource, Snapshot projectSnapshot) { + // assume that the resource has already been saved + return getPastMeasures(resource.getId(), projectSnapshot); + } + + public List<MeasureModel> getPastMeasures(int resourceId, Snapshot projectSnapshot) { + // TODO improvement : select only some columns + // TODO support measure on characteristics + String hql = "select m from " + MeasureModel.class.getSimpleName() + " m, " + Snapshot.class.getSimpleName() + " s " + + "where m.snapshotId=s.id and m.metricId in (:metricIds) and m.ruleId=null and m.rulePriority=null and m.characteristic=null " + + "and (s.rootId=:rootSnapshotId or s.id=:rootSnapshotId) and s.resourceId=:resourceId and s.status=:status"; + return session.createQuery(hql) + .setParameter("metricIds", metricByIds.keySet()) + .setParameter("rootSnapshotId", ObjectUtils.defaultIfNull(projectSnapshot.getRootId(), projectSnapshot.getId())) + .setParameter("resourceId", resourceId) + .setParameter("status", Snapshot.STATUS_PROCESSED) + .getResultList(); + } +} diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PeriodLocator.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PeriodLocator.java index ef1acbcbf22..5acf136782f 100644 --- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PeriodLocator.java +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/PeriodLocator.java @@ -19,7 +19,6 @@ */ package org.sonar.plugins.core.timemachine; -import org.apache.commons.lang.StringUtils; import org.apache.commons.lang.time.DateUtils; import org.sonar.api.BatchExtension; import org.sonar.api.database.DatabaseSession; @@ -28,7 +27,7 @@ import org.sonar.api.database.model.Snapshot; import java.util.Date; import java.util.List; -public final class PeriodLocator implements BatchExtension { +public class PeriodLocator implements BatchExtension { private Snapshot projectSnapshot; // TODO replace by PersistenceManager private DatabaseSession session; @@ -37,11 +36,6 @@ public final class PeriodLocator implements BatchExtension { this.session = session; } - // currently not used - boolean acceptProperty(String property) { - return doAcceptProperty(property); - } - Snapshot locate(int days) { List<Snapshot> snapshots = loadSnapshotsFromDatabase(); return getNearestToTarget(snapshots, projectSnapshot.getCreatedAt(), days); @@ -76,8 +70,4 @@ public final class PeriodLocator implements BatchExtension { static long distance(Date d1, Date d2) { return Math.abs(d1.getTime() - d2.getTime()); } - - static Boolean doAcceptProperty(String property) { - return StringUtils.trimToEmpty(property).matches("[0-9]+d"); - } } diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/TimeMachineConfiguration.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/TimeMachineConfiguration.java index 4e219294a09..a940ce514a7 100644 --- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/TimeMachineConfiguration.java +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/TimeMachineConfiguration.java @@ -19,36 +19,84 @@ */ package org.sonar.plugins.core.timemachine; +import com.google.common.collect.Lists; import org.apache.commons.configuration.Configuration; +import org.apache.commons.lang.StringUtils; import org.sonar.api.BatchExtension; import org.sonar.api.CoreProperties; +import org.sonar.api.database.DatabaseSession; import org.sonar.api.database.model.Snapshot; +import java.util.List; + public final class TimeMachineConfiguration implements BatchExtension { - private Configuration configuration; - private PeriodLocator periodLocator; + private static final int NUMBER_OF_VARIATION_TARGETS = 3; + + private final Configuration configuration; + private List<VariationTarget> variationTargets; - public TimeMachineConfiguration(Configuration configuration, PeriodLocator periodLocator) { + public TimeMachineConfiguration(Configuration configuration, DatabaseSession session, PeriodLocator periodLocator) { this.configuration = configuration; - this.periodLocator = periodLocator; + initVariationTargets(periodLocator, session); } - boolean skipTendencies() { - return configuration.getBoolean(CoreProperties.SKIP_TENDENCIES_PROPERTY, CoreProperties.SKIP_TENDENCIES_DEFAULT_VALUE); + /** + * for unit tests + */ + TimeMachineConfiguration(Configuration configuration, List<VariationTarget> variationTargets) { + this.configuration = configuration; + this.variationTargets = variationTargets; } - int getTendencyPeriodInDays() { - return configuration.getInt(CoreProperties.CORE_TENDENCY_DEPTH_PROPERTY, CoreProperties.CORE_TENDENCY_DEPTH_DEFAULT_VALUE); + private void initVariationTargets(PeriodLocator periodLocator, DatabaseSession session) { + variationTargets = Lists.newLinkedList(); + for (int index = 1; index <= NUMBER_OF_VARIATION_TARGETS; index++) { + VariationTarget target = loadVariationTarget(index, periodLocator); + if (target != null) { + save(target, session); + variationTargets.add(target); + } + } + } + + private void save(VariationTarget target, DatabaseSession session) { + Snapshot projectSnapshot = target.getProjectSnapshot(); + switch (target.getIndex()) { + case 1: + projectSnapshot.setVarMode1("PERIOD_IN_DAYS"); + break; + case 2: + projectSnapshot.setVarMode2("PERIOD_IN_DAYS"); + break; + case 3: + projectSnapshot.setVarMode3("PERIOD_IN_DAYS"); + break; + } + session.save(projectSnapshot); } - Integer getDiffPeriodInDays(int index) { - String property = configuration.getString("sonar.timemachine.diff" + index); - return property == null ? null : Integer.valueOf(property); + private VariationTarget loadVariationTarget(int index, PeriodLocator periodLocator) { + String property = configuration.getString("sonar.timemachine.variation" + index); + if (StringUtils.isNotBlank(property)) { + // todo manage non-integer values + Snapshot projectSnapshot = periodLocator.locate(Integer.parseInt(property)); + if (projectSnapshot != null) { + return new VariationTarget(index, projectSnapshot); + } + } + return null; + } + + public boolean skipTendencies() { + return configuration.getBoolean(CoreProperties.SKIP_TENDENCIES_PROPERTY, CoreProperties.SKIP_TENDENCIES_DEFAULT_VALUE); + } + + public int getTendencyPeriodInDays() { + return configuration.getInt(CoreProperties.CORE_TENDENCY_DEPTH_PROPERTY, CoreProperties.CORE_TENDENCY_DEPTH_DEFAULT_VALUE); } - Snapshot getProjectSnapshotForDiffValues(int index) { - Integer days = getDiffPeriodInDays(index); - return days == null ? null : periodLocator.locate(days); + public List<VariationTarget> getVariationTargets() { + return variationTargets; } } diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationDecorator.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationDecorator.java index aa656b01db0..08c5e34837a 100644 --- a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationDecorator.java +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationDecorator.java @@ -20,13 +20,12 @@ package org.sonar.plugins.core.timemachine; import com.google.common.collect.Maps; -import org.apache.commons.configuration.Configuration; -import org.apache.commons.lang.ObjectUtils; import org.sonar.api.batch.*; -import org.sonar.api.database.DatabaseSession; import org.sonar.api.database.model.MeasureModel; -import org.sonar.api.database.model.Snapshot; -import org.sonar.api.measures.*; +import org.sonar.api.measures.Measure; +import org.sonar.api.measures.MeasuresFilters; +import org.sonar.api.measures.Metric; +import org.sonar.api.measures.RuleMeasure; import org.sonar.api.qualitymodel.Characteristic; import org.sonar.api.resources.Project; import org.sonar.api.resources.Resource; @@ -40,63 +39,43 @@ import java.util.Map; @DependedUpon(DecoratorBarriers.END_OF_TIME_MACHINE) public class VariationDecorator implements Decorator { - private Snapshot[] projectTargetSnapshots; - private Map<Integer, Metric> metricByIds; - private DatabaseSession session; + private List<VariationTarget> targets; + private PastMeasuresLoader pastMeasuresLoader; - public VariationDecorator(DatabaseSession session, PeriodLocator periodLocator, Configuration configuration, MetricFinder metricFinder) { - this.session = session; - Snapshot snapshot = periodLocator.locate(5); - projectTargetSnapshots = new Snapshot[]{snapshot}; - initMetrics(metricFinder.findAll()); + public VariationDecorator(PastMeasuresLoader pastMeasuresLoader, TimeMachineConfiguration configuration) { + this(pastMeasuresLoader, configuration.getVariationTargets()); } - /** - * only for unit tests - */ - VariationDecorator(DatabaseSession session, Snapshot[] projectTargetSnapshots, Collection<Metric> metrics) { - this.session = session; - this.projectTargetSnapshots = projectTargetSnapshots; - initMetrics(metrics); - } - - private void initMetrics(Collection<Metric> metrics) { - this.metricByIds = Maps.newHashMap(); - for (Metric metric : metrics) { - if (metric.isNumericType()) { - metricByIds.put(metric.getId(), metric); - } - } + VariationDecorator(PastMeasuresLoader pastMeasuresLoader, List<VariationTarget> targets) { + this.pastMeasuresLoader = pastMeasuresLoader; + this.targets = targets; } public boolean shouldExecuteOnProject(Project project) { - return true; + return project.isLatestAnalysis(); } @DependsUpon public Collection<Metric> dependsUponMetrics() { - return metricByIds.values(); + return pastMeasuresLoader.getMetrics(); } - static boolean shouldCalculateDiffValues(Resource resource) { - // measures on files are currently purged, so past measures are not available + static boolean shouldCalculateVariations(Resource resource) { + // measures on files are currently purged, so past measures are not available on files return !ResourceUtils.isEntity(resource); } public void decorate(Resource resource, DecoratorContext context) { - if (shouldCalculateDiffValues(resource)) { - for (int index = 0; index < projectTargetSnapshots.length; index++) { - Snapshot projectTargetSnapshot = projectTargetSnapshots[index]; - if (projectTargetSnapshot != null) { - calculateDiffValues(resource, context, index, projectTargetSnapshot); - } + if (shouldCalculateVariations(resource)) { + for (VariationTarget target : targets) { + calculateVariation(resource, context, target); } } } - private void calculateDiffValues(Resource resource, DecoratorContext context, int index, Snapshot projectTargetSnapshot) { - List<MeasureModel> pastMeasures = selectPastMeasures(resource.getId(), projectTargetSnapshot); - compareWithPastMeasures(context, index, pastMeasures); + private void calculateVariation(Resource resource, DecoratorContext context, VariationTarget target) { + List<MeasureModel> pastMeasures = pastMeasuresLoader.getPastMeasures(resource, target.getProjectSnapshot()); + compareWithPastMeasures(context, target.getIndex(), pastMeasures); } void compareWithPastMeasures(DecoratorContext context, int index, List<MeasureModel> pastMeasures) { @@ -109,51 +88,21 @@ public class VariationDecorator implements Decorator { for (Measure measure : context.getMeasures(MeasuresFilters.all())) { // compare with past measure MeasureModel pastMeasure = pastMeasuresByKey.get(new MeasureKey(measure)); - if (updateDiffValue(measure, pastMeasure, index)) { - context.saveMeasure(measure); - } + updateVariation(measure, pastMeasure, index); + context.saveMeasure(measure); } } - boolean updateDiffValue(Measure measure, MeasureModel pastMeasure, int index) { - boolean updated = false; + void updateVariation(Measure measure, MeasureModel pastMeasure, int index) { if (pastMeasure != null && pastMeasure.getValue() != null && measure.getValue() != null) { - double diff = (measure.getValue().doubleValue() - pastMeasure.getValue().doubleValue()); - updated = true; - switch (index) { - case 0: - measure.setDiffValue1(diff); - break; - case 1: - measure.setDiffValue2(diff); - break; - case 2: - measure.setDiffValue3(diff); - break; - default: - updated = false; - } + double variation = (measure.getValue().doubleValue() - pastMeasure.getValue().doubleValue()); + measure.setVariation(index, variation); } - return updated; - } - - List<MeasureModel> selectPastMeasures(int resourceId, Snapshot projectTargetSnapshot) { - // improvements : keep query in cache ? select only some columns ? - // TODO support measure on rules and characteristics - String hql = "select m from " + MeasureModel.class.getSimpleName() + " m, " + Snapshot.class.getSimpleName() + " s " + - "where m.snapshotId=s.id and m.metricId in (:metricIds) and m.ruleId=null and m.rulePriority=null and m.characteristic=null " - + "and (s.rootId=:rootSnapshotId or s.id=:rootSnapshotId) and s.resourceId=:resourceId and s.status=:status"; - return session.createQuery(hql) - .setParameter("metricIds", metricByIds.keySet()) - .setParameter("rootSnapshotId", ObjectUtils.defaultIfNull(projectTargetSnapshot.getRootId(), projectTargetSnapshot.getId())) - .setParameter("resourceId", resourceId) - .setParameter("status", Snapshot.STATUS_PROCESSED) - .getResultList(); } @Override public String toString() { - return getClass().toString(); + return getClass().getSimpleName(); } static class MeasureKey { @@ -182,17 +131,26 @@ public class VariationDecorator implements Decorator { @Override public boolean equals(Object o) { - if (this == o) return true; - if (o == null || getClass() != o.getClass()) return false; + if (this == o) { + return true; + } + if (o == null || getClass() != o.getClass()) { + return false; + } MeasureKey that = (MeasureKey) o; - - if (characteristic != null ? !characteristic.equals(that.characteristic) : that.characteristic != null) + if (characteristic != null ? !characteristic.equals(that.characteristic) : that.characteristic != null) { return false; - if (!metricId.equals(that.metricId)) return false; - if (priority != that.priority) return false; - if (ruleId != null ? !ruleId.equals(that.ruleId) : that.ruleId != null) return false; - + } + if (!metricId.equals(that.metricId)) { + return false; + } + if (priority != that.priority) { + return false; + } + if (ruleId != null ? !ruleId.equals(that.ruleId) : that.ruleId != null) { + return false; + } return true; } @@ -205,4 +163,4 @@ public class VariationDecorator implements Decorator { return result; } } -} +}
\ No newline at end of file diff --git a/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationTarget.java b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationTarget.java new file mode 100644 index 00000000000..185f85c53f4 --- /dev/null +++ b/plugins/sonar-core-plugin/src/main/java/org/sonar/plugins/core/timemachine/VariationTarget.java @@ -0,0 +1,47 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2009 SonarSource SA + * mailto:contact AT sonarsource DOT com + * + * Sonar is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * Sonar is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ +package org.sonar.plugins.core.timemachine; + +import org.sonar.api.database.model.Snapshot; + +import java.util.Date; + +public final class VariationTarget { + + private int index; + private Snapshot projectSnapshot; + + public VariationTarget(int index, Snapshot projectSnapshot) { + this.index = index; + this.projectSnapshot = projectSnapshot; + } + + public int getIndex() { + return index; + } + + public Snapshot getProjectSnapshot() { + return projectSnapshot; + } + + public Date getDate() { + return projectSnapshot.getCreatedAt(); + } +} diff --git a/plugins/sonar-core-plugin/src/main/resources/org/sonar/plugins/core/widgets/rules.html.erb b/plugins/sonar-core-plugin/src/main/resources/org/sonar/plugins/core/widgets/rules.html.erb index 83ff6fed336..c62d9c4c78e 100644 --- a/plugins/sonar-core-plugin/src/main/resources/org/sonar/plugins/core/widgets/rules.html.erb +++ b/plugins/sonar-core-plugin/src/main/resources/org/sonar/plugins/core/widgets/rules.html.erb @@ -3,19 +3,17 @@ <tr> <td valign="top" width="50%"> <div class="dashbox"> + <h3>Violations</h3> + <div class="big marginbottom10"> + <%= format_measure(Metric::VIOLATIONS, :url => url_for(:controller => 'drilldown', :action => 'violations', :id => @project.key)) -%> <%= trend_icon(Metric::VIOLATIONS) -%> + </div> <h3>Rules compliance</h3> <div class="big"> <%= format_measure(Metric::VIOLATIONS_DENSITY, :url => url_for_drilldown(Metric::WEIGHTED_VIOLATIONS, {:highlight => Metric::WEIGHTED_VIOLATIONS})) -%> <%= trend_icon(Metric::VIOLATIONS_DENSITY) -%> </div> - </div> + </div> </td> <td valign="top" width="50%" nowrap> - <div class="dashbox"> - <h3>Violations</h3> - <div class="big"> - <%= format_measure(Metric::VIOLATIONS, :url => url_for(:controller => 'drilldown', :action => 'violations', :id => @project.key)) -%> <%= trend_icon(Metric::VIOLATIONS) -%> - </div> - </div> <% blocker_violations = @snapshot.measure(Metric::BLOCKER_VIOLATIONS) critical_violations = @snapshot.measure(Metric::CRITICAL_VIOLATIONS) diff --git a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/NewViolationsDecoratorTest.java b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/NewViolationsDecoratorTest.java index 6ad62671bb4..fc62ae583ef 100644 --- a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/NewViolationsDecoratorTest.java +++ b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/NewViolationsDecoratorTest.java @@ -19,11 +19,6 @@ */ package org.sonar.plugins.core.timemachine; -import static org.hamcrest.Matchers.is; -import static org.junit.Assert.assertThat; -import static org.mockito.Mockito.mock; -import static org.mockito.Mockito.when; - import org.apache.commons.lang.time.DateUtils; import org.junit.Before; import org.junit.Test; @@ -35,6 +30,12 @@ import org.sonar.api.rules.Violation; import java.util.Arrays; import java.util.Date; +import java.util.List; + +import static org.hamcrest.Matchers.is; +import static org.junit.Assert.assertThat; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; public class NewViolationsDecoratorTest { @@ -48,35 +49,43 @@ public class NewViolationsDecoratorTest { } @Test - public void decoratorDefinition() { - assertThat(decorator.shouldExecuteOnProject(new Project("project")), is(true)); + public void shouldExecuteIfLastAnalysis() { + Project project = mock(Project.class); + + when(project.isLatestAnalysis()).thenReturn(false); + assertThat(decorator.shouldExecuteOnProject(project), is(false)); + + when(project.isLatestAnalysis()).thenReturn(true); + assertThat(decorator.shouldExecuteOnProject(project), is(true)); + } + + @Test + public void shouldBeDependedUponMetric() { assertThat(decorator.generatesMetric(), is(CoreMetrics.NEW_VIOLATIONS)); - assertThat(decorator.toString(), is(NewViolationsDecorator.class.getSimpleName())); } @Test - public void shouldCalculate() { + public void shouldCountViolationsAfterDate() { Date date1 = new Date(); Date date2 = DateUtils.addDays(date1, -20); Project project = new Project("project"); project.setAnalysisDate(date1); Violation violation1 = new Violation(null).setCreatedAt(date1); Violation violation2 = new Violation(null).setCreatedAt(date2); - when(context.getViolations()).thenReturn(Arrays.asList(violation1, violation2)); - when(context.getProject()).thenReturn(project); + List<Violation> violations = Arrays.asList(violation1, violation2); - assertThat(decorator.calculate(context, 10), is(1)); - assertThat(decorator.calculate(context, 30), is(2)); + assertThat(decorator.countViolationsAfterDate(violations, DateUtils.addDays(date1, -10)), is(1)); + assertThat(decorator.countViolationsAfterDate(violations, DateUtils.addDays(date1, -30)), is(2)); } @Test public void shouldSumChildren() { - Measure measure1 = new Measure(CoreMetrics.NEW_VIOLATIONS).setDiffValue1(1.0).setDiffValue2(1.0).setDiffValue3(3.0); - Measure measure2 = new Measure(CoreMetrics.NEW_VIOLATIONS).setDiffValue1(1.0).setDiffValue2(2.0).setDiffValue3(3.0); + Measure measure1 = new Measure(CoreMetrics.NEW_VIOLATIONS).setVariation1(1.0).setVariation2(1.0).setVariation3(3.0); + Measure measure2 = new Measure(CoreMetrics.NEW_VIOLATIONS).setVariation1(1.0).setVariation2(2.0).setVariation3(3.0); when(context.getChildrenMeasures(CoreMetrics.NEW_VIOLATIONS)).thenReturn(Arrays.asList(measure1, measure2)); - assertThat(decorator.sumChildren(context, 0), is(2.0)); - assertThat(decorator.sumChildren(context, 1), is(3.0)); - assertThat(decorator.sumChildren(context, 2), is(6.0)); + assertThat(decorator.sumChildren(context, 1), is(2)); + assertThat(decorator.sumChildren(context, 2), is(3)); + assertThat(decorator.sumChildren(context, 3), is(6)); } } diff --git a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PastMeasuresLoaderTest.java b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PastMeasuresLoaderTest.java new file mode 100644 index 00000000000..73959db7844 --- /dev/null +++ b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PastMeasuresLoaderTest.java @@ -0,0 +1,95 @@ +/* + * Sonar, open source software quality management tool. + * Copyright (C) 2009 SonarSource SA + * mailto:contact AT sonarsource DOT com + * + * Sonar is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 3 of the License, or (at your option) any later version. + * + * Sonar is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with Sonar; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02 + */ +package org.sonar.plugins.core.timemachine; + +import org.junit.Test; +import org.sonar.api.database.model.MeasureModel; +import org.sonar.api.database.model.Snapshot; +import org.sonar.api.measures.Metric; +import org.sonar.jpa.test.AbstractDbUnitTestCase; + +import java.util.Arrays; +import java.util.List; + +import static org.hamcrest.CoreMatchers.anyOf; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.junit.internal.matchers.IsCollectionContaining.hasItems; + +public class PastMeasuresLoaderTest extends AbstractDbUnitTestCase { + + private static final int PROJECT_SNAPSHOT_ID = 1000; + private static final int PROJECT_ID = 1; + private static final int FILE_ID = 3; + + @Test + public void shouldGetPastResourceMeasures() { + setupData("shared"); + + List<Metric> metrics = selectMetrics(); + Snapshot projectSnapshot = getSession().getSingleResult(Snapshot.class, "id", PROJECT_SNAPSHOT_ID); + + PastMeasuresLoader loader = new PastMeasuresLoader(getSession(), metrics); + List<MeasureModel> measures = loader.getPastMeasures(FILE_ID, projectSnapshot); + assertThat(measures.size(), is(2)); + + for (MeasureModel measure : measures) { + assertThat(measure.getId(), anyOf(is(5L), is(6L))); + assertThat(measure.getValue(), anyOf(is(5.0), is(60.0))); + } + } + + @Test + public void shouldGetPastProjectMeasures() { + setupData("shared"); + + List<Metric> metrics = selectMetrics(); + Snapshot projectSnapshot = getSession().getSingleResult(Snapshot.class, "id", PROJECT_SNAPSHOT_ID); + + PastMeasuresLoader loader = new PastMeasuresLoader(getSession(), metrics); + List<MeasureModel> measures = loader.getPastMeasures(PROJECT_ID, projectSnapshot); + assertThat(measures.size(), is(2)); + + for (MeasureModel measure : measures) { + assertThat(measure.getId(), anyOf(is(1L), is(2L))); + assertThat(measure.getValue(), anyOf(is(60.0), is(80.0))); + } + } + + @Test + public void shouldKeepOnlyNumericalMetrics() { + Metric ncloc = new Metric("ncloc", Metric.ValueType.INT); + ncloc.setId(1); + Metric complexity = new Metric("complexity", Metric.ValueType.INT); + complexity.setId(2); + Metric data = new Metric("data", Metric.ValueType.DATA); + data.setId(3); + List<Metric> metrics = Arrays.asList(ncloc, complexity, data); + + PastMeasuresLoader loader = new PastMeasuresLoader(getSession(), metrics); + + assertThat(loader.getMetrics().size(), is(2)); + assertThat(loader.getMetrics(), hasItems(ncloc, complexity)); + } + + private List<Metric> selectMetrics() { + return getSession().getResults(Metric.class); + } +} diff --git a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PeriodLocatorTest.java b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PeriodLocatorTest.java index 54ceea82043..d0d99fed25f 100644 --- a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PeriodLocatorTest.java +++ b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/PeriodLocatorTest.java @@ -41,22 +41,6 @@ public class PeriodLocatorTest extends AbstractDbUnitTestCase { private static SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd"); @Test - public void shouldAcceptProperty() { - assertThat(PeriodLocator.doAcceptProperty("1d"), is(true)); - assertThat(PeriodLocator.doAcceptProperty("2d"), is(true)); - assertThat(PeriodLocator.doAcceptProperty("500d"), is(true)); - assertThat(PeriodLocator.doAcceptProperty(" 500d "), is(true)); - } - - @Test - public void shouldNotAcceptProperty() { - assertThat(PeriodLocator.doAcceptProperty(null), is(false)); - assertThat(PeriodLocator.doAcceptProperty(""), is(false)); - assertThat(PeriodLocator.doAcceptProperty("1.2.3"), is(false)); - assertThat(PeriodLocator.doAcceptProperty("2010-05-18"), is(false)); - } - - @Test public void shouldLoadSnapshotsFromDatabase() { setupData("shouldLoadSnapshotsFromDatabase"); diff --git a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/TimeMachineConfigurationTest.java b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/TimeMachineConfigurationTest.java index 4d42f736373..755088e2f5b 100644 --- a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/TimeMachineConfigurationTest.java +++ b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/TimeMachineConfigurationTest.java @@ -19,15 +19,17 @@ */ package org.sonar.plugins.core.timemachine; -import static org.hamcrest.CoreMatchers.is; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.nullValue; - import org.apache.commons.configuration.PropertiesConfiguration; import org.junit.Test; import org.sonar.api.CoreProperties; +import org.sonar.jpa.test.AbstractDbUnitTestCase; + +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.verify; -public class TimeMachineConfigurationTest { +public class TimeMachineConfigurationTest extends AbstractDbUnitTestCase { @Test public void shouldSkipTendencies() { @@ -43,11 +45,14 @@ public class TimeMachineConfigurationTest { } @Test - public void shouldReturnDiffPeriodInDays() { + public void shouldGetPeriodVariationTargets() { PropertiesConfiguration conf = new PropertiesConfiguration(); - conf.setProperty("sonar.timemachine.diff0", "30"); - assertThat(new TimeMachineConfiguration(conf, null).getDiffPeriodInDays(0), is(30)); - assertThat(new TimeMachineConfiguration(conf, null).getDiffPeriodInDays(1), nullValue()); + conf.setProperty("sonar.timemachine.variation1", "7"); + conf.setProperty("sonar.timemachine.variation2", "30"); + PeriodLocator periodLocator = mock(PeriodLocator.class); + new TimeMachineConfiguration(conf, getSession(), periodLocator).getVariationTargets(); + verify(periodLocator).locate(7); + verify(periodLocator).locate(30); } } diff --git a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/VariationDecoratorTest.java b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/VariationDecoratorTest.java index 7e2e86db8fc..b4f93aa97db 100644 --- a/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/VariationDecoratorTest.java +++ b/plugins/sonar-core-plugin/src/test/java/org/sonar/plugins/core/timemachine/VariationDecoratorTest.java @@ -20,70 +20,23 @@ package org.sonar.plugins.core.timemachine; import org.junit.Test; -import org.sonar.api.database.model.MeasureModel; -import org.sonar.api.database.model.Snapshot; -import org.sonar.api.measures.Metric; import org.sonar.api.resources.*; import org.sonar.jpa.test.AbstractDbUnitTestCase; -import java.util.List; - -import static org.hamcrest.CoreMatchers.anyOf; import static org.hamcrest.CoreMatchers.is; import static org.hamcrest.MatcherAssert.assertThat; public class VariationDecoratorTest extends AbstractDbUnitTestCase { - private static final int PROJECT_SNAPSHOT_ID = 1000; - private static final int PROJECT_ID = 1; - private static final int FILE_ID = 3; - - @Test - public void shouldSelectPastResourceMeasures() { - setupData("shared"); - - List<Metric> metrics = selectMetrics(); - Snapshot projectSnapshot = getSession().getSingleResult(Snapshot.class, "id", PROJECT_SNAPSHOT_ID); - - VariationDecorator decorator = new VariationDecorator(getSession(), new Snapshot[0], metrics); - List<MeasureModel> measures = decorator.selectPastMeasures(FILE_ID, projectSnapshot); - assertThat(measures.size(), is(2)); - - for (MeasureModel measure : measures) { - assertThat(measure.getId(), anyOf(is(5L), is(6L))); - assertThat(measure.getValue(), anyOf(is(5.0), is(60.0))); - } - } - @Test - public void shouldSelectPastProjectMeasures() { - setupData("shared"); - - List<Metric> metrics = selectMetrics(); - Snapshot projectSnapshot = getSession().getSingleResult(Snapshot.class, "id", PROJECT_SNAPSHOT_ID); - - VariationDecorator decorator = new VariationDecorator(getSession(), new Snapshot[0], metrics); - List<MeasureModel> measures = decorator.selectPastMeasures(PROJECT_ID, projectSnapshot); - assertThat(measures.size(), is(2)); - - for (MeasureModel measure : measures) { - assertThat(measure.getId(), anyOf(is(1L), is(2L))); - assertThat(measure.getValue(), anyOf(is(60.0), is(80.0))); - } - } - - @Test - public void shouldNotCalculateDiffValuesOnFiles() { - assertThat(VariationDecorator.shouldCalculateDiffValues(new Project("foo")), is(true)); - assertThat(VariationDecorator.shouldCalculateDiffValues(new JavaPackage("org.foo")), is(true)); - assertThat(VariationDecorator.shouldCalculateDiffValues(new Directory("org/foo")), is(true)); - - assertThat(VariationDecorator.shouldCalculateDiffValues(new JavaFile("org.foo.Bar")), is(false)); - assertThat(VariationDecorator.shouldCalculateDiffValues(new JavaFile("org.foo.Bar", true)), is(false)); - assertThat(VariationDecorator.shouldCalculateDiffValues(new File("org/foo/Bar.php")), is(false)); + public void shouldNotCalculateVariationsOnFiles() { + assertThat(VariationDecorator.shouldCalculateVariations(new Project("foo")), is(true)); + assertThat(VariationDecorator.shouldCalculateVariations(new JavaPackage("org.foo")), is(true)); + assertThat(VariationDecorator.shouldCalculateVariations(new Directory("org/foo")), is(true)); + + assertThat(VariationDecorator.shouldCalculateVariations(new JavaFile("org.foo.Bar")), is(false)); + assertThat(VariationDecorator.shouldCalculateVariations(new JavaFile("org.foo.Bar", true)), is(false)); + assertThat(VariationDecorator.shouldCalculateVariations(new File("org/foo/Bar.php")), is(false)); } - private List<Metric> selectMetrics() { - return getSession().getResults(Metric.class); - } } diff --git a/plugins/sonar-core-plugin/src/test/resources/org/sonar/plugins/core/timemachine/VariationDecoratorTest/shared.xml b/plugins/sonar-core-plugin/src/test/resources/org/sonar/plugins/core/timemachine/PastMeasuresLoaderTest/shared.xml index 69fc7dcba5b..69fc7dcba5b 100644 --- a/plugins/sonar-core-plugin/src/test/resources/org/sonar/plugins/core/timemachine/VariationDecoratorTest/shared.xml +++ b/plugins/sonar-core-plugin/src/test/resources/org/sonar/plugins/core/timemachine/PastMeasuresLoaderTest/shared.xml |