org.citrusframework.xml.actions.Receive Maven / Gradle / Ivy
/*
* Copyright the original author or authors.
*
* 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 org.citrusframework.xml.actions;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.stream.Stream;
import jakarta.xml.bind.annotation.XmlAccessType;
import jakarta.xml.bind.annotation.XmlAccessorType;
import jakarta.xml.bind.annotation.XmlAttribute;
import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import jakarta.xml.bind.annotation.XmlType;
import org.citrusframework.TestActionBuilder;
import org.citrusframework.TestActor;
import org.citrusframework.actions.ReceiveMessageAction;
import org.citrusframework.exceptions.CitrusRuntimeException;
import org.citrusframework.spi.ReferenceResolver;
import org.citrusframework.spi.ReferenceResolverAware;
import org.citrusframework.util.StringUtils;
import org.citrusframework.validation.json.JsonMessageValidationContext;
import org.citrusframework.validation.json.JsonPathMessageValidationContext;
import org.citrusframework.validation.script.ScriptValidationContext;
import org.citrusframework.validation.xml.XmlMessageValidationContext;
import org.citrusframework.xml.actions.script.ScriptDefinitionType;
@XmlRootElement(name = "receive")
public class Receive implements TestActionBuilder, ReferenceResolverAware {
private final ReceiveMessageAction.ReceiveMessageActionBuilder, ?, ?> builder;
private String actor;
private ReferenceResolver referenceResolver;
private final List validators = new ArrayList<>();
private final XmlMessageValidationContext.Builder xmlValidationContext = new XmlMessageValidationContext.Builder();
private final JsonMessageValidationContext.Builder jsonValidationContext = new JsonMessageValidationContext.Builder();
@XmlElement(name = "ignore")
protected List ignores;
@XmlElement(name = "validate")
protected List validates;
@XmlElement(name = "namespace")
protected List namespaces;
public Receive() {
this(new ReceiveMessageAction.Builder());
}
public Receive(ReceiveMessageAction.ReceiveMessageActionBuilder, ?, ?> builder) {
this.builder = builder;
}
@XmlElement
public Receive setDescription(String value) {
builder.description(value);
return this;
}
public List getIgnores() {
if (ignores == null) {
ignores = new ArrayList<>();
}
return this.ignores;
}
public List getValidates() {
if (validates == null) {
validates = new ArrayList<>();
}
return this.validates;
}
public List getNamespaces() {
if (namespaces == null) {
namespaces = new ArrayList<>();
}
return this.namespaces;
}
@XmlAttribute(name = "validator")
public Receive setValidator(String validator) {
if (StringUtils.hasText(validator)) {
validators.add(validator);
}
return this;
}
@XmlAttribute(name = "validators")
public Receive setValidators(String messageValidatorExpression) {
if (StringUtils.hasText(messageValidatorExpression)) {
Stream.of(messageValidatorExpression.split(","))
.map(String::trim)
.forEach(validators::add);
}
return this;
}
@XmlAttribute(name = "header-validator")
public Receive setHeaderValidator(String headerValidator) {
if (StringUtils.hasText(headerValidator)) {
builder.getHeaderValidationContext().addHeaderValidator(headerValidator);
}
return this;
}
@XmlAttribute(name = "header-validators")
public Receive setHeaderValidators(String headerValidatorExpression) {
if (StringUtils.hasText(headerValidatorExpression)) {
Stream.of(headerValidatorExpression.split(","))
.map(String::trim)
.forEach(builder.getHeaderValidationContext()::addHeaderValidator);
}
return this;
}
@XmlElement(required = true)
public Receive setMessage(Message message) {
MessageSupport.configureMessage(builder, message);
if (message.schema != null || message.schemaRepository != null) {
xmlValidationContext.schemaValidation(message.isSchemaValidation())
.schema(message.schema)
.schemaRepository(message.schemaRepository);
jsonValidationContext
.schemaValidation(message.isSchemaValidation())
.schema(message.schema)
.schemaRepository(message.schemaRepository);
} else if (message.isSchemaValidation() != null && !message.isSchemaValidation()) {
xmlValidationContext.schemaValidation(message.isSchemaValidation());
jsonValidationContext.schemaValidation(message.isSchemaValidation());
}
return this;
}
private void addScriptValidationContext() {
ScriptValidationContext.Builder context = null;
for (Validate validateElement : getValidates()) {
ScriptDefinitionType scriptElement = validateElement.getScript();
// check for nested validate script child node
if (scriptElement != null) {
if (context != null) {
throw new CitrusRuntimeException("Found multiple validation script definitions - " +
"only supporting a single validation script for message validation");
}
String type = scriptElement.getType();
context = new ScriptValidationContext.Builder().scriptType(type);
String filePath = scriptElement.getFile();
if (StringUtils.hasText(filePath)) {
context.scriptResource(filePath);
if (scriptElement.getCharset() != null) {
context.scriptResourceCharset(scriptElement.getCharset());
}
} else if (scriptElement.getValue() != null) {
context.script(scriptElement.getValue().trim());
}
builder.validate(context);
}
}
}
private void addJsonValidationContext() {
Set ignoreExpressions = new HashSet<>();
for (Ignore ignoreValue : getIgnores()) {
ignoreExpressions.add(ignoreValue.path);
}
ignoreExpressions.forEach(jsonValidationContext::ignore);
builder.validate(jsonValidationContext);
JsonPathMessageValidationContext.Builder jsonPathContext = new JsonPathMessageValidationContext.Builder();
//check for validate elements, these elements can either have script, jsonPath or namespace validation information
//for now we only handle jsonPath validation
Map validateJsonPathExpressions = new HashMap<>();
for (Validate validateElement : getValidates()) {
extractJsonPathValidateExpressions(validateElement, validateJsonPathExpressions);
}
if (!validateJsonPathExpressions.isEmpty()) {
jsonPathContext.expressions(validateJsonPathExpressions);
builder.validate(jsonPathContext);
}
}
private void extractJsonPathValidateExpressions(Validate validateElement, Map validateJsonPathExpressions) {
//check for jsonPath validation - old style with direct attribute
String pathExpression = validateElement.path;
if (JsonPathMessageValidationContext.isJsonPathExpression(pathExpression)) {
validateJsonPathExpressions.put(pathExpression, validateElement.value);
}
//check for jsonPath validation elements - new style preferred
for (Validate.JsonPath jsonPathElement : validateElement.getJsonPaths()) {
String expression = jsonPathElement.expression;
if (StringUtils.hasText(expression)) {
validateJsonPathExpressions.put(expression, jsonPathElement.value);
}
}
}
private void addXmlValidationContext() {
Set ignoreExpressions = new HashSet<>();
for (Ignore ignore : getIgnores()) {
ignoreExpressions.add(ignore.path);
}
ignoreExpressions.forEach(xmlValidationContext::ignore);
//check for validate elements, these elements can either have script, xpath or namespace validation information
//for now we only handle namespace validation
Map validateNamespaces = new HashMap<>();
for (Validate validateElement : getValidates()) {
//check for namespace validation elements
for (Validate.Namespace namespaceElement : validateElement.getNamespaces()) {
validateNamespaces.put(namespaceElement.prefix, namespaceElement.value);
}
}
xmlValidationContext.namespaces(validateNamespaces);
//Catch namespace declarations for namespace context
Map namespaces = new HashMap<>();
for (Namespace namespace : getNamespaces()) {
namespaces.put(namespace.prefix, namespace.value);
}
xmlValidationContext.namespaceContext(namespaces);
//check for validate elements, these elements can either have script, xpath or namespace validation information
//for now we only handle xpath validation
Map validateXpathExpressions = new HashMap<>();
for (Validate validateElement : getValidates()) {
extractXPathValidateExpressions(validateElement, validateXpathExpressions);
}
if (!validateXpathExpressions.isEmpty()) {
builder.validate(xmlValidationContext.xpath().expressions(validateXpathExpressions));
} else {
builder.validate(xmlValidationContext);
}
}
private void extractXPathValidateExpressions(Validate validateElement, Map validateXpathExpressions) {
//check for xpath validation - old style with direct attribute
String pathExpression = validateElement.path;
if (StringUtils.hasText(pathExpression) && !JsonPathMessageValidationContext.isJsonPathExpression(pathExpression)) {
//construct pathExpression with explicit result-type, like boolean:/TestMessage/Value
if (validateElement.resultType != null) {
pathExpression = validateElement.resultType + ":" + pathExpression;
}
validateXpathExpressions.put(pathExpression, validateElement.value);
}
//check for xpath validation elements - new style preferred
for (Validate.Xpath xpathElement : validateElement.getXpaths()) {
String expression = xpathElement.expression;
if (StringUtils.hasText(expression)) {
//construct expression with explicit result-type, like boolean:/TestMessage/Value
if (xpathElement.resultType != null) {
expression = xpathElement.resultType + ":" + expression;
}
validateXpathExpressions.put(expression, xpathElement.value);
}
}
}
@XmlElement
public Receive setExtract(Message.Extract value) {
MessageSupport.configureExtract(builder, value);
return this;
}
@XmlAttribute
public Receive setEndpoint(String value) {
builder.endpoint(value);
return this;
}
@XmlAttribute
public Receive setActor(String value) {
this.actor = value;
return this;
}
@XmlAttribute
public Receive setTimeout(Integer value) {
builder.timeout(value);
return this;
}
@XmlAttribute
public Receive setSelect(String value) {
builder.selector(value);
return this;
}
@XmlElement
public Receive setSelector(Selector selector) {
if (selector.selectorValue != null) {
builder.selector(selector.selectorValue);
}
if (selector.elements != null) {
Map selectorElements = new HashMap<>();
for (Selector.Element element : selector.elements) {
selectorElements.put(element.name, element.value);
}
builder.selector(selectorElements);
}
return this;
}
@Override
public ReceiveMessageAction build() {
if (referenceResolver != null) {
if (actor != null) {
builder.actor(referenceResolver.resolve(actor, TestActor.class));
}
}
validators.forEach(builder::validator);
addXmlValidationContext();
addJsonValidationContext();
addScriptValidationContext();
return doBuild();
}
/**
* Subclasses may add additional building logic here.
* @return
*/
protected ReceiveMessageAction doBuild() {
return builder.build();
}
@Override
public void setReferenceResolver(ReferenceResolver referenceResolver) {
builder.setReferenceResolver(referenceResolver);
this.referenceResolver = referenceResolver;
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "", propOrder = {
"elements",
"selectorValue"
})
public static class Selector {
@XmlElement(name = "element")
protected List elements;
@XmlElement(name = "value")
protected String selectorValue;
public List getElements() {
if (elements == null) {
elements = new ArrayList<>();
}
return this.elements;
}
public String getSelectorValue() {
return selectorValue;
}
public void setSelectorValue(String value) {
this.selectorValue = value;
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class Element {
@XmlAttribute(name = "name", required = true)
protected String name;
@XmlAttribute(name = "value", required = true)
protected String value;
public String getName() {
return name;
}
public void setName(String value) {
this.name = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class Ignore {
@XmlAttribute(name = "path", required = true)
protected String path;
public String getPath() {
return path;
}
public void setPath(String value) {
this.path = value;
}
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class Namespace {
@XmlAttribute(name = "prefix", required = true)
protected String prefix;
@XmlAttribute(name = "value", required = true)
protected String value;
public String getPrefix() {
return prefix;
}
public void setPrefix(String value) {
this.prefix = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "", propOrder = {
"script",
"xpaths",
"jsonPaths",
"namespaces"
})
public static class Validate {
protected ScriptDefinitionType script;
@XmlElement(name = "xpath")
protected List xpaths;
@XmlElement(name = "json-path")
protected List jsonPaths;
@XmlElement(name = "namespace")
protected List namespaces;
@XmlAttribute(name = "path")
protected String path;
@XmlAttribute(name = "value")
protected String value;
@XmlAttribute(name = "result-type")
protected String resultType;
public ScriptDefinitionType getScript() {
return script;
}
public void setScript(ScriptDefinitionType value) {
this.script = value;
}
public List getXpaths() {
if (xpaths == null) {
xpaths = new ArrayList<>();
}
return this.xpaths;
}
public List getJsonPaths() {
if (jsonPaths == null) {
jsonPaths = new ArrayList<>();
}
return this.jsonPaths;
}
public List getNamespaces() {
if (namespaces == null) {
namespaces = new ArrayList<>();
}
return this.namespaces;
}
public String getPath() {
return path;
}
public void setPath(String value) {
this.path = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
public String getResultType() {
return resultType;
}
public void setResultType(String value) {
this.resultType = value;
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class JsonPath {
@XmlAttribute(name = "expression", required = true)
protected String expression;
@XmlAttribute(name = "value", required = true)
protected String value;
public String getExpression() {
return expression;
}
public void setExpression(String value) {
this.expression = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class Namespace {
@XmlAttribute(name = "prefix", required = true)
protected String prefix;
@XmlAttribute(name = "value", required = true)
protected String value;
public String getPrefix() {
return prefix;
}
public void setPrefix(String value) {
this.prefix = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
}
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "")
public static class Xpath {
@XmlAttribute(name = "expression", required = true)
protected String expression;
@XmlAttribute(name = "value", required = true)
protected String value;
@XmlAttribute(name = "result-type")
protected String resultType;
public String getExpression() {
return expression;
}
public void setExpression(String value) {
this.expression = value;
}
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
public String getResultType() {
return resultType;
}
public void setResultType(String value) {
this.resultType = value;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy