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

com.palantir.proxy.util.ProxyUtils Maven / Gradle / Ivy

There is a newer version: 0.1152.0
Show newest version
/*
 * (c) Copyright 2018 Palantir Technologies Inc. All rights reserved.
 *
 * 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 com.palantir.proxy.util;

import com.palantir.proxy.exception.ProxyException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.Set;

public final class ProxyUtils {
    private ProxyUtils() {
        // Utility class, should never be instantiated
    }

    @SuppressWarnings({"checkstyle:AbbreviationAsWordInName", "checkstyle:IllegalThrows"}) // Avoiding API breaks
    public static Object invokeAndUnwrapITEs(Object object, Method method, Object[] args) throws Throwable {
        Object returnValue;
        try {
            returnValue = method.invoke(object, args);
        } catch (InvocationTargetException e) {
            throw e.getCause();
        }
        return returnValue;
    }

    public static boolean isHashCode(Method method) throws ProxyException {
        try {
            return method.equals(Object.class.getMethod("hashCode", new Class[] {}));
        } catch (NoSuchMethodException nsme) {
            throw new ProxyException(nsme);
        }
    }

    public static boolean isEquals(Method method) throws ProxyException {
        try {
            return method.equals(Object.class.getMethod("equals", Object.class));
        } catch (NoSuchMethodException nsme) {
            throw new ProxyException(nsme);
        }
    }

    public static boolean isToString(Method method) throws ProxyException {
        try {
            return method.equals(Object.class.getMethod("toString", new Class[] {}));
        } catch (NoSuchMethodException nsme) {
            throw new ProxyException(nsme);
        }
    }

    public static boolean isProxiableMethod(Method method) throws ProxyException {
        return !isHashCode(method) && !isEquals(method) && !isToString(method);
    }

    public static Object invokeUnproxiableMethod(Method method, Object proxyObject, Object[] args)
            throws ProxyException {
        if (ProxyUtils.isEquals(method)) {
            return proxyObject == args[0];
        } else if (ProxyUtils.isHashCode(method)) {
            return System.identityHashCode(proxyObject);
        } else if (ProxyUtils.isToString(method)) {
            return proxyObject.getClass().getName() + "@" + Integer.toHexString(proxyObject.hashCode());
        } else {
            throw new IllegalArgumentException(
                    "Method should be not be invoked directly on proxyObject: " + method.toString());
        }
    }

    /**
     * Returns a new proxy for a given interface class, plus all interfaces from the delegate class.
     *
     * @param iface main interface to proxy
     * @param delegate delegate class whose interfaces ot proxy
     * @param handler proxy invocation handler
     * @return a new proxy instance that implements the specified interface as well as all the
     *         interfaces from the delegate class
     */
    public static  T newProxy(Class iface, Class delegate, InvocationHandler handler) {
        checkIsInterface(iface);
        return iface.cast(
                Proxy.newProxyInstance(iface.getClassLoader(), ProxyUtils.interfaces(iface, delegate), handler));
    }

    /**
     * Returns a set of interfaces implemented by a collection of classes, unioned with the interfaces in the
     * additionalInterfaces set.
     *
     * @return the set of interfaces for the specified classes
     * @throws IllegalArgumentException if the specified iface is not an interface
     */
    public static Class[] interfaces(
            Class iface, Collection> additionalInterfaces, Class delegateClass) {
        checkIsInterface(iface);
        Set> interfaces = new LinkedHashSet>();
        interfaces.add(iface);
        interfaces.addAll(additionalInterfaces);
        if (delegateClass.isInterface()) {
            interfaces.add(delegateClass);
        }
        interfaces.addAll(Arrays.asList(delegateClass.getInterfaces()));

        checkAreAllInterfaces(interfaces);
        return interfaces.toArray(new Class[interfaces.size()]);
    }

    /**
     * Returns a set of interfaces implemented by a collection of classes.
     *
     * @return the set of interfaces for the specified classes
     * @throws IllegalArgumentException if the specified iface is not an interface
     */
    public static Class[] interfaces(Class iface, Class delegateClass) {
        return interfaces(iface, Collections.>emptySet(), delegateClass);
    }

    private static void checkIsInterface(Class iface) {
        if (!iface.isInterface()) {
            throw new IllegalArgumentException(iface + " is not an interface");
        }
    }

    private static void checkAreAllInterfaces(Set> interfaces) {
        for (Class possibleInterface : interfaces) {
            checkIsInterface(possibleInterface);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy