All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.commons.configuration.XMLConfiguration Maven / Gradle / Ivy

There is a newer version: 0.40.13
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.commons.configuration;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.io.StringReader;
import java.io.StringWriter;
import java.io.Writer;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.OutputKeys;
import javax.xml.transform.Result;
import javax.xml.transform.Source;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.TransformerFactoryConfigurationError;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;

import org.apache.commons.configuration.resolver.DefaultEntityResolver;
import org.apache.commons.configuration.resolver.EntityRegistry;
import org.apache.commons.configuration.tree.ConfigurationNode;
import org.apache.commons.logging.LogFactory;
import org.w3c.dom.Attr;
import org.w3c.dom.CDATASection;
import org.w3c.dom.DOMException;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.NodeList;
import org.w3c.dom.Text;
import org.xml.sax.EntityResolver;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import org.xml.sax.helpers.DefaultHandler;

/**
 * 

A specialized hierarchical configuration class that is able to parse XML * documents.

* *

The parsed document will be stored keeping its structure. The class also * tries to preserve as much information from the loaded XML document as * possible, including comments and processing instructions. These will be * contained in documents created by the {@code save()} methods, too.

* *

Like other file based configuration classes this class maintains the name * and path to the loaded configuration file. These properties can be altered * using several setter methods, but they are not modified by {@code save()} * and {@code load()} methods. If XML documents contain relative paths to * other documents (e.g. to a DTD), these references are resolved based on the * path set for this configuration.

* *

By inheriting from {@link AbstractConfiguration} this class * provides some extended functionality, e.g. interpolation of property values. * Like in {@link PropertiesConfiguration} property values can * contain delimiter characters (the comma ',' per default) and are then split * into multiple values. This works for XML attributes and text content of * elements as well. The delimiter can be escaped by a backslash. As an example * consider the following XML fragment:

* *

*

 * <config>
 *   <array>10,20,30,40</array>
 *   <scalar>3\,1415</scalar>
 *   <cite text="To be or not to be\, this is the question!"/>
 * </config>
 * 
*

*

Here the content of the {@code array} element will be split at * the commas, so the {@code array} key will be assigned 4 values. In the * {@code scalar} property and the {@code text} attribute of the * {@code cite} element the comma is escaped, so that no splitting is * performed.

* *

The configuration API allows setting multiple values for a single attribute, * e.g. something like the following is legal (assuming that the default * expression engine is used): *

 * XMLConfiguration config = new XMLConfiguration();
 * config.addProperty("test.dir[@name]", "C:\\Temp\\");
 * config.addProperty("test.dir[@name]", "D:\\Data\\");
 * 

* *

Because in XML such a constellation is not directly supported (an attribute * can appear only once for a single element), the values are concatenated to a * single value. If delimiter parsing is enabled (refer to the * {@link #setDelimiterParsingDisabled(boolean)} method), the * current list delimiter character will be used as separator. Otherwise the * pipe symbol ("|") will be used for this purpose. No matter which character is * used as delimiter, it can always be escaped with a backslash. A backslash * itself can also be escaped with another backslash. Consider the following * example fragment from a configuration file: *

 * <directories names="C:\Temp\\|D:\Data\"/>
 * 
* Here the backslash after Temp is escaped. This is necessary because it * would escape the list delimiter (the pipe symbol assuming that list delimiter * parsing is disabled) otherwise. So this attribute would have two values.

* *

Note: You should ensure that the delimiter parsing disabled * property is always consistent when you load and save a configuration file. * Otherwise the values of properties can become corrupted.

* *

Whitespace in the content of XML documents is trimmed per default. In most * cases this is desired. However, sometimes whitespace is indeed important and * should be treated as part of the value of a property as in the following * example: *

 *   <indent>    </indent>
 * 

* *

Per default the spaces in the {@code indent} element will be trimmed * resulting in an empty element. To tell {@code XMLConfiguration} that * spaces are relevant the {@code xml:space} attribute can be used, which is * defined in the XML * specification. This will look as follows: *

 *   <indent xml:space="preserve">    </indent>
 * 
* The value of the {@code indent} property will now contain the spaces.

* *

{@code XMLConfiguration} implements the {@link FileConfiguration} * interface and thus provides full support for loading XML documents from * different sources like files, URLs, or streams. A full description of these * features can be found in the documentation of * {@link AbstractFileConfiguration}.

* *

Note:Configuration objects of this type can be read concurrently * by multiple threads. However if one of these threads modifies the object, * synchronization has to be performed manually.

* * @since commons-configuration 1.0 * * @author Jörg Schaible * @version $Id: XMLConfiguration.java 1231721 2012-01-15 18:32:07Z oheger $ */ public class XMLConfiguration extends AbstractHierarchicalFileConfiguration implements EntityResolver, EntityRegistry { /** * The serial version UID. */ private static final long serialVersionUID = 2453781111653383552L; /** Constant for the default root element name. */ private static final String DEFAULT_ROOT_NAME = "configuration"; /** Constant for the name of the space attribute.*/ private static final String ATTR_SPACE = "xml:space"; /** Constant for the xml:space value for preserving whitespace.*/ private static final String VALUE_PRESERVE = "preserve"; /** Constant for the delimiter for multiple attribute values.*/ private static final char ATTR_VALUE_DELIMITER = '|'; /** Schema Langauge key for the parser */ private static final String JAXP_SCHEMA_LANGUAGE = "http://java.sun.com/xml/jaxp/properties/schemaLanguage"; /** Schema Language for the parser */ private static final String W3C_XML_SCHEMA = "http://www.w3.org/2001/XMLSchema"; /** The document from this configuration's data source. */ private Document document; /** Stores the name of the root element. */ private String rootElementName; /** Stores the public ID from the DOCTYPE.*/ private String publicID; /** Stores the system ID from the DOCTYPE.*/ private String systemID; /** Stores the document builder that should be used for loading.*/ private DocumentBuilder documentBuilder; /** Stores a flag whether DTD or Schema validation should be performed.*/ private boolean validating; /** Stores a flag whether DTD or Schema validation is used */ private boolean schemaValidation; /** A flag whether attribute splitting is disabled.*/ private boolean attributeSplittingDisabled; /** The EntityResolver to use */ private EntityResolver entityResolver = new DefaultEntityResolver(); /** * Creates a new instance of {@code XMLConfiguration}. */ public XMLConfiguration() { super(); setLogger(LogFactory.getLog(XMLConfiguration.class)); } /** * Creates a new instance of {@code XMLConfiguration} and copies the * content of the passed in configuration into this object. Note that only * the data of the passed in configuration will be copied. If, for instance, * the other configuration is a {@code XMLConfiguration}, too, * things like comments or processing instructions will be lost. * * @param c the configuration to copy * @since 1.4 */ public XMLConfiguration(HierarchicalConfiguration c) { super(c); clearReferences(getRootNode()); setRootElementName(getRootNode().getName()); setLogger(LogFactory.getLog(XMLConfiguration.class)); } /** * Creates a new instance of{@code XMLConfiguration}. The * configuration is loaded from the specified file * * @param fileName the name of the file to load * @throws ConfigurationException if the file cannot be loaded */ public XMLConfiguration(String fileName) throws ConfigurationException { super(fileName); setLogger(LogFactory.getLog(XMLConfiguration.class)); } /** * Creates a new instance of {@code XMLConfiguration}. * The configuration is loaded from the specified file. * * @param file the file * @throws ConfigurationException if an error occurs while loading the file */ public XMLConfiguration(File file) throws ConfigurationException { super(file); setLogger(LogFactory.getLog(XMLConfiguration.class)); } /** * Creates a new instance of {@code XMLConfiguration}. * The configuration is loaded from the specified URL. * * @param url the URL * @throws ConfigurationException if loading causes an error */ public XMLConfiguration(URL url) throws ConfigurationException { super(url); setLogger(LogFactory.getLog(XMLConfiguration.class)); } /** * Returns the name of the root element. If this configuration was loaded * from a XML document, the name of this document's root element is * returned. Otherwise it is possible to set a name for the root element * that will be used when this configuration is stored. * * @return the name of the root element */ public String getRootElementName() { if (getDocument() == null) { return (rootElementName == null) ? DEFAULT_ROOT_NAME : rootElementName; } else { return getDocument().getDocumentElement().getNodeName(); } } /** * Sets the name of the root element. This name is used when this * configuration object is stored in an XML file. Note that setting the name * of the root element works only if this configuration has been newly * created. If the configuration was loaded from an XML file, the name * cannot be changed and an {@code UnsupportedOperationException} * exception is thrown. Whether this configuration has been loaded from an * XML document or not can be found out using the {@code getDocument()} * method. * * @param name the name of the root element */ public void setRootElementName(String name) { if (getDocument() != null) { throw new UnsupportedOperationException("The name of the root element " + "cannot be changed when loaded from an XML document!"); } rootElementName = name; getRootNode().setName(name); } /** * Returns the {@code DocumentBuilder} object that is used for * loading documents. If no specific builder has been set, this method * returns null. * * @return the {@code DocumentBuilder} for loading new documents * @since 1.2 */ public DocumentBuilder getDocumentBuilder() { return documentBuilder; } /** * Sets the {@code DocumentBuilder} object to be used for loading * documents. This method makes it possible to specify the exact document * builder. So an application can create a builder, configure it for its * special needs, and then pass it to this method. * * @param documentBuilder the document builder to be used; if undefined, a * default builder will be used * @since 1.2 */ public void setDocumentBuilder(DocumentBuilder documentBuilder) { this.documentBuilder = documentBuilder; } /** * Returns the public ID of the DOCTYPE declaration from the loaded XML * document. This is null if no document has been loaded yet or if * the document does not contain a DOCTYPE declaration with a public ID. * * @return the public ID * @since 1.3 */ public String getPublicID() { return publicID; } /** * Sets the public ID of the DOCTYPE declaration. When this configuration is * saved, a DOCTYPE declaration will be constructed that contains this * public ID. * * @param publicID the public ID * @since 1.3 */ public void setPublicID(String publicID) { this.publicID = publicID; } /** * Returns the system ID of the DOCTYPE declaration from the loaded XML * document. This is null if no document has been loaded yet or if * the document does not contain a DOCTYPE declaration with a system ID. * * @return the system ID * @since 1.3 */ public String getSystemID() { return systemID; } /** * Sets the system ID of the DOCTYPE declaration. When this configuration is * saved, a DOCTYPE declaration will be constructed that contains this * system ID. * * @param systemID the system ID * @since 1.3 */ public void setSystemID(String systemID) { this.systemID = systemID; } /** * Returns the value of the validating flag. * * @return the validating flag * @since 1.2 */ public boolean isValidating() { return validating; } /** * Sets the value of the validating flag. This flag determines whether * DTD/Schema validation should be performed when loading XML documents. This * flag is evaluated only if no custom {@code DocumentBuilder} was set. * * @param validating the validating flag * @since 1.2 */ public void setValidating(boolean validating) { if (!schemaValidation) { this.validating = validating; } } /** * Returns the value of the schemaValidation flag. * * @return the schemaValidation flag * @since 1.7 */ public boolean isSchemaValidation() { return schemaValidation; } /** * Sets the value of the schemaValidation flag. This flag determines whether * DTD or Schema validation should be used. This * flag is evaluated only if no custom {@code DocumentBuilder} was set. * If set to true the XML document must contain a schemaLocation definition * that provides resolvable hints to the required schemas. * * @param schemaValidation the validating flag * @since 1.7 */ public void setSchemaValidation(boolean schemaValidation) { this.schemaValidation = schemaValidation; if (schemaValidation) { this.validating = true; } } /** * Sets a new EntityResolver. Setting this will cause RegisterEntityId to have no * effect. * @param resolver The EntityResolver to use. * @since 1.7 */ public void setEntityResolver(EntityResolver resolver) { this.entityResolver = resolver; } /** * Returns the EntityResolver. * @return The EntityResolver. * @since 1.7 */ public EntityResolver getEntityResolver() { return this.entityResolver; } /** * Returns the flag whether attribute splitting is disabled. * * @return the flag whether attribute splitting is disabled * @see #setAttributeSplittingDisabled(boolean) * @since 1.6 */ public boolean isAttributeSplittingDisabled() { return attributeSplittingDisabled; } /** *

* Sets a flag whether attribute splitting is disabled. *

*

* The Configuration API allows adding multiple values to an attribute. This * is problematic when storing the configuration because in XML an attribute * can appear only once with a single value. To solve this problem, per * default multiple attribute values are concatenated using a special * separator character and split again when the configuration is loaded. The * separator character is either the list delimiter character (see * {@link #setListDelimiter(char)}) or the pipe symbol ("|") if * list delimiter parsing is disabled. *

*

* In some constellations the splitting of attribute values can have * undesired effects, especially if list delimiter parsing is disabled and * attributes may contain the "|" character. In these cases it is * possible to disable the attribute splitting mechanism by calling this * method with a boolean value set to false. If attribute splitting * is disabled, the values of attributes will not be processed, but stored * as configuration properties exactly as they are returned by the XML * parser. *

*

* Note that in this mode multiple attribute values cannot be handled * correctly. It is possible to create a {@code XMLConfiguration} * object, add multiple values to an attribute and save it. When the * configuration is loaded again and attribute splitting is disabled, the * attribute will only have a single value, which is the concatenation of * all values set before. So it lies in the responsibility of the * application to carefully set the values of attributes. *

*

* As is true for the {@link #setDelimiterParsingDisabled(boolean)} method, * this method must be called before the configuration is loaded. So it * can't be used together with one of the constructors expecting the * specification of the file to load. Instead the default constructor has to * be used, then {@code setAttributeSplittingDisabled(false)} has to be * called, and finally the configuration can be loaded using one of its * {@code load()} methods. *

* * @param attributeSplittingDisabled true for disabling attribute * splitting, false for enabling it * @see #setDelimiterParsingDisabled(boolean) * @since 1.6 */ public void setAttributeSplittingDisabled(boolean attributeSplittingDisabled) { this.attributeSplittingDisabled = attributeSplittingDisabled; } /** * Returns the XML document this configuration was loaded from. The return * value is null if this configuration was not loaded from a XML * document. * * @return the XML document this configuration was loaded from */ public Document getDocument() { return document; } /** * Removes all properties from this configuration. If this configuration * was loaded from a file, the associated DOM document is also cleared. */ @Override public void clear() { super.clear(); setRoot(new Node()); document = null; } /** * Initializes this configuration from an XML document. * * @param document the document to be parsed * @param elemRefs a flag whether references to the XML elements should be set */ public void initProperties(Document document, boolean elemRefs) { if (document.getDoctype() != null) { setPublicID(document.getDoctype().getPublicId()); setSystemID(document.getDoctype().getSystemId()); } constructHierarchy(getRoot(), document.getDocumentElement(), elemRefs, true); getRootNode().setName(document.getDocumentElement().getNodeName()); if (elemRefs) { getRoot().setReference(document.getDocumentElement()); } } /** * Helper method for building the internal storage hierarchy. The XML * elements are transformed into node objects. * * @param node the actual node * @param element the actual XML element * @param elemRefs a flag whether references to the XML elements should be set * @param trim a flag whether the text content of elements should be trimmed; * this controls the whitespace handling */ private void constructHierarchy(Node node, Element element, boolean elemRefs, boolean trim) { boolean trimFlag = shouldTrim(element, trim); processAttributes(node, element, elemRefs); StringBuilder buffer = new StringBuilder(); NodeList list = element.getChildNodes(); for (int i = 0; i < list.getLength(); i++) { org.w3c.dom.Node w3cNode = list.item(i); if (w3cNode instanceof Element) { Element child = (Element) w3cNode; Node childNode = new XMLNode(child.getTagName(), elemRefs ? child : null); constructHierarchy(childNode, child, elemRefs, trimFlag); node.addChild(childNode); handleDelimiters(node, childNode, trimFlag); } else if (w3cNode instanceof Text) { Text data = (Text) w3cNode; buffer.append(data.getData()); } } String text = buffer.toString(); if (trimFlag) { text = text.trim(); } if (text.length() > 0 || (!node.hasChildren() && node != getRoot())) { node.setValue(text); } } /** * Helper method for constructing node objects for the attributes of the * given XML element. * * @param node the actual node * @param element the actual XML element * @param elemRefs a flag whether references to the XML elements should be set */ private void processAttributes(Node node, Element element, boolean elemRefs) { NamedNodeMap attributes = element.getAttributes(); for (int i = 0; i < attributes.getLength(); ++i) { org.w3c.dom.Node w3cNode = attributes.item(i); if (w3cNode instanceof Attr) { Attr attr = (Attr) w3cNode; List values; if (isAttributeSplittingDisabled()) { values = Collections.singletonList(attr.getValue()); } else { values = PropertyConverter.split(attr.getValue(), isDelimiterParsingDisabled() ? ATTR_VALUE_DELIMITER : getListDelimiter()); } for (String value : values) { Node child = new XMLNode(attr.getName(), elemRefs ? element : null); child.setValue(value); node.addAttribute(child); } } } } /** * Deals with elements whose value is a list. In this case multiple child * elements must be added. * * @param parent the parent element * @param child the child element * @param trim flag whether texts of elements should be trimmed */ private void handleDelimiters(Node parent, Node child, boolean trim) { if (child.getValue() != null) { List values; if (isDelimiterParsingDisabled()) { values = new ArrayList(); values.add(child.getValue().toString()); } else { values = PropertyConverter.split(child.getValue().toString(), getListDelimiter(), trim); } if (values.size() > 1) { Iterator it = values.iterator(); // Create new node for the original child's first value Node c = createNode(child.getName()); c.setValue(it.next()); // Copy original attributes to the new node for (ConfigurationNode ndAttr : child.getAttributes()) { ndAttr.setReference(null); c.addAttribute(ndAttr); } parent.remove(child); parent.addChild(c); // add multiple new children while (it.hasNext()) { c = new XMLNode(child.getName(), null); c.setValue(it.next()); parent.addChild(c); } } else if (values.size() == 1) { // we will have to replace the value because it might // contain escaped delimiters child.setValue(values.get(0)); } } } /** * Checks whether the content of the current XML element should be trimmed. * This method checks whether a {@code xml:space} attribute is * present and evaluates its value. See * http://www.w3.org/TR/REC-xml/#sec-white-space for more details. * * @param element the current XML element * @param currentTrim the current trim flag * @return a flag whether the content of this element should be trimmed */ private boolean shouldTrim(Element element, boolean currentTrim) { Attr attr = element.getAttributeNode(ATTR_SPACE); if (attr == null) { return currentTrim; } else { return !VALUE_PRESERVE.equals(attr.getValue()); } } /** * Creates the {@code DocumentBuilder} to be used for loading files. * This implementation checks whether a specific * {@code DocumentBuilder} has been set. If this is the case, this * one is used. Otherwise a default builder is created. Depending on the * value of the validating flag this builder will be a validating or a non * validating {@code DocumentBuilder}. * * @return the {@code DocumentBuilder} for loading configuration * files * @throws ParserConfigurationException if an error occurs * @since 1.2 */ protected DocumentBuilder createDocumentBuilder() throws ParserConfigurationException { if (getDocumentBuilder() != null) { return getDocumentBuilder(); } else { DocumentBuilderFactory factory = DocumentBuilderFactory .newInstance(); if (isValidating()) { factory.setValidating(true); if (isSchemaValidation()) { factory.setNamespaceAware(true); factory.setAttribute(JAXP_SCHEMA_LANGUAGE, W3C_XML_SCHEMA); } } DocumentBuilder result = factory.newDocumentBuilder(); result.setEntityResolver(this.entityResolver); if (isValidating()) { // register an error handler which detects validation errors result.setErrorHandler(new DefaultHandler() { @Override public void error(SAXParseException ex) throws SAXException { throw ex; } }); } return result; } } /** * Creates a DOM document from the internal tree of configuration nodes. * * @return the new document * @throws ConfigurationException if an error occurs */ protected Document createDocument() throws ConfigurationException { try { if (document == null) { DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder(); Document newDocument = builder.newDocument(); Element rootElem = newDocument.createElement(getRootElementName()); newDocument.appendChild(rootElem); document = newDocument; } XMLBuilderVisitor builder = new XMLBuilderVisitor(document, isDelimiterParsingDisabled() ? (char) 0 : getListDelimiter(), isAttributeSplittingDisabled()); builder.processDocument(getRoot()); initRootElementText(document, getRootNode().getValue()); return document; } catch (DOMException domEx) { throw new ConfigurationException(domEx); } catch (ParserConfigurationException pex) { throw new ConfigurationException(pex); } } /** * Sets the text of the root element of a newly created XML Document. * * @param doc the document * @param value the new text to be set */ private void initRootElementText(Document doc, Object value) { Element elem = doc.getDocumentElement(); NodeList children = elem.getChildNodes(); // Remove all existing text nodes for (int i = 0; i < children.getLength(); i++) { org.w3c.dom.Node nd = children.item(i); if (nd.getNodeType() == org.w3c.dom.Node.TEXT_NODE) { elem.removeChild(nd); } } if (value != null) { // Add a new text node elem.appendChild(doc.createTextNode(String.valueOf(value))); } } /** * Creates a new node object. This implementation returns an instance of the * {@code XMLNode} class. * * @param name the node's name * @return the new node */ @Override protected Node createNode(String name) { return new XMLNode(name, null); } /** * Loads the configuration from the given input stream. * * @param in the input stream * @throws ConfigurationException if an error occurs */ @Override public void load(InputStream in) throws ConfigurationException { load(new InputSource(in)); } /** * Load the configuration from the given reader. * Note that the {@code clear()} method is not called, so * the properties contained in the loaded file will be added to the * actual set of properties. * * @param in An InputStream. * * @throws ConfigurationException if an error occurs */ public void load(Reader in) throws ConfigurationException { load(new InputSource(in)); } /** * Loads a configuration file from the specified input source. * @param source the input source * @throws ConfigurationException if an error occurs */ private void load(InputSource source) throws ConfigurationException { try { URL sourceURL = getDelegate().getURL(); if (sourceURL != null) { source.setSystemId(sourceURL.toString()); } DocumentBuilder builder = createDocumentBuilder(); Document newDocument = builder.parse(source); Document oldDocument = document; document = null; initProperties(newDocument, oldDocument == null); document = (oldDocument == null) ? newDocument : oldDocument; } catch (SAXParseException spe) { throw new ConfigurationException("Error parsing " + source.getSystemId(), spe); } catch (Exception e) { this.getLogger().debug("Unable to load the configuraton", e); throw new ConfigurationException("Unable to load the configuration", e); } } /** * Saves the configuration to the specified writer. * * @param writer the writer used to save the configuration * @throws ConfigurationException if an error occurs */ public void save(Writer writer) throws ConfigurationException { try { Transformer transformer = createTransformer(); Source source = new DOMSource(createDocument()); Result result = new StreamResult(writer); transformer.transform(source, result); } catch (TransformerException e) { throw new ConfigurationException("Unable to save the configuration", e); } catch (TransformerFactoryConfigurationError e) { throw new ConfigurationException("Unable to save the configuration", e); } } /** * Validate the document against the Schema. * @throws ConfigurationException if the validation fails. */ public void validate() throws ConfigurationException { try { Transformer transformer = createTransformer(); Source source = new DOMSource(createDocument()); StringWriter writer = new StringWriter(); Result result = new StreamResult(writer); transformer.transform(source, result); Reader reader = new StringReader(writer.getBuffer().toString()); DocumentBuilder builder = createDocumentBuilder(); builder.parse(new InputSource(reader)); } catch (SAXException e) { throw new ConfigurationException("Validation failed", e); } catch (IOException e) { throw new ConfigurationException("Validation failed", e); } catch (TransformerException e) { throw new ConfigurationException("Validation failed", e); } catch (ParserConfigurationException pce) { throw new ConfigurationException("Validation failed", pce); } } /** * Creates and initializes the transformer used for save operations. This * base implementation initializes all of the default settings like * indention mode and the DOCTYPE. Derived classes may overload this method * if they have specific needs. * * @return the transformer to use for a save operation * @throws TransformerException if an error occurs * @since 1.3 */ protected Transformer createTransformer() throws TransformerException { Transformer transformer = TransformerFactory.newInstance() .newTransformer(); transformer.setOutputProperty(OutputKeys.INDENT, "yes"); if (getEncoding() != null) { transformer.setOutputProperty(OutputKeys.ENCODING, getEncoding()); } if (getPublicID() != null) { transformer.setOutputProperty(OutputKeys.DOCTYPE_PUBLIC, getPublicID()); } if (getSystemID() != null) { transformer.setOutputProperty(OutputKeys.DOCTYPE_SYSTEM, getSystemID()); } return transformer; } /** * Creates a copy of this object. The new configuration object will contain * the same properties as the original, but it will lose any connection to a * source document (if one exists). This is to avoid race conditions if both * the original and the copy are modified and then saved. * * @return the copy */ @Override public Object clone() { XMLConfiguration copy = (XMLConfiguration) super.clone(); // clear document related properties copy.document = null; copy.setDelegate(copy.createDelegate()); // clear all references in the nodes, too clearReferences(copy.getRootNode()); return copy; } /** * Creates the file configuration delegate for this object. This implementation * will return an instance of a class derived from {@code FileConfigurationDelegate} * that deals with some specialties of {@code XMLConfiguration}. * @return the delegate for this object */ @Override protected FileConfigurationDelegate createDelegate() { return new XMLFileConfigurationDelegate(); } /** * Adds a collection of nodes directly to this configuration. This * implementation ensures that the nodes to be added are of the correct node * type (they have to be converted to {@code XMLNode} if necessary). * * @param key the key where the nodes are to be added * @param nodes the collection with the new nodes * @since 1.5 */ @Override public void addNodes(String key, Collection nodes) { if (nodes != null && !nodes.isEmpty()) { Collection xmlNodes; xmlNodes = new ArrayList(nodes.size()); for (ConfigurationNode node : nodes) { xmlNodes.add(convertToXMLNode(node)); } super.addNodes(key, xmlNodes); } else { super.addNodes(key, nodes); } } /** * Converts the specified node into a {@code XMLNode} if necessary. * This is required for nodes that are directly added, e.g. by * {@code addNodes()}. If the passed in node is already an instance * of {@code XMLNode}, it is directly returned, and conversion * stops. Otherwise a new {@code XMLNode} is created, and the * children are also converted. * * @param node the node to be converted * @return the converted node */ private XMLNode convertToXMLNode(ConfigurationNode node) { if (node instanceof XMLNode) { return (XMLNode) node; } XMLNode nd = (XMLNode) createNode(node.getName()); nd.setValue(node.getValue()); nd.setAttribute(node.isAttribute()); for (ConfigurationNode child : node.getChildren()) { nd.addChild(convertToXMLNode(child)); } for (ConfigurationNode attr : node.getAttributes()) { nd.addAttribute(convertToXMLNode(attr)); } return nd; } /** *

* Registers the specified DTD URL for the specified public identifier. *

*

* {@code XMLConfiguration} contains an internal * {@code EntityResolver} implementation. This maps * {@code PUBLICID}'s to URLs (from which the resource will be * loaded). A common use case for this method is to register local URLs * (possibly computed at runtime by a class loader) for DTDs. This allows * the performance advantage of using a local version without having to * ensure every {@code SYSTEM} URI on every processed XML document is * local. This implementation provides only basic functionality. If more * sophisticated features are required, using * {@link #setDocumentBuilder(DocumentBuilder)} to set a custom * {@code DocumentBuilder} (which also can be initialized with a * custom {@code EntityResolver}) is recommended. *

*

* Note: This method will have no effect when a custom * {@code DocumentBuilder} has been set. (Setting a custom * {@code DocumentBuilder} overrides the internal implementation.) *

*

* Note: This method must be called before the * configuration is loaded. So the default constructor of * {@code XMLConfiguration} should be used, the location of the * configuration file set, {@code registerEntityId()} called, and * finally the {@code load()} method can be invoked. *

* * @param publicId Public identifier of the DTD to be resolved * @param entityURL The URL to use for reading this DTD * @throws IllegalArgumentException if the public ID is undefined * @since 1.5 */ public void registerEntityId(String publicId, URL entityURL) { if (entityResolver instanceof EntityRegistry) { ((EntityRegistry) entityResolver).registerEntityId(publicId, entityURL); } } /** * Resolves the requested external entity. This is the default * implementation of the {@code EntityResolver} interface. It checks * the passed in public ID against the registered entity IDs and uses a * local URL if possible. * * @param publicId the public identifier of the entity being referenced * @param systemId the system identifier of the entity being referenced * @return an input source for the specified entity * @throws SAXException if a parsing exception occurs * @since 1.5 * @deprecated Use getEntityResolver().resolveEntity() */ @Deprecated public InputSource resolveEntity(String publicId, String systemId) throws SAXException { try { return entityResolver.resolveEntity(publicId, systemId); } catch (IOException e) { throw new SAXException(e); } } /** * Returns a map with the entity IDs that have been registered using the * {@code registerEntityId()} method. * * @return a map with the registered entity IDs */ public Map getRegisteredEntities() { if (entityResolver instanceof EntityRegistry) { return ((EntityRegistry) entityResolver).getRegisteredEntities(); } return new HashMap(); } /** * A specialized {@code Node} class that is connected with an XML * element. Changes on a node are also performed on the associated element. */ class XMLNode extends Node { /** * The serial version UID. */ private static final long serialVersionUID = -4133988932174596562L; /** * Creates a new instance of {@code XMLNode} and initializes it * with a name and the corresponding XML element. * * @param name the node's name * @param elem the XML element */ public XMLNode(String name, Element elem) { super(name); setReference(elem); } /** * Sets the value of this node. If this node is associated with an XML * element, this element will be updated, too. * * @param value the node's new value */ @Override public void setValue(Object value) { super.setValue(value); if (getReference() != null && document != null) { if (isAttribute()) { updateAttribute(); } else { updateElement(value); } } } /** * Updates the associated XML elements when a node is removed. */ @Override protected void removeReference() { if (getReference() != null) { Element element = (Element) getReference(); if (isAttribute()) { updateAttribute(); } else { org.w3c.dom.Node parentElem = element.getParentNode(); if (parentElem != null) { parentElem.removeChild(element); } } } } /** * Updates the node's value if it represents an element node. * * @param value the new value */ private void updateElement(Object value) { Text txtNode = findTextNodeForUpdate(); if (value == null) { // remove text if (txtNode != null) { ((Element) getReference()).removeChild(txtNode); } } else { if (txtNode == null) { String newValue = isDelimiterParsingDisabled() ? value.toString() : PropertyConverter.escapeDelimiters(value.toString(), getListDelimiter()); txtNode = document.createTextNode(newValue); if (((Element) getReference()).getFirstChild() != null) { ((Element) getReference()).insertBefore(txtNode, ((Element) getReference()).getFirstChild()); } else { ((Element) getReference()).appendChild(txtNode); } } else { String newValue = isDelimiterParsingDisabled() ? value.toString() : PropertyConverter.escapeDelimiters(value.toString(), getListDelimiter()); txtNode.setNodeValue(newValue); } } } /** * Updates the node's value if it represents an attribute. * */ private void updateAttribute() { XMLBuilderVisitor.updateAttribute(getParent(), getName(), getListDelimiter(), isAttributeSplittingDisabled()); } /** * Returns the only text node of this element for update. This method is * called when the element's text changes. Then all text nodes except * for the first are removed. A reference to the first is returned or * null if there is no text node at all. * * @return the first and only text node */ private Text findTextNodeForUpdate() { Text result = null; Element elem = (Element) getReference(); // Find all Text nodes NodeList children = elem.getChildNodes(); Collection textNodes = new ArrayList(); for (int i = 0; i < children.getLength(); i++) { org.w3c.dom.Node nd = children.item(i); if (nd instanceof Text) { if (result == null) { result = (Text) nd; } else { textNodes.add(nd); } } } // We don't want CDATAs if (result instanceof CDATASection) { textNodes.add(result); result = null; } // Remove all but the first Text node for (org.w3c.dom.Node tn : textNodes) { elem.removeChild(tn); } return result; } } /** * A concrete {@code BuilderVisitor} that can construct XML * documents. */ static class XMLBuilderVisitor extends BuilderVisitor { /** Stores the document to be constructed. */ private Document document; /** Stores the list delimiter.*/ private char listDelimiter = AbstractConfiguration. getDefaultListDelimiter(); /** True if attributes should not be split */ private boolean isAttributeSplittingDisabled; /** * Creates a new instance of {@code XMLBuilderVisitor}. * * @param doc the document to be created * @param listDelimiter the delimiter for attribute properties with multiple values * @param isAttributeSplittingDisabled true if attribute splitting is disabled. */ public XMLBuilderVisitor(Document doc, char listDelimiter, boolean isAttributeSplittingDisabled) { document = doc; this.listDelimiter = listDelimiter; this.isAttributeSplittingDisabled = isAttributeSplittingDisabled; } /** * Processes the node hierarchy and adds new nodes to the document. * * @param rootNode the root node */ public void processDocument(Node rootNode) { rootNode.visit(this, null); } /** * Inserts a new node. This implementation ensures that the correct * XML element is created and inserted between the given siblings. * * @param newNode the node to insert * @param parent the parent node * @param sibling1 the first sibling * @param sibling2 the second sibling * @return the new node */ @Override protected Object insert(Node newNode, Node parent, Node sibling1, Node sibling2) { if (newNode.isAttribute()) { updateAttribute(parent, getElement(parent), newNode.getName(), listDelimiter, isAttributeSplittingDisabled); return null; } else { Element elem = document.createElement(newNode.getName()); if (newNode.getValue() != null) { String txt = newNode.getValue().toString(); if (listDelimiter != 0) { txt = PropertyConverter.escapeListDelimiter(txt, listDelimiter); } elem.appendChild(document.createTextNode(txt)); } if (sibling2 == null) { getElement(parent).appendChild(elem); } else if (sibling1 != null) { getElement(parent).insertBefore(elem, getElement(sibling1).getNextSibling()); } else { getElement(parent).insertBefore(elem, getElement(parent).getFirstChild()); } return elem; } } /** * Helper method for updating the value of the specified node's * attribute with the given name. * * @param node the affected node * @param elem the element that is associated with this node * @param name the name of the affected attribute * @param listDelimiter the delimiter for attributes with multiple values * @param isAttributeSplittingDisabled true if attribute splitting is disabled. */ private static void updateAttribute(Node node, Element elem, String name, char listDelimiter, boolean isAttributeSplittingDisabled) { if (node != null && elem != null) { boolean hasAttribute = false; List attrs = node.getAttributes(name); StringBuilder buf = new StringBuilder(); char delimiter = (listDelimiter != 0) ? listDelimiter : ATTR_VALUE_DELIMITER; for (ConfigurationNode attr : attrs) { if (attr.getValue() != null) { hasAttribute = true; if (buf.length() > 0) { buf.append(delimiter); } String value = isAttributeSplittingDisabled ? attr.getValue().toString() : PropertyConverter.escapeDelimiters(attr.getValue().toString(), delimiter); buf.append(value); } attr.setReference(elem); } if (!hasAttribute) { elem.removeAttribute(name); } else { elem.setAttribute(name, buf.toString()); } } } /** * Updates the value of the specified attribute of the given node. * Because there can be multiple child nodes representing this attribute * the new value is determined by iterating over all those child nodes. * * @param node the affected node * @param name the name of the attribute * @param listDelimiter the delimiter for attributes with multiple values * @param isAttributeSplittingDisabled true if attributes splitting is disabled. */ static void updateAttribute(Node node, String name, char listDelimiter, boolean isAttributeSplittingDisabled) { if (node != null) { updateAttribute(node, (Element) node.getReference(), name, listDelimiter, isAttributeSplittingDisabled); } } /** * Helper method for accessing the element of the specified node. * * @param node the node * @return the element of this node */ private Element getElement(Node node) { // special treatment for root node of the hierarchy return (node.getName() != null && node.getReference() != null) ? (Element) node .getReference() : document.getDocumentElement(); } } /** * A special implementation of the {@code FileConfiguration} interface that is * used internally to implement the {@code FileConfiguration} methods * for {@code XMLConfiguration}, too. */ private class XMLFileConfigurationDelegate extends FileConfigurationDelegate { @Override public void load(InputStream in) throws ConfigurationException { XMLConfiguration.this.load(in); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy