org.conqat.engine.sourcecode.coverage.DecisionProbe Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of teamscale-commons Show documentation
Show all versions of teamscale-commons Show documentation
Provides common DTOs for Teamscale
package org.conqat.engine.sourcecode.coverage;
import org.conqat.lib.commons.test.IndexValueClass;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import org.conqat.lib.commons.assertion.CCSMAssert;
import org.conqat.lib.commons.collections.CollectionUtils;
import org.conqat.lib.commons.collections.UnmodifiableList;
import org.conqat.lib.commons.js_export.ExportToTypeScript;
import org.conqat.lib.commons.string.StringUtils;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonProperty;
/**
* Probe for a decision, e.g. an "if" statement, which carries two execution
* counters
*/
@ExportToTypeScript
@IndexValueClass(containedInBackup = true)
public class DecisionProbe extends CoverageProbeBase {
private static final long serialVersionUID = 1L;
@JsonProperty("trueExecutionCount")
private int trueExecutionCount;
@JsonProperty("falseExecutionCount")
private int falseExecutionCount;
@JsonProperty("configurations")
private final List configurations = new ArrayList<>();
@JsonProperty("conditions")
private final List conditions = new ArrayList<>();
@JsonCreator
public DecisionProbe(@JsonProperty("line") int line, @JsonProperty("trueExecutionCount") int trueExecutionCount,
@JsonProperty("falseExecutionCount") int falseExecutionCount) {
super(line);
this.trueExecutionCount = trueExecutionCount;
this.falseExecutionCount = falseExecutionCount;
}
/** @see #falseExecutionCount */
public int getFalseExecutionCount() {
return falseExecutionCount;
}
/** @see #trueExecutionCount */
public int getTrueExecutionCount() {
return trueExecutionCount;
}
/** Adds the given configuration to this probe */
public void addConfiguration(DecisionProbeConfiguration configuration) {
configurations.add(configuration);
}
/** Adds the given condition to this probe */
public void addCondition(McDcCondition condition) {
conditions.add(condition);
}
/** @see #conditions */
public UnmodifiableList getConditions() {
return CollectionUtils.asUnmodifiable(conditions);
}
/** @see #configurations */
public UnmodifiableList getConfigurations() {
return CollectionUtils.asUnmodifiable(configurations);
}
@Override
public String toString() {
StringBuilder result = new StringBuilder("Decision in line " + getLine() + "\n");
result.append(StringUtils.concat(configurations, "\n"));
result.append(StringUtils.concat(conditions, "\n"));
return result.toString();
}
@Override
@JsonIgnore
public int getCoverableCount() {
// CTC++ counts one coverable point for each possible outcome of the overall
// decision (true/false) + one coverable point for each atomic condition
return 2 + getConditions().size();
}
@Override
@JsonIgnore
public int getCoveredCount() {
int coveredCount = 0;
if (trueExecutionCount > 0) {
coveredCount++;
}
if (falseExecutionCount > 0) {
coveredCount++;
}
for (McDcCondition condition : conditions) {
if (condition.isFulfilled()) {
coveredCount++;
}
}
return coveredCount;
}
@Override
public boolean mergeWith(CoverageProbeBase other) {
CCSMAssert.isInstanceOf(other, DecisionProbe.class);
DecisionProbe sourceDecisionProbe = (DecisionProbe) other;
trueExecutionCount += sourceDecisionProbe.trueExecutionCount;
falseExecutionCount += sourceDecisionProbe.falseExecutionCount;
mergeIntoList(configurations, sourceDecisionProbe.configurations,
DecisionProbeConfiguration::haveSameDescriptionAndDecisionValue,
(target, source) -> target.increaseExecutionCount(source.getExecutionCount()));
mergeIntoList(conditions, sourceDecisionProbe.conditions,
(target, source) -> target.getDescription().equals(source.getDescription()),
(target, source) -> target.setFulfilled(target.isFulfilled() || source.isFulfilled()));
// we deem merging of decision probes to be unsafe, as we do not really parse
// the content of the probes, but rather use string equality only
return false;
}
/**
* Merges all elements from source into target. For this a matching element
* w.r.t. equalsChecker is searched. If this is found, the merge function is
* applied, otherwise the element is just added to the target list.
*/
private static void mergeIntoList(List targets, List sources, BiFunction equalsChecker,
BiConsumer targetSourceMerger) {
for (T source : sources) {
Optional match = targets.stream().filter(target -> equalsChecker.apply(source, target)).findFirst();
if (!match.isPresent()) {
targets.add(source);
} else {
targetSourceMerger.accept(match.get(), source);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy