From 53325f5637042322f1b810ad32c8ee3bad2915f4 Mon Sep 17 00:00:00 2001 From: nickl- Date: Sun, 29 Oct 2017 23:42:49 +0200 Subject: [PATCH] Fixes to javassist.scopepool Applied parameters and squashed all compiler warnings. Changed loops to enhanced for where appropriate. Fixed SoftValueHashMap: Made it a raw type. Overwritten implementation of the `get` method correctly unwrapped the reference but the methods `containsValue` and `entrySet` needed to be fixed. Implemented the remaining outstanding methods from the `java.util.Map` contract `values`, `putAll` and `keySet`. Dropped `AbstractMap` inheritance as we were not inheriting anything anymore. Changed internal map to `ConcurrentHashMap' to be atomic and parallel safe. Found original unit tests from jboss retro which are now updated to reflect the javassist versions of the scoped pool classes. Added tests for the SoftValueHashMap including test which overloads the heap to ensure the cache cleans itself appropriately. --- .../javassist/scopedpool/ScopedClassPool.java | 26 +- .../scopedpool/ScopedClassPoolRepository.java | 2 +- .../ScopedClassPoolRepositoryImpl.java | 47 ++-- .../scopedpool/SoftValueHashMap.java | 132 +++++++--- src/test/scoped/ScopedRepositoryTestCase.java | 242 ++++++++++++++++++ src/test/scoped/TestAnnotation.java | 34 +++ .../UnscopedAnnotationDefaultUsage.java | 29 +++ src/test/scoped/UnscopedAnnotationUsage.java | 29 +++ .../FullyScopedAnnotationDefaultUsage.java | 29 +++ .../jar1/FullyScopedAnnotationUsage.java | 29 +++ .../jar1/ScopedAnnotationDefaultUsage.java | 31 +++ .../scoped/jar1/ScopedAnnotationUsage.java | 31 +++ .../scoped/jar1/ScopedTestAnnotation.java | 34 +++ src/test/scoped/jar1/TestClass1.java | 26 ++ 14 files changed, 650 insertions(+), 71 deletions(-) create mode 100644 src/test/scoped/ScopedRepositoryTestCase.java create mode 100644 src/test/scoped/TestAnnotation.java create mode 100644 src/test/scoped/UnscopedAnnotationDefaultUsage.java create mode 100644 src/test/scoped/UnscopedAnnotationUsage.java create mode 100644 src/test/scoped/jar1/FullyScopedAnnotationDefaultUsage.java create mode 100644 src/test/scoped/jar1/FullyScopedAnnotationUsage.java create mode 100644 src/test/scoped/jar1/ScopedAnnotationDefaultUsage.java create mode 100644 src/test/scoped/jar1/ScopedAnnotationUsage.java create mode 100644 src/test/scoped/jar1/ScopedTestAnnotation.java create mode 100644 src/test/scoped/jar1/TestClass1.java diff --git a/src/main/javassist/scopedpool/ScopedClassPool.java b/src/main/javassist/scopedpool/ScopedClassPool.java index bcda0ded..fb87709d 100644 --- a/src/main/javassist/scopedpool/ScopedClassPool.java +++ b/src/main/javassist/scopedpool/ScopedClassPool.java @@ -16,9 +16,9 @@ package javassist.scopedpool; +import java.lang.ref.Reference; import java.lang.ref.WeakReference; import java.security.ProtectionDomain; -import java.util.Iterator; import java.util.Map; import javassist.CannotCompileException; import javassist.ClassPool; @@ -37,11 +37,11 @@ import javassist.NotFoundException; public class ScopedClassPool extends ClassPool { protected ScopedClassPoolRepository repository; - protected WeakReference classLoader; + protected Reference classLoader; protected LoaderClassPath classPath; - protected SoftValueHashMap softcache = new SoftValueHashMap(); + protected Map softcache = new SoftValueHashMap(); boolean isBootstrapCl = true; @@ -59,10 +59,10 @@ public class ScopedClassPool extends ClassPool { * the original class pool * @param repository * the repository - *@deprecated */ protected ScopedClassPool(ClassLoader cl, ClassPool src, - ScopedClassPoolRepository repository) { + ScopedClassPoolRepository repository) + { this(cl, src, repository, false); } @@ -78,11 +78,12 @@ public class ScopedClassPool extends ClassPool { * @param isTemp * Whether this is a temporary pool used to resolve references */ - protected ScopedClassPool(ClassLoader cl, ClassPool src, ScopedClassPoolRepository repository, boolean isTemp) + protected ScopedClassPool(ClassLoader cl, ClassPool src, + ScopedClassPoolRepository repository, boolean isTemp) { super(src); this.repository = repository; - this.classLoader = new WeakReference(cl); + this.classLoader = new WeakReference(cl); if (cl != null) { classPath = new LoaderClassPath(cl); this.insertClassPath(classPath); @@ -110,7 +111,7 @@ public class ScopedClassPool extends ClassPool { } protected ClassLoader getClassLoader0() { - return (ClassLoader)classLoader.get(); + return classLoader.get(); } /** @@ -163,6 +164,7 @@ public class ScopedClassPool extends ClassPool { * the class name * @return the class */ + @Override protected CtClass getCached(String classname) { CtClass clazz = getCachedLocally(classname); if (clazz == null) { @@ -186,11 +188,9 @@ public class ScopedClassPool extends ClassPool { } if (!isLocal) { - Map registeredCLs = repository.getRegisteredCLs(); + Map registeredCLs = repository.getRegisteredCLs(); synchronized (registeredCLs) { - Iterator it = registeredCLs.values().iterator(); - while (it.hasNext()) { - ScopedClassPool pool = (ScopedClassPool)it.next(); + for (ScopedClassPool pool:registeredCLs.values()) { if (pool.isUnloadedClassLoader()) { repository.unregisterClassLoader(pool .getClassLoader()); @@ -289,7 +289,7 @@ public class ScopedClassPool extends ClassPool { * @throws CannotCompileException * for any error */ - public Class toClass(CtClass ct, ClassLoader loader, ProtectionDomain domain) + public Class toClass(CtClass ct, ClassLoader loader, ProtectionDomain domain) throws CannotCompileException { // We need to pass up the classloader stored in this pool, as the // default implementation uses the Thread context cl. diff --git a/src/main/javassist/scopedpool/ScopedClassPoolRepository.java b/src/main/javassist/scopedpool/ScopedClassPoolRepository.java index 497501be..33e47124 100644 --- a/src/main/javassist/scopedpool/ScopedClassPoolRepository.java +++ b/src/main/javassist/scopedpool/ScopedClassPoolRepository.java @@ -81,7 +81,7 @@ public interface ScopedClassPoolRepository { * * @return the registered classloaders. */ - Map getRegisteredCLs(); + Map getRegisteredCLs(); /** * This method will check to see if a register classloader has been diff --git a/src/main/javassist/scopedpool/ScopedClassPoolRepositoryImpl.java b/src/main/javassist/scopedpool/ScopedClassPoolRepositoryImpl.java index a9a3c822..8c20d97c 100644 --- a/src/main/javassist/scopedpool/ScopedClassPoolRepositoryImpl.java +++ b/src/main/javassist/scopedpool/ScopedClassPoolRepositoryImpl.java @@ -18,7 +18,7 @@ package javassist.scopedpool; import java.util.ArrayList; import java.util.Collections; -import java.util.Iterator; +import java.util.List; import java.util.Map; import java.util.WeakHashMap; @@ -43,8 +43,8 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository boolean pruneWhenCached; /** The registered classloaders */ - protected Map registeredCLs = Collections - .synchronizedMap(new WeakHashMap()); + protected Map registeredCLs = Collections + .synchronizedMap(new WeakHashMap()); /** The default class pool */ protected ClassPool classpool; @@ -76,6 +76,7 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository * * @return the prune. */ + @Override public boolean isPrune() { return prune; } @@ -85,6 +86,7 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository * * @param prune a new value. */ + @Override public void setPrune(boolean prune) { this.prune = prune; } @@ -96,10 +98,12 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository * @param src the original classpool. * @return the classpool */ + @Override public ScopedClassPool createScopedClassPool(ClassLoader cl, ClassPool src) { return factory.create(cl, src, this); } + @Override public ClassPool findClassPool(ClassLoader cl) { if (cl == null) return registerClassLoader(ClassLoader.getSystemClassLoader()); @@ -113,6 +117,7 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository * @param ucl the classloader. * @return the classpool */ + @Override public ClassPool registerClassLoader(ClassLoader ucl) { synchronized (registeredCLs) { // FIXME: Probably want to take this method out later @@ -121,7 +126,7 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository // a // ClassPool.classpath if (registeredCLs.containsKey(ucl)) { - return (ClassPool)registeredCLs.get(ucl); + return registeredCLs.get(ucl); } ScopedClassPool pool = createScopedClassPool(ucl, classpool); registeredCLs.put(ucl, pool); @@ -132,7 +137,8 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository /** * Get the registered classloaders. */ - public Map getRegisteredCLs() { + @Override + public Map getRegisteredCLs() { clearUnregisteredClassLoaders(); return registeredCLs; } @@ -141,34 +147,31 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository * This method will check to see if a register classloader has been * undeployed (as in JBoss) */ + @Override public void clearUnregisteredClassLoaders() { - ArrayList toUnregister = null; + List toUnregister = null; synchronized (registeredCLs) { - Iterator it = registeredCLs.values().iterator(); - while (it.hasNext()) { - ScopedClassPool pool = (ScopedClassPool)it.next(); - if (pool.isUnloadedClassLoader()) { - it.remove(); - ClassLoader cl = pool.getClassLoader(); + for (Map.Entry reg:registeredCLs.entrySet()) { + if (reg.getValue().isUnloadedClassLoader()) { + ClassLoader cl = reg.getValue().getClassLoader(); if (cl != null) { - if (toUnregister == null) { - toUnregister = new ArrayList(); - } + if (toUnregister == null) + toUnregister = new ArrayList(); toUnregister.add(cl); } + registeredCLs.remove(reg.getKey()); } } - if (toUnregister != null) { - for (int i = 0; i < toUnregister.size(); i++) { - unregisterClassLoader((ClassLoader)toUnregister.get(i)); - } - } + if (toUnregister != null) + for (ClassLoader cl:toUnregister) + unregisterClassLoader(cl); } } + @Override public void unregisterClassLoader(ClassLoader cl) { synchronized (registeredCLs) { - ScopedClassPool pool = (ScopedClassPool)registeredCLs.remove(cl); + ScopedClassPool pool = registeredCLs.remove(cl); if (pool != null) pool.close(); } @@ -178,10 +181,12 @@ public class ScopedClassPoolRepositoryImpl implements ScopedClassPoolRepository // Noop - this is the end } + @Override public void setClassPoolFactory(ScopedClassPoolFactory factory) { this.factory = factory; } + @Override public ScopedClassPoolFactory getClassPoolFactory() { return factory; } diff --git a/src/main/javassist/scopedpool/SoftValueHashMap.java b/src/main/javassist/scopedpool/SoftValueHashMap.java index df6ae599..1154e62e 100644 --- a/src/main/javassist/scopedpool/SoftValueHashMap.java +++ b/src/main/javassist/scopedpool/SoftValueHashMap.java @@ -18,10 +18,14 @@ package javassist.scopedpool; import java.lang.ref.ReferenceQueue; import java.lang.ref.SoftReference; -import java.util.AbstractMap; -import java.util.HashMap; +import java.util.AbstractMap.SimpleImmutableEntry; +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashSet; +import java.util.List; import java.util.Map; import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; /** * This Map will remove entries when the value in the map has been cleaned from @@ -30,21 +34,21 @@ import java.util.Set; * @version $Revision: 1.4 $ * @author Bill Burke */ -public class SoftValueHashMap extends AbstractMap implements Map { - private static class SoftValueRef extends SoftReference { - public Object key; +public class SoftValueHashMap implements Map { + private static class SoftValueRef extends SoftReference { + public K key; - private SoftValueRef(Object key, Object val, ReferenceQueue q) { + private SoftValueRef(K key, V val, ReferenceQueue q) { super(val, q); this.key = key; } - private static SoftValueRef create(Object key, Object val, - ReferenceQueue q) { + private static SoftValueRef create( + K key, V val, ReferenceQueue q) { if (val == null) return null; else - return new SoftValueRef(key, val, q); + return new SoftValueRef(key, val, q); } } @@ -52,30 +56,37 @@ public class SoftValueHashMap extends AbstractMap implements Map { /** * Returns a set of the mappings contained in this hash table. */ - public Set entrySet() { + @Override + public Set> entrySet() { processQueue(); - return hash.entrySet(); + Set> ret = new HashSet>(); + for (Entry> e:hash.entrySet()) + ret.add(new SimpleImmutableEntry ( + e.getKey(), e.getValue().get())); + return ret; } /* Hash table mapping WeakKeys to values */ - private Map hash; + private Map> hash; /* Reference queue for cleared WeakKeys */ - private ReferenceQueue queue = new ReferenceQueue(); + private ReferenceQueue queue = new ReferenceQueue(); /* * Remove all invalidated entries from the map, that is, remove all entries * whose values have been discarded. */ private void processQueue() { - SoftValueRef ref; - while ((ref = (SoftValueRef)queue.poll()) != null) { - if (ref == (SoftValueRef)hash.get(ref.key)) { - // only remove if it is the *exact* same WeakValueRef - // - hash.remove(ref.key); + Object ref; + if (!hash.isEmpty()) + while ((ref = queue.poll()) != null) + if (ref instanceof SoftValueRef) { + @SuppressWarnings("rawtypes") + SoftValueRef que =(SoftValueRef) ref; + if (ref == hash.get(que.key)) + // only remove if it is the *exact* same SoftValueRef + hash.remove(que.key); } - } } /* -- Constructors -- */ @@ -95,7 +106,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * factor is nonpositive */ public SoftValueHashMap(int initialCapacity, float loadFactor) { - hash = new HashMap(initialCapacity, loadFactor); + hash = new ConcurrentHashMap>(initialCapacity, loadFactor); } /** @@ -109,7 +120,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * If the initial capacity is less than zero */ public SoftValueHashMap(int initialCapacity) { - hash = new HashMap(initialCapacity); + hash = new ConcurrentHashMap>(initialCapacity); } /** @@ -117,7 +128,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * initial capacity and the default load factor, which is 0.75. */ public SoftValueHashMap() { - hash = new HashMap(); + hash = new ConcurrentHashMap>(); } /** @@ -129,7 +140,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * * @param t the map whose mappings are to be placed in this map. */ - public SoftValueHashMap(Map t) { + public SoftValueHashMap(Map t) { this(Math.max(2 * t.size(), 11), 0.75f); putAll(t); } @@ -142,6 +153,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * Map interface, the time required by this operation is * linear in the size of the map. */ + @Override public int size() { processQueue(); return hash.size(); @@ -150,6 +162,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { /** * Returns true if this map contains no key-value mappings. */ + @Override public boolean isEmpty() { processQueue(); return hash.isEmpty(); @@ -162,6 +175,7 @@ public class SoftValueHashMap extends AbstractMap implements Map { * @param key * The key whose presence in this map is to be tested. */ + @Override public boolean containsKey(Object key) { processQueue(); return hash.containsKey(key); @@ -177,12 +191,10 @@ public class SoftValueHashMap extends AbstractMap implements Map { * @param key * The key whose associated value, if any, is to be returned. */ - public Object get(Object key) { + @Override + public V get(Object key) { processQueue(); - SoftReference ref = (SoftReference)hash.get(key); - if (ref != null) - return ref.get(); - return null; + return valueOrNull(hash.get(key)); } /** @@ -200,12 +212,10 @@ public class SoftValueHashMap extends AbstractMap implements Map { * @return The previous value to which this key was mapped, or * null if if there was no mapping for the key */ - public Object put(Object key, Object value) { + @Override + public V put(K key, V value) { processQueue(); - Object rtn = hash.put(key, SoftValueRef.create(key, value, queue)); - if (rtn != null) - rtn = ((SoftReference)rtn).get(); - return rtn; + return valueOrNull(hash.put(key, SoftValueRef.create(key, value, queue))); } /** @@ -218,16 +228,66 @@ public class SoftValueHashMap extends AbstractMap implements Map { * @return The value to which this key was mapped, or null if * there was no mapping for the key. */ - public Object remove(Object key) { + @Override + public V remove(Object key) { processQueue(); - return hash.remove(key); + return valueOrNull(hash.remove(key)); } /** * Removes all mappings from this map. */ + @Override public void clear() { processQueue(); hash.clear(); } + + /* + * Check whether the supplied value exists. + * @param Object the value to compare. + * @return true if it was found or null. + */ + @Override + public boolean containsValue(Object arg0) { + processQueue(); + if (null == arg0) + return false; + + for (SoftValueRef e:hash.values()) + if (null != e && arg0.equals(e.get())) + return true; + return false; + } + + /* {@inheritDoc} */ + @Override + public Set keySet() { + processQueue(); + return hash.keySet(); + } + + /* {@inheritDoc} */ + @Override + public void putAll(Map arg0) { + processQueue(); + for (K key:arg0.keySet()) + put(key, arg0.get(key)); + } + + /* {@inheritDoc} */ + @Override + public Collection values() { + processQueue(); + List ret = new ArrayList(); + for (SoftValueRef e:hash.values()) + ret.add(e.get()); + return ret; + } + + private V valueOrNull(SoftValueRef rtn) { + if (null == rtn) + return null; + return rtn.get(); + } } diff --git a/src/test/scoped/ScopedRepositoryTestCase.java b/src/test/scoped/ScopedRepositoryTestCase.java new file mode 100644 index 00000000..9dca4f56 --- /dev/null +++ b/src/test/scoped/ScopedRepositoryTestCase.java @@ -0,0 +1,242 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped; + +import java.io.File; +import java.lang.reflect.Method; +import java.math.BigDecimal; +import java.net.URL; +import java.net.URLClassLoader; +import java.util.Arrays; +import java.util.Map; +import java.util.stream.LongStream; + +import javassist.ClassPool; +import javassist.CtClass; +import javassist.CtConstructor; +import javassist.CtField; +import javassist.CtMethod; +import javassist.scopedpool.ScopedClassPool; +import javassist.scopedpool.ScopedClassPoolRepository; +import javassist.scopedpool.ScopedClassPoolRepositoryImpl; +import javassist.scopedpool.SoftValueHashMap; +import junit.framework.TestCase; + + +/** + * ScopedRepositoryTest. + * + * @author Adrian Brock + * @version $Revision$ + */ +public class ScopedRepositoryTestCase extends TestCase +{ + private static final ScopedClassPoolRepository repository = ScopedClassPoolRepositoryImpl.getInstance(); + + public void testJDKClasses() throws Exception + { + ClassPool poolClass = repository.findClassPool(Class.class.getClassLoader()); + assertNotNull(poolClass); + ClassPool poolString = repository.findClassPool(String.class.getClassLoader()); + assertNotNull(poolString); + assertEquals(poolClass, poolString); + } + + public void testScopedClasses() throws Exception + { + ClassLoader cl = getURLClassLoader("test-classes14-jar1"); + ClassPool pool1 = repository.findClassPool(cl); + CtClass clazz = pool1.get("scoped.jar1.TestClass1"); + assertNotNull(clazz); + ClassPool poolClass = repository.findClassPool(Class.class.getClassLoader()); + assertNotNull(poolClass); + assertNotSame(pool1, poolClass); + } + + public void testUnscopedAnnotationUsage() throws Exception + { + CtClass clazz = getCtClass(UnscopedAnnotationUsage.class); + checkTestAnnotation(clazz, "notDefault"); + } + + public void testUnscopedAnnotationDefaultUsage() throws Exception + { + CtClass clazz = getCtClass(UnscopedAnnotationDefaultUsage.class); + checkTestAnnotation(clazz, "defaultValue"); + } + + public void testScopedAnnotationUsage() throws Exception + { + ClassLoader cl = getURLClassLoader("test-classes14-jar1"); + CtClass clazz = getCtClass("scoped.jar1.ScopedAnnotationUsage", cl); + checkTestAnnotation(clazz, "notDefault"); + } + + public void testScopedAnnotationDefaultUsage() throws Exception + { + ClassLoader cl = getURLClassLoader("test-classes14-jar1"); + CtClass clazz = getCtClass("scoped.jar1.ScopedAnnotationDefaultUsage", cl); + checkTestAnnotation(clazz, "defaultValue"); + } + + public void testFullyScopedAnnotationUsage() throws Exception + { + ClassLoader cl = getURLClassLoader("test-classes14-jar1"); + CtClass clazz = getCtClass("scoped.jar1.FullyScopedAnnotationUsage", cl); + checkScopedAnnotation(cl, clazz, "notDefault"); + } + + public void testFullyScopedAnnotationDefaultUsage() throws Exception + { + ClassLoader cl = getURLClassLoader("test-classes14-jar1"); + CtClass clazz = getCtClass("scoped.jar1.FullyScopedAnnotationDefaultUsage", cl); + checkScopedAnnotation(cl, clazz, "defaultValue"); + } + + public void testSoftValueHashMap() throws Exception { + Map> map = new SoftValueHashMap<>(); + Class cls = this.getClass(); + assertTrue(map.put(cls.getName(), cls) == null); + assertTrue(map.put(cls.getName(), cls) == cls); + assertTrue(map.size() == 1); + assertTrue(map.get(cls.getName()) == cls); + assertTrue(map.values().iterator().next() == cls); + assertTrue(map.entrySet().iterator().next().getValue() == cls); + assertTrue(map.containsValue(cls)); + assertTrue(map.remove(cls.getName()) == cls); + assertTrue(map.size() == 0); + } + + public void testSoftCache() throws Exception { + // Overload the heap to test that the map auto cleans + Map map = new SoftValueHashMap<>(); + // 12+8*30000000 = +- 252 MB + long[] data = LongStream.range(0, 30000000).toArray(); + int current = map.size(); + while (current <= map.size()) { + current = map.size(); + for (int ii = 0; ii < 5; ii++) { + map.put(current+"-"+ii, Arrays.copyOf(data, data.length)); + } + } + assertTrue(current > map.size()); + } + + protected CtClass getCtClass(Class clazz) throws Exception + { + return getCtClass(clazz.getName(), clazz.getClassLoader()); + } + + protected CtClass getCtClass(String name, ClassLoader cl) throws Exception + { + ClassPool pool = repository.findClassPool(cl); + assertNotNull(pool); + CtClass clazz = pool.get(name); + assertNotNull(clazz); + return clazz; + } + + protected void checkTestAnnotation(CtClass ctClass, String value) throws Exception + { + checkTestAnnotation(ctClass.getAnnotations(), value); + checkTestAnnotation(getFieldAnnotations(ctClass), value); + checkTestAnnotation(getConstructorAnnotations(ctClass), value); + checkTestAnnotation(getConstructorParameterAnnotations(ctClass), value); + checkTestAnnotation(getMethodAnnotations(ctClass), value); + checkTestAnnotation(getMethodParameterAnnotations(ctClass), value); + } + + protected void checkTestAnnotation(Object[] annotations, String value) throws Exception + { + assertNotNull(annotations); + assertEquals(1, annotations.length); + assertNotNull(annotations[0]); + assertTrue(annotations[0] instanceof TestAnnotation); + TestAnnotation annotation = (TestAnnotation) annotations[0]; + assertEquals(value, annotation.something()); + } + + protected void checkScopedAnnotation(ClassLoader cl, CtClass ctClass, String value) throws Exception + { + Class annotationClass = cl.loadClass("scoped.jar1.ScopedTestAnnotation"); + checkScopedAnnotation(annotationClass, ctClass.getAnnotations(), value); + checkScopedAnnotation(annotationClass, getFieldAnnotations(ctClass), value); + checkScopedAnnotation(annotationClass, getConstructorAnnotations(ctClass), value); + checkScopedAnnotation(annotationClass, getConstructorParameterAnnotations(ctClass), value); + checkScopedAnnotation(annotationClass, getMethodAnnotations(ctClass), value); + checkScopedAnnotation(annotationClass, getMethodParameterAnnotations(ctClass), value); + } + + protected void checkScopedAnnotation(Class annotationClass, Object[] annotations, String value) throws Exception + { + assertNotNull(annotations); + assertEquals(1, annotations.length); + assertNotNull(annotations[0]); + assertTrue(annotationClass.isInstance(annotations[0])); + + Method method = annotationClass.getMethod("something", new Class[0]); + assertEquals(value, method.invoke(annotations[0], (Object[]) null)); + } + + protected Object[] getFieldAnnotations(CtClass clazz) throws Exception + { + CtField field = clazz.getField("aField"); + assertNotNull(field); + return field.getAnnotations(); + } + + protected Object[] getMethodAnnotations(CtClass clazz) throws Exception + { + CtMethod method = clazz.getMethod("doSomething", "(I)V"); + assertNotNull(method); + return method.getAnnotations(); + } + + protected Object[] getMethodParameterAnnotations(CtClass clazz) throws Exception + { + CtMethod method = clazz.getMethod("doSomething", "(I)V"); + assertNotNull(method); + Object[] paramAnnotations = method.getParameterAnnotations(); + assertNotNull(paramAnnotations); + assertEquals(1, paramAnnotations.length); + return (Object[]) paramAnnotations[0]; + } + + protected Object[] getConstructorAnnotations(CtClass clazz) throws Exception + { + CtConstructor constructor = clazz.getConstructor("(I)V"); + assertNotNull(constructor); + return constructor.getAnnotations(); + } + + protected Object[] getConstructorParameterAnnotations(CtClass clazz) throws Exception + { + CtConstructor constructor = clazz.getConstructor("(I)V"); + assertNotNull(constructor); + Object[] paramAnnotations = constructor.getParameterAnnotations(); + assertNotNull(paramAnnotations); + assertEquals(1, paramAnnotations.length); + return (Object[]) paramAnnotations[0]; + } + + protected ClassLoader getURLClassLoader(String context) throws Exception + { + String output = "."; + File file = new File(output + File.separator + context); + URL url = file.toURI().toURL(); + return new URLClassLoader(new URL[] { url }); + } +} diff --git a/src/test/scoped/TestAnnotation.java b/src/test/scoped/TestAnnotation.java new file mode 100644 index 00000000..dd5c7b8c --- /dev/null +++ b/src/test/scoped/TestAnnotation.java @@ -0,0 +1,34 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * TestAnnotation. + * + * @author Adrian Brock + * @version $Revision$ + */ +@Retention(value = RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER, ElementType.CONSTRUCTOR }) +public @interface TestAnnotation +{ + String something() default "defaultValue"; +} diff --git a/src/test/scoped/UnscopedAnnotationDefaultUsage.java b/src/test/scoped/UnscopedAnnotationDefaultUsage.java new file mode 100644 index 00000000..bcbecfe6 --- /dev/null +++ b/src/test/scoped/UnscopedAnnotationDefaultUsage.java @@ -0,0 +1,29 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped; + +@TestAnnotation +public class UnscopedAnnotationDefaultUsage +{ + @TestAnnotation + public int aField; + + @TestAnnotation + public UnscopedAnnotationDefaultUsage(@TestAnnotation int param) {} + + @TestAnnotation + public void doSomething(@TestAnnotation int param) {} +} diff --git a/src/test/scoped/UnscopedAnnotationUsage.java b/src/test/scoped/UnscopedAnnotationUsage.java new file mode 100644 index 00000000..e63db330 --- /dev/null +++ b/src/test/scoped/UnscopedAnnotationUsage.java @@ -0,0 +1,29 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped; + +@TestAnnotation(something="notDefault") +public class UnscopedAnnotationUsage +{ + @TestAnnotation(something="notDefault") + public int aField; + + @TestAnnotation(something="notDefault") + public UnscopedAnnotationUsage(@TestAnnotation(something="notDefault") int param) {} + + @TestAnnotation(something="notDefault") + public void doSomething(@TestAnnotation(something="notDefault") int param) {} +} diff --git a/src/test/scoped/jar1/FullyScopedAnnotationDefaultUsage.java b/src/test/scoped/jar1/FullyScopedAnnotationDefaultUsage.java new file mode 100644 index 00000000..663d18f8 --- /dev/null +++ b/src/test/scoped/jar1/FullyScopedAnnotationDefaultUsage.java @@ -0,0 +1,29 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +@ScopedTestAnnotation +public class FullyScopedAnnotationDefaultUsage +{ + @ScopedTestAnnotation + public int aField; + + @ScopedTestAnnotation + public FullyScopedAnnotationDefaultUsage(@ScopedTestAnnotation int value) {} + + @ScopedTestAnnotation + public void doSomething(@ScopedTestAnnotation int value) {} +} diff --git a/src/test/scoped/jar1/FullyScopedAnnotationUsage.java b/src/test/scoped/jar1/FullyScopedAnnotationUsage.java new file mode 100644 index 00000000..c44aeb75 --- /dev/null +++ b/src/test/scoped/jar1/FullyScopedAnnotationUsage.java @@ -0,0 +1,29 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +@ScopedTestAnnotation(something="notDefault") +public class FullyScopedAnnotationUsage +{ + @ScopedTestAnnotation(something="notDefault") + public int aField; + + @ScopedTestAnnotation(something="notDefault") + public FullyScopedAnnotationUsage(@ScopedTestAnnotation(something="notDefault") int param) {} + + @ScopedTestAnnotation(something="notDefault") + public void doSomething(@ScopedTestAnnotation(something="notDefault") int param) {} +} diff --git a/src/test/scoped/jar1/ScopedAnnotationDefaultUsage.java b/src/test/scoped/jar1/ScopedAnnotationDefaultUsage.java new file mode 100644 index 00000000..284b0868 --- /dev/null +++ b/src/test/scoped/jar1/ScopedAnnotationDefaultUsage.java @@ -0,0 +1,31 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +import scoped.TestAnnotation; + +@TestAnnotation +public class ScopedAnnotationDefaultUsage +{ + @TestAnnotation + public int aField; + + @TestAnnotation + public ScopedAnnotationDefaultUsage(@TestAnnotation int param) {} + + @TestAnnotation + public void doSomething(@TestAnnotation int param) {} +} diff --git a/src/test/scoped/jar1/ScopedAnnotationUsage.java b/src/test/scoped/jar1/ScopedAnnotationUsage.java new file mode 100644 index 00000000..23400f27 --- /dev/null +++ b/src/test/scoped/jar1/ScopedAnnotationUsage.java @@ -0,0 +1,31 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +import scoped.TestAnnotation; + +@TestAnnotation(something="notDefault") +public class ScopedAnnotationUsage +{ + @TestAnnotation(something="notDefault") + public int aField; + + @TestAnnotation(something="notDefault") + public ScopedAnnotationUsage(@TestAnnotation(something="notDefault") int param) {} + + @TestAnnotation(something="notDefault") + public void doSomething(@TestAnnotation(something="notDefault") int param) {} +} diff --git a/src/test/scoped/jar1/ScopedTestAnnotation.java b/src/test/scoped/jar1/ScopedTestAnnotation.java new file mode 100644 index 00000000..25a36a58 --- /dev/null +++ b/src/test/scoped/jar1/ScopedTestAnnotation.java @@ -0,0 +1,34 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +/** + * TestAnnotation. + * + * @author Adrian Brock + * @version $Revision$ + */ +@Retention(value = RetentionPolicy.RUNTIME) +@Target({ ElementType.TYPE, ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER, ElementType.CONSTRUCTOR }) +public @interface ScopedTestAnnotation +{ + String something() default "defaultValue"; +} diff --git a/src/test/scoped/jar1/TestClass1.java b/src/test/scoped/jar1/TestClass1.java new file mode 100644 index 00000000..a538bf77 --- /dev/null +++ b/src/test/scoped/jar1/TestClass1.java @@ -0,0 +1,26 @@ +/* + * Javassist, a Java-bytecode translator toolkit. + * Copyright (C) 1999- Shigeru Chiba. All Rights Reserved. + * + * The contents of this file are subject to the Mozilla Public License Version + * 1.1 (the "License"); you may not use this file except in compliance with + * the License. Alternatively, the contents of this file may be used under + * the terms of the GNU Lesser General Public License Version 2.1 or later, + * or the Apache License Version 2.0. + * + * Software distributed under the License is distributed on an "AS IS" basis, + * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License + * for the specific language governing rights and limitations under the + * License. + */ +package scoped.jar1; + +/** + * TestClass1. + * + * @author Adrian Brock + * @version $Revision$ + */ +public class TestClass1 +{ +} -- 2.39.5