/* * JBoss, Home of Professional Open Source * Copyright 2006, JBoss Inc., and individual contributors as indicated * by the @authors tag. See the copyright.txt in the distribution for a * full listing of individual contributors. * * This is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this software; if not, write to the Free * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ package org.jboss.deployers.vfs.plugins.annotations; import java.net.URL; import java.util.ArrayList; import java.util.List; import javassist.ClassPath; import javassist.ClassPool; import javassist.LoaderClassPath; import org.jboss.classloading.spi.dependency.Module; import org.jboss.deployers.plugins.annotations.GenericAnnotationResourceVisitor; import org.jboss.deployers.spi.DeploymentException; import org.jboss.deployers.spi.annotations.AnnotationEnvironment; import org.jboss.deployers.spi.deployer.DeploymentStages; import org.jboss.deployers.structure.spi.DeploymentUnit; import org.jboss.deployers.vfs.spi.deployer.AbstractOptionalVFSRealDeployer; import org.jboss.deployers.vfs.spi.structure.VFSDeploymentUnit; import org.jboss.virtual.VirtualFile; /** * A POST_CLASSLOADER deployer which creates AnnotationEnvironment for sub-deployments. * * @author <a href="mailto:ales.justin@jboss.com">Ales Justin</a> */ public class AnnotationEnvironmentDeployer extends AbstractOptionalVFSRealDeployer<Module> { private boolean forceAnnotations; private boolean keepAnnotations; private boolean checkInterfaces; public AnnotationEnvironmentDeployer() { super(Module.class); setStage(DeploymentStages.POST_CLASSLOADER); setOutput(AnnotationEnvironment.class); checkInterfaces = true; } /** * Should we force all annotations to be available. * * @param forceAnnotations the force annotations flag */ public void setForceAnnotations(boolean forceAnnotations) { this.forceAnnotations = forceAnnotations; } /** * Set the keep annotations flag. * * @param keepAnnotations the keep annotations flag */ public void setKeepAnnotations(boolean keepAnnotations) { this.keepAnnotations = keepAnnotations; } /** * Should we check interfaces for annotations as well. * * @param checkInterfaces the check interfaces flag */ public void setCheckInterfaces(boolean checkInterfaces) { this.checkInterfaces = checkInterfaces; } /** * Create GenericAnnotationResourceVisitor. * * Can be used change existing GARV's filter. * Or determin if we need to force/keep annotations. * * @param unit the deployment unit * @param pool the class pool * @param classLoader the classloader * @return new generic annotation visitor */ protected GenericAnnotationResourceVisitor createGenericAnnotationResourceVisitor(DeploymentUnit unit, ClassPool pool, ClassLoader classLoader) { GenericAnnotationResourceVisitor visitor = new GenericAnnotationResourceVisitor(pool, classLoader); visitor.setForceAnnotations(forceAnnotations); visitor.setKeepAnnotations(keepAnnotations); visitor.setCheckInterfaces(checkInterfaces); return visitor; } /** * Create class pool. * * @param classLoader the class loader * @return javassist class pool */ protected ClassPool createClassPool(ClassLoader classLoader) { ClassPool pool = new ClassPool(); ClassPath classPath = new LoaderClassPath(classLoader); pool.insertClassPath(classPath); return pool; } /** * Visit module. * * Util method to add some behavior to Module * before we visit it. * * @param unit the deployment unit * @param module the module * @param visitor the resource visitor * @throws DeploymentException for any error */ protected void visitModule(VFSDeploymentUnit unit, Module module, GenericAnnotationResourceVisitor visitor) throws DeploymentException { try { module.visit(visitor, visitor.getFilter(), null, getUrls(unit)); } catch (Exception e) { throw DeploymentException.rethrowAsDeploymentException("Exception visiting module", e); } } /** * Get the matching urls. * * @param unit the deployment unit * @return matching urls * @throws Exception for any error */ protected URL[] getUrls(VFSDeploymentUnit unit) throws Exception { List<VirtualFile> classpath = unit.getClassPath(); if (classpath != null && classpath.isEmpty() == false) { List<URL> urls = new ArrayList<URL>(); VirtualFile root = unit.getRoot(); for (VirtualFile cp : classpath) { VirtualFile check = cp; while(check != null && check.equals(root) == false) check = check.getParent(); if (check != null) urls.add(cp.toURL()); } if (urls.isEmpty() == false) { if (log.isTraceEnabled()) log.trace("Explicit urls: " + urls); return urls.toArray(new URL[urls.size()]); } } return new URL[0]; } public void deploy(VFSDeploymentUnit unit, Module module) throws DeploymentException { if (module == null) { VFSDeploymentUnit parent = unit.getParent(); while(parent != null && module == null) { module = parent.getAttachment(Module.class); parent = parent.getParent(); } if (module == null) throw new IllegalArgumentException("No module in deployment unit's hierarchy: " + unit.getName()); } if (log.isTraceEnabled()) log.trace("Creating AnnotationEnvironment for " + unit.getName() + ", module: " + module + ", force annotations: " + forceAnnotations); ClassLoader classLoader = unit.getClassLoader(); ClassPool pool = createClassPool(classLoader); GenericAnnotationResourceVisitor visitor = createGenericAnnotationResourceVisitor(unit, pool, classLoader); // something in javassist uses TCL ClassLoader tcl = Thread.currentThread().getContextClassLoader(); Thread.currentThread().setContextClassLoader(classLoader); try { visitModule(unit, module, visitor); } finally { Thread.currentThread().setContextClassLoader(tcl); } unit.addAttachment(AnnotationEnvironment.class, visitor.getEnv()); } }