
com.hazelcast.config.AbstractConfigBuilder Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2016, Hazelcast, Inc. All Rights Reserved.
*
* 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 com.hazelcast.config;
import com.hazelcast.logging.ILogger;
import com.hazelcast.logging.Logger;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import javax.xml.namespace.NamespaceContext;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathFactory;
import java.io.InputStream;
import java.net.URL;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Properties;
import java.util.Set;
import static com.hazelcast.config.XmlElements.IMPORT;
/**
* Contains logic for replacing system variables in the XML file and importing XML files from different locations.
*/
public abstract class AbstractConfigBuilder extends AbstractXmlConfigHelper {
protected enum ConfigType {
SERVER("hazelcast"),
CLIENT("hazelcast-client");
final String name;
ConfigType(String name) {
this.name = name;
}
}
private static final ILogger LOGGER = Logger.getLogger(AbstractConfigBuilder.class);
private final Set currentlyImportedFiles = new HashSet();
private final XPath xpath;
public AbstractConfigBuilder() {
final XPathFactory fac = XPathFactory.newInstance();
this.xpath = fac.newXPath();
xpath.setNamespaceContext(new NamespaceContext() {
@Override public String getNamespaceURI(String prefix) {
return "hz".equals(prefix) ? xmlns : null;
}
@Override public String getPrefix(String namespaceURI) { return null; }
@Override public Iterator getPrefixes(String namespaceURI) { return null; }
});
}
protected void process(Node root) throws Exception {
traverseChildsAndReplaceVariables(root);
replaceImportElementsWithActualFileContents(root);
}
private void replaceImportElementsWithActualFileContents(Node root) throws Exception {
Document document = root.getOwnerDocument();
NodeList misplacedImports = (NodeList) xpath.evaluate(
String.format("//hz:%s/parent::*[not(self::hz:%s)]", IMPORT.name, getXmlType().name),
document, XPathConstants.NODESET);
if (misplacedImports.getLength() > 0) {
throw new InvalidConfigurationException(
" element can appear only in the top level of the XML");
}
NodeList importTags = (NodeList) xpath.evaluate(
String.format("/hz:%s/hz:%s", this.getXmlType().name, IMPORT.name), document, XPathConstants.NODESET);
for (Node node : asElementIterable(importTags)) {
loadAndReplaceImportElement(root, node);
}
}
private void loadAndReplaceImportElement(Node root, Node node) throws Exception {
NamedNodeMap attributes = node.getAttributes();
Node resourceAtrribute = attributes.getNamedItem("resource");
String resource = resourceAtrribute.getTextContent();
URL url = ConfigLoader.locateConfig(resource);
if (url == null) {
throw new InvalidConfigurationException("Failed to load resource: " + resource);
}
if (!currentlyImportedFiles.add(url.getPath())) {
throw new InvalidConfigurationException("Cyclic loading of resource " + url.getPath() + " is detected !");
}
Document doc = parse(url.openStream());
Element importedRoot = doc.getDocumentElement();
traverseChildsAndReplaceVariables(importedRoot);
replaceImportElementsWithActualFileContents(importedRoot);
for (Node fromImportedDoc : childElements(importedRoot)) {
Node importedNode = root.getOwnerDocument().importNode(fromImportedDoc, true);
root.insertBefore(importedNode, node);
}
root.removeChild(node);
}
/**
* Reads xml from InputStream and parses
*
* @param inputStream
* @return Document after parsing xml
* @throws Exception
*/
protected abstract Document parse(InputStream inputStream) throws Exception;
/**
* @return system properties
*/
protected abstract Properties getProperties();
/**
* @return ConfigType of current config class as enum value.
*/
protected abstract ConfigType getXmlType();
private void traverseChildsAndReplaceVariables(Node root) {
NamedNodeMap attributes = root.getAttributes();
if (attributes != null) {
for (int k = 0; k < attributes.getLength(); k++) {
Node attribute = attributes.item(k);
replaceVariables(attribute);
}
}
if (root.getNodeValue() != null) {
replaceVariables(root);
}
final NodeList childNodes = root.getChildNodes();
for (int k = 0; k < childNodes.getLength(); k++) {
Node child = childNodes.item(k);
traverseChildsAndReplaceVariables(child);
}
}
private void replaceVariables(Node node) {
String value = node.getNodeValue();
StringBuilder sb = new StringBuilder(value);
int endIndex = -1;
int startIndex = sb.indexOf("${");
while (startIndex > -1) {
endIndex = sb.indexOf("}", startIndex);
if (endIndex == -1) {
LOGGER.warning("Bad variable syntax. Could not find a closing curly bracket '}' on node: "
+ node.getLocalName());
break;
}
String variable = sb.substring(startIndex + 2, endIndex);
String variableReplacement = getProperties().getProperty(variable);
if (variableReplacement != null) {
sb.replace(startIndex, endIndex + 1, variableReplacement);
endIndex = startIndex + variableReplacement.length();
} else {
LOGGER.warning("Could not find a value for property '" + variable + "' on node: "
+ node.getLocalName());
}
startIndex = sb.indexOf("${", endIndex);
}
node.setNodeValue(sb.toString());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy