org.conqat.engine.sourcecode.coverage.ProbeCoverageInfo Maven / Gradle / Ivy
/*
* Copyright (c) CQSE GmbH
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.conqat.engine.sourcecode.coverage;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;
import org.conqat.lib.commons.collections.CollectionUtils;
import org.conqat.lib.commons.collections.UnmodifiableList;
import org.conqat.lib.commons.test.IndexValueClass;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import io.swagger.v3.oas.annotations.media.ArraySchema;
import io.swagger.v3.oas.annotations.media.Schema;
/**
* Holds coverage information for a file represented as "probes". See {@link CoverageProbeBase} for
* a description of probes.
*
* We use this object both to represent covered and coverable information. The flag
* {@link #uploaded} differs between the two cases.
*/
@IndexValueClass(containedInBackup = true)
public class ProbeCoverageInfo implements Serializable {
private static final long serialVersionUID = 1L;
/** The name of the JSON property name for {@link #probes}. */
private static final String PROBES_PROPERTY = "probes";
/** The name of the JSON property name for {@link #uploaded}. */
private static final String UPLOADED_PROPERTY = "uploaded";
@JsonProperty(PROBES_PROPERTY)
@Schema(implementation = Probes.class)
private final ArrayList probes;
/**
* Signals whether this is coverage info uploaded externally. See also class comment
*/
@JsonProperty(UPLOADED_PROPERTY)
private final boolean uploaded;
public ProbeCoverageInfo(boolean uploaded) {
this.probes = new ArrayList<>();
this.uploaded = uploaded;
}
@JsonCreator
public ProbeCoverageInfo(@JsonProperty(UPLOADED_PROPERTY) boolean uploaded,
@JsonProperty(PROBES_PROPERTY) Collection probes) {
this(uploaded);
this.probes.addAll(probes);
}
/** Constructor. */
public ProbeCoverageInfo(ProbeCoverageInfo coverageProbeInfo) {
this.probes = new ArrayList<>(coverageProbeInfo.probes);
this.uploaded = coverageProbeInfo.uploaded;
}
/** Adds the given probe */
public void addProbe(CoverageProbeBase probe) {
probes.add(probe);
}
public UnmodifiableList getProbes() {
return CollectionUtils.asUnmodifiable(probes);
}
/** Overwrites the probes of this coverage info. */
public void setProbes(List probes) {
this.probes.clear();
this.probes.addAll(probes);
}
/** @see #uploaded */
public boolean isUploaded() {
return uploaded;
}
/** Returns the number of coverable entities of this probe coverage */
public int getCoverableCount() {
int coverableCount = 0;
for (CoverageProbeBase probe : probes) {
coverableCount += probe.getCoverableCount();
}
return coverableCount;
}
/** Returns the number of covered entities of this probe coverage */
public int getCoveredCount() {
int coveredCount = 0;
for (CoverageProbeBase probe : probes) {
coveredCount += probe.getCoveredCount();
}
return coveredCount;
}
/**
* Merges the results from the other coverage into this one. This is done by comparing the content
* of {@link #probes} where the same type and line is used and merging those probes. Note that probe
* merging is non-trivial and hence error-prone, unless in very simple cases.
*
* This operation is "almost" symmetric, i.e. it does not really matter, what is merged into what,
* but the this
instance will be modified to store the result.
*
* @return whether the merge was safe, i.e. if this returns true
we are sure that the
* result of the merge was correct, otherwise this might be off.
*/
public boolean mergeWith(ProbeCoverageInfo otherCoverage) {
if (otherCoverage.probes.isEmpty()) {
return true;
}
if (probes.isEmpty()) {
probes.addAll(otherCoverage.probes);
return true;
}
return mergeProbes(otherCoverage.probes, probes);
}
/**
* Merges the given source probes into the target probes (i.e. the target collection will be
* modified).
*
* @return whether the merge was safe, i.e. if this returns true
we are sure that the
* result of the merge was correct, otherwise this might be off.
*/
private static boolean mergeProbes(Collection sourceProbes,
Collection targetProbes) {
Map> targetProbesByLine = targetProbes.stream()
.collect(Collectors.groupingBy(CoverageProbeBase::getLine));
boolean safe = true;
for (CoverageProbeBase sourceProbe : sourceProbes) {
List targetProbeCandidates = targetProbesByLine.get(sourceProbe.getLine());
Optional matchedProbe = findFirstMatchingProbe(sourceProbe, targetProbeCandidates);
if (matchedProbe.isPresent()) {
safe = matchedProbe.get().mergeWith(sourceProbe) && safe;
} else {
targetProbes.add(sourceProbe);
}
}
return safe;
}
private static Optional findFirstMatchingProbe(CoverageProbeBase sourceProbe,
List targetCandidates) {
if (targetCandidates == null) {
return Optional.empty();
}
return targetCandidates.stream().filter(targetProbe -> targetProbe.getClass() == sourceProbe.getClass())
.findFirst();
}
/*
* The following interfaces are required to correctly generate the OpenAPI spec. Otherwise, it is
* not possible to set the correct value type of the child metrics map.
*
* See: https://github.com/swagger-api/swagger-core/issues/3080
*
* See: https://stackoverflow.com/a/68752621
*/
@Schema(oneOf = { DecisionProbe.class, SimpleStatementCoverageProbe.class })
public interface Probe {
}
/** Describes a list of metric values for the OpenAPI specification. */
@ArraySchema(schema = @Schema(implementation = Probe.class))
public interface Probes extends List {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy