5 changed files with 279 additions and 1 deletions
@ -0,0 +1,105 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2011 the original author or authors. |
||||||
|
* |
||||||
|
* 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 org.springframework.instrument.classloading.websphere; |
||||||
|
|
||||||
|
import java.lang.instrument.ClassFileTransformer; |
||||||
|
import java.lang.reflect.Constructor; |
||||||
|
import java.lang.reflect.Field; |
||||||
|
import java.lang.reflect.InvocationHandler; |
||||||
|
import java.lang.reflect.InvocationTargetException; |
||||||
|
import java.lang.reflect.Method; |
||||||
|
import java.lang.reflect.Proxy; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import org.springframework.util.Assert; |
||||||
|
|
||||||
|
/** |
||||||
|
* |
||||||
|
* Reflective wrapper around a WebSphere 7 class loader. Used to |
||||||
|
* encapsulate the classloader-specific methods (discovered and |
||||||
|
* called through reflection) from the load-time weaver. |
||||||
|
* |
||||||
|
* @author Costin Leau |
||||||
|
*/ |
||||||
|
class WebSphereClassLoaderAdapter { |
||||||
|
|
||||||
|
private static final String COMPOUND_CLASS_LOADER_NAME = "com.ibm.ws.classloader.CompoundClassLoader"; |
||||||
|
private static final String CLASS_PRE_PROCESSOR_NAME = "com.ibm.websphere.classloader.ClassLoaderInstancePreDefinePlugin"; |
||||||
|
private static final String PLUGINS_FIELD = "preDefinePlugins"; |
||||||
|
|
||||||
|
private ClassLoader classLoader; |
||||||
|
private Class<?> wsPreProcessorClass; |
||||||
|
private Method addPreDefinePlugin; |
||||||
|
private Constructor<? extends ClassLoader> cloneConstructor; |
||||||
|
private Field transformerList; |
||||||
|
|
||||||
|
public WebSphereClassLoaderAdapter(ClassLoader classLoader) { |
||||||
|
Class<?> wsCompoundClassLoaderClass = null; |
||||||
|
try { |
||||||
|
wsCompoundClassLoaderClass = classLoader.loadClass(COMPOUND_CLASS_LOADER_NAME); |
||||||
|
cloneConstructor = classLoader.getClass().getDeclaredConstructor(wsCompoundClassLoaderClass); |
||||||
|
cloneConstructor.setAccessible(true); |
||||||
|
|
||||||
|
wsPreProcessorClass = classLoader.loadClass(CLASS_PRE_PROCESSOR_NAME); |
||||||
|
addPreDefinePlugin = classLoader.getClass().getMethod("addPreDefinePlugin", wsPreProcessorClass); |
||||||
|
transformerList = wsCompoundClassLoaderClass.getDeclaredField(PLUGINS_FIELD); |
||||||
|
transformerList.setAccessible(true); |
||||||
|
|
||||||
|
} catch (Exception ex) { |
||||||
|
throw new IllegalStateException( |
||||||
|
"Could not initialize WebSphere LoadTimeWeaver because WebSphere 7 API classes are not available", |
||||||
|
ex); |
||||||
|
} |
||||||
|
Assert.isInstanceOf(wsCompoundClassLoaderClass, classLoader, "ClassLoader must be instance of [" |
||||||
|
+ COMPOUND_CLASS_LOADER_NAME + "]"); |
||||||
|
this.classLoader = classLoader; |
||||||
|
} |
||||||
|
|
||||||
|
public ClassLoader getClassLoader() { |
||||||
|
return this.classLoader; |
||||||
|
} |
||||||
|
|
||||||
|
public void addTransformer(ClassFileTransformer transformer) { |
||||||
|
Assert.notNull(transformer, "ClassFileTransformer must not be null"); |
||||||
|
try { |
||||||
|
InvocationHandler adapter = new WebSphereClassPreDefinePlugin(transformer); |
||||||
|
Object adapterInstance = Proxy.newProxyInstance(this.wsPreProcessorClass.getClassLoader(), |
||||||
|
new Class[] { this.wsPreProcessorClass }, adapter); |
||||||
|
this.addPreDefinePlugin.invoke(this.classLoader, adapterInstance); |
||||||
|
|
||||||
|
} catch (InvocationTargetException ex) { |
||||||
|
throw new IllegalStateException("WebSphere addPreDefinePlugin method threw exception", ex.getCause()); |
||||||
|
} catch (Exception ex) { |
||||||
|
throw new IllegalStateException("Could not invoke WebSphere addPreDefinePlugin method", ex); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@SuppressWarnings("unchecked") |
||||||
|
public ClassLoader getThrowawayClassLoader() { |
||||||
|
try { |
||||||
|
ClassLoader loader = (ClassLoader) cloneConstructor.newInstance(getClassLoader()); |
||||||
|
// clear out the transformers (copied as well)
|
||||||
|
List list = (List) transformerList.get(loader); |
||||||
|
list.clear(); |
||||||
|
return loader; |
||||||
|
} catch (InvocationTargetException ex) { |
||||||
|
throw new IllegalStateException("WebSphere CompoundClassLoader constructor failed", ex.getCause()); |
||||||
|
} catch (Exception ex) { |
||||||
|
throw new IllegalStateException("Could not construct WebSphere CompoundClassLoader", ex); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
@ -0,0 +1,93 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2011 the original author or authors. |
||||||
|
* |
||||||
|
* 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 org.springframework.instrument.classloading.websphere; |
||||||
|
|
||||||
|
import java.lang.instrument.ClassFileTransformer; |
||||||
|
import java.lang.reflect.InvocationHandler; |
||||||
|
import java.lang.reflect.Method; |
||||||
|
import java.security.CodeSource; |
||||||
|
|
||||||
|
import org.springframework.util.FileCopyUtils; |
||||||
|
|
||||||
|
/** |
||||||
|
* Adapter that implements WebSphere 7.0 ClassPreProcessPlugin interface, delegating to a |
||||||
|
* standard JDK {@link ClassFileTransformer} underneath. |
||||||
|
* |
||||||
|
* <p>To avoid compile time checks again the vendor API, a dynamic proxy is |
||||||
|
* being used. |
||||||
|
* |
||||||
|
* @author Costin Leau |
||||||
|
*/ |
||||||
|
class WebSphereClassPreDefinePlugin implements InvocationHandler { |
||||||
|
|
||||||
|
private final ClassFileTransformer transformer; |
||||||
|
|
||||||
|
private class Dummy { |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Creates a new {@link WebSphereClassPreDefinePlugin}. |
||||||
|
* |
||||||
|
* @param transformer the {@link ClassFileTransformer} to be adapted (must |
||||||
|
* not be <code>null</code>) |
||||||
|
*/ |
||||||
|
public WebSphereClassPreDefinePlugin(ClassFileTransformer transformer) { |
||||||
|
this.transformer = transformer; |
||||||
|
ClassLoader classLoader = transformer.getClass().getClassLoader(); |
||||||
|
|
||||||
|
// first force the full class loading of the weaver by invoking transformation on a dummy class
|
||||||
|
try { |
||||||
|
String dummyClass = Dummy.class.getName().replace('.', '/'); |
||||||
|
byte[] bytes = FileCopyUtils.copyToByteArray(classLoader.getResourceAsStream(dummyClass + ".class")); |
||||||
|
transformer.transform(classLoader, dummyClass, null, null, bytes); |
||||||
|
} catch (Throwable ex) { |
||||||
|
throw new IllegalArgumentException("Cannot load transformer", ex); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { |
||||||
|
String name = method.getName(); |
||||||
|
|
||||||
|
if ("equals".equals(name)) { |
||||||
|
return (Boolean.valueOf(proxy == args[0])); |
||||||
|
} else if ("hashCode".equals(name)) { |
||||||
|
return hashCode(); |
||||||
|
} else if ("toString".equals(name)) { |
||||||
|
return toString(); |
||||||
|
} else if ("transformClass".equals(name)) { |
||||||
|
return transform((String) args[0], (byte[]) args[1], (CodeSource) args[2], (ClassLoader) args[3]); |
||||||
|
} else { |
||||||
|
throw new IllegalArgumentException("Unknown method: " + method); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
public byte[] transform(String className, byte[] classfileBuffer, CodeSource codeSource, ClassLoader classLoader) |
||||||
|
throws Exception { |
||||||
|
// NB: WebSphere passes className as "." without class while the
|
||||||
|
// transformer expects a VM, "/" format
|
||||||
|
byte[] result = transformer.transform(classLoader, className.replace('.', '/'), null, null, classfileBuffer); |
||||||
|
|
||||||
|
return (result != null ? result : classfileBuffer); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public String toString() { |
||||||
|
StringBuilder builder = new StringBuilder(getClass().getName()); |
||||||
|
builder.append(" for transformer: "); |
||||||
|
builder.append(this.transformer); |
||||||
|
return builder.toString(); |
||||||
|
} |
||||||
|
} |
||||||
@ -0,0 +1,68 @@ |
|||||||
|
/* |
||||||
|
* Copyright 2011 the original author or authors. |
||||||
|
* |
||||||
|
* 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 org.springframework.instrument.classloading.websphere; |
||||||
|
|
||||||
|
import java.lang.instrument.ClassFileTransformer; |
||||||
|
|
||||||
|
import org.springframework.instrument.classloading.LoadTimeWeaver; |
||||||
|
import org.springframework.util.Assert; |
||||||
|
import org.springframework.util.ClassUtils; |
||||||
|
|
||||||
|
/** |
||||||
|
* {@link LoadTimeWeaver} implementation for WebSphere instrumentable classloader. |
||||||
|
* |
||||||
|
* <p><b>NOTE:</b> Requires WebSphere Application Server version 7.0.0 or higher. |
||||||
|
* |
||||||
|
* @author Costin Leau |
||||||
|
* @since 3.1 |
||||||
|
*/ |
||||||
|
public class WebSphereLoadTimeWeaver implements LoadTimeWeaver { |
||||||
|
|
||||||
|
private final WebSphereClassLoaderAdapter classLoader; |
||||||
|
|
||||||
|
/** |
||||||
|
* Create a new instance of the {@link WebSphereLoadTimeWeaver} class using |
||||||
|
* the default {@link ClassLoader class loader}. |
||||||
|
* @see org.springframework.util.ClassUtils#getDefaultClassLoader() |
||||||
|
*/ |
||||||
|
public WebSphereLoadTimeWeaver() { |
||||||
|
this(ClassUtils.getDefaultClassLoader()); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Create a new instance of the {@link WebSphereLoadTimeWeaver} class using |
||||||
|
* the supplied {@link ClassLoader}. |
||||||
|
* @param classLoader the <code>ClassLoader</code> to delegate to for |
||||||
|
* weaving (must not be <code>null</code>) |
||||||
|
*/ |
||||||
|
public WebSphereLoadTimeWeaver(ClassLoader classLoader) { |
||||||
|
Assert.notNull(classLoader, "ClassLoader must not be null"); |
||||||
|
this.classLoader = new WebSphereClassLoaderAdapter(classLoader); |
||||||
|
} |
||||||
|
|
||||||
|
public void addTransformer(ClassFileTransformer transformer) { |
||||||
|
this.classLoader.addTransformer(transformer); |
||||||
|
} |
||||||
|
|
||||||
|
public ClassLoader getInstrumentableClassLoader() { |
||||||
|
return this.classLoader.getClassLoader(); |
||||||
|
} |
||||||
|
|
||||||
|
public ClassLoader getThrowawayClassLoader() { |
||||||
|
return this.classLoader.getThrowawayClassLoader(); |
||||||
|
} |
||||||
|
} |
||||||
Loading…
Reference in new issue