org.evosuite.coverage.io.output.OutputCoverageGoal Maven / Gradle / Ivy
The newest version!
/**
* Copyright (C) 2010-2018 Gordon Fraser, Andrea Arcuri and EvoSuite
* contributors
*
* This file is part of EvoSuite.
*
* EvoSuite 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.0 of the License, or
* (at your option) any later version.
*
* EvoSuite 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 Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with EvoSuite. If not, see .
*/
package org.evosuite.coverage.io.output;
import org.evosuite.Properties;
import org.evosuite.TestGenerationContext;
import org.evosuite.assertion.Inspector;
import org.evosuite.assertion.InspectorManager;
import org.evosuite.setup.DependencyAnalysis;
import org.evosuite.testcase.TestFitnessFunction;
import org.evosuite.testcase.statements.MethodStatement;
import org.objectweb.asm.Type;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.Serializable;
import java.lang.reflect.Array;
import java.lang.reflect.InvocationTargetException;
import java.util.*;
import static org.evosuite.coverage.io.IOCoverageConstants.*;
/**
* A single output coverage goal.
* Evaluates the value depending on the return type of the method.
*
* @author Gordon Fraser, Andre Mis, Jose Miguel Rojas
*/
public class OutputCoverageGoal implements Serializable, Comparable {
private static final long serialVersionUID = 3539419075883329059L;
private static final Logger logger = LoggerFactory.getLogger(OutputCoverageGoal.class);
private final String className;
private final String methodName;
private final String type;
private final String valueDescriptor;
private final Number numericValue;
/**
* Can be used to create an arbitrary {@code OutputCoverageGoal} trying to cover the
* method such that it returns a given {@code value}
*
*
* If the method returns a boolean, this goal will try to cover the method with either {@code true} or {@code false}
* If the given branch is {@code null}, this goal will try to cover the root branch
* of the method identified by the given name - meaning it will just try to
* call the method at hand
*
*
* Otherwise this goal will try to reach the given branch and if value is
* true, make the branchInstruction jump and visa versa
*
* @param className a {@link java.lang.String} object.
* @param methodName a {@link java.lang.String} object.
* @param type a {@link java.lang.String} object.
* @param valueDescriptor a value descriptor.
*/
public OutputCoverageGoal(String className, String methodName, Type type, String valueDescriptor) {
this(className, methodName, type, valueDescriptor, null);
}
public OutputCoverageGoal(String className, String methodName, Type type, String valueDescriptor, Number numericValue) {
if (className == null || methodName == null)
throw new IllegalArgumentException("null given");
this.className = className;
this.methodName = methodName;
this.type = type.toString();
this.valueDescriptor = valueDescriptor;
this.numericValue = numericValue;
}
/**
* @return the className
*/
public String getClassName() {
return className;
}
/**
* @return the methodName
*/
public String getMethodName() {
return methodName;
}
/**
* @return the type
*/
public Type getType() {
return Type.getType(type);
}
/**
* @return the value
*/
public String getValueDescriptor() {
return valueDescriptor;
}
public Number getNumericValue() { return numericValue; }
// inherited from Object
/**
* {@inheritDoc}
*
* Readable representation
*/
@Override
public String toString() {
return className + "." + methodName + ":" + valueDescriptor;
}
/**
* {@inheritDoc}
*/
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + className.hashCode();
result = prime * result + methodName.hashCode();
result = prime * result + (type == null ? 0 : type.hashCode());
result = prime * result + (valueDescriptor == null ? 0 : valueDescriptor.hashCode());
return result;
}
/**
* {@inheritDoc}
*/
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
OutputCoverageGoal other = (OutputCoverageGoal) obj;
if (!this.methodName.equals(other.methodName) && this.className.equals(other.className))
return false;
if ((this.type == null && other.type != null) || (this.type != null && other.type == null))
return false;
if (type != null && !this.type.equals(other.type))
return false;
if ((this.valueDescriptor == null && other.valueDescriptor != null) || (this.valueDescriptor != null && other.valueDescriptor == null))
return false;
if (valueDescriptor != null && !this.valueDescriptor.equals(other.valueDescriptor))
return false;
return true;
}
@Override
public int compareTo(OutputCoverageGoal o) {
int diff = className.compareTo(o.className);
if (diff == 0) {
int diff2 = methodName.compareTo(o.methodName);
if (diff2 == 0) {
int diff3 = type.compareTo(o.type);
if (diff3 == 0)
return this.valueDescriptor.compareTo(o.valueDescriptor);
else
return diff3;
} else
return diff2;
} else
return diff;
}
public static Set createGoalsFromObject(String className, String methodName, String methodDesc, Object returnValue) {
Set goals = new LinkedHashSet<>();
if (! DependencyAnalysis.isTargetClassName(className))
return goals;
if (methodName.equals("hashCode"))
return goals;
String methodNameWithDesc = methodName + methodDesc;
Type returnType = Type.getReturnType(methodDesc);
switch (returnType.getSort()) {
case Type.BOOLEAN:
String desc = ((boolean) returnValue) ? BOOL_TRUE : BOOL_FALSE;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, desc));
break;
case Type.CHAR:
char c = (char) returnValue;
if (Character.isAlphabetic(c))
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, CHAR_ALPHA));
else if (Character.isDigit(c))
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, CHAR_DIGIT));
else
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, CHAR_OTHER));
break;
case Type.BYTE:
case Type.SHORT:
case Type.INT:
case Type.FLOAT:
case Type.LONG:
case Type.DOUBLE:
assert (returnValue instanceof Number);
if(isJavaNumber(returnValue)) {
double value = ((Number) returnValue).doubleValue();
String numDesc = (value < 0) ? NUM_NEGATIVE : (value == 0) ? NUM_ZERO : NUM_POSITIVE;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, numDesc, (Number)returnValue));
}
break;
case Type.ARRAY:
if (returnValue == null)
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, REF_NULL));
else {
String arrDesc = (Array.getLength(returnValue) == 0) ? ARRAY_EMPTY : ARRAY_NONEMPTY;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, arrDesc));
}
break;
case Type.OBJECT:
if (returnValue == null)
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, REF_NULL));
else {
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, REF_NONNULL));
if (returnType.getClassName().equals("java.lang.String")) {
String valDesc = ((String)returnValue).isEmpty() ? STRING_EMPTY : STRING_NONEMPTY;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, valDesc));
break;
}
for(Inspector inspector : InspectorManager.getInstance().getInspectors(returnValue.getClass())) {
String insp = inspector.getMethodCall() + Type.getMethodDescriptor(inspector.getMethod());
try {
Object val = inspector.getValue(returnValue);
if (val instanceof Boolean) {
String valDesc = ((boolean)val) ? BOOL_TRUE : BOOL_FALSE;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, REF_NONNULL + ":" + returnType.getClassName() + ":" + insp + ":" + valDesc));
} else if (isJavaNumber(val)) {
double dv = ((Number) val).doubleValue();
String valDesc = (dv < 0) ? NUM_NEGATIVE : (dv == 0) ? NUM_ZERO : NUM_POSITIVE;
goals.add(new OutputCoverageGoal(className, methodNameWithDesc, returnType, REF_NONNULL + ":" + returnType.getClassName() + ":" + insp + ":" + valDesc));
}
} catch (InvocationTargetException e) {
// Exceptions in inspectors can happen
logger.debug(e.getMessage(), e);
} catch(IllegalAccessException e) {
logger.warn(e.getMessage(), e);
}
}
}
break;
default:
// IGNORE
// TODO: what to do with the sort for METHOD?
break;
}
return goals;
}
/**
* The SUT could have classes extending Number. Calling doubleValue()
* on those would lead to many problems, like for example security and
* loop counter checks.
*
* @param val
* @return
*/
private static boolean isJavaNumber(Object val){
return val instanceof Number && val.getClass().getName().startsWith("java.");
}
// private void writeObject(ObjectOutputStream oos) throws IOException {
// oos.defaultWriteObject();
// // Write/save additional fields
// if (branch != null)
// oos.writeInt(branch.getActualBranchId());
// else
// oos.writeInt(-1);
// }
//
// // assumes "static java.util.Date aDate;" declared
// private void readObject(ObjectInputStream ois) throws ClassNotFoundException,
// IOException {
// ois.defaultReadObject();
//
// int branchId = ois.readInt();
// if (branchId >= 0)
// this.branch = BranchPool.getBranch(branchId);
// else
// this.branch = null;
// }
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy