/**
* Copyright 2015-2016 Red Hat, Inc, and 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.wildfly.swarm.jaxrs.internal;
import java.io.IOException;
import java.io.InputStream;
import java.util.Map;
import java.util.logging.Logger;
import javax.ws.rs.core.Application;
import org.jboss.shrinkwrap.api.Archive;
import org.jboss.shrinkwrap.api.ArchiveEvent;
import org.jboss.shrinkwrap.api.ArchiveEventHandler;
import org.jboss.shrinkwrap.api.ArchivePath;
import org.jboss.shrinkwrap.api.ArchivePaths;
import org.jboss.shrinkwrap.api.Node;
import org.jboss.shrinkwrap.api.asset.ArchiveAsset;
import org.jboss.shrinkwrap.api.asset.Asset;
import org.jboss.shrinkwrap.api.asset.ByteArrayAsset;
import org.jboss.shrinkwrap.impl.base.container.WebContainerBase;
import org.jboss.shrinkwrap.impl.base.spec.WebArchiveImpl;
import org.objectweb.asm.ClassReader;
import org.wildfly.swarm.jaxrs.JAXRSArchive;
import org.wildfly.swarm.jaxrs.JAXRSMessages;
import org.wildfly.swarm.undertow.descriptors.WebXmlAsset;
/**
* @author Bob McWhirter
* @author Ken Finnigan
*/
public class JAXRSArchiveImpl extends WebContainerBase<JAXRSArchive> implements JAXRSArchive {
// -------------------------------------------------------------------------------------||
// Class Members ----------------------------------------------------------------------||
// -------------------------------------------------------------------------------------||
/**
* Create a new JAXRS Archive with any type storage engine as backing.
*
* @param delegate The storage backing.
*/
public JAXRSArchiveImpl(Archive<?> delegate) throws IOException {
super(JAXRSArchive.class, delegate);
addGeneratedApplication();
addFaviconExceptionHandler();
}
@Override
public JAXRSArchive addResource(Class<?> resource) {
addClass(resource);
return covarientReturn();
}
private static boolean hasApplicationPathAnnotation(ArchivePath path, Asset asset) {
if (asset == null) {
return false;
}
if (asset instanceof ArchiveAsset) {
return hasApplicationPathAnnotation(((ArchiveAsset) asset).getArchive());
}
if (!path.get().endsWith(".class")) {
return false;
}
try (InputStream in = asset.openStream()) {
ClassReader reader = new ClassReader(in);
ApplicationPathAnnotationSeekingClassVisitor visitor = new ApplicationPathAnnotationSeekingClassVisitor();
reader.accept(visitor, 0);
return visitor.isFound();
} catch (IOException ignored) {
}
return false;
}
private static boolean hasApplicationPathAnnotation(Archive<?> archive) {
Map<ArchivePath, Node> content = archive.getContent();
for (Map.Entry<ArchivePath, Node> entry : content.entrySet()) {
Node node = entry.getValue();
Asset asset = node.getAsset();
if (hasApplicationPathAnnotation(node.getPath(), asset)) {
return true;
}
}
return false;
}
private static boolean hasApplicationServletMapping(Archive<?> archive) {
Node webXmlNode = archive.get(PATH_WEB_XML);
if (webXmlNode != null) {
return hasApplicationServletMapping(webXmlNode.getAsset());
}
return false;
}
private static boolean hasApplicationServletMapping(Asset asset) {
if (asset == null) {
return false;
}
WebXmlAsset webXmlAsset;
if (asset instanceof WebXmlAsset) {
webXmlAsset = (WebXmlAsset) asset;
} else {
try {
webXmlAsset = new WebXmlAsset(asset.openStream());
} catch (Exception e) {
JAXRSMessages.MESSAGES.unableToParseWebXml(e);
return false;
}
}
return !webXmlAsset.getServletMapping(Application.class.getName()).isEmpty();
}
/**
* See also JAX-RS spec, section 2.3.2 Servlet.
*
* @param archive
* @return <code>true</code> if there is an {@link javax.ws.rs.core.Application} subclass annotated with {@link javax.ws.rs.ApplicationPath} or web.xml with
* mapping for <code>javax.ws.rs.core.Application</code> servlet, <code>false</code> otherwise
*/
private boolean hasApplicationPathOrServletMapping(Archive<?> archive) {
return hasApplicationServletMapping(archive) || hasApplicationPathAnnotation(archive);
}
protected void addGeneratedApplication() throws IOException {
if (!hasApplicationPathOrServletMapping(getArchive())) {
String name = "org.wildfly.swarm.generated.WildFlySwarmDefaultJAXRSApplication";
String path = "WEB-INF/classes/" + name.replace('.', '/') + ".class";
byte[] generatedApp;
generatedApp = ApplicationFactory2.create(name, "/");
add(new ByteArrayAsset(generatedApp), path);
addHandlers(new ApplicationHandler(this, path));
}
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.ContainerBase#getManifestPath()
*/
@Override
protected ArchivePath getManifestPath() {
return PATH_MANIFEST;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.ContainerBase#getClassesPath()
*/
@Override
protected ArchivePath getClassesPath() {
return PATH_CLASSES;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.ContainerBase#getResourcePath()
*/
@Override
protected ArchivePath getResourcePath() {
return PATH_RESOURCE;
}
public static boolean isJAXRS(Archive<?> archive) {
Map<ArchivePath, Node> content = archive.getContent();
for (Map.Entry<ArchivePath, Node> entry : content.entrySet()) {
Node node = entry.getValue();
Asset asset = node.getAsset();
if (isJAXRS(node.getPath(), asset)) {
return true;
}
}
return false;
}
private static boolean isJAXRS(ArchivePath path, Asset asset) {
if (asset == null) {
return false;
}
if (asset instanceof ArchiveAsset) {
return isJAXRS(((ArchiveAsset) asset).getArchive());
}
if (!path.get().endsWith(".class")) {
return false;
}
try (InputStream in = asset.openStream()) {
ClassReader reader = new ClassReader(in);
JAXRSAnnotationSeekingClassVisitor visitor = new JAXRSAnnotationSeekingClassVisitor();
reader.accept(visitor, 0);
return visitor.isFound();
} catch (IOException ignored) {
}
return false;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.ContainerBase#getLibraryPath()
*/
@Override
protected ArchivePath getLibraryPath() {
return PATH_LIBRARY;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.WebContainerBase#getWebPath()
*/
@Override
protected ArchivePath getWebPath() {
return PATH_WEB;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.WebContainerBase#getWebInfPath()
*/
@Override
protected ArchivePath getWebInfPath() {
return PATH_WEB_INF;
}
/**
* {@inheritDoc}
*
* @see org.jboss.shrinkwrap.impl.base.container.WebContainerBase#getWebInfPath()
*/
@Override
protected ArchivePath getServiceProvidersPath() {
return PATH_SERVICE_PROVIDERS;
}
// -------------------------------------------------------------------------------------||
// Required Implementations -----------------------------------------------------------||
// -------------------------------------------------------------------------------------||
@SuppressWarnings("unused")
private static final Logger log = Logger.getLogger(WebArchiveImpl.class.getName());
/**
* Path to the web inside of the Archive.
*/
private static final ArchivePath PATH_WEB = ArchivePaths.root();
/**
* Path to the WEB-INF inside of the Archive.
*/
private static final ArchivePath PATH_WEB_INF = ArchivePaths.create("WEB-INF");
/**
* Path to the web.xml descriptor.
*/
private static final ArchivePath PATH_WEB_XML = ArchivePaths.create(PATH_WEB_INF, "web.xml");
/**
* Path to the resources inside of the Archive.
*/
private static final ArchivePath PATH_RESOURCE = ArchivePaths.create(PATH_WEB_INF, "classes");
/**
* Path to the libraries inside of the Archive.
*/
private static final ArchivePath PATH_LIBRARY = ArchivePaths.create(PATH_WEB_INF, "lib");
/**
* Path to the classes inside of the Archive.
*/
private static final ArchivePath PATH_CLASSES = ArchivePaths.create(PATH_WEB_INF, "classes");
/**
* Path to the manifests inside of the Archive.
*/
private static final ArchivePath PATH_MANIFEST = ArchivePaths.create("META-INF");
/**
* Path to web archive service providers.
*/
private static final ArchivePath PATH_SERVICE_PROVIDERS = ArchivePaths.create(PATH_CLASSES, "META-INF/services");
public static class ApplicationHandler implements ArchiveEventHandler {
public ApplicationHandler(JAXRSArchive archive, String path) {
this.archive = archive;
this.path = path;
}
@Override
public void handle(ArchiveEvent event) {
Asset asset = event.getAsset();
if ((PATH_WEB_XML.equals(event.getPath()) && hasApplicationServletMapping(asset))
|| hasApplicationPathAnnotation(event.getPath(), asset)) {
this.archive.delete(this.path);
}
}
private final JAXRSArchive archive;
private final String path;
}
}