Skip to content

Commit 6653f25

Browse files
author
Costin Leau
committed
SPR-6394
SPR-6333 + eliminated compile-time dependency on GlassFish custom API for the GlassFish LTW + added support for GlassFish 3
1 parent 5a158fb commit 6653f25

File tree

8 files changed

+159
-84
lines changed

8 files changed

+159
-84
lines changed

org.springframework.context/.classpath

-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@
1111
<classpathentry combineaccessrules="false" kind="src" path="/org.springframework.expression"/>
1212
<classpathentry combineaccessrules="false" kind="src" path="/org.springframework.instrument"/>
1313
<classpathentry kind="lib" path="/org.springframework.asm/target/artifacts/org.springframework.asm.jar" sourcepath="/org.springframework.asm/target/artifacts/org.springframework.asm-sources.jar"/>
14-
<classpathentry kind="var" path="IVY_CACHE/com.sun.enterprise/com.springsource.com.sun.enterprise.loader/1.0.0/com.springsource.com.sun.enterprise.loader-1.0.0.jar"/>
1514
<classpathentry kind="var" path="IVY_CACHE/edu.emory.mathcs.backport/com.springsource.edu.emory.mathcs.backport/3.0.0/com.springsource.edu.emory.mathcs.backport-3.0.0.jar" sourcepath="/IVY_CACHE/edu.emory.mathcs.backport/com.springsource.edu.emory.mathcs.backport/3.0.0/com.springsource.edu.emory.mathcs.backport-sources-3.0.0.jar"/>
1615
<classpathentry kind="var" path="IVY_CACHE/javax.ejb/com.springsource.javax.ejb/3.0.0/com.springsource.javax.ejb-3.0.0.jar" sourcepath="/IVY_CACHE/javax.ejb/com.springsource.javax.ejb/3.0.0/com.springsource.javax.ejb-sources-3.0.0.jar"/>
1716
<classpathentry kind="var" path="IVY_CACHE/javax.inject/com.springsource.javax.inject/1.0.0/com.springsource.javax.inject-1.0.0.jar" sourcepath="/IVY_CACHE/javax.inject/com.springsource.javax.inject/1.0.0/com.springsource.javax.inject-sources-1.0.0.jar"/>

org.springframework.context/context.iml

-9
Original file line numberDiff line numberDiff line change
@@ -78,15 +78,6 @@
7878
<SOURCES />
7979
</library>
8080
</orderEntry>
81-
<orderEntry type="module-library">
82-
<library>
83-
<CLASSES>
84-
<root url="jar://$IVY_CACHE$/com.sun.enterprise/com.springsource.com.sun.enterprise.loader/1.0.0/com.springsource.com.sun.enterprise.loader-1.0.0.jar!/" />
85-
</CLASSES>
86-
<JAVADOC />
87-
<SOURCES />
88-
</library>
89-
</orderEntry>
9081
<orderEntry type="module-library">
9182
<library>
9283
<CLASSES>

org.springframework.context/ivy.xml

-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,6 @@
3030
</publications>
3131

3232
<dependencies>
33-
<dependency org="com.sun.enterprise" name="com.springsource.com.sun.enterprise.loader" rev="1.0.0" conf="provided->compile"/>
3433
<dependency org="edu.emory.mathcs.backport" name="com.springsource.edu.emory.mathcs.backport" rev="3.0.0" conf="optional, backport-util-concurrent->compile"/>
3534
<dependency org="javax.annotation" name="com.springsource.javax.annotation" rev="1.0.0" conf="provided, ejb->compile"/>
3635
<dependency org="javax.ejb" name="com.springsource.javax.ejb" rev="3.0.0" conf="provided, ejb->compile"/>

org.springframework.context/src/main/java/org/springframework/instrument/classloading/glassfish/ClassTransformerAdapter.java

+3-6
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323
import javax.persistence.spi.ClassTransformer;
2424

2525
/**
26-
* Adapter that implements the JPA ClassTransformer interface (as required by GlassFish)
26+
* Adapter that implements the JPA ClassTransformer interface (as required by GlassFish V1 and V2)
2727
* based on a given JDK 1.5 ClassFileTransformer.
2828
*
2929
* @author Costin Leau
@@ -34,7 +34,6 @@ class ClassTransformerAdapter implements ClassTransformer {
3434

3535
private final ClassFileTransformer classFileTransformer;
3636

37-
3837
/**
3938
* Build a new ClassTransformerAdapter for the given ClassFileTransformer.
4039
* @param classFileTransformer the JDK 1.5 ClassFileTransformer to wrap
@@ -43,15 +42,13 @@ public ClassTransformerAdapter(ClassFileTransformer classFileTransformer) {
4342
this.classFileTransformer = classFileTransformer;
4443
}
4544

46-
4745
public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
4846
ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {
4947

50-
byte[] result = this.classFileTransformer.transform(
51-
loader, className, classBeingRedefined, protectionDomain, classfileBuffer);
48+
byte[] result = this.classFileTransformer.transform(loader, className, classBeingRedefined, protectionDomain,
49+
classfileBuffer);
5250

5351
// If no transformation was done, return null.
5452
return (result == classfileBuffer ? null : result);
5553
}
56-
5754
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
/*
2+
* Copyright 2006-2009 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* http://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package org.springframework.instrument.classloading.glassfish;
17+
18+
import java.lang.instrument.ClassFileTransformer;
19+
import java.lang.reflect.InvocationTargetException;
20+
import java.lang.reflect.Method;
21+
22+
/**
23+
* Reflective wrapper around the GlassFish class loader. Used to
24+
* encapsulate the classloader-specific methods (discovered and
25+
* called through reflection) from the load-time weaver.
26+
*
27+
* <p/> Supports GlassFish V1, V2 and V3 (currently in beta).
28+
*
29+
* @author Costin Leau
30+
* @since 3.0.0
31+
*/
32+
class GlassFishClassLoaderAdapter {
33+
34+
static final String INSTRUMENTABLE_CLASSLOADER_GLASSFISH_V2 = "com.sun.enterprise.loader.InstrumentableClassLoader";
35+
static final String INSTRUMENTABLE_CLASSLOADER_GLASSFISH_V3 = "org.glassfish.api.deployment.InstrumentableClassLoader";
36+
private static final String CLASS_TRANSFORMER = "javax.persistence.spi.ClassTransformer";
37+
38+
private final ClassLoader classLoader;
39+
private final Method addTransformer;
40+
private final Method copy;
41+
private final boolean glassFishV3;
42+
43+
public GlassFishClassLoaderAdapter(ClassLoader classLoader) {
44+
Class<?> instrumentableLoaderClass = null;
45+
boolean glassV3 = false;
46+
try {
47+
// try the V1/V2 API first
48+
instrumentableLoaderClass = classLoader.loadClass(INSTRUMENTABLE_CLASSLOADER_GLASSFISH_V2);
49+
} catch (ClassNotFoundException ex) {
50+
// fall back to V3
51+
try {
52+
instrumentableLoaderClass = classLoader.loadClass(INSTRUMENTABLE_CLASSLOADER_GLASSFISH_V3);
53+
glassV3 = true;
54+
} catch (ClassNotFoundException cnfe) {
55+
throw new IllegalStateException(
56+
"Could not initialize GlassFish LoadTimeWeaver because GlassFish (V1, V2 or V3) API classes are not available",
57+
ex);
58+
}
59+
}
60+
try {
61+
Class<?> classTransformerClass = (glassV3 ? ClassFileTransformer.class : classLoader
62+
.loadClass(CLASS_TRANSFORMER));
63+
64+
addTransformer = instrumentableLoaderClass.getMethod("addTransformer", classTransformerClass);
65+
copy = instrumentableLoaderClass.getMethod("copy");
66+
67+
} catch (Exception ex) {
68+
throw new IllegalStateException(
69+
"Could not initialize GlassFish LoadTimeWeaver because GlassFish API classes are not available", ex);
70+
}
71+
72+
ClassLoader clazzLoader = null;
73+
// Detect transformation-aware ClassLoader by traversing the hierarchy
74+
// (as in GlassFish, Spring can be loaded by the WebappClassLoader).
75+
for (ClassLoader cl = classLoader; cl != null && clazzLoader == null; cl = cl.getParent()) {
76+
if (instrumentableLoaderClass.isInstance(cl)) {
77+
clazzLoader = cl;
78+
}
79+
}
80+
81+
if (clazzLoader == null) {
82+
throw new IllegalArgumentException(classLoader + " and its parents are not suitable ClassLoaders: " + "A ["
83+
+ instrumentableLoaderClass.getName() + "] implementation is required.");
84+
}
85+
86+
this.classLoader = clazzLoader;
87+
this.glassFishV3 = glassV3;
88+
}
89+
90+
public void addTransformer(ClassFileTransformer transformer) {
91+
try {
92+
addTransformer.invoke(classLoader, (glassFishV3 ? transformer : new ClassTransformerAdapter(transformer)));
93+
} catch (InvocationTargetException ex) {
94+
throw new IllegalStateException("GlassFish addTransformer method threw exception ", ex.getCause());
95+
} catch (Exception ex) {
96+
throw new IllegalStateException("Could not invoke GlassFish addTransformer method", ex);
97+
}
98+
}
99+
100+
public ClassLoader getClassLoader() {
101+
return this.classLoader;
102+
}
103+
104+
public ClassLoader getThrowawayClassLoader() {
105+
try {
106+
return (ClassLoader) copy.invoke(classLoader, (Object[]) null);
107+
} catch (InvocationTargetException ex) {
108+
throw new IllegalStateException("GlassFish copy method threw exception ", ex.getCause());
109+
} catch (Exception ex) {
110+
throw new IllegalStateException("Could not invoke GlassFish copy method", ex);
111+
}
112+
}
113+
}

org.springframework.context/src/main/java/org/springframework/instrument/classloading/glassfish/GlassFishLoadTimeWeaver.java

+11-37
Original file line numberDiff line numberDiff line change
@@ -18,28 +18,26 @@
1818

1919
import java.lang.instrument.ClassFileTransformer;
2020

21-
import com.sun.enterprise.loader.InstrumentableClassLoader;
22-
2321
import org.springframework.instrument.classloading.LoadTimeWeaver;
2422
import org.springframework.util.Assert;
2523
import org.springframework.util.ClassUtils;
2624

2725
/**
2826
* {@link LoadTimeWeaver} implementation for GlassFish's
29-
* {@link InstrumentableClassLoader}.
27+
* {@link InstrumentableClassLoader}.
28+
*
29+
* <p/>Since Spring 3.0.0, GlassFish V3 is supported as well.
3030
*
3131
* @author Costin Leau
3232
* @author Juergen Hoeller
3333
* @since 2.0.1
34-
* @see com.sun.enterprise.loader.InstrumentableClassLoader
3534
*/
3635
public class GlassFishLoadTimeWeaver implements LoadTimeWeaver {
3736

38-
private final InstrumentableClassLoader classLoader;
39-
37+
private final GlassFishClassLoaderAdapter classLoader;
4038

4139
/**
42-
* Create a new instance of the <code>GlassFishLoadTimeWeaver</code> class
40+
* Creates a new instance of the <code>GlassFishLoadTimeWeaver</code> class
4341
* using the default {@link ClassLoader}.
4442
* @see #GlassFishLoadTimeWeaver(ClassLoader)
4543
*/
@@ -48,49 +46,25 @@ public GlassFishLoadTimeWeaver() {
4846
}
4947

5048
/**
51-
* Create a new instance of the <code>GlassFishLoadTimeWeaver</code> class.
49+
* Creates a new instance of the <code>GlassFishLoadTimeWeaver</code> class.
5250
* @param classLoader the specific {@link ClassLoader} to use; must not be <code>null</code>
5351
* @throws IllegalArgumentException if the supplied <code>classLoader</code> is <code>null</code>;
5452
* or if the supplied <code>classLoader</code> is not an {@link InstrumentableClassLoader}
5553
*/
5654
public GlassFishLoadTimeWeaver(ClassLoader classLoader) {
5755
Assert.notNull(classLoader, "ClassLoader must not be null");
58-
InstrumentableClassLoader icl = determineClassLoader(classLoader);
59-
if (icl == null) {
60-
throw new IllegalArgumentException(classLoader + " and its parents are not suitable ClassLoaders: " +
61-
"An [" + InstrumentableClassLoader.class.getName() + "] implementation is required.");
62-
}
63-
this.classLoader = icl;
64-
}
65-
66-
/**
67-
* Determine the GlassFish {@link InstrumentableClassLoader} for the given
68-
* {@link ClassLoader}.
69-
* @param classLoader the <code>ClassLoader</code> to check
70-
* @return the <code>InstrumentableClassLoader</code>, or <code>null</code> if none found
71-
*/
72-
protected InstrumentableClassLoader determineClassLoader(ClassLoader classLoader) {
73-
// Detect transformation-aware ClassLoader by traversing the hierarchy
74-
// (as in GlassFish, Spring can be loaded by the WebappClassLoader).
75-
for (ClassLoader cl = classLoader; cl != null; cl = cl.getParent()) {
76-
if (cl instanceof InstrumentableClassLoader) {
77-
return (InstrumentableClassLoader) cl;
78-
}
79-
}
80-
return null;
56+
this.classLoader = new GlassFishClassLoaderAdapter(classLoader);
8157
}
8258

83-
8459
public void addTransformer(ClassFileTransformer transformer) {
85-
this.classLoader.addTransformer(new ClassTransformerAdapter(transformer));
60+
this.classLoader.addTransformer(transformer);
8661
}
8762

8863
public ClassLoader getInstrumentableClassLoader() {
89-
return (ClassLoader) this.classLoader;
64+
return this.classLoader.getClassLoader();
9065
}
9166

9267
public ClassLoader getThrowawayClassLoader() {
93-
return this.classLoader.copy();
68+
return this.classLoader.getThrowawayClassLoader();
9469
}
95-
96-
}
70+
}

0 commit comments

Comments
 (0)