com.yahoo.vespa.model.builder.xml.dom.DomConfigPayloadBuilder Maven / Gradle / Ivy
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.model.builder.xml.dom;
import com.yahoo.collections.Tuple2;
import com.yahoo.config.FileReference;
import com.yahoo.config.ModelReference;
import com.yahoo.config.UrlReference;
import com.yahoo.text.XML;
import com.yahoo.vespa.config.ConfigDefinition;
import com.yahoo.vespa.config.ConfigDefinitionKey;
import com.yahoo.vespa.config.ConfigPayloadBuilder;
import com.yahoo.vespa.config.util.ConfigUtils;
import com.yahoo.yolean.Exceptions;
import org.w3c.dom.Element;
import java.util.List;
import java.util.Optional;
import java.util.regex.Pattern;
/**
* Builder that transforms xml config to a slime tree representation of the config. The root element of the xml config
* must be named 'config' and have a 'name' attribute that matches the name of the {@link ConfigDefinition}. The values
* are not validated against their types. That task is moved to the builders.
*
* @author Ulf Lilleengen
*/
public class DomConfigPayloadBuilder {
private static final Pattern namePattern = ConfigDefinition.namePattern;
private static final Pattern namespacePattern = ConfigDefinition.namespacePattern;
/** The config definition, not null if not found */
private final ConfigDefinition configDefinition;
public DomConfigPayloadBuilder(ConfigDefinition configDefinition) {
this.configDefinition = configDefinition;
}
/**
* Builds a {@link ConfigPayloadBuilder} representing the input 'config' xml element.
*
* @param configE the 'config' xml element
* @return a new payload builder built from xml
*/
public ConfigPayloadBuilder build(Element configE) {
parseConfigName(configE);
ConfigPayloadBuilder payloadBuilder = new ConfigPayloadBuilder(configDefinition);
for (Element child : XML.getChildren(configE)) {
parseElement(child, payloadBuilder, null);
}
return payloadBuilder;
}
public static ConfigDefinitionKey parseConfigName(Element configE) {
if (!configE.getNodeName().equals("config")) {
throw new IllegalArgumentException("The root element must be 'config', but was '" + configE.getNodeName() + "'");
}
if (!configE.hasAttribute("name")) {
throw new IllegalArgumentException
("The 'config' element must have a 'name' attribute that matches the name of the config definition");
}
String elementString = configE.getAttribute("name");
if (!elementString.contains(".")) {
throw new IllegalArgumentException("The config name '" + elementString +
"' contains illegal characters. Only names with the pattern " +
namespacePattern.pattern() + "." + namePattern.pattern() + " are legal.");
}
Tuple2 t = ConfigUtils.getNameAndNamespaceFromString(elementString);
String xmlName = t.first;
String xmlNamespace = t.second;
if (!validName(xmlName)) {
throw new IllegalArgumentException("The config name '" + xmlName +
"' contains illegal characters. Only names with the pattern " +
namePattern.toString() + " are legal.");
}
if (!validNamespace(xmlNamespace)) {
throw new IllegalArgumentException("The config namespace '" + xmlNamespace +
"' contains illegal characters. Only namespaces with the pattern " +
namespacePattern.toString() + " are legal.");
}
return new ConfigDefinitionKey(xmlName, xmlNamespace);
}
private static boolean validName(String name) {
if (name == null) return false;
return namePattern.matcher(name).matches();
}
private static boolean validNamespace(String namespace) {
if (namespace == null) return false;
return namespacePattern.matcher(namespace).matches();
}
private String extractName(Element element) {
String initial = element.getNodeName();
if (initial.indexOf('-') < 0) {
return initial;
}
StringBuilder buf = new StringBuilder();
boolean upcase = false;
for (char ch : initial.toCharArray()) {
if (ch == '-') {
upcase = true;
} else if (upcase && ch >= 'a' && ch <= 'z') {
buf.append((char)('A' + ch - 'a'));
upcase = false;
} else {
buf.append(ch);
upcase = false;
}
}
return buf.toString();
}
/** Parse leaf value in an xml tree. */
private void parseLeaf(Element element, ConfigPayloadBuilder payloadBuilder, String parentName) {
String name = extractName(element);
String value = XML.getValue(element);
if (value == null) {
throw new IllegalArgumentException("Element '" + name + "' must have either children or a value");
}
if ("item".equals(name)) {
if (parentName == null)
throw new IllegalArgumentException("- is a reserved keyword for array and map elements");
if (element.hasAttribute("key")) {
payloadBuilder.getMap(parentName).put(element.getAttribute("key"), value);
} else {
payloadBuilder.getArray(parentName).append(value);
}
}
else if (element.hasAttribute("model-id") || element.hasAttribute("url") || element.hasAttribute("path")) {
// special syntax for "model" fields
var model = ModelReference.unresolved(modelElement("model-id", element),
modelElement("url", element).map(UrlReference::new),
modelElement("path", element).map(FileReference::new));
payloadBuilder.setField(name, model.toString());
}
else { // leaf value:
value
payloadBuilder.setField(name, value);
}
}
private Optional modelElement(String attributeName, Element element) {
Optional value = XML.attribute(attributeName, element);
if (value.isPresent() && value.get().contains(" "))
throw new IllegalArgumentException("The value of " + attributeName + " on " + element.getTagName() +
"cannot contain space");
return value;
}
private void parseComplex(Element element, List children, ConfigPayloadBuilder payloadBuilder, String parentName) {
String name = extractName(element);
// Inner value
if ("item".equals(name)) {
// Reserved item means array/map element as struct
if (element.hasAttribute("key")) {
ConfigPayloadBuilder childPayloadBuilder = payloadBuilder.getMap(parentName).get(element.getAttribute("key"));
for (Element child : children) {
parseElement(child, childPayloadBuilder, parentName);
}
} else {
ConfigPayloadBuilder.Array array = payloadBuilder.getArray(parentName);
ConfigPayloadBuilder childPayloadBuilder = array.append();
for (Element child : children) {
parseElement(child, childPayloadBuilder, parentName);
}
}
} else {
int numMatching = 0;
for (Element child : children) {
numMatching += ("item".equals(child.getTagName())) ? 1 : 0;
}
if (numMatching == 0) {
// struct, e.g.
ConfigPayloadBuilder p = payloadBuilder.getObject(name);
//Cursor struct = node.setObject(name);
for (Element child : children)
parseElement(child, p, name);
} else if (numMatching == children.size()) {
// Array with -
for (Element child : children) {
parseElement(child, payloadBuilder, name);
}
} else {
throw new IllegalArgumentException("
- is a reserved keyword for array and map elements");
}
}
}
/**
* Adds the values and children (recursively) in the given xml element to the given {@link ConfigPayloadBuilder}.
* @param currElem The element representing a config parameter.
* @param payloadBuilder The builder to use when adding elements.
*/
private void parseElement(Element currElem, ConfigPayloadBuilder payloadBuilder, String parentName) {
List
children = XML.getChildren(currElem);
try {
if (children.isEmpty()) {
parseLeaf(currElem, payloadBuilder, parentName);
} else {
parseComplex(currElem, children, payloadBuilder, parentName);
}
} catch (Exception exception) {
throw new IllegalArgumentException("Error parsing element at " + XML.getNodePath(currElem, " > ") +
": " + Exceptions.toMessageString(exception));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy