All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.internal.classpath.CallInterceptingMetaClassTest.groovy Maven / Gradle / Ivy

/*
 * Copyright 2023 the original author or authors.
 *
 * 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.gradle.internal.classpath

import org.gradle.internal.classpath.intercept.DefaultCallSiteDecorator
import org.gradle.internal.classpath.intercept.DefaultCallSiteInterceptorSet
import org.gradle.internal.classpath.intercept.CallSiteInterceptorSet
import spock.lang.Specification

import static org.gradle.internal.classpath.BasicCallInterceptionTestInterceptorsDeclaration.TEST_GENERATED_CLASSES_PACKAGE
import static org.gradle.internal.classpath.GroovyCallInterceptorsProvider.ClassLoaderSourceGroovyCallInterceptorsProvider
import static org.gradle.internal.classpath.InstrumentedGroovyCallsTracker.CallKind.GET_PROPERTY
import static org.gradle.internal.classpath.InstrumentedGroovyCallsTracker.CallKind.INVOKE_METHOD
import static org.gradle.internal.classpath.InstrumentedGroovyCallsTracker.CallKind.SET_PROPERTY
import static org.gradle.internal.instrumentation.api.types.BytecodeInterceptorFilter.ALL

class CallInterceptingMetaClassTest extends Specification {

    CallSiteInterceptorSet callInterceptors = new DefaultCallSiteInterceptorSet(new ClassLoaderSourceGroovyCallInterceptorsProvider(this.class.classLoader, TEST_GENERATED_CLASSES_PACKAGE));
    private MetaClass originalMetaClass = null
    private static InterceptorTestReceiver instance = null

    // We don't want to interfere with other tests that modify the meta class
    def interceptorTestReceiverClassLock = new ClassBasedLock(InterceptorTestReceiver)

    def callTracker = new DefaultInstrumentedGroovyCallsTracker()

    def setup() {
        interceptorTestReceiverClassLock.lock()
        originalMetaClass = InterceptorTestReceiver.metaClass
        def interceptors = callInterceptors.getCallInterceptors(ALL)
        def callSiteResolver = new DefaultCallSiteDecorator(interceptors)
        GroovySystem.metaClassRegistry.setMetaClass(
            InterceptorTestReceiver,
            new CallInterceptingMetaClass(GroovySystem.metaClassRegistry, InterceptorTestReceiver, InterceptorTestReceiver.metaClass, callTracker, callSiteResolver)
        )
        instance = new InterceptorTestReceiver()
    }

    def cleanup() {
        GroovySystem.metaClassRegistry.setMetaClass(InterceptorTestReceiver, originalMetaClass)
        instance = null
        interceptorTestReceiverClassLock.unlock()
    }

    def 'intercepts a dynamic call with no argument'() {
        when:
        withEntryPoint(INVOKE_METHOD, "test") {
            instance.invokeMethod("test", [].toArray())
        }

        then:
        instance.intercepted == "test()"
    }

    def 'intercepts a dynamic call with argument'() {
        when:
        withEntryPoint(INVOKE_METHOD, "test") {
            instance.invokeMethod("test", [].toArray())
        }

        then:
        instance.intercepted == "test()"
    }

    def 'intercepts a dynamic call to a non-existent method'() {
        when:
        withEntryPoint(INVOKE_METHOD, "nonExistent") {
            instance.nonExistent("test")
        }

        then:
        instance.intercepted == "nonExistent(String)-non-existent"

        when: "calling it outside the entry point scope"
        instance.nonExistent("test")

        then: "it should throw an exception as usual"
        thrown(MissingMethodException)
    }

    def 'intercepts only one call in a succession with a single entry point'() {
        when:
        withEntryPoint(INVOKE_METHOD, "test") {
            instance.test()
            assert instance.intercepted == "test()"
            instance.intercepted = null
            instance.test()
            instance.test(instance) // also call a different signature
        }

        then:
        instance.intercepted == null
    }

    def 'method invocation with #name on a metaClass is intercepted: #intercepted'() {
        MissingMethodException thrown = null

        when:
        withEntryPoint(INVOKE_METHOD, "test") {
            try {
                closure()
            } catch (MissingMethodException e) {
                thrown = e
            }
        }

        then:
        (thrown != null) == missing
        instance.intercepted == (intercepted ? "test()" : null)

        where:
        name                                                                  | closure                                                                                       | intercepted | missing
        "invokeMethod(receiver, methodName, (Object[]) arguments)"            | { instance.metaClass.invokeMethod(instance, "test", [].toArray()) }                           | true        | false
        "invokeMethod(receiver, methodName, (Object) arguments)"              | { instance.metaClass.invokeMethod(instance, "test", (Object) [].toArray()) }                  | true        | false
        "invokeMethod(sender, receiver, methodName, arguments, false, false)" | { instance.metaClass.invokeMethod(getClass(), instance, "test", [].toArray(), false, false) } | true        | false

        // These should not be intercepted because of isCallToSuper and fromInsideClass
        "invokeMethod(sender, receiver, methodName, arguments, true, false)"  | { instance.metaClass.invokeMethod(getClass(), instance, "test", [].toArray(), true, false) }  | false       | true
        "invokeMethod(sender, receiver, methodName, arguments, false, true)"  | { instance.metaClass.invokeMethod(getClass(), instance, "test", [].toArray(), false, true) }  | false       | true
        "invokeMethod(sender, receiver, methodName, arguments, true, true)"   | { instance.metaClass.invokeMethod(getClass(), instance, "test", [].toArray(), true, true) }   | false       | true
    }

    def 'non-intercepted invokeMethod calls invoke the original method'() {
        when:
        withEntryPoint(INVOKE_METHOD, "callNonIntercepted") {
            instance.callNonIntercepted()
        }

        then:
        instance.intercepted == "callNotIntercepted()-not-intercepted"
    }

    def 'a metamethod obtained within an entry point scope does not break out of the scope'() {
        MetaMethod method = null

        when:
        withEntryPoint(INVOKE_METHOD, "test") {
            method = instance.metaClass.getMetaMethod("test")
        }
        method.invoke(instance, [].toArray())

        then: 'the call should not be intercepted'
        instance.intercepted == null
    }

    def 'successive pickMethod invocations in the entry point scope return the intercepted method'() {
        when:
        def method1 = null
        def method2 = null
        withEntryPoint(INVOKE_METHOD, "test") {
            method1 = instance.metaClass.pickMethod("test", new Class[]{})
            method2 = instance.metaClass.pickMethod("test", new Class[]{InterceptorTestReceiver})
        }
        // not in the scope of an entry point, so should not be an intercepted method
        def method3 = instance.metaClass.pickMethod("test", new Class[]{String})

        then:
        method1 instanceof CallInterceptingMetaClass.InterceptedMetaMethod
        method2 instanceof CallInterceptingMetaClass.InterceptedMetaMethod
        !(method3 instanceof CallInterceptingMetaClass.InterceptedMetaMethod)
    }

    def 'intercepts invokeMethod in a closure'() {
        given:
        def closure = {
            testVararg()
        }
        closure.delegate = instance

        when:
        withEntryPoint(INVOKE_METHOD, "testVararg") {
            closure()
        }

        then:
        instance.intercepted == "testVararg(Object...)"
    }

    def 'intercepts invokeMethod in a nested closure'() {
        given:
        def closure = {
            withEntryPoint(INVOKE_METHOD, "testVararg") {
                testVararg()
            }
        }
        closure.delegate = instance

        when:
        closure()

        then:
        instance.intercepted == "testVararg(Object...)"
    }

    def 'intercepts getMetaMethod and pickMethod for matching signatures only'() {
        when:
        def methodByArgs = withEntryPoint(INVOKE_METHOD, name) {
            instance.metaClass.getMetaMethod(name, args.toArray())
        }
        then:
        (methodByArgs instanceof CallInterceptingMetaClass.InterceptedMetaMethod) == intercepted

        and:
        def methodByTypes = withEntryPoint(INVOKE_METHOD, name) {
            instance.metaClass.pickMethod(name, args.collect { it == null ? null : it.class }.toArray(new Class[0]) as Class[])
        }
        intercepted == methodByTypes instanceof CallInterceptingMetaClass.InterceptedMetaMethod

        where:
        name             | args                                                           | intercepted

        "test"           | []                                                             | true
        "test"           | [new InterceptorTestReceiver()]                                | true
        "test"           | [null]                                                         | true
        "testVararg"     | [new InterceptorTestReceiver(), new InterceptorTestReceiver()] | true
        "testVararg"     | [new InterceptorTestReceiver(), null]                          | true
        "testVararg"     | [new InterceptorTestReceiver[]{new InterceptorTestReceiver()}] | true
        // can intercept non-existent methods:
        "nonExistent"    | ["test"]                                                       | true

        "unrelated"      | [1, 2, 3]                                                      | false
        "notIntercepted" | []                                                             | false
    }

    def 'property access via #method property from closure is intercepted: #intercepted'() {
        MissingPropertyException missingPropertyException = null

        when:
        def result = withEntryPoint(GET_PROPERTY, "testString") {
            try {
                propertyAccess()
            } catch (MissingPropertyException e) {
                missingPropertyException = e
            }
        }

        then:
        missing == (missingPropertyException != null)
        missing || (result == (intercepted ? "testString-intercepted" : "testString"))
        instance.intercepted == (intercepted ? "getTestString()" : null)

        where:
        method                                             | propertyAccess                                                                       | intercepted | missing
        "getProperty(Object, String)"                      | { instance.metaClass.getProperty(instance, "testString") }                           | true        | false
        "getProperty(Class, Object, String, false, false)" | { instance.metaClass.getProperty(getClass(), instance, "testString", false, false) } | true        | false

        // These are not supported because of isCallToSuper or fromInsideClass
        "getProperty(Class, Object, String, true, false)"  | { instance.metaClass.getProperty(getClass(), instance, "testString", true, false) }  | false       | true
        "getProperty(Class, Object, String, false, true)"  | { instance.metaClass.getProperty(getClass(), instance, "testString", false, true) }  | false       | false
        "getProperty(Class, Object, String, true, true)"   | { instance.metaClass.getProperty(getClass(), instance, "testString", true, true) }   | false       | true
    }

    def 'only one property read is intercepted per entry point'() {
        when:
        def result = withEntryPoint(GET_PROPERTY, "testString") {
            [
                instance.metaClass.getProperty(instance, "testString"),
                instance.metaClass.getProperty(instance, "testString")
            ]
        }

        then:
        result == ["testString-intercepted", "testString"]
    }

    def 'intercepts setting #type property #method from closure'() {
        given:
        setExpr.delegate = instance

        when:
        withEntryPoint(SET_PROPERTY, propertyName) {
            setExpr()
        }

        then:
        instance.intercepted == intercepting

        where:
        method            | type      | intercepting                                  | propertyName          | setExpr
        "directly"        | "string"  | "setTestString(String)"                       | "testString"          | { testString = "!" }
        "directly"        | "boolean" | "setTestFlag(boolean)"                        | "testFlag"            | { testFlag = true }
        "directly"        | "string"  | "setNonExistentProperty(String)-non-existent" | "nonExistentProperty" | { nonExistentProperty = "!" }
        "via setProperty" | "string"  | "setTestString(String)"                       | "testString"          | { instance.metaClass.setProperty(null, instance, "testString", "!", false, false) }
        "via setProperty" | "boolean" | "setTestFlag(boolean)"                        | "testFlag"            | { instance.metaClass.setProperty(null, instance, "testFlag", true, false, false) }
        "via setProperty" | "string"  | "setNonExistentProperty(String)-non-existent" | "nonExistentProperty" | { instance.metaClass.setProperty(null, instance, "nonExistentProperty", "!", false, false) }
    }

    def 'intercepts getMetaProperty for matching properties'() {
        MetaProperty property = null

        when:
        withEntryPoint(GET_PROPERTY, propertyName) {
            property = instance.metaClass.getMetaProperty(propertyName)
        }

        then:
        property != null
        (property instanceof CallInterceptingMetaClass.InterceptedMetaProperty) == shouldIntercept

        where:
        propertyName | shouldIntercept
        "testString" | true
        "metaClass"  | false
    }

    def 'successive calls to getMetaProperty in the scope of an entry point all return the intercepted property'() {
        List properties = []

        when:
        withEntryPoint(GET_PROPERTY, "testString") {
            properties.add(instance.metaClass.getMetaProperty("testString"))
            properties.add(instance.metaClass.getMetaProperty("testString"))
        }
        withEntryPoint(SET_PROPERTY, "testString") {
            properties.add(instance.metaClass.getMetaProperty("testString"))
            properties.add(instance.metaClass.getMetaProperty("testString"))
        }

        then:
        properties.size() == 4
        properties.every { it instanceof CallInterceptingMetaClass.InterceptedMetaProperty }
    }

    private Object withEntryPoint(InstrumentedGroovyCallsTracker.CallKind kind, String name, Closure call) {
        def entryPoint = callTracker.enterCall("from-test", name, kind)
        try {
            return call()
        } finally {
            callTracker.leaveCall(entryPoint)
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy