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

jodd.proxetta.Proxetta Maven / Gradle / Ivy

The newest version!
// Copyright (c) 2003-present, Jodd Team (http://jodd.org)
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.

package jodd.proxetta;

import jodd.proxetta.impl.InvokeProxetta;
import jodd.proxetta.impl.ProxyProxetta;
import jodd.proxetta.impl.WrapperProxetta;

import java.io.File;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

/**
 * Proxetta creates dynamic proxy classes in the run-time.
 * 

* To wrap a class with proxy Proxetta needs a target class (or its name or InputStream) * and one or more {@link jodd.proxetta.ProxyAspect proxy aspects} that will be applied to target. * Proxetta will examine target class and check if there are any methods to wrap, as defined by aspects pointcut. * If there is at least one matched method, new proxy class will be created that extends target class. *

* If no matching method founded, Proxetta may or may not create an empty proxy class. * This behaviour is defined by forced mode during creation. *

* There are several options that describes how proxy class will be named. By default, * proxy class name is created from target class name by adding default suffix. Suffix * can be changed, also, name can be variable, so each time class is created it will have a new name. *

* It is also possible to set proxy simple class name and/or package name. This is useful when * proxyfing JDK classes or any other that can't be loaded by some classloader. Requested proxy name * can be in the following forms: *

    *
  • .Foo (starting with a dot) - proxy package name is equal to target package, just proxy simple class name is set.
  • *
  • foo. (ending with a dot) - proxy package is set, proxy simple name is create from target simple class name.
  • *
  • foo.Foo - full proxy class name is specified.
  • *
* @see ProxettaFactory */ public abstract class Proxetta { /** * Creates a new instance of {@link WrapperProxetta}. */ public static WrapperProxetta wrapperProxetta() { return new WrapperProxetta(); } /** * Creates a new instance of {@link ProxyProxetta}. */ public static ProxyProxetta proxyProxetta() { return new ProxyProxetta(); } /** * Creates a new instance of {@link InvokeProxetta}. */ public static InvokeProxetta invokeProxetta() { return new InvokeProxetta(); } @SuppressWarnings("unchecked") protected T _this() { return (T) this; } // ---------------------------------------------------------------- properties protected boolean forced; protected ClassLoader classLoader; protected boolean variableClassName; protected String classNameSuffix; protected File debugFolder; protected final List proxyAspectList = new ArrayList<>(); // ---------------------------------------------------------------- aspects /** * Adds an aspect. */ public T withAspect(final A proxyAspect) { proxyAspectList.add(proxyAspect); return _this(); } public T withAspects(final A... aspects) { Collections.addAll(proxyAspectList, aspects); return _this(); } public A[] getAspects(final A[] array) { return proxyAspectList.toArray(array); } /** * Specifies 'forced' mode. If true, new proxy class will be created even if there are no * matching pointcuts. If false, new proxy class will be created only if there is at least one * matching pointcut - otherwise, original class will be returned. */ public T setForced(final boolean forced) { this.forced = forced; return _this(); } public boolean isForced() { return forced; } /** * Specifies classloaders for loading created classes. * If classloader not specified, default one will be used. */ public T setClassLoader(final ClassLoader classLoader) { this.classLoader = classLoader; return _this(); } /** * Returns specified classloader for loading created classes. * If classloader is not specified, returns null. */ public ClassLoader getClassLoader() { return classLoader; } /** * Sets variable proxy class name so every time when new proxy class is created * its name will be different,so one classloader may load it without a problem. * Otherwise, sets constant proxy class name so each time created proxy * class will have the same name. Such class can be loaded only once by a classloader. *

* This prevents "java.lang.LinkageError: duplicate class definition" errors. */ public T setVariableClassName(final boolean variableClassName) { this.variableClassName = variableClassName; return _this(); } public boolean isVariableClassName() { return variableClassName; } /** * Specifies custom classname suffix to be added to the class name of created proxy. * Warning: when class name suffix is not used, full classname has to be * specified that differs from target class name. */ public T setClassNameSuffix(final String suffix) { this.classNameSuffix = suffix; return _this(); } public String getClassNameSuffix() { return classNameSuffix; } /** * Specifies the debug folder where all created classes will be * written to, for debugging purposes. */ public T setDebugFolder(final String debugFolder) { this.debugFolder = new File(debugFolder); return _this(); } public T setDebugFolder(final File debugFolder) { this.debugFolder = debugFolder; return _this(); } /** * Returns debug folder or {@code null} if debug folder does not exist. */ public File getDebugFolder() { return debugFolder; } // ---------------------------------------------------------------- builder /** * Creates {@link ProxettaFactory} with of this Proxetta. */ public abstract ProxettaFactory proxy(); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy