org.jboss.cdi.tck.shrinkwrap.ArchiveBuilder Maven / Gradle / Ivy
/*
* JBoss, Home of Professional Open Source
* Copyright 2010, Red Hat, Inc., and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* 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.jboss.cdi.tck.shrinkwrap;
import java.io.File;
import java.io.FilenameFilter;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.enterprise.inject.spi.Extension;
import org.jboss.arquillian.container.test.api.Deployment;
import org.jboss.arquillian.container.test.api.RunAsClient;
import org.jboss.arquillian.container.test.api.ShouldThrowException;
import org.jboss.cdi.tck.AbstractTest;
import org.jboss.cdi.tck.TestGroups;
import org.jboss.cdi.tck.api.Configuration;
import org.jboss.cdi.tck.cdi.Sections;
import org.jboss.cdi.tck.impl.ConfigurationFactory;
import org.jboss.cdi.tck.impl.ConfigurationImpl;
import org.jboss.cdi.tck.impl.PropertiesBasedConfigurationBuilder;
import org.jboss.cdi.tck.spi.Beans;
import org.jboss.cdi.tck.util.Timer;
import org.jboss.cdi.tck.util.Versions;
import org.jboss.cdi.tck.util.annotated.AnnotatedWrapper;
import org.jboss.shrinkwrap.api.Archive;
import org.jboss.shrinkwrap.api.ArchivePath;
import org.jboss.shrinkwrap.api.ArchivePaths;
import org.jboss.shrinkwrap.api.Filter;
import org.jboss.shrinkwrap.api.ShrinkWrap;
import org.jboss.shrinkwrap.api.asset.Asset;
import org.jboss.shrinkwrap.api.asset.ClassAsset;
import org.jboss.shrinkwrap.api.asset.ClassLoaderAsset;
import org.jboss.shrinkwrap.api.asset.EmptyAsset;
import org.jboss.shrinkwrap.api.asset.StringAsset;
import org.jboss.shrinkwrap.api.container.ClassContainer;
import org.jboss.shrinkwrap.api.container.LibraryContainer;
import org.jboss.shrinkwrap.api.container.ManifestContainer;
import org.jboss.shrinkwrap.api.container.ResourceContainer;
import org.jboss.shrinkwrap.api.spec.JavaArchive;
import org.jboss.shrinkwrap.api.spec.WebArchive;
import org.jboss.shrinkwrap.descriptor.api.Descriptors;
import org.jboss.shrinkwrap.descriptor.api.beans11.BeanDiscoveryMode;
import org.jboss.shrinkwrap.descriptor.api.beans11.BeansDescriptor;
import org.jboss.shrinkwrap.descriptor.api.ejbjar31.EjbJarDescriptor;
import org.jboss.shrinkwrap.descriptor.api.persistence20.PersistenceDescriptor;
import org.jboss.shrinkwrap.descriptor.api.webapp30.WebAppDescriptor;
import org.jboss.shrinkwrap.descriptor.api.webcommon30.WebAppVersionType;
/**
* Abstract ShrinkWrap archive builder for CDI TCK Arquillian test.
*
* This is a base class for builders that try to solve most JBoss Test Harness to Arquillian migration issues. The main goal was to use CDI TCK
* 1.0 tests with minimum code changes.
*
*
* Note that all Arquillian tests running in as-client mode (including tests using {@link ShouldThrowException}) must not contain testing related stuff
* (anything that depends on Arquillian, TestNG, incl. test class itself) since Arquillian is not enriching as-client test archives. That's why
* {@link #isAsClientMode} has to be properly set.
*
*
* In case of {@link #isTestArchive} set to false
this archive may not include any testing related stuff as it is probably part of another test
* archive.
*
*
* @param Self type to enable abstract builder pattern
* @param Final shrinkwrap archive
* @author Martin Kouba
*/
public abstract class ArchiveBuilder, A extends Archive> {
private static final Logger logger = Logger.getLogger(ArchiveBuilder.class.getName());
private static final JavaArchive supportLibrary = buildSupportLibrary();
private static final JavaArchive incontainerLibrary = buildIncontainerLibrary();
public static final String DEFAULT_EJB_VERSION = "3.1";
public static final String DEBUG_MODE_PROPERTY = "debugMode";
private static final String DOLLAR_SIGN = "$";
private static final String INNER_CLASS_MARKER = "\\" + DOLLAR_SIGN;
private static final String INNER_CLASS_MATCH_PATTERN_BASE = ".*" + AssetUtil.DELIMITER_RESOURCE_PATH + "%s" + INNER_CLASS_MARKER + ".+\\.class";
private String name;
private Boolean isAsClientMode = null;
private boolean isTestArchive = true;
private Class> testClazz = null;
private boolean isDebugMode = false;
protected ResourceDescriptor beansXml = null;
protected BeansDescriptor beansDescriptor = null;
protected ResourceDescriptor webXml = null;
protected WebAppDescriptor webXmlDescriptor = null;
protected PersistenceDescriptor persistenceDescriptor = null;
protected EjbJarDescriptor ejbJarDescriptor = null;
protected ResourceDescriptor ejbJarXml = null;
protected List manifestResources = null;
protected List resources = null;
protected List webResources = null;
protected Set packages = null;
protected Set> classes = null;
protected Set excludedClasses = null;
protected List libraries = null;
protected List shrinkWrapLibraries = null;
protected List serviceProviders = null;
public void readSystemProperties() {
String debugMode = System.getProperty(DEBUG_MODE_PROPERTY);
if (debugMode != null && !("false".equals(debugMode))) {
debugMode();
}
}
/**
* Set the name of the archive.
*
* @param name
* @return
*/
public T withName(String name) {
this.name = name;
return self();
}
/**
* Add beans.xml
located in src/main/resource/{testPackagePath}.
*
*
* Do not use this in new tests - use {@link #withBeansXml(BeansDescriptor)} instead.
*
*
* @param beansXml
* @return self
*/
public T withBeansXml(String beansXml) {
this.beansXml = new ResourceDescriptor(beansXml, "beans.xml");
return self();
}
/**
* Add beans.xml
descriptor created with shrinkwrap-descriptors.
*
* @param beansDescriptor
* @return self
*/
public T withBeansXml(BeansDescriptor beansDescriptor) {
this.beansDescriptor = beansDescriptor;
return self();
}
/**
* Add CDI extension. This method does not add the specified extension class to the archive.
*
* @param extensionClass
* @return self
*/
public T withExtension(Class extends Extension> extensionClass) {
return withServiceProvider(new ServiceProviderDescriptor(Extension.class, extensionClass));
}
/**
* Add CDI extensions. This method does not add the specified extension classes to the archive.
*
* @param extensionClasses
* @return self
*/
public T withExtensions(Class extends Extension>... extensionClasses) {
return withServiceProvider(new ServiceProviderDescriptor(Extension.class, extensionClasses));
}
/**
* Add service provider.
*
* @param serviceProvider
* @return
*/
public T withServiceProvider(ServiceProviderDescriptor serviceProvider) {
if (serviceProviders == null)
serviceProviders = new ArrayList();
serviceProviders.add(serviceProvider);
return self();
}
/**
* Add class to archive.
*
* @param clazz
* @return self
*/
public T withClass(Class> clazz) {
if (this.classes == null)
this.classes = new HashSet>();
this.classes.add(clazz);
return self();
}
/**
* Add classes to archive.
*
* @param classes
* @return self
*/
public T withClasses(Class>... classes) {
for (Class> clazz : classes) {
withClass(clazz);
}
return self();
}
/**
* Specified class must be excluded from final archive unless also added via {@link #withClass(Class)} or {@link #withClasses(Class...)}. Useful for
* exluding some classes from package added via {@link #withPackage(Package)}.
*
* Avoid using this feature if possible - the implementation has negative performance effects.
*
* @param clazz Fully qualified class name
* @return self
*/
public T withExcludedClass(String clazz) {
if (this.excludedClasses == null)
this.excludedClasses = new HashSet();
this.excludedClasses.add(clazz);
return self();
}
/**
* Specified classes must be excluded from final archive unless also added via {@link #withClass(Class)} or {@link #withClasses(Class...)}. Useful for
* exluding some classes from package added via {@link #withPackage(Package)}.
*
* Avoid using this feature if possible - the implementation has negative performance effects.
*
* @param classes Fully qualified class names
* @return self
*/
public T withExcludedClasses(String... classes) {
for (String clazz : classes) {
withExcludedClass(clazz);
}
return self();
}
/**
* Add all classes in the test class package to archive and set test class definition for configuration purpose.
*
* @param testClazz
* @return self
*/
public T withTestClassPackage(Class> testClazz) {
return withTestClassDefinition(testClazz).withPackage(testClazz.getPackage());
}
/**
* Add test class to archive and set test class definition for configuration purpose.
*
* @param testClazz
* @return self
*/
public T withTestClass(Class> testClazz) {
return withTestClassDefinition(testClazz).withClass(testClazz);
}
/**
* Set test class definition for configuration purpose. Do not add it to final archive.
*
* Always use this for as-client test archives, e.g. deployment method annotated with {@link ShouldThrowException}.
*
* @param test
* @return self
*/
public T withTestClassDefinition(Class> testClazz) {
if (this.testClazz != null)
throw new IllegalStateException("Cannot set more than one test class definition!");
this.testClazz = testClazz;
return self();
}
/**
* Add package (that is its content). Subpackages are not included.
*
* @param pack
* @return self
*/
public T withPackage(Package pack) {
if (this.packages == null)
this.packages = new HashSet();
this.packages.add(pack);
return self();
}
/**
* Add resource to META-INF.
*
* @param source
* @return self
*/
public T withManifestResource(String source) {
return withManifestResource(source, null, true);
}
/**
* Add resource to META-INF.
*
* @param source
* @param useTestPackageToLocateSource
* @return self
*/
public T withManifestResource(String source, boolean useTestPackageToLocateSource) {
return withManifestResource(source, null, useTestPackageToLocateSource);
}
/**
* Add resource to META-INF.
*
* @param source
* @param target
* @param useTestPackageToLocateSource
* @return self
*/
public T withManifestResource(String source, String target, boolean useTestPackageToLocateSource) {
if (this.manifestResources == null)
this.manifestResources = new ArrayList();
this.manifestResources.add(new ResourceDescriptor(source, target, useTestPackageToLocateSource));
return self();
}
/**
* Add resource.
*
* @param source
* @return self
*/
public T withResource(String source) {
return withResource(source, null, true);
}
/**
* Add resource.
*
* @param source
* @param useTestPackageToLocateSource
* @return self
*/
public T withResource(String source, boolean useTestPackageToLocateSource) {
return withResource(source, null, useTestPackageToLocateSource);
}
/**
* Add resource.
*
* @param source
* @param target
* @param useTestPackageToLocateSource
* @return self
*/
public T withResource(String source, String target, boolean useTestPackageToLocateSource) {
if (this.resources == null)
this.resources = new ArrayList();
this.resources.add(new ResourceDescriptor(source, target, useTestPackageToLocateSource));
return self();
}
/**
* Add web resource.
*
* @param source
* @return self
*/
public T withWebResource(String source) {
return withWebResource(source, null, true);
}
/**
* Add web resource.
*
* @param source
* @param target
* @return self
*/
public T withWebResource(String source, String target) {
return withWebResource(source, target, true);
}
/**
* Add web resource.
*
* @param source
* @param useTestPackageToLocateSource
* @return self
*/
public T withWebResource(String source, boolean useTestPackageToLocateSource) {
return withWebResource(source, null, useTestPackageToLocateSource);
}
/**
* Add web resource.
*
* @param source
* @return
*/
public T withWebResource(String source, String target, boolean useTestPackageToLocateSource) {
if (this.webResources == null)
this.webResources = new ArrayList();
this.webResources.add(new ResourceDescriptor(source, target, useTestPackageToLocateSource));
return self();
}
/**
* Add ejb-jar.xml located in src/main/resource/{testPackagePath}.
*
* @param ejbJarXml
* @return
*/
public T withEjbJarXml(String ejbJarXml) {
this.ejbJarXml = new ResourceDescriptor(ejbJarXml, "ejb-jar.xml");
return self();
}
/**
* Add ejb-jar.xml descriptor created with shrinkwrap-descriptors.
*
* @param ejbJarXml
* @return
*/
public T withEjbJarXml(EjbJarDescriptor descriptor) {
if (descriptor.getVersion() == null) {
// CDITCK-316 always set the version attribute
descriptor.version(DEFAULT_EJB_VERSION);
}
this.ejbJarDescriptor = descriptor;
return self();
}
/**
* Add web.xml located in src/main/resource/{testPackagePath}.
*
*
* Do not use this in new tests - use {@link #withWebXml(WebAppDescriptor)} instead.
*
*
* @param webXml
* @return
*/
public T withWebXml(String webXml) {
this.webXml = new ResourceDescriptor(webXml);
return self();
}
/**
* Add web.xml
descriptor created with shrinkwrap-descriptors.
*
* @param webXml
* @return
*/
public T withWebXml(WebAppDescriptor webXml) {
if (webXml.getVersion() == null) {
// CDITCK-316 always set the version attribute
webXml.version(WebAppVersionType._3_0);
}
this.webXmlDescriptor = webXml;
return self();
}
/**
* Add default persistence.xml.
*
* @return self
*/
public T withDefaultPersistenceXml() {
return withPersistenceXml(Descriptors.create(PersistenceDescriptor.class).createPersistenceUnit().name("test")
.jtaDataSource(ConfigurationFactory.get().getTestDataSource()).up());
}
/**
* Add persistence.xml descriptor created with shrinkwrap-descriptors.
*
* @param persistenceXml
* @return self
*/
public T withPersistenceXml(PersistenceDescriptor persistenceDescriptor) {
if (persistenceDescriptor.getVersion() == null || persistenceDescriptor.getVersion().isEmpty()) {
// CDITCK-316 always set the version attribute
persistenceDescriptor.version("2.0");
}
this.persistenceDescriptor = persistenceDescriptor;
return self();
}
/**
* Add library.
*
* @param library
* @return self
*/
public T withLibrary(File library) {
if (this.libraries == null)
this.libraries = new ArrayList();
this.libraries.add(new LibraryDescriptor(library));
return self();
}
/**
* Add bean library that consists of defined bean classes; automatically include empty beans.xml
and if any of defined classes implements
* {@link Extension} add corresponding service provider.
*
* @param beanClasses
* @return self
*/
public T withBeanLibrary(Class>... beanClasses) {
return withLibrary(null, true, beanClasses);
}
/**
* Add bean library that consists of defined bean classes; automatically include empty beans.xml
and if any of defined classes implements
* {@link Extension} add corresponding service provider.
*
* @param beanClasses
* @return self
*/
public T withBeanLibrary(BeansDescriptor beansDescriptor, Class>... beanClasses) {
return withBeanLibrary(null, beansDescriptor, beanClasses);
}
/**
* Add bean library that consists of defined bean classes; automatically include empty beans.xml
and if any of defined classes implements
* {@link Extension} add corresponding service provider.
*
* @param beanClasses
* @return self
*/
public T withBeanLibrary(String name, Class>... beanClasses) {
return withBeanLibrary(name, null, beanClasses);
}
/**
* Add bean library that consists of defined bean classes; automatically include empty beans.xml
and if any of defined classes implements
* {@link Extension} add corresponding service provider.
*
* @param beanClasses
* @return self
*/
public T withBeanLibrary(String name, BeansDescriptor beansDescriptor, Class>... beanClasses) {
return withLibrary(name, beansDescriptor, true, beanClasses);
}
/**
* Add library that consists of defined classes.
*
* @param classes
* @return self
*/
public T withLibrary(Class>... classes) {
return withLibrary(null, false, classes);
}
/**
* Add library that consists of defined classes. Include empty beans.xml if necessary.
*
* @param serviceProvider
* @param omitBeansXml
* @param classes
* @return self
*/
public T withLibrary(BeansDescriptor beansDescriptor, boolean includeEmptyBeanXml, Class>... classes) {
return withLibrary(null, beansDescriptor, includeEmptyBeanXml, classes);
}
/**
* Add library that consists of defined classes. Include empty beans.xml if necessary.
*
* @param serviceProvider
* @param omitBeansXml
* @param classes
* @return self
*/
public T withLibrary(String name, BeansDescriptor beansDescriptor, boolean includeEmptyBeanXml, Class>... classes) {
if (libraries == null)
libraries = new ArrayList(5);
List> extensions = new ArrayList>();
for (Class> clazz : classes) {
if (Extension.class.isAssignableFrom(clazz)) {
extensions.add(clazz);
}
}
ServiceProviderDescriptor serviceProvider = extensions.isEmpty() ? null : new ServiceProviderDescriptor(Extension.class,
extensions.toArray(new Class>[extensions.size()]));
this.libraries.add(beansDescriptor != null ? new LibraryDescriptor(name, serviceProvider, beansDescriptor, classes) : new LibraryDescriptor(name,
serviceProvider, includeEmptyBeanXml, classes));
return self();
}
/**
* Add the specified ShrinkWrap library.
*
* @param library
* @return self
*/
public T withLibrary(JavaArchive library) {
if (shrinkWrapLibraries == null)
shrinkWrapLibraries = new ArrayList(5);
shrinkWrapLibraries.add(library);
return self();
}
/**
* Add specified ShrinkWrap libraries.
*
* @param libraries
* @return
*/
public T withLibraries(JavaArchive... libraries) {
for (JavaArchive library : libraries) {
withLibrary(library);
}
return self();
}
/**
* @return true
if building as-client mode archive, false
otherwise
*/
public Boolean isAsClientMode() {
return isAsClientMode != null ? isAsClientMode : false;
}
/**
* @param isAsClientMode
* @see #resolveAsClientMode()
*/
public T setAsClientMode(boolean isAsClientMode) {
this.isAsClientMode = isAsClientMode;
return self();
}
/**
* @return true
if TCK specific infrastructure (porting package, utils, etc.) should be automatically added, false
otherwise
* @see #resolveAsClientMode()
*/
public boolean isTestArchive() {
return isTestArchive;
}
/**
* Mark this archive as non-testing. TCK specific infrastructure (porting package, utils, etc.) will not be automatically added.
*/
public T notTestArchive() {
this.isTestArchive = false;
return self();
}
/**
* Enable debug mode. Basically shows the content of the default beans.xml and built ShrinkWrap archive in the log.
*/
public T debugMode() {
this.isDebugMode = true;
return self();
}
/**
* @return name of final archive
*/
public String getName() {
return name;
}
/**
* @return self to enable generic builder
*/
public abstract T self();
/**
* @return ShrinkWrap archive
*/
public A build() {
readSystemProperties();
long start = System.currentTimeMillis();
if (isTestArchive()) {
if (testClazz == null)
throw new IllegalStateException("Test class must be set!");
resolveAsClientMode();
// Support library
withLibrary(supportLibrary);
// Default libraries
addDefaultLibraries();
if (!isAsClientMode()) {
withLibrary(incontainerLibrary);
}
}
A archive = buildInternal();
if (this.isDebugMode) {
logger.info(archive.toString(true));
}
logger.log(Level.INFO, "Test archive built [info: {0}, time: {1} ms]",
new Object[] { testClazz != null ? testClazz.getName() : archive.getName(), Long.valueOf(System.currentTimeMillis() - start) });
return archive;
}
/**
* @return concrete shrinkwrap archive
*/
protected abstract A buildInternal();
/**
* Process packages. Exclude classes specified via {@link #withExcludedClass(Class)}. If in as-client mode, filter test class.
*
* @param archive
*/
protected & ClassContainer>> void processPackages(final P archive) {
if (packages == null)
return;
// Avoid using URLPackageScanner if possible - it has negative
// performance effects
if ((excludedClasses == null || excludedClasses.isEmpty()) && !isAsClientMode()) {
archive.addPackages(false, packages.toArray(new Package[packages.size()]));
} else {
final String testClazzName = testClazz.getName();
final ClassLoader cl = testClazz.getClassLoader();
final ClassLoader clToUse = (cl != null ? cl : ClassLoader.getSystemClassLoader());
final URLPackageScanner.Callback callback = new URLPackageScanner.Callback() {
@Override
public void classFound(String className) {
if (isAsClientMode() && (testClazzName.equals(className) || className.startsWith(testClazzName))) {
return;
}
if (excludedClasses != null && !excludedClasses.isEmpty()) {
for (String exludeClassName : excludedClasses) {
// Handle annonymous inner classes
if (className.equals(exludeClassName))
return;
else if (className.startsWith(exludeClassName) && className.contains("$"))
return;
}
}
// This is a performance WORKAROUND - adding classes via
// ClassContainer.addClass() is much slower
// Actually the performace loss is caused by inner classes
// handling - which is not necessary here
// See also
// org.jboss.shrinkwrap.impl.base.container.ContainerBase
// addPackage() and getClassesPath() methods
ArchivePath classesPath = resolveClassesPath(archive);
if (classesPath != null) {
ArchivePath classNamePath = AssetUtil.getFullPathForClassResource(className);
archive.add(new ClassLoaderAsset(classNamePath.get().substring(1), clToUse), ArchivePaths.create(classesPath, classNamePath));
} else {
archive.addClass(className);
}
}
};
for (Package pack : packages) {
URLPackageScanner.newInstance(false, clToUse, callback, pack.getName()).scanPackage();
}
}
}
/**
* Process classes.
*
* @param archive
*/
protected
& ClassContainer>> void processClasses(P archive) {
if (classes == null || classes.isEmpty()) {
return;
}
// Optimize if all classes come from the same package
if (isSinglePackage(classes)) {
Package classesPackage = null;
final List simpleNames = new ArrayList(classes.size());
for (Class> clazz : classes) {
if (skipClassName(clazz.getName())) {
continue;
}
simpleNames.add(clazz.getSimpleName());
if (classesPackage == null) {
classesPackage = clazz.getPackage();
}
Asset resource = new ClassAsset(clazz);
ArchivePath location = ArchivePaths.create(resolveClassesPath(archive), AssetUtil.getFullPathForClassResource(clazz));
archive.add(resource, location);
}
if (simpleNames.isEmpty()) {
return;
}
// Quite naive way of handling inner classes
// The reason for this is that similar code would be normally called
// for each class - see also ContainerBase.addClasses()
archive.addPackages(false, new Filter() {
@Override
public boolean include(ArchivePath path) {
String pathStr = path.get();
// Filter out irrelevant filenames
for (String simpleName : simpleNames) {
if (isInnerClassCandidate(simpleName, pathStr)) {
// Match inner class filename
// .*/SimpleName\\$.+\\.class
if (pathStr.matches(String.format(INNER_CLASS_MATCH_PATTERN_BASE, simpleName))) {
return true;
}
}
}
return false;
}
}, classesPackage);
} else {
for (Class> clazz : classes) {
if (skipClassName(clazz.getName())) {
continue;
}
archive.addClass(clazz);
}
}
}
/**
* Process libraries.
*
* @param archive
*/
protected void processLibraries(LibraryContainer> archive) {
if (libraries != null) {
for (LibraryDescriptor descriptor : libraries) {
if (descriptor.getFileDescriptor() != null) {
archive.addAsLibrary(descriptor.getFileDescriptor());
} else {
archive.addAsLibrary(descriptor.buildJarArchive());
}
}
}
if (shrinkWrapLibraries != null) {
archive.addAsLibraries(shrinkWrapLibraries);
}
}
/**
* Process resources.
*
* @param archive
*/
protected void processResources(ResourceContainer> archive) {
if (resources == null)
return;
for (ResourceDescriptor resource : resources) {
if (resource.getTarget() == null) {
archive.addAsResource(resource.getSource());
} else {
archive.addAsResource(resource.getSource(), resource.getTarget());
}
}
}
/**
* Process manifest resources.
*
* @param archive
*/
protected void processManifestResources(ManifestContainer> archive) {
if (manifestResources != null) {
for (ResourceDescriptor resource : manifestResources) {
if (resource.getTarget() == null) {
archive.addAsManifestResource(resource.getSource());
} else {
archive.addAsManifestResource(resource.getSource(), resource.getTarget());
}
}
}
if (serviceProviders != null) {
for (ServiceProviderDescriptor descriptor : serviceProviders) {
archive.addAsServiceProvider(descriptor.getServiceInterface(), descriptor.getServiceImplementations());
}
}
}
/**
* @return corresponding beans descriptor asset
*/
protected Asset getBeansDescriptorAsset() {
Asset asset = null;
if (beansDescriptor != null) {
if (beansDescriptor.getBeanDiscoveryMode() == null || beansDescriptor.getBeanDiscoveryMode().isEmpty()) {
beansDescriptor.beanDiscoveryMode(BeanDiscoveryMode._ALL.toString()).version(Versions.v1_1);
}
asset = new StringAsset(beansDescriptor.exportAsString());
} else if (beansXml != null) {
asset = new ClassLoaderAsset(beansXml.getSource());
} else {
asset = new StringAsset(
Descriptors.create(BeansDescriptor.class).beanDiscoveryMode(BeanDiscoveryMode._ALL.toString()).version(Versions.v1_1).exportAsString());
}
if (this.isDebugMode) {
logger.info("\n" + AssetUtil.readAssetContent(asset));
}
return asset;
}
/**
* @return
*/
protected String getBeansDescriptorTarget() {
String target = null;
if (beansDescriptor != null) {
target = beansDescriptor.getDescriptorName();
} else if (beansXml != null) {
target = beansXml.getTarget();
} else {
target = "beans.xml";
}
return target;
}
/**
* Internal service provider descriptor.
*
* @author Martin Kouba
*/
protected class ServiceProviderDescriptor {
private Class> serviceInterface;
private Class>[] serviceImplementations;
public ServiceProviderDescriptor(Class> serviceInterface, Class>... serviceImplementations) {
super();
this.serviceInterface = serviceInterface;
this.serviceImplementations = serviceImplementations;
}
public Class> getServiceInterface() {
return serviceInterface;
}
public Class>[] getServiceImplementations() {
return serviceImplementations;
}
}
/**
* Internal resource descriptor.
*
* @author Martin Kouba
*/
protected class ResourceDescriptor {
private String source;
private String target;
private boolean useTestPackageToLocateSource = true;
public ResourceDescriptor(String source) {
super();
this.source = source;
}
public ResourceDescriptor(String source, String target) {
super();
this.source = source;
this.target = target;
}
public ResourceDescriptor(String source, String target, boolean useTestPackageToLocateSource) {
super();
this.source = source;
this.target = target;
this.useTestPackageToLocateSource = useTestPackageToLocateSource;
}
public String getSource() {
return useTestPackageToLocateSource ? getTestPackagePath() + source : source;
}
public String getPlainSource() {
return source;
}
public String getTarget() {
return target;
}
}
/**
* Internal library descriptor.
*
* @author Martin Kouba
*/
protected class LibraryDescriptor {
private String name;
private File fileDescriptor = null;
private List> libraryClasses = null;
protected BeansDescriptor beansDescriptor = null;
private boolean includeEmptyBeanXml = false;
private List serviceProviders;
public LibraryDescriptor(File fileDescriptor) {
super();
this.fileDescriptor = fileDescriptor;
}
public LibraryDescriptor(String name, ServiceProviderDescriptor serviceProvider, BeansDescriptor beansDescriptor, Class>... classes) {
super();
if (serviceProvider != null) {
this.serviceProviders = new ArrayList();
this.serviceProviders.add(serviceProvider);
}
this.beansDescriptor = beansDescriptor;
if (beansDescriptor.getBeanDiscoveryMode() == null || beansDescriptor.getBeanDiscoveryMode().isEmpty()) {
beansDescriptor.beanDiscoveryMode(BeanDiscoveryMode._ALL.toString()).version(Versions.v1_1);
}
this.libraryClasses = Arrays.asList(classes);
this.name = name;
}
/**
* @param name
* @param serviceProvider
* @param includeEmptyBeanXml Automatically include empty beans.xml to promote the lib to BDA
* @param classes
*/
public LibraryDescriptor(String name, ServiceProviderDescriptor serviceProvider, boolean includeEmptyBeanXml, Class>... classes) {
super();
if (serviceProvider != null) {
this.serviceProviders = new ArrayList();
this.serviceProviders.add(serviceProvider);
}
this.includeEmptyBeanXml = includeEmptyBeanXml;
this.libraryClasses = Arrays.asList(classes);
this.name = name;
}
public String getName() {
return name;
}
public List> getBeanClasses() {
return libraryClasses;
}
public ResourceDescriptor getBeansXml() {
return beansXml;
}
public boolean isOmitBeanXml() {
return includeEmptyBeanXml;
}
public File getFileDescriptor() {
return fileDescriptor;
}
public List getServiceProviders() {
return serviceProviders;
}
/**
* @return shrinkwrap jar archive
*/
public JavaArchive buildJarArchive() {
JavaArchive library = null;
if (name != null) {
library = ShrinkWrap.create(JavaArchive.class, name);
} else {
library = ShrinkWrap.create(JavaArchive.class);
}
for (Class> clazz : libraryClasses) {
library.addClass(clazz);
}
if (serviceProviders != null) {
for (ServiceProviderDescriptor serviceProvider : serviceProviders) {
library.addAsServiceProvider(serviceProvider.getServiceInterface(), serviceProvider.getServiceImplementations());
}
}
if (beansDescriptor != null) {
library.addAsManifestResource(new StringAsset(beansDescriptor.exportAsString()), beansDescriptor.getDescriptorName());
} else if (includeEmptyBeanXml) {
library.addAsManifestResource(EmptyAsset.INSTANCE, ArchivePaths.create("beans.xml"));
}
return library;
}
}
/**
* Add default libraries from lib directory specified with org.jboss.cdi.tck.libraryDirectory
property in cdi-tck.properties.
*/
private void addDefaultLibraries() {
File directory = new File(ConfigurationFactory.get(true).getLibraryDirectory());
logger.log(Level.FINE, "Scanning default library dir {0}", directory.getPath());
if (directory.isDirectory()) {
for (File file : directory.listFiles(new FilenameFilter() {
public boolean accept(File dir, String name) {
return name.endsWith(".jar");
}
})) {
logger.log(Level.FINE, "Adding default library {0}", file.getName());
withLibrary(file);
}
}
}
private boolean skipClassName(String className) {
return ((isAsClientMode() && testClazz.getName().equals(className)) || (excludedClasses != null && excludedClasses.contains(className)));
}
private boolean isSinglePackage(Set> classes) {
String packageName = null;
for (Class> clazz : classes) {
if (packageName == null) {
packageName = getPackageName(clazz);
} else {
if (!packageName.equals(getPackageName(clazz))) {
return false;
}
}
}
return true;
}
private String getPackageName(Class> clazz) {
return clazz.getPackage() != null ? clazz.getPackage().getName() : "";
}
private String getTestPackagePath() {
return this.testClazz.getPackage().getName().replace('.', '/').concat("/");
}
/**
* Try to resolve as-client mode automatically unless it was set already.
*
* Set as-client mode to true
provided that:
*
* - test class is annotated with {@link RunAsClient}
* - the deployment method on test class definition is annotated with {@link ShouldThrowException} or {@link Deployment#testable()} is false
*
*
* @throws IllegalStateException If multiple deployments detected and as-client mode not set
* @see #setAsClientMode(boolean)
*/
private void resolveAsClientMode() {
if (this.isAsClientMode != null) {
return;
}
if (testClazz.isAnnotationPresent(RunAsClient.class)) {
setAsClientMode(true);
}
Method deploymentMethod = null;
for (Method method : testClazz.getMethods()) {
if (Modifier.isStatic(method.getModifiers()) && method.isAnnotationPresent(Deployment.class)) {
if (deploymentMethod != null && this.isAsClientMode == null) {
throw new IllegalStateException("Multi-deployment test class and as-client mode not set");
}
deploymentMethod = method;
if (method.isAnnotationPresent(ShouldThrowException.class) || !method.getAnnotation(Deployment.class).testable()) {
setAsClientMode(true);
}
}
}
}
/**
* @param archive
* @return Base Path for the ClassContainer resources
*/
private ArchivePath resolveClassesPath(Archive> archive) {
if (archive instanceof WebArchive) {
return ArchivePaths.create("WEB-INF/classes");
} else if (archive instanceof JavaArchive) {
return ArchivePaths.create("/");
}
return null;
}
private static JavaArchive buildSupportLibrary() {
long start = System.currentTimeMillis();
JavaArchive supportLib = ShrinkWrap.create(JavaArchive.class, "cdi-tck-support.jar")
// CDI TCK properties
.addAsResource(PropertiesBasedConfigurationBuilder.RESOURCE_BUNDLE)
// org.jboss.cdi.tck.api
.addPackage(Configuration.class.getPackage())
// org.jboss.cdi.tck.spi
.addPackage(Beans.class.getPackage())
// org.jboss.cdi.tck.impl
.addClasses(ConfigurationFactory.class, ConfigurationImpl.class, PropertiesBasedConfigurationBuilder.class)
// Util packages
.addPackage(Timer.class.getPackage()).addPackage(AnnotatedWrapper.class.getPackage());
logger.log(Level.INFO, "Support library built [time: {0} ms]", System.currentTimeMillis() - start);
return supportLib;
}
private static JavaArchive buildIncontainerLibrary() {
long start = System.currentTimeMillis();
JavaArchive supportLib = ShrinkWrap.create(JavaArchive.class, "cdi-tck-incontainer.jar").addClasses(AbstractTest.class, Sections.class,
TestGroups.class);
logger.log(Level.INFO, "Incontainer library built [time: {0} ms]", System.currentTimeMillis() - start);
return supportLib;
}
private boolean isInnerClassCandidate(String simpleName, String path) {
return path.contains(DOLLAR_SIGN) && path.contains(simpleName);
}
protected String getSha1OfTestClass() {
if (testClazz != null) {
MessageDigest messageDigest = null;
try {
messageDigest = MessageDigest.getInstance("SHA-1");
} catch (NoSuchAlgorithmException e) {
return null;
}
messageDigest.update(testClazz.getName().getBytes());
byte[] digest = messageDigest.digest();
StringBuilder hexString = new StringBuilder();
for (int i = 0; i < digest.length; i++) {
hexString.append(Integer.toHexString(0xFF & digest[i]));
}
return hexString.toString();
}
return null;
}
}