001// Licensed under the Apache License, Version 2.0 (the "License");
002// you may not use this file except in compliance with the License.
003// You may obtain a copy of the License at
004//
005// http://www.apache.org/licenses/LICENSE-2.0
006//
007// Unless required by applicable law or agreed to in writing, software
008// distributed under the License is distributed on an "AS IS" BASIS,
009// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
010// See the License for the specific language governing permissions and
011// limitations under the License.
012
013package org.apache.tapestry5.plastic;
014
015import java.lang.annotation.Annotation;
016import java.lang.reflect.Method;
017import java.util.List;
018import java.util.Set;
019
020/**
021 * The representation of a class while it is being instrumented and transformed. PlasticClass allows
022 * for an imperative style of development: the PlasticClass is provided to other objects; they can query it
023 * for relevant fields or methods, and invoke methods that modify the class in various ways. Ultimately, the
024 * end result is a {@link ClassInstantiator} used to create instances of the fully instrumented and transformed class.
025 *
026 * The terminology is that a class that is being transformed is "plastic", but the end result is a normal concrete class
027 * (albeit in a different class loader).
028 *
029 * Implements {@link AnnotationAccess} to provide access to annotations on the type itself.
030 *
031 * This class is expressly <em>not thread safe</em>; only a single thread should be responsible for operating on a
032 * PlasticClass.
033 *
034 * TODO: what about annotation inheritance?
035 */
036@SuppressWarnings("rawtypes")
037public interface PlasticClass extends AnnotationAccess
038{
039    /**
040     * Returns the fully qualified class name of the class being transformed.
041     */
042    String getClassName();
043
044    /**
045     * Matches all fields (claimed or not) that have the given annotation. Returns the fields in sorted order.
046     *
047     * @return Unmodifiable List of fields.
048     */
049    <T extends Annotation> List<PlasticField> getFieldsWithAnnotation(Class<T> annotationType);
050
051    /**
052     * Returns all non-introduced fields, in sorted order by name.
053     *
054     * @return Unmodifiable list of fields.
055     */
056    List<PlasticField> getAllFields();
057
058    /**
059     * Returns all unclaimed fields, in sorted order by name. This does not include introduced fields.
060     *
061     * @return Unmodifiable list of fields.
062     * @see PlasticField#claim(Object)
063     */
064    List<PlasticField> getUnclaimedFields();
065
066    /**
067     * Introduces a new private field into the class.
068     *
069     * @param typeName      the Java class name for the field, or (possibly) a primitive type name or an array
070     * @param suggestedName the suggested name for the field, which may be modified to ensure that the field name
071     *                      is unique
072     * @return PlasticField for the introduced field
073     */
074    PlasticField introduceField(String typeName, String suggestedName);
075
076    /**
077     * Convenience method that uses a Java class rather than a type name.
078     */
079    PlasticField introduceField(Class fieldType, String suggestedName);
080
081    /**
082     * Introduces a new private method into the class, ensuring that the method name is unique.
083     *
084     * @param typeName       return type of method
085     * @param suggestedName  suggested name for the method; the actual method name may be modified to ensure uniqueness
086     * @param argumentTypes  types of any arguments (may be null)
087     * @param exceptionTypes type of any checked exceptions (may be null)
088     * @return new method, with default implementation
089     */
090    PlasticMethod introducePrivateMethod(String typeName, String suggestedName, String[] argumentTypes,
091                                         String[] exceptionTypes);
092
093    /**
094     * Matches methods with the given annotation.
095     *
096     * @return Unmodifiable list of methods, in sorted order.
097     */
098    <T extends Annotation> List<PlasticMethod> getMethodsWithAnnotation(Class<T> annotationType);
099
100    /**
101     * Returns all methods of the class, in sorted order. This does not include static methods,
102     * or any {@linkplain #introduceMethod(MethodDescription) introduced methods}.
103     *
104     * @return Unmodifiable list of methods.
105     */
106    List<PlasticMethod> getMethods();
107
108    /**
109     * Returns an existing method declared in this class, or introduces a new method into this class.
110     * The method is created with default behavior. If the method overrides a non-private, non-abstract method
111     * implemented in a <em>transformed</em> super class, the the default behavior is to invoke that method and return
112     * its value. Otherwise, the default behavior is to ignore parameters and return 0, false, or null. Void methods
113     * will invoke the super-class implementation (if it exists) and return no value.
114     *
115     * It is allowed for the method description to indicate an abstract method; however the abstract flag will be
116     * removed, and a non-abstract method will be created.
117     *
118     * @param description describes the method name, visibility, return value, etc.
119     * @return a new (or previously created) PlasticMethod for the method
120     * @throws IllegalArgumentException if the method is abstract or static
121     */
122    PlasticMethod introduceMethod(MethodDescription description);
123
124    /**
125     * Returns an existing method declared in this class, or introduces a new method into this class.
126     * The method is created with default behavior.
127     *
128     * It is allowed for the method description to indicate an abstract method; however the abstract flag will be
129     * removed, and a non-abstract method will be created.
130     *
131     * @param description describes the method name, visibility, return value, etc.
132     * @param callback    used to create the implementation of the method
133     * @return a new (or previously created) PlasticMethod for the method
134     * @throws IllegalArgumentException if the method is abstract or static
135     */
136    PlasticMethod introduceMethod(MethodDescription description, InstructionBuilderCallback callback);
137
138    /**
139     * A convenience that creates a {@link MethodDescription} from the Method and introduces that. This is often
140     * invoked when walking the methods of an interface and introducing each of those methods.
141     *
142     * Introduced methods are always concrete, not abstract. The abstract flag on the method modifiers will always be
143     * stripped off, which is handy when {@linkplain #introduceInterface(Class) introducing methods from an interface}.
144     *
145     * @param method to introduce
146     * @return new (or previously created) PlasticMethod
147     */
148    PlasticMethod introduceMethod(Method method);
149
150    /**
151     * Introduces each method defined by the interface into the class. Determines which new methods must
152     * be introduced in order to ensure that all methods of the interface are implemented. The newly introduced methods,
153     * if any, are returned.
154     */
155    Set<PlasticMethod> introduceInterface(Class interfaceType);
156
157    /**
158     * Introduces the interface, and then invokes {@link PlasticMethod#delegateTo(PlasticField)} on each method
159     * defined by the interface.
160     *
161     * @param interfaceType defines the interface to proxy
162     * @param field         field containing an object to delegate to
163     * @return this plastic class, for further configuration
164     */
165    PlasticClass proxyInterface(Class interfaceType, PlasticField field);
166
167    /**
168     * Conditionally adds an implementation of <code>toString()</code> to the class, but only if it is not already
169     * present in the class, or in a (transformed) super-class.
170     *
171     * @param toStringValue the fixed value to be returned from invoking toString()
172     * @return this plastic class, for further configuration
173     */
174    PlasticClass addToString(String toStringValue);
175
176    /**
177     * Returns true if this class has an implementation of the indicated method, or a super-class provides
178     * a non-abstract implementation.
179     */
180    boolean isMethodImplemented(MethodDescription description);
181
182    /**
183     * Returns true if this class, or a super-class, implements the indicated interface.
184     *
185     * @param interfaceType
186     * @return true if the interface is implemented
187     */
188    boolean isInterfaceImplemented(Class interfaceType);
189
190    /**
191     * Returns the name of the super-class of the class being transformed.
192     */
193    String getSuperClassName();
194
195    /**
196     * Adds the callback for execution when an instance of the class is instantiated.
197     *
198     * @param callback to execute at instance construction time
199     */
200    PlasticClass onConstruct(ConstructorCallback callback);
201}