From 8278fd360ef3eda3652068697f1442fc5b8c10e7 Mon Sep 17 00:00:00 2001 From: Raffi Khatchadourian Date: Tue, 23 Apr 2024 15:40:58 -0400 Subject: [PATCH 1/3] Move variable closer. --- .../callgraph/PythonTrampolineTargetSelector.java | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java index 5ea98489..17eadc16 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java @@ -117,13 +117,6 @@ public IMethod getCalleeTarget(CGNode caller, CallSiteReference site, IClass rec PythonSummary x = new PythonSummary(tr, call.getNumberOfTotalParameters()); IClass filter = ((PythonInstanceMethodTrampoline) receiver).getRealClass(); - // Are we calling a static method? - boolean staticMethodReceiver = filter.getAnnotations().contains(make(STATIC_METHOD)); - logger.fine( - staticMethodReceiver - ? "Found static method receiver: " + filter - : "Method is not static: " + filter); - int v = call.getNumberOfTotalParameters() + 1; x.addStatement( @@ -145,6 +138,13 @@ public IMethod getCalleeTarget(CGNode caller, CallSiteReference site, IClass rec int v1; + // Are we calling a static method? + boolean staticMethodReceiver = filter.getAnnotations().contains(make(STATIC_METHOD)); + logger.fine( + staticMethodReceiver + ? "Found static method receiver: " + filter + : "Method is not static: " + filter); + // only add self if the receiver isn't static. if (!staticMethodReceiver) { v1 = v + 2; From 54f03c0bcfca9483df28dbeea00e61a54d064d15 Mon Sep 17 00:00:00 2001 From: Raffi Khatchadourian Date: Thu, 25 Apr 2024 12:38:59 -0400 Subject: [PATCH 2/3] Support class methods (#100) https://docs.python.org/3/library/functions.html#classmethod * Add comment. For https://github.com/wala/ML/issues/107. * Add javadoc. --- .../wala/cast/python/parser/PythonParser.java | 9 +- .../python/ml/test/TestTensorflow2Model.java | 25 ++ .../data/tf2_test_class_method.py | 11 + .../data/tf2_test_class_method2.py | 11 + .../data/tf2_test_class_method3.py | 15 + .../data/tf2_test_class_method4.py | 15 + .../data/tf2_test_class_method5.py | 15 + .../python/client/PythonAnalysisEngine.java | 10 +- ...onClassMethodTrampolineTargetSelector.java | 131 +++++++++ .../PythonConstructorTargetSelector.java | 25 ++ ...stanceMethodTrampolineTargetSelector.java} | 258 ++++++++++-------- .../PythonMethodTrampolineTargetSelector.java | 118 ++++++++ .../wala/cast/python/types/PythonTypes.java | 6 + .../com/ibm/wala/cast/python/types/Util.java | 53 ++++ .../com/ibm/wala/cast/python/util/Util.java | 31 +++ 15 files changed, 606 insertions(+), 127 deletions(-) create mode 100644 com.ibm.wala.cast.python.test/data/tf2_test_class_method.py create mode 100644 com.ibm.wala.cast.python.test/data/tf2_test_class_method2.py create mode 100644 com.ibm.wala.cast.python.test/data/tf2_test_class_method3.py create mode 100644 com.ibm.wala.cast.python.test/data/tf2_test_class_method4.py create mode 100644 com.ibm.wala.cast.python.test/data/tf2_test_class_method5.py create mode 100644 com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonClassMethodTrampolineTargetSelector.java rename com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/{PythonTrampolineTargetSelector.java => PythonInstanceMethodTrampolineTargetSelector.java} (63%) create mode 100644 com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonMethodTrampolineTargetSelector.java diff --git a/com.ibm.wala.cast.python.jython3/source/com/ibm/wala/cast/python/parser/PythonParser.java b/com.ibm.wala.cast.python.jython3/source/com/ibm/wala/cast/python/parser/PythonParser.java index 4ae07fc6..fd3b7ff8 100644 --- a/com.ibm.wala.cast.python.jython3/source/com/ibm/wala/cast/python/parser/PythonParser.java +++ b/com.ibm.wala.cast.python.jython3/source/com/ibm/wala/cast/python/parser/PythonParser.java @@ -10,6 +10,7 @@ *****************************************************************************/ package com.ibm.wala.cast.python.parser; +import static com.ibm.wala.cast.python.util.Util.CLASS_METHOD_ANNOTATION_NAME; import static com.ibm.wala.cast.python.util.Util.DYNAMIC_ANNOTATION_KEY; import static com.ibm.wala.cast.python.util.Util.STATIC_METHOD_ANNOTATION_NAME; import static com.ibm.wala.cast.python.util.Util.getNameStream; @@ -1263,8 +1264,12 @@ public int getKind() { @Override public CAstNode getAST() { if (function instanceof FunctionDef) { - // Only add object metadata for non-static methods. - if (isMethod && !staticMethod) { + + boolean classMethod = + getNameStream(annotations).anyMatch(s -> s.equals(CLASS_METHOD_ANNOTATION_NAME)); + + // Only add object metadata for non-static and non-class methods. + if (isMethod && !staticMethod && !classMethod) { CAst Ast = PythonParser.this.Ast; CAstNode[] newNodes = new CAstNode[nodes.length + 2]; diff --git a/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java b/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java index ed9c5a2f..b1cb904a 100644 --- a/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java +++ b/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java @@ -3500,6 +3500,31 @@ public void testStaticMethod12() throws ClassHierarchyException, CancelException expectedTensorParameterValueNumbers); } + @Test + public void testClassMethod() throws ClassHierarchyException, CancelException, IOException { + test("tf2_test_class_method.py", "MyClass.the_class_method", 1, 1, 3); + } + + @Test + public void testClassMethod2() throws ClassHierarchyException, CancelException, IOException { + test("tf2_test_class_method2.py", "MyClass.the_class_method", 1, 1, 3); + } + + @Test + public void testClassMethod3() throws ClassHierarchyException, CancelException, IOException { + test("tf2_test_class_method3.py", "MyClass.f", 1, 1, 2); + } + + @Test + public void testClassMethod4() throws ClassHierarchyException, CancelException, IOException { + test("tf2_test_class_method4.py", "MyClass.f", 1, 1, 2); + } + + @Test + public void testClassMethod5() throws ClassHierarchyException, CancelException, IOException { + test("tf2_test_class_method5.py", "MyClass.f", 1, 1, 2); + } + private void test( String filename, String functionName, diff --git a/com.ibm.wala.cast.python.test/data/tf2_test_class_method.py b/com.ibm.wala.cast.python.test/data/tf2_test_class_method.py new file mode 100644 index 00000000..310569a5 --- /dev/null +++ b/com.ibm.wala.cast.python.test/data/tf2_test_class_method.py @@ -0,0 +1,11 @@ +import tensorflow as tf + + +class MyClass: + + @classmethod + def the_class_method(cls, x): + assert isinstance(x, tf.Tensor) + + +MyClass.the_class_method(tf.constant(1)) diff --git a/com.ibm.wala.cast.python.test/data/tf2_test_class_method2.py b/com.ibm.wala.cast.python.test/data/tf2_test_class_method2.py new file mode 100644 index 00000000..e968a7a0 --- /dev/null +++ b/com.ibm.wala.cast.python.test/data/tf2_test_class_method2.py @@ -0,0 +1,11 @@ +import tensorflow as tf + + +class MyClass: + + @classmethod + def the_class_method(cls, x): + assert isinstance(x, tf.Tensor) + + +MyClass().the_class_method(tf.constant(1)) diff --git a/com.ibm.wala.cast.python.test/data/tf2_test_class_method3.py b/com.ibm.wala.cast.python.test/data/tf2_test_class_method3.py new file mode 100644 index 00000000..629dbb84 --- /dev/null +++ b/com.ibm.wala.cast.python.test/data/tf2_test_class_method3.py @@ -0,0 +1,15 @@ +import tensorflow as tf + + +class MyClass: + + def f(x): + assert isinstance(x, tf.Tensor) + + @classmethod + def the_class_method(cls, x): + assert isinstance(x, tf.Tensor) + cls.f(x) + + +MyClass().the_class_method(tf.constant(1)) diff --git a/com.ibm.wala.cast.python.test/data/tf2_test_class_method4.py b/com.ibm.wala.cast.python.test/data/tf2_test_class_method4.py new file mode 100644 index 00000000..aca19cee --- /dev/null +++ b/com.ibm.wala.cast.python.test/data/tf2_test_class_method4.py @@ -0,0 +1,15 @@ +import tensorflow as tf + + +class MyClass: + + def f(x): + assert isinstance(x, tf.Tensor) + + @classmethod + def the_class_method(cls, x): + assert isinstance(x, tf.Tensor) + cls.f(x) + + +MyClass.the_class_method(tf.constant(1)) diff --git a/com.ibm.wala.cast.python.test/data/tf2_test_class_method5.py b/com.ibm.wala.cast.python.test/data/tf2_test_class_method5.py new file mode 100644 index 00000000..aca19cee --- /dev/null +++ b/com.ibm.wala.cast.python.test/data/tf2_test_class_method5.py @@ -0,0 +1,15 @@ +import tensorflow as tf + + +class MyClass: + + def f(x): + assert isinstance(x, tf.Tensor) + + @classmethod + def the_class_method(cls, x): + assert isinstance(x, tf.Tensor) + cls.f(x) + + +MyClass.the_class_method(tf.constant(1)) diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/client/PythonAnalysisEngine.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/client/PythonAnalysisEngine.java index 5317344e..10bd1053 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/client/PythonAnalysisEngine.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/client/PythonAnalysisEngine.java @@ -6,10 +6,11 @@ import com.ibm.wala.cast.ipa.callgraph.AstContextInsensitiveSSAContextInterpreter; import com.ibm.wala.cast.ir.ssa.AstIRFactory; import com.ibm.wala.cast.loader.AstDynamicField; +import com.ibm.wala.cast.python.ipa.callgraph.PythonClassMethodTrampolineTargetSelector; import com.ibm.wala.cast.python.ipa.callgraph.PythonConstructorTargetSelector; +import com.ibm.wala.cast.python.ipa.callgraph.PythonInstanceMethodTrampolineTargetSelector; import com.ibm.wala.cast.python.ipa.callgraph.PythonSSAPropagationCallGraphBuilder; import com.ibm.wala.cast.python.ipa.callgraph.PythonScopeMappingInstanceKeys; -import com.ibm.wala.cast.python.ipa.callgraph.PythonTrampolineTargetSelector; import com.ibm.wala.cast.python.ipa.summaries.BuiltinFunctions; import com.ibm.wala.cast.python.ipa.summaries.PythonComprehensionTrampolines; import com.ibm.wala.cast.python.ipa.summaries.PythonSuper; @@ -299,9 +300,10 @@ public boolean isReferenceType() { protected void addBypassLogic(IClassHierarchy cha, AnalysisOptions options) { options.setSelector( - new PythonTrampolineTargetSelector( - new PythonConstructorTargetSelector( - new PythonComprehensionTrampolines(options.getMethodTargetSelector())), + new PythonInstanceMethodTrampolineTargetSelector( + new PythonClassMethodTrampolineTargetSelector( + new PythonConstructorTargetSelector( + new PythonComprehensionTrampolines(options.getMethodTargetSelector()))), this)); BuiltinFunctions builtins = new BuiltinFunctions(cha); diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonClassMethodTrampolineTargetSelector.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonClassMethodTrampolineTargetSelector.java new file mode 100644 index 00000000..519d1695 --- /dev/null +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonClassMethodTrampolineTargetSelector.java @@ -0,0 +1,131 @@ +/****************************************************************************** + * Copyright (c) 2018 IBM Corporation. + * All rights reserved. This program and the accompanying materials + * are made available under the terms of the Eclipse Public License v1.0 + * which accompanies this distribution, and is available at + * http://www.eclipse.org/legal/epl-v10.html + * + * Contributors: + * IBM Corporation - initial API and implementation + *****************************************************************************/ +package com.ibm.wala.cast.python.ipa.callgraph; + +import static com.ibm.wala.cast.python.types.Util.getGlobalName; +import static com.ibm.wala.cast.python.types.Util.makeGlobalRef; +import static com.ibm.wala.cast.python.util.Util.isClassMethod; + +import com.ibm.wala.cast.ir.ssa.AstGlobalRead; +import com.ibm.wala.cast.loader.DynamicCallSiteReference; +import com.ibm.wala.cast.python.ipa.summaries.PythonSummary; +import com.ibm.wala.cast.python.ir.PythonLanguage; +import com.ibm.wala.cast.python.ssa.PythonInvokeInstruction; +import com.ibm.wala.cast.python.types.PythonTypes; +import com.ibm.wala.classLoader.CallSiteReference; +import com.ibm.wala.classLoader.IClass; +import com.ibm.wala.core.util.strings.Atom; +import com.ibm.wala.ipa.callgraph.CGNode; +import com.ibm.wala.ipa.callgraph.MethodTargetSelector; +import com.ibm.wala.ipa.cha.IClassHierarchy; +import com.ibm.wala.ssa.SSAInstructionFactory; +import com.ibm.wala.ssa.SSAReturnInstruction; +import com.ibm.wala.types.FieldReference; +import com.ibm.wala.util.collections.HashMapFactory; +import com.ibm.wala.util.collections.Pair; +import java.util.Map; +import java.util.logging.Logger; + +/** + * A trampoline for class + * methods that are not called using object instances. + * + * @author Raffi Khatchadourian + */ +public class PythonClassMethodTrampolineTargetSelector + extends PythonMethodTrampolineTargetSelector { + + protected static final Logger LOGGER = + Logger.getLogger(PythonClassMethodTrampolineTargetSelector.class.getName()); + + public PythonClassMethodTrampolineTargetSelector(MethodTargetSelector base) { + super(base); + } + + @Override + protected boolean shouldProcess(CGNode caller, CallSiteReference site, IClass receiver) { + IClassHierarchy cha = receiver.getClassHierarchy(); + + // Are we calling a class method? + boolean classMethodReceiver = isClassMethod(receiver); + + // Is the caller a trampoline? + boolean trampoline = + caller + .getMethod() + .getSelector() + .getName() + .startsWith(Atom.findOrCreateAsciiAtom("trampoline")); + + return classMethodReceiver + && !cha.isSubclassOf(receiver, cha.lookupClass(PythonTypes.trampoline)) + && !trampoline; + } + + @SuppressWarnings("unchecked") + @Override + protected void populate( + PythonSummary x, int v, IClass receiver, PythonInvokeInstruction call, Logger logger) { + Map names = HashMapFactory.make(); + SSAInstructionFactory insts = PythonLanguage.Python.instructionFactory(); + + // Read the class from the global scope. + String globalName = getGlobalName(receiver.getReference()); + FieldReference globalRef = makeGlobalRef(receiver.getClassLoader(), globalName); + int globalReadRes = v++; + int pc = 0; + + x.addStatement(new AstGlobalRead(pc++, globalReadRes, globalRef)); + + int getInstRes = v++; + + // Read the field from the class corresponding to the called method. + FieldReference method = + FieldReference.findOrCreate( + PythonTypes.Root, Atom.findOrCreateUnicodeAtom("the_class_method"), PythonTypes.Root); + + x.addStatement(insts.GetInstruction(pc++, getInstRes, globalReadRes, method)); + + int i = 0; + int paramSize = Math.max(2, call.getNumberOfPositionalParameters() + 1); + int[] params = new int[paramSize]; + params[i++] = getInstRes; + params[i++] = globalReadRes; + + for (int j = 1; j < call.getNumberOfPositionalParameters(); j++) params[i++] = j + 1; + + int ki = 0, ji = call.getNumberOfPositionalParameters() + 1; + Pair[] keys = new Pair[0]; + + if (call.getKeywords() != null) { + keys = new Pair[call.getKeywords().size()]; + + for (String k : call.getKeywords()) { + names.put(ji, Atom.findOrCreateUnicodeAtom(k)); + keys[ki++] = Pair.make(k, ji++); + } + } + + CallSiteReference ref = new DynamicCallSiteReference(call.getCallSite().getDeclaredTarget(), 2); + + int except = v++; + int invokeResult = v++; + + x.addStatement(new PythonInvokeInstruction(pc++, invokeResult, except, ref, params, keys)); + x.addStatement(new SSAReturnInstruction(pc++, invokeResult, false)); + x.setValueNames(names); + } + + @Override + protected Logger getLogger() { + return LOGGER; + } +} diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonConstructorTargetSelector.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonConstructorTargetSelector.java index 72151d80..6312868c 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonConstructorTargetSelector.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonConstructorTargetSelector.java @@ -10,6 +10,10 @@ *****************************************************************************/ package com.ibm.wala.cast.python.ipa.callgraph; +import static com.ibm.wala.cast.python.types.Util.getGlobalName; +import static com.ibm.wala.cast.python.types.Util.makeGlobalRef; + +import com.ibm.wala.cast.ir.ssa.AstGlobalRead; import com.ibm.wala.cast.loader.DynamicCallSiteReference; import com.ibm.wala.cast.python.ipa.summaries.PythonInstanceMethodTrampoline; import com.ibm.wala.cast.python.ipa.summaries.PythonSummarizedFunction; @@ -137,6 +141,27 @@ public IMethod getCalleeTarget(CGNode caller, CallSiteReference site, IClass rec PythonTypes.Root))); pc++; + // Add a metadata variable that refers to the declaring class. + // NOTE: Per https://docs.python.org/3/library/functions.html#classmethod, "[i]f a class + // method is called for a derived class, the derived class object is passed as the + // implied first argument." I'm unsure whether `receiver` can refer to the derived + // class especially in light of https://github.com/wala/ML/issues/107. + int classVar = v++; + String globalName = getGlobalName(r); + FieldReference globalRef = makeGlobalRef(receiver.getClassLoader(), globalName); + + ctor.addStatement(new AstGlobalRead(pc++, classVar, globalRef)); + + ctor.addStatement( + insts.PutInstruction( + pc++, + f, + classVar, + FieldReference.findOrCreate( + PythonTypes.Root, + Atom.findOrCreateUnicodeAtom("$class"), + PythonTypes.Root))); + ctor.addStatement( insts.PutInstruction( pc, diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonInstanceMethodTrampolineTargetSelector.java similarity index 63% rename from com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java rename to com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonInstanceMethodTrampolineTargetSelector.java index 17eadc16..90cb5fe9 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonTrampolineTargetSelector.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonInstanceMethodTrampolineTargetSelector.java @@ -11,18 +11,18 @@ package com.ibm.wala.cast.python.ipa.callgraph; import static com.ibm.wala.cast.python.types.PythonTypes.STATIC_METHOD; +import static com.ibm.wala.cast.python.types.Util.getDeclaringClassTypeReference; +import static com.ibm.wala.cast.python.util.Util.isClassMethod; import static com.ibm.wala.types.annotations.Annotation.make; import com.ibm.wala.cast.ipa.callgraph.ScopeMappingInstanceKeys.ScopeMappingInstanceKey; import com.ibm.wala.cast.loader.DynamicCallSiteReference; import com.ibm.wala.cast.python.client.PythonAnalysisEngine; import com.ibm.wala.cast.python.ipa.summaries.PythonInstanceMethodTrampoline; -import com.ibm.wala.cast.python.ipa.summaries.PythonSummarizedFunction; import com.ibm.wala.cast.python.ipa.summaries.PythonSummary; import com.ibm.wala.cast.python.ir.PythonLanguage; import com.ibm.wala.cast.python.ssa.PythonInvokeInstruction; import com.ibm.wala.cast.python.types.PythonTypes; -import com.ibm.wala.cast.types.AstMethodReference; import com.ibm.wala.classLoader.CallSiteReference; import com.ibm.wala.classLoader.IClass; import com.ibm.wala.classLoader.IMethod; @@ -38,7 +38,6 @@ import com.ibm.wala.ssa.SSAReturnInstruction; import com.ibm.wala.types.ClassLoaderReference; import com.ibm.wala.types.FieldReference; -import com.ibm.wala.types.MethodReference; import com.ibm.wala.types.TypeName; import com.ibm.wala.types.TypeReference; import com.ibm.wala.util.collections.HashMapFactory; @@ -47,10 +46,11 @@ import java.util.Map; import java.util.logging.Logger; -public class PythonTrampolineTargetSelector implements MethodTargetSelector { +public class PythonInstanceMethodTrampolineTargetSelector + extends PythonMethodTrampolineTargetSelector { private static final Logger logger = - Logger.getLogger(PythonTrampolineTargetSelector.class.getName()); + Logger.getLogger(PythonInstanceMethodTrampolineTargetSelector.class.getName()); /** * The method name that is used for Python callables. @@ -70,139 +70,140 @@ public class PythonTrampolineTargetSelector implements MethodTargetSelector { */ private static final String CALLABLE_METHOD_NAME_FOR_KERAS_MODELS = "call"; - private final MethodTargetSelector base; - private PythonAnalysisEngine engine; - public PythonTrampolineTargetSelector( + public PythonInstanceMethodTrampolineTargetSelector( MethodTargetSelector base, PythonAnalysisEngine pythonAnalysisEngine) { - this.base = base; + super(base); this.engine = pythonAnalysisEngine; } - private final Map, IMethod> codeBodies = HashMapFactory.make(); + @Override + protected boolean shouldProcess(CGNode caller, CallSiteReference site, IClass receiver) { + IClassHierarchy cha = receiver.getClassHierarchy(); + return cha.isSubclassOf(receiver, cha.lookupClass(PythonTypes.trampoline)) + || this.isCallable(receiver); + } - @SuppressWarnings("unchecked") @Override public IMethod getCalleeTarget(CGNode caller, CallSiteReference site, IClass receiver) { - if (receiver != null) { - logger.fine("Getting callee target for receiver: " + receiver); - logger.fine("Calling method name is: " + caller.getMethod().getName()); - - IClassHierarchy cha = receiver.getClassHierarchy(); - final boolean callable = receiver.getReference().equals(PythonTypes.object); - - if (cha.isSubclassOf(receiver, cha.lookupClass(PythonTypes.trampoline)) || callable) { - PythonInvokeInstruction call = (PythonInvokeInstruction) caller.getIR().getCalls(site)[0]; + if (isCallable(receiver)) { + logger.fine("Encountered callable."); - if (callable) { - logger.fine("Encountered callable."); + PythonInvokeInstruction call = this.getCall(caller, site); - // It's a callable. Change the receiver. - receiver = getCallable(caller, cha, call); + // It's a callable. Change the receiver. + receiver = getCallable(caller, receiver.getClassHierarchy(), call); - if (receiver == null) return null; // not found. - else logger.fine("Substituting the receiver with one derived from a callable."); - } + if (receiver == null) return null; // not found. + else logger.fine("Substituting the receiver with one derived from a callable."); + } - Pair key = Pair.make(receiver, call.getNumberOfTotalParameters()); - - if (!codeBodies.containsKey(key)) { - Map names = HashMapFactory.make(); - MethodReference tr = - MethodReference.findOrCreate( - receiver.getReference(), - Atom.findOrCreateUnicodeAtom("trampoline" + call.getNumberOfTotalParameters()), - AstMethodReference.fnDesc); - PythonSummary x = new PythonSummary(tr, call.getNumberOfTotalParameters()); - IClass filter = ((PythonInstanceMethodTrampoline) receiver).getRealClass(); - - int v = call.getNumberOfTotalParameters() + 1; - - x.addStatement( - PythonLanguage.Python.instructionFactory() - .GetInstruction( - 0, - v, - 1, - FieldReference.findOrCreate( - PythonTypes.Root, - Atom.findOrCreateUnicodeAtom("$function"), - PythonTypes.Root))); - - int v0 = v + 1; - - x.addStatement( - PythonLanguage.Python.instructionFactory() - .CheckCastInstruction(1, v0, v, filter.getReference(), true)); - - int v1; - - // Are we calling a static method? - boolean staticMethodReceiver = filter.getAnnotations().contains(make(STATIC_METHOD)); - logger.fine( - staticMethodReceiver - ? "Found static method receiver: " + filter - : "Method is not static: " + filter); - - // only add self if the receiver isn't static. - if (!staticMethodReceiver) { - v1 = v + 2; - - x.addStatement( - PythonLanguage.Python.instructionFactory() - .GetInstruction( - 1, - v1, - 1, - FieldReference.findOrCreate( - PythonTypes.Root, - Atom.findOrCreateUnicodeAtom("$self"), - PythonTypes.Root))); - } else v1 = v + 1; - - int i = 0; - int paramSize = - Math.max( - staticMethodReceiver ? 1 : 2, - call.getNumberOfPositionalParameters() + (staticMethodReceiver ? 0 : 1)); - int[] params = new int[paramSize]; - params[i++] = v0; - - if (!staticMethodReceiver) params[i++] = v1; - - for (int j = 1; j < call.getNumberOfPositionalParameters(); j++) params[i++] = j + 1; - - int ki = 0, ji = call.getNumberOfPositionalParameters() + 1; - Pair[] keys = new Pair[0]; - - if (call.getKeywords() != null) { - keys = new Pair[call.getKeywords().size()]; - - for (String k : call.getKeywords()) { - names.put(ji, Atom.findOrCreateUnicodeAtom(k)); - keys[ki++] = Pair.make(k, ji++); - } - } - - int result = v1 + 1; - int except = v1 + 2; - - CallSiteReference ref = - new DynamicCallSiteReference(call.getCallSite().getDeclaredTarget(), 2); - - x.addStatement(new PythonInvokeInstruction(2, result, except, ref, params, keys)); - x.addStatement(new SSAReturnInstruction(3, result, false)); - x.setValueNames(names); - - codeBodies.put(key, new PythonSummarizedFunction(tr, x, receiver)); - } + return super.getCalleeTarget(caller, site, receiver); + } - return codeBodies.get(key); + @SuppressWarnings("unchecked") + @Override + protected void populate( + PythonSummary x, int v, IClass receiver, PythonInvokeInstruction call, Logger logger) { + Map names = HashMapFactory.make(); + IClass filter = ((PythonInstanceMethodTrampoline) receiver).getRealClass(); + + x.addStatement( + PythonLanguage.Python.instructionFactory() + .GetInstruction( + 0, + v, + 1, + FieldReference.findOrCreate( + PythonTypes.Root, + Atom.findOrCreateUnicodeAtom("$function"), + PythonTypes.Root))); + + int v0 = v + 1; + + x.addStatement( + PythonLanguage.Python.instructionFactory() + .CheckCastInstruction(1, v0, v, filter.getReference(), true)); + + int v1; + + // Are we calling a static method? + boolean staticMethodReceiver = filter.getAnnotations().contains(make(STATIC_METHOD)); + logger.fine( + staticMethodReceiver + ? "Found static method receiver: " + filter + : "Method is not static: " + filter); + + // Are we calling a class method? If so, it would be using an object instance instead of a + // class on the LHS. + boolean classMethodReceiver = isClassMethod(receiver); + + // only add self if the receiver isn't static or a class method. + if (!staticMethodReceiver && !classMethodReceiver) { + v1 = v + 2; + + x.addStatement( + PythonLanguage.Python.instructionFactory() + .GetInstruction( + 1, + v1, + 1, + FieldReference.findOrCreate( + PythonTypes.Root, Atom.findOrCreateUnicodeAtom("$self"), PythonTypes.Root))); + } else if (classMethodReceiver) { + // Add a class reference. + v1 = v + 2; + + x.addStatement( + PythonLanguage.Python.instructionFactory() + .GetInstruction( + 1, + v1, + 1, + FieldReference.findOrCreate( + PythonTypes.Root, Atom.findOrCreateUnicodeAtom("$class"), PythonTypes.Root))); + + int v2 = v + 3; + TypeReference reference = getDeclaringClassTypeReference(filter.getReference()); + + x.addStatement( + PythonLanguage.Python.instructionFactory() + .CheckCastInstruction(1, v2, v1++, reference, true)); + } else v1 = v + 1; + + int i = 0; + int paramSize = + Math.max( + staticMethodReceiver ? 1 : 2, + call.getNumberOfPositionalParameters() + (staticMethodReceiver ? 0 : 1)); + int[] params = new int[paramSize]; + params[i++] = v0; + + if (!staticMethodReceiver) params[i++] = v1; + + for (int j = 1; j < call.getNumberOfPositionalParameters(); j++) params[i++] = j + 1; + + int ki = 0, ji = call.getNumberOfPositionalParameters() + 1; + Pair[] keys = new Pair[0]; + + if (call.getKeywords() != null) { + keys = new Pair[call.getKeywords().size()]; + + for (String k : call.getKeywords()) { + names.put(ji, Atom.findOrCreateUnicodeAtom(k)); + keys[ki++] = Pair.make(k, ji++); } } - return base.getCalleeTarget(caller, site, receiver); + int result = v1 + 1; + int except = v1 + 2; + + CallSiteReference ref = new DynamicCallSiteReference(call.getCallSite().getDeclaredTarget(), 2); + + x.addStatement(new PythonInvokeInstruction(2, result, except, ref, params, keys)); + x.addStatement(new SSAReturnInstruction(3, result, false)); + x.setValueNames(names); } /** @@ -328,4 +329,19 @@ private static AllocationSiteInNode getAllocationSiteInNode(ConstantKey const public PythonAnalysisEngine getEngine() { return engine; } + + @Override + protected Logger getLogger() { + return logger; + } + + /** + * Returns true iff the given {@link IClass} represents a Python callable object. + * + * @param receiver The {@link IClass} in question. + * @return True iff the given {@link IClass} represents a Python callable object. + */ + private boolean isCallable(IClass receiver) { + return receiver != null && receiver.getReference().equals(PythonTypes.object); + } } diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonMethodTrampolineTargetSelector.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonMethodTrampolineTargetSelector.java new file mode 100644 index 00000000..fcfc539d --- /dev/null +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/ipa/callgraph/PythonMethodTrampolineTargetSelector.java @@ -0,0 +1,118 @@ +package com.ibm.wala.cast.python.ipa.callgraph; + +import com.ibm.wala.cast.python.ipa.summaries.PythonSummarizedFunction; +import com.ibm.wala.cast.python.ipa.summaries.PythonSummary; +import com.ibm.wala.cast.python.ssa.PythonInvokeInstruction; +import com.ibm.wala.cast.types.AstMethodReference; +import com.ibm.wala.classLoader.CallSiteReference; +import com.ibm.wala.classLoader.IClass; +import com.ibm.wala.classLoader.IMethod; +import com.ibm.wala.core.util.strings.Atom; +import com.ibm.wala.ipa.callgraph.CGNode; +import com.ibm.wala.ipa.callgraph.MethodTargetSelector; +import com.ibm.wala.types.MethodReference; +import com.ibm.wala.util.collections.HashMapFactory; +import com.ibm.wala.util.collections.Pair; +import java.util.Map; +import java.util.logging.Logger; + +public abstract class PythonMethodTrampolineTargetSelector implements MethodTargetSelector { + + protected final MethodTargetSelector base; + + protected final Map, IMethod> codeBodies = HashMapFactory.make(); + + public PythonMethodTrampolineTargetSelector(MethodTargetSelector base) { + super(); + this.base = base; + } + + @Override + public IMethod getCalleeTarget(CGNode caller, CallSiteReference site, IClass receiver) { + if (receiver != null) { + Logger logger = this.getLogger(); + + logger.fine("Getting callee target for receiver: " + receiver); + logger.fine("Calling method name is: " + caller.getMethod().getName()); + + if (this.shouldProcess(caller, site, receiver)) { + PythonInvokeInstruction call = this.getCall(caller, site); + Pair key = this.makeKey(receiver, call); + + if (!codeBodies.containsKey(key)) { + MethodReference tr = + MethodReference.findOrCreate( + receiver.getReference(), + Atom.findOrCreateUnicodeAtom("trampoline" + call.getNumberOfTotalParameters()), + AstMethodReference.fnDesc); + PythonSummary x = new PythonSummary(tr, call.getNumberOfTotalParameters()); + int v = call.getNumberOfTotalParameters() + 1; + + populate(x, v, receiver, call, logger); + + codeBodies.put(key, new PythonSummarizedFunction(tr, x, receiver)); + } + + return codeBodies.get(key); + } + } + + return base.getCalleeTarget(caller, site, receiver); + } + + /** + * Returns the {@link PythonInvokeInstruction} at the given {@link CallSiteReference} within the + * given {@link CGNode}. + * + * @param caller The calling {@link CGNode}. + * @param site A {@link CallSiteReference} within the given {@link CGNode}. + * @return The {@link PythonInvokeInstruction} at the given {@link CallSiteReference} within the + * given {@link CGNode}. + */ + protected PythonInvokeInstruction getCall(CGNode caller, CallSiteReference site) { + return (PythonInvokeInstruction) caller.getIR().getCalls(site)[0]; + } + + /** + * Returns a unique {@link Pair} for the given {@link Receiver} and {@link + * PythonInvokeInstruction}. + * + * @return A unique {@link Pair} for the given {@link Receiver} and {@link + * PythonInvokeInstruction}. + */ + private Pair makeKey(IClass receiver, PythonInvokeInstruction call) { + return Pair.make(receiver, call.getNumberOfTotalParameters()); + } + + /** + * The {@link Logger} to be used. + * + * @return The {@link Logger} to be used. + */ + protected abstract Logger getLogger(); + + /** + * True iff this {@link PythonMethodTrampolineTargetSelector} should handle the given {@link + * CGNode}, {@link CallSiteReference}, {@link IClass} combination. If the combination is not to be + * processed, the next target selector will be used. + * + * @return True iff this {@link PythonMethodTrampolineTargetSelector} should handle the given + * {@link CGNode}, {@link CallSiteReference}, {@link IClass} combination. + */ + protected abstract boolean shouldProcess(CGNode caller, CallSiteReference site, IClass receiver); + + /** + * Populate the given {@link PythonSummary} that will be used as the trampoline. At the completion + * of this method, the given {@link PythonInvokeInstruction} will be the last instruction. + * + *

This fill the trampoline body that eventually invokes the original method. + * + * @param x The {@link PythonSummary} representing the trampoline to fill. + * @param v The starting variable number in the SSA. + * @param receiver The receiver of the original call. + * @param call The original call. + * @param logger The {@link Logger} to use. + */ + protected abstract void populate( + PythonSummary x, int v, IClass receiver, PythonInvokeInstruction call, Logger logger); +} diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/PythonTypes.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/PythonTypes.java index 1c2a89f6..74154afe 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/PythonTypes.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/PythonTypes.java @@ -10,6 +10,7 @@ *****************************************************************************/ package com.ibm.wala.cast.python.types; +import static com.ibm.wala.cast.python.util.Util.CLASS_METHOD_ANNOTATION_NAME; import static com.ibm.wala.cast.python.util.Util.STATIC_METHOD_ANNOTATION_NAME; import com.ibm.wala.cast.tree.CAstType; @@ -90,6 +91,11 @@ public class PythonTypes extends AstTypeReference { TypeReference.findOrCreate( pythonLoader, TypeName.findOrCreate("L" + STATIC_METHOD_ANNOTATION_NAME)); + /** https://docs.python.org/3/library/functions.html#classmethod. */ + public static final TypeReference CLASS_METHOD = + TypeReference.findOrCreate( + pythonLoader, TypeName.findOrCreate("L" + CLASS_METHOD_ANNOTATION_NAME)); + /** A {@link CAstType} representing a dynamic annotation (decorator). */ public static final CAstType CAST_DYNAMIC_ANNOTATION = new CAstType() { diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/Util.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/Util.java index 626b2015..2063e243 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/Util.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/types/Util.java @@ -1,9 +1,17 @@ package com.ibm.wala.cast.python.types; +import com.ibm.wala.cast.types.AstTypeReference; +import com.ibm.wala.classLoader.IClassLoader; +import com.ibm.wala.core.util.strings.Atom; +import com.ibm.wala.types.FieldReference; +import com.ibm.wala.types.MethodReference; import com.ibm.wala.types.TypeName; +import com.ibm.wala.types.TypeReference; public class Util { + private static final String GLOBAL_KEYWORD = "global"; + /** * Returns the filename portion of the given {@link TypeName} representing a Python type. * @@ -20,5 +28,50 @@ public static String getFilename(final TypeName typeName) { return ret; } + /** + * creates a reference to a global named globalName. the declaring type and type of the global are + * both the root type. + */ + public static FieldReference makeGlobalRef(IClassLoader loader, String globalName) { + TypeReference rootTypeRef = + TypeReference.findOrCreate(loader.getReference(), AstTypeReference.rootTypeName); + return FieldReference.findOrCreate( + rootTypeRef, Atom.findOrCreateUnicodeAtom(GLOBAL_KEYWORD + " " + globalName), rootTypeRef); + } + + /** + * Returns the {@link TypeReference} of the given {@link TypeReference}'s declaring class. + * + * @param reference The {@link TypeReference} for which to extract the {@link TypeReference} of + * the declaring class. + * @return The {@link TypeReference} of the given {@link TypeReference}'s declaring class. + */ + public static TypeReference getDeclaringClassTypeReference(TypeReference reference) { + TypeName name = reference.getName(); + Atom packageName = name.getPackage(); + name = TypeName.findOrCreate("L" + packageName.toString()); + return TypeReference.findOrCreate(reference.getClassLoader(), name); + } + + /** + * Returns the global name of the given {@link MethodReference}'s declaring class. + * + * @param methodReference The {@link MethodReference} for which to extract the global script name. + * @return The global name of the given {@link MethodReference}'s declaring class. + */ + public static String getGlobalName(MethodReference methodReference) { + return getGlobalName(methodReference.getDeclaringClass()); + } + + /** + * Returns the global name of the given {@link TypeReference}. + * + * @param typeReference The {@link TypeReference} for which to extract the global script name. + * @return The global name of the given {@link TypeReference}. + */ + public static String getGlobalName(TypeReference typeReference) { + return typeReference.getName().getPackage().toString(); + } + private Util() {} } diff --git a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/util/Util.java b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/util/Util.java index 4ad4f151..92c7aa20 100644 --- a/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/util/Util.java +++ b/com.ibm.wala.cast.python/source/com/ibm/wala/cast/python/util/Util.java @@ -2,12 +2,16 @@ import static com.google.common.collect.Iterables.concat; import static com.ibm.wala.cast.python.types.PythonTypes.CAST_DYNAMIC_ANNOTATION; +import static com.ibm.wala.cast.python.types.PythonTypes.CLASS_METHOD; +import static com.ibm.wala.types.annotations.Annotation.make; import static java.util.Collections.emptyList; import static java.util.stream.Collectors.toList; import com.ibm.wala.cast.python.ipa.callgraph.PytestEntrypointBuilder; +import com.ibm.wala.cast.python.ipa.summaries.PythonInstanceMethodTrampoline; import com.ibm.wala.cast.tree.CAstAnnotation; import com.ibm.wala.cast.tree.CAstNode; +import com.ibm.wala.classLoader.IClass; import com.ibm.wala.ipa.callgraph.Entrypoint; import com.ibm.wala.ipa.callgraph.propagation.PropagationCallGraphBuilder; import java.io.File; @@ -28,6 +32,9 @@ public class Util { /** Name of the annotation (decorator) that marks methods as static. */ public static final String STATIC_METHOD_ANNOTATION_NAME = "staticmethod"; + /** Name of the annotation (decorator) that marks methods as a class method. */ + public static final String CLASS_METHOD_ANNOTATION_NAME = "classmethod"; + /** * Add Pytest entrypoints to the given {@link PropagationCallGraphBuilder}. * @@ -86,5 +93,29 @@ public static Stream getNameStream(Collection annotation .map(String.class::cast); } + /** + * Returns true iff the given {@link IClass} represents a Python class method. + * + * @param method The {@link IClass} in question. + * @return True iff the given {@link IClass} represents a Python class method. + * @apiNote Python methods and functions are represented using {@link IClass}. + * @implNote This method will log whether the given {@link IClass} is a class method or not. + */ + public static boolean isClassMethod(IClass method) { + // If it's a trampoline. + if (method instanceof PythonInstanceMethodTrampoline) + // Use the "real class." + method = ((PythonInstanceMethodTrampoline) method).getRealClass(); + + boolean ret = method.getAnnotations().contains(make(CLASS_METHOD)); + + LOGGER.fine( + ret ? "Found class method: " + method : "Method: " + method + " is not a class method."); + + return ret; + } + private Util() {} } From c4c95916f7371af12a48b6a314b572e91d8113a3 Mon Sep 17 00:00:00 2001 From: Raffi Khatchadourian Date: Thu, 25 Apr 2024 16:10:26 -0400 Subject: [PATCH 3/3] Class methods are only supported for Jython3. --- .../python/ml/test/TestTensorflow2Model.java | 92 ++++++++++++++++++- 1 file changed, 88 insertions(+), 4 deletions(-) diff --git a/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java b/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java index b1cb904a..79f910cd 100644 --- a/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java +++ b/com.ibm.wala.cast.python.ml.test/source/com/ibm/wala/cast/python/ml/test/TestTensorflow2Model.java @@ -3502,7 +3502,28 @@ public void testStaticMethod12() throws ClassHierarchyException, CancelException @Test public void testClassMethod() throws ClassHierarchyException, CancelException, IOException { - test("tf2_test_class_method.py", "MyClass.the_class_method", 1, 1, 3); + int expectNumberofTensorParameters; + int expectedNumberOfTensorVariables; + int[] expectedTensorParameterValueNumbers; + + // Class methods are only supported for Jython3. + if (usesJython3Testing()) { + expectNumberofTensorParameters = 1; + expectedNumberOfTensorVariables = 1; + expectedTensorParameterValueNumbers = new int[] {3}; + } else { + // NOTE: Remove this case once https://github.com/wala/ML/issues/147 is fixed. + expectNumberofTensorParameters = 1; + expectedNumberOfTensorVariables = 1; + expectedTensorParameterValueNumbers = new int[] {2}; + } + + test( + "tf2_test_class_method.py", + "MyClass.the_class_method", + expectNumberofTensorParameters, + expectedNumberOfTensorVariables, + expectedTensorParameterValueNumbers); } @Test @@ -3512,17 +3533,80 @@ public void testClassMethod2() throws ClassHierarchyException, CancelException, @Test public void testClassMethod3() throws ClassHierarchyException, CancelException, IOException { - test("tf2_test_class_method3.py", "MyClass.f", 1, 1, 2); + int expectNumberofTensorParameters; + int expectedNumberOfTensorVariables; + int[] expectedTensorParameterValueNumbers; + + // Class methods are only supported for Jython3. + if (usesJython3Testing()) { + expectNumberofTensorParameters = 1; + expectedNumberOfTensorVariables = 1; + expectedTensorParameterValueNumbers = new int[] {3}; + } else { + // NOTE: Remove this case once https://github.com/wala/ML/issues/147 is fixed. + expectNumberofTensorParameters = 0; + expectedNumberOfTensorVariables = 0; + expectedTensorParameterValueNumbers = new int[] {}; + } + + test( + "tf2_test_class_method3.py", + "MyClass.f", + expectNumberofTensorParameters, + expectedNumberOfTensorVariables, + expectedTensorParameterValueNumbers); } @Test public void testClassMethod4() throws ClassHierarchyException, CancelException, IOException { - test("tf2_test_class_method4.py", "MyClass.f", 1, 1, 2); + int expectNumberofTensorParameters; + int expectedNumberOfTensorVariables; + int[] expectedTensorParameterValueNumbers; + + // Class methods are only supported for Jython3. + if (usesJython3Testing()) { + expectNumberofTensorParameters = 1; + expectedNumberOfTensorVariables = 1; + expectedTensorParameterValueNumbers = new int[] {3}; + } else { + // NOTE: Remove this case once https://github.com/wala/ML/issues/147 is fixed. + expectNumberofTensorParameters = 0; + expectedNumberOfTensorVariables = 0; + expectedTensorParameterValueNumbers = new int[] {}; + } + + test( + "tf2_test_class_method4.py", + "MyClass.f", + expectNumberofTensorParameters, + expectedNumberOfTensorVariables, + expectedTensorParameterValueNumbers); } @Test public void testClassMethod5() throws ClassHierarchyException, CancelException, IOException { - test("tf2_test_class_method5.py", "MyClass.f", 1, 1, 2); + int expectNumberofTensorParameters; + int expectedNumberOfTensorVariables; + int[] expectedTensorParameterValueNumbers; + + // Class methods are only supported for Jython3. + if (usesJython3Testing()) { + expectNumberofTensorParameters = 1; + expectedNumberOfTensorVariables = 1; + expectedTensorParameterValueNumbers = new int[] {3}; + } else { + // NOTE: Remove this case once https://github.com/wala/ML/issues/147 is fixed. + expectNumberofTensorParameters = 0; + expectedNumberOfTensorVariables = 0; + expectedTensorParameterValueNumbers = new int[] {}; + } + + test( + "tf2_test_class_method5.py", + "MyClass.f", + expectNumberofTensorParameters, + expectedNumberOfTensorVariables, + expectedTensorParameterValueNumbers); } private void test(