mirror of
https://github.com/alibaba/testable-mock.git
synced 2025-01-10 20:30:11 +08:00
implement member method call substitution
This commit is contained in:
parent
9c5be87f42
commit
83df20c36b
@ -1,7 +1,7 @@
|
|||||||
package com.alibaba.testable.handler;
|
package com.alibaba.testable.handler;
|
||||||
|
|
||||||
import com.alibaba.testable.model.TravelStatus;
|
|
||||||
import com.alibaba.testable.util.ClassUtil;
|
import com.alibaba.testable.util.ClassUtil;
|
||||||
|
import com.alibaba.testable.util.StringUtil;
|
||||||
import org.objectweb.asm.ClassReader;
|
import org.objectweb.asm.ClassReader;
|
||||||
import org.objectweb.asm.ClassWriter;
|
import org.objectweb.asm.ClassWriter;
|
||||||
import org.objectweb.asm.Opcodes;
|
import org.objectweb.asm.Opcodes;
|
||||||
@ -21,6 +21,12 @@ public class TestableClassHandler implements Opcodes {
|
|||||||
|
|
||||||
private static final String CONSTRUCTOR = "<init>";
|
private static final String CONSTRUCTOR = "<init>";
|
||||||
private static final String TESTABLE_NE = "n/e";
|
private static final String TESTABLE_NE = "n/e";
|
||||||
|
private static final String TESTABLE_W = "w";
|
||||||
|
private static final String TESTABLE_F = "f";
|
||||||
|
private static final String CONSTRUCTOR_DESC_PREFIX = "(Ljava/lang/Class;";
|
||||||
|
private static final String METHOD_DESC_PREFIX = "(Ljava/lang/Object;Ljava/lang/String;";
|
||||||
|
private static final String OBJECT_DESC = "Ljava/lang/Object;";
|
||||||
|
private static final String METHOD_DESC_POSTFIX = ")Ljava/lang/Object;";
|
||||||
|
|
||||||
public byte[] getBytes(String className) throws IOException {
|
public byte[] getBytes(String className) throws IOException {
|
||||||
ClassReader cr = new ClassReader(className);
|
ClassReader cr = new ClassReader(className);
|
||||||
@ -46,38 +52,50 @@ public class TestableClassHandler implements Opcodes {
|
|||||||
|
|
||||||
private void transformMethod(ClassNode cn, MethodNode mn, List<String> methodNames) {
|
private void transformMethod(ClassNode cn, MethodNode mn, List<String> methodNames) {
|
||||||
AbstractInsnNode[] instructions = mn.instructions.toArray();
|
AbstractInsnNode[] instructions = mn.instructions.toArray();
|
||||||
TravelStatus status = TravelStatus.INIT;
|
|
||||||
String target = "";
|
|
||||||
int rangeStart = 0;
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
do {
|
do {
|
||||||
if (instructions[i].getOpcode() == Opcodes.NEW) {
|
if (instructions[i].getOpcode() == Opcodes.INVOKESPECIAL) {
|
||||||
TypeInsnNode node = (TypeInsnNode)instructions[i];
|
|
||||||
if (!SYS_CLASSES.contains(node.desc)) {
|
|
||||||
target = node.desc;
|
|
||||||
status = TravelStatus.NEW_REP;
|
|
||||||
rangeStart = i;
|
|
||||||
}
|
|
||||||
} else if (instructions[i].getOpcode() == Opcodes.INVOKESPECIAL) {
|
|
||||||
MethodInsnNode node = (MethodInsnNode)instructions[i];
|
MethodInsnNode node = (MethodInsnNode)instructions[i];
|
||||||
if (methodNames.contains(node.name) && cn.name.equals(node.owner)) {
|
int rangeEnd = i;
|
||||||
status = TravelStatus.MEM_REP;
|
if (cn.name.equals(node.owner) && methodNames.contains(node.name)) {
|
||||||
} else if (TravelStatus.NEW_REP == status && CONSTRUCTOR.equals(node.name) && target.equals(node.owner)) {
|
int rangeStart = getMemberMethodStart(instructions, rangeEnd);
|
||||||
instructions = replaceNewOps(mn, instructions, rangeStart, i);
|
instructions = replaceMemberCallOps(mn, instructions, rangeStart, rangeEnd);
|
||||||
|
i = rangeStart;
|
||||||
|
} else if (CONSTRUCTOR.equals(node.name) && !SYS_CLASSES.contains(node.owner)) {
|
||||||
|
int rangeStart = getConstructorStart(instructions, node.owner, rangeEnd);
|
||||||
|
instructions = replaceNewOps(mn, instructions, rangeStart, rangeEnd);
|
||||||
i = rangeStart;
|
i = rangeStart;
|
||||||
status = TravelStatus.INIT;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
i++;
|
i++;
|
||||||
} while (i < instructions.length);
|
} while (i < instructions.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private int getConstructorStart(AbstractInsnNode[] instructions, String target, int rangeEnd) {
|
||||||
|
for (int i = rangeEnd - 1; i > 0; i--) {
|
||||||
|
if (instructions[i].getOpcode() == Opcodes.NEW && ((TypeInsnNode)instructions[i]).desc.equals(target)) {
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int getMemberMethodStart(AbstractInsnNode[] instructions, int rangeEnd) {
|
||||||
|
for (int i = rangeEnd - 1; i > 0; i--) {
|
||||||
|
if (instructions[i].getOpcode() == Opcodes.ALOAD && ((VarInsnNode)instructions[i]).var == 0) {
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
private AbstractInsnNode[] replaceNewOps(MethodNode mn, AbstractInsnNode[] instructions, int start, int end) {
|
private AbstractInsnNode[] replaceNewOps(MethodNode mn, AbstractInsnNode[] instructions, int start, int end) {
|
||||||
String classType = ((TypeInsnNode)instructions[start]).desc;
|
String classType = ((TypeInsnNode)instructions[start]).desc;
|
||||||
String paramTypes = ((MethodInsnNode)instructions[end]).desc;
|
String constructorDesc = ((MethodInsnNode)instructions[end]).desc;
|
||||||
mn.instructions.insertBefore(instructions[start], new LdcInsnNode(Type.getType("L" + classType + ";")));
|
mn.instructions.insertBefore(instructions[start], new LdcInsnNode(Type.getType("L" + classType + ";")));
|
||||||
InsnList il = new InsnList();
|
InsnList il = new InsnList();
|
||||||
il.add(new MethodInsnNode(INVOKESTATIC, TESTABLE_NE, "w", ClassUtil.generateTargetDesc(paramTypes), false));
|
il.add(new MethodInsnNode(INVOKESTATIC, TESTABLE_NE, TESTABLE_W,
|
||||||
|
getConstructorSubstitutionDesc(constructorDesc), false));
|
||||||
il.add(new TypeInsnNode(CHECKCAST, classType));
|
il.add(new TypeInsnNode(CHECKCAST, classType));
|
||||||
mn.instructions.insertBefore(instructions[end], il);
|
mn.instructions.insertBefore(instructions[end], il);
|
||||||
mn.instructions.remove(instructions[start]);
|
mn.instructions.remove(instructions[start]);
|
||||||
@ -87,4 +105,29 @@ public class TestableClassHandler implements Opcodes {
|
|||||||
return mn.instructions.toArray();
|
return mn.instructions.toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private String getConstructorSubstitutionDesc(String constructorDesc) {
|
||||||
|
int paramCount = ClassUtil.getParameterCount(constructorDesc);
|
||||||
|
return CONSTRUCTOR_DESC_PREFIX + StringUtil.repeat(OBJECT_DESC, paramCount) + METHOD_DESC_POSTFIX;
|
||||||
|
}
|
||||||
|
|
||||||
|
private AbstractInsnNode[] replaceMemberCallOps(MethodNode mn, AbstractInsnNode[] instructions, int start, int end) {
|
||||||
|
String methodDesc = ((MethodInsnNode)instructions[end]).desc;
|
||||||
|
String returnType = ClassUtil.getReturnType(methodDesc);
|
||||||
|
String methodName = ((MethodInsnNode)instructions[end]).name;
|
||||||
|
mn.instructions.insert(instructions[start], new LdcInsnNode(methodName));
|
||||||
|
InsnList il = new InsnList();
|
||||||
|
il.add(new MethodInsnNode(INVOKESTATIC, TESTABLE_NE, TESTABLE_F,
|
||||||
|
getMethodSubstitutionDesc(methodDesc), false));
|
||||||
|
il.add(new TypeInsnNode(CHECKCAST, returnType));
|
||||||
|
mn.instructions.insertBefore(instructions[end], il);
|
||||||
|
mn.instructions.remove(instructions[end]);
|
||||||
|
mn.maxStack += 1;
|
||||||
|
return mn.instructions.toArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getMethodSubstitutionDesc(String methodDesc) {
|
||||||
|
int paramCount = ClassUtil.getParameterCount(methodDesc);
|
||||||
|
return METHOD_DESC_PREFIX + StringUtil.repeat(OBJECT_DESC, paramCount) + METHOD_DESC_POSTFIX;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
package com.alibaba.testable.model;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @author flin
|
|
||||||
*/
|
|
||||||
|
|
||||||
public enum TravelStatus {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Initialized
|
|
||||||
*/
|
|
||||||
INIT,
|
|
||||||
/**
|
|
||||||
* Processing member method replacement
|
|
||||||
*/
|
|
||||||
MEM_REP,
|
|
||||||
/**
|
|
||||||
* Processing member operation replacement
|
|
||||||
*/
|
|
||||||
NEW_REP
|
|
||||||
|
|
||||||
}
|
|
@ -30,7 +30,7 @@ public class ClassUtil {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static String generateTargetDesc(String paramTypes) {
|
public static int getParameterCount(String paramTypes) {
|
||||||
int paramCount = 0;
|
int paramCount = 0;
|
||||||
boolean travelingClass = false;
|
boolean travelingClass = false;
|
||||||
for (byte b : paramTypes.getBytes()) {
|
for (byte b : paramTypes.getBytes()) {
|
||||||
@ -49,15 +49,11 @@ public class ClassUtil {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return "(Ljava/lang/Class;" + repeat("Ljava/lang/Object;", paramCount) + ")Ljava/lang/Object;";
|
return paramCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static String repeat(String text, int times) {
|
public static String getReturnType(String desc) {
|
||||||
StringBuilder sb = new StringBuilder();
|
return null;
|
||||||
for (int i = 0; i < times; i++) {
|
|
||||||
sb.append(text);
|
|
||||||
}
|
|
||||||
return sb.toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,16 @@
|
|||||||
|
package com.alibaba.testable.util;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author flin
|
||||||
|
*/
|
||||||
|
public class StringUtil {
|
||||||
|
|
||||||
|
public static String repeat(String text, int times) {
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
for (int i = 0; i < times; i++) {
|
||||||
|
sb.append(text);
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -9,13 +9,13 @@ class ClassUtilTest {
|
|||||||
@Test
|
@Test
|
||||||
void should_able_to_generate_target_desc() {
|
void should_able_to_generate_target_desc() {
|
||||||
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;)Ljava/lang/Object;",
|
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;)Ljava/lang/Object;",
|
||||||
ClassUtil.generateTargetDesc("(Ljava/lang/String;)V"));
|
ClassUtil.getParameterCount("(Ljava/lang/String;)V"));
|
||||||
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
||||||
ClassUtil.generateTargetDesc("(Ljava/lang/String;IDLjava/lang/String;ZLjava/net/URL;)V"));
|
ClassUtil.getParameterCount("(Ljava/lang/String;IDLjava/lang/String;ZLjava/net/URL;)V"));
|
||||||
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
||||||
ClassUtil.generateTargetDesc("(ZLjava/lang/String;IJFDCSBZ)V"));
|
ClassUtil.getParameterCount("(ZLjava/lang/String;IJFDCSBZ)V"));
|
||||||
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
assertEquals("(Ljava/lang/Class;Ljava/lang/Object;Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;",
|
||||||
ClassUtil.generateTargetDesc("(Ljava/lang/String;[I[Ljava/lang/String;)V"));
|
ClassUtil.getParameterCount("(Ljava/lang/String;[I[Ljava/lang/String;)V"));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user