/* * JBoss, Home of Professional Open Source. * Copyright 2011, Red Hat, Inc., and individual contributors * as indicated by the @author tags. See the copyright.txt file 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.as.controller.parsing; import static javax.xml.stream.XMLStreamConstants.END_ELEMENT; import static org.jboss.as.controller.logging.ControllerLogger.ROOT_LOGGER; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.ADD; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.EXTENSION; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP_ADDR; import static org.jboss.as.controller.parsing.ParseUtils.invalidAttributeValue; import static org.jboss.as.controller.parsing.ParseUtils.readStringAttributeElement; import static org.jboss.as.controller.parsing.ParseUtils.requireNamespace; import static org.jboss.as.controller.parsing.ParseUtils.requireNoAttributes; import static org.jboss.as.controller.parsing.ParseUtils.unexpectedElement; import java.util.HashSet; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Set; import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; import java.util.concurrent.Future; import javax.xml.stream.XMLStreamException; import org.jboss.as.controller.logging.ControllerLogger; import org.jboss.as.controller.Extension; import org.jboss.as.controller.extension.ExtensionRegistry; import org.jboss.dmr.ModelNode; import org.jboss.modules.Module; import org.jboss.modules.ModuleIdentifier; import org.jboss.modules.ModuleLoadException; import org.jboss.modules.ModuleLoader; import org.jboss.staxmapper.XMLExtendedStreamReader; import org.jboss.staxmapper.XMLExtendedStreamWriter; import org.jboss.staxmapper.XMLMapper; import org.wildfly.security.manager.WildFlySecurityManager; /** * Parsing and marshalling logic related to the {@code extension} element in standalone.xml and domain.xml. * * @author <a href="mailto:david.lloyd@redhat.com">David M. Lloyd</a> * @author <a href="mailto:darran.lofthouse@jboss.com">Darran Lofthouse</a> * @author Brian Stansberry (c) 2011 Red Hat Inc. */ public class ExtensionXml { private final ModuleLoader moduleLoader; private final ExecutorService bootExecutor; private final ExtensionRegistry extensionRegistry; public ExtensionXml(final ModuleLoader loader, final ExecutorService executorService, final ExtensionRegistry extensionRegistry) { moduleLoader = loader; bootExecutor = executorService; this.extensionRegistry = extensionRegistry; } public void writeExtensions(final XMLExtendedStreamWriter writer, final ModelNode modelNode) throws XMLStreamException { Set<String> keys = modelNode.keys(); if (keys.size() > 0) { writer.writeStartElement(Element.EXTENSIONS.getLocalName()); for (final String extension : keys) { writer.writeEmptyElement(Element.EXTENSION.getLocalName()); writer.writeAttribute(Attribute.MODULE.getLocalName(), extension); } writer.writeEndElement(); } } public void parseExtensions(final XMLExtendedStreamReader reader, final ModelNode address, final Namespace expectedNs, final List<ModelNode> list) throws XMLStreamException { long start = System.currentTimeMillis(); requireNoAttributes(reader); final Set<String> found = new HashSet<String>(); final XMLMapper xmlMapper = reader.getXMLMapper(); final Map<String, Future<XMLStreamException>> loadFutures = bootExecutor != null ? new LinkedHashMap<String, Future<XMLStreamException>>() : null; while (reader.hasNext() && reader.nextTag() != END_ELEMENT) { requireNamespace(reader, expectedNs); final Element element = Element.forName(reader.getLocalName()); if (element != Element.EXTENSION) { throw unexpectedElement(reader); } // One attribute && require no content final String moduleName = readStringAttributeElement(reader, Attribute.MODULE.getLocalName()); if (!found.add(moduleName)) { // duplicate module name throw invalidAttributeValue(reader, 0); } if (loadFutures != null) { // Load the module asynchronously Callable<XMLStreamException> callable = new Callable<XMLStreamException>() { @Override public XMLStreamException call() throws Exception { return loadModule(moduleName, xmlMapper); } }; Future<XMLStreamException> future = bootExecutor.submit(callable); loadFutures.put(moduleName, future); } else { // Load the module from this thread XMLStreamException xse = loadModule(moduleName, xmlMapper); if (xse != null) { throw xse; } addExtensionAddOperation(address, list, moduleName); } } if (loadFutures != null) { for (Map.Entry<String, Future<XMLStreamException>> entry : loadFutures.entrySet()) { try { XMLStreamException xse = entry.getValue().get(); if (xse != null) { throw xse; } } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw ControllerLogger.ROOT_LOGGER.moduleLoadingInterrupted(entry.getKey()); } catch (ExecutionException e) { throw ControllerLogger.ROOT_LOGGER.failedToLoadModule(e, entry.getKey()); } addExtensionAddOperation(address, list, entry.getKey()); } } long elapsed = System.currentTimeMillis() - start; if (ROOT_LOGGER.isDebugEnabled()) { ROOT_LOGGER.debugf("Parsed extensions in [%d] ms", elapsed); } } private void addExtensionAddOperation(ModelNode address, List<ModelNode> list, String moduleName) { final ModelNode add = new ModelNode(); add.get(OP_ADDR).set(address).add(EXTENSION, moduleName); add.get(OP).set(ADD); list.add(add); } private XMLStreamException loadModule(final String moduleName, final XMLMapper xmlMapper) throws XMLStreamException { // Register element handlers for this extension try { final Module module = moduleLoader.loadModule(ModuleIdentifier.fromString(moduleName)); boolean initialized = false; for (final Extension extension : module.loadService(Extension.class)) { ClassLoader oldTccl = WildFlySecurityManager.setCurrentContextClassLoaderPrivileged(extension.getClass()); try { extension.initializeParsers(extensionRegistry.getExtensionParsingContext(moduleName, xmlMapper)); } finally { WildFlySecurityManager.setCurrentContextClassLoaderPrivileged(oldTccl); } if (!initialized) { initialized = true; } } if (!initialized) { throw ControllerLogger.ROOT_LOGGER.notFound("META-INF/services/", Extension.class.getName(), module.getIdentifier()); } return null; } catch (final ModuleLoadException e) { throw ControllerLogger.ROOT_LOGGER.failedToLoadModule(e); } } }