/* * 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.spi.deployer; import java.io.InputStream; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.jboss.deployers.vfs.spi.structure.VFSDeploymentUnit; import org.jboss.util.xml.JBossEntityResolver; import org.jboss.virtual.VirtualFile; import org.w3c.dom.Document; import org.xml.sax.InputSource; /** * JAXPDeployer is an abstract deployer that overrides parse to run a jaxp * parse of the VirtualFile passed to parse. * * @see #parse(VFSDeploymentUnit, VirtualFile, Object) * @see #doParse(VFSDeploymentUnit, VirtualFile) * * @param <T> the expected type * @author <a href="adrian@jboss.com">Adrian Brock</a> * @author Scott.Stark@jboss.org * @author <a href="ales.justin@jboss.com">Ales Justin</a> * @version $Revision: 60707 $ */ public abstract class JAXPDeployer<T> extends UnmarshallerFactoryDeployer<T, Boolean> { /** Use a namespace aware parser */ private boolean useNamespaceAwareParser = true; /** A flag indicating if deployment descriptors should be validated */ private boolean validateDTDs; /** The document builder factory */ private DocumentBuilderFactory documentBuilderFactory; /** * Create a new JAXPDeployer. * * @param output the output * @throws IllegalArgumentException for a null output */ public JAXPDeployer(Class<T> output) { super(output); } /** * Get the useNamespaceAwareParser. * * @return the useNamespaceAwareParser. */ public boolean isUseNamespaceAwareParser() { return useNamespaceAwareParser; } /** * Set the useNamespaceAwareParser. * * @param useNamespaceAwareParser the useNamespaceAwareParser. */ public void setUseNamespaceAwareParser(boolean useNamespaceAwareParser) { this.useNamespaceAwareParser = useNamespaceAwareParser; } /** * Get the validateDTDs. * * @return the validateDTDs. */ public boolean isValidateDTDs() { return validateDTDs; } /** * Set the validateDTDs. * * @param validateDTDs the validateDTDs. */ public void setValidateDTDs(boolean validateDTDs) { this.validateDTDs = validateDTDs; } /** * Get the documentBuilderFactory. * * @return the documentBuilderFactory. * @throws IllegalStateException if the create method has not been invoked */ protected DocumentBuilderFactory getDocumentBuilderFactory() { if (documentBuilderFactory == null) throw new IllegalStateException("Document builder factory has not been constructed"); return documentBuilderFactory; } /** * Create lifecycle * * @throws Exception for any problem */ public void create() throws Exception { documentBuilderFactory = DocumentBuilderFactory.newInstance(); documentBuilderFactory.setNamespaceAware(useNamespaceAwareParser); documentBuilderFactory.setValidating(validateDTDs); } /** * Destroy lifecycle */ public void destroy() { documentBuilderFactory = null; } protected UnmarshallerFactory<Boolean> createUnmarshallerFactory() { return new UnmarshallerFactory<Boolean>() { public void setFeature(String featureName, Boolean flag) throws Exception { getDocumentBuilderFactory().setFeature(featureName, flag); } }; } protected Boolean fromString(String value) { return Boolean.valueOf(value); } @Override protected T parse(VFSDeploymentUnit unit, VirtualFile file, T root) throws Exception { // First look for an existing Document attachment Document document = unit.getAttachment(Document.class); if( document == null ) { // Next parse the metadata file document = doParse(unit, file); } // Transform the document into a T instance return parse(unit, file, document); } /** * Do the parsing * * @param unit the deployment unit * @param file the metadata file * @return the document * @throws Exception for any error */ protected Document doParse(VFSDeploymentUnit unit, VirtualFile file) throws Exception { if (file == null) throw new IllegalArgumentException("Null file"); log.debug("Parsing: " + file.getName()); DocumentBuilder parser = getDocumentBuilderFactory().newDocumentBuilder(); InputStream is = openStreamAndValidate(file); try { InputSource source = new InputSource(is); source.setSystemId(file.toURI().toString()); parser.setEntityResolver(new JBossEntityResolver()); return parser.parse(source); } finally { try { is.close(); } catch (Exception ignored) { } } } /** * Parse a deployment * * @param unit the deployment unit * @param file the metadata file * @param document the document * @return the metadata * @throws Exception for any error */ protected abstract T parse(VFSDeploymentUnit unit, VirtualFile file, Document document) throws Exception; }