org.apache.servicemix.validation.ValidationEndpoint Maven / Gradle / Ivy
/*
* 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.servicemix.validation;
import java.io.IOException;
import javax.jbi.JBIException;
import javax.jbi.messaging.Fault;
import javax.jbi.messaging.MessageExchange;
import javax.jbi.messaging.MessagingException;
import javax.jbi.messaging.NormalizedMessage;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.Source;
import javax.xml.transform.TransformerException;
import javax.xml.transform.dom.DOMResult;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamSource;
import javax.xml.validation.Schema;
import javax.xml.validation.SchemaFactory;
import javax.xml.validation.Validator;
import org.apache.servicemix.common.endpoints.ProviderEndpoint;
import org.apache.servicemix.common.util.MessageUtil;
import org.apache.servicemix.jbi.exception.FaultException;
import org.apache.servicemix.jbi.jaxp.SourceTransformer;
import org.apache.servicemix.jbi.jaxp.StringSource;
import org.apache.servicemix.validation.handler.CountingErrorHandlerFactory;
import org.apache.servicemix.validation.handler.MessageAwareErrorHandler;
import org.apache.servicemix.validation.handler.MessageAwareErrorHandlerFactory;
import org.springframework.core.io.Resource;
import org.xml.sax.SAXException;
/**
* @org.apache.xbean.XBean element="endpoint"
* @author lhein
*/
public class ValidationEndpoint extends ProviderEndpoint implements
ValidationEndpointType {
public static final String FAULT_FLOW = "FAULT_FLOW";
public static final String FAULT_JBI = "FAULT_JBI";
public static final String TAG_RESULT_START = "";
public static final String TAG_RESULT_END = " ";
public static final String TAG_WARNING_START = "";
public static final String TAG_WARNING_END = " ";
public static final String TAG_ERROR_START = "";
public static final String TAG_ERROR_END = " ";
public static final String TAG_FATAL_START = "";
public static final String TAG_FATAL_END = " ";
private String handlingErrorMethod = FAULT_JBI;
private Schema schema;
private String schemaLanguage = "http://www.w3.org/2001/XMLSchema";
private Source schemaSource;
private Resource schemaResource;
private Resource noNamespaceSchemaResource;
private MessageAwareErrorHandlerFactory errorHandlerFactory = new CountingErrorHandlerFactory();
private SourceTransformer sourceTransformer = new SourceTransformer();
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.ProviderEndpoint#start()
*/
@Override
public void start() throws Exception {
super.start();
try {
if (schema == null) {
SchemaFactory factory = SchemaFactory
.newInstance(schemaLanguage);
if (schemaSource == null) {
if (schemaResource == null) {
if (noNamespaceSchemaResource == null) {
throw new JBIException("You must specify schema, schemaSource, schemaResource or noNamespaceSchemaResource property.");
}
// Don't instantiate the schema here
schema = factory.newSchema();
} else {
if (schemaResource.getURL() == null) {
schemaSource = new StreamSource(schemaResource.getInputStream());
} else {
schemaSource = new StreamSource(schemaResource.getInputStream(), schemaResource.getURL().toExternalForm());
}
schema = factory.newSchema(schemaSource);
}
}
}
} catch (IOException e) {
throw new JBIException("Failed to load schema: " + e, e);
} catch (SAXException e) {
throw new JBIException("Failed to load schema: " + e, e);
}
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.ProviderEndpoint#stop()
*/
@Override
public void stop() throws Exception {
super.stop();
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.ProviderEndpoint#processInOnly(javax.jbi.messaging.MessageExchange,
* javax.jbi.messaging.NormalizedMessage)
*/
@Override
protected void processInOnly(MessageExchange exchange, NormalizedMessage in)
throws Exception {
NormalizedMessage out = exchange.createMessage();
Fault fault = exchange.createFault();
this.startValidation(exchange, in, out, fault);
if (fault.getContent() != null) {
throw new RuntimeException(sourceTransformer.contentToString(fault));
}
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.ProviderEndpoint#processInOut(javax.jbi.messaging.MessageExchange,
* javax.jbi.messaging.NormalizedMessage,
* javax.jbi.messaging.NormalizedMessage)
*/
@Override
protected void processInOut(MessageExchange exchange, NormalizedMessage in,
NormalizedMessage out) throws Exception {
Fault fault = exchange.createFault();
this.startValidation(exchange, in, out, fault);
if (fault.getContent() != null) {
exchange.setFault(fault);
}
}
/**
* does the validation
*
* @param exchange
* @param in
* @param out
* @param fault
* @throws Exception
*/
public void startValidation(MessageExchange exchange, NormalizedMessage in,
NormalizedMessage out, Fault fault) throws Exception {
Validator validator = schema.newValidator();
if (noNamespaceSchemaResource != null) {
logger.info("Enabling validation for noNamespace-XML documents.");
validator.setFeature("http://xml.org/sax/features/validation", true);
validator.setFeature("http://apache.org/xml/features/validation/schema", true);
validator.setProperty("http://apache.org/xml/properties/schema/external-noNamespaceSchemaLocation", noNamespaceSchemaResource.getURL().toExternalForm());
}
// create a new errorHandler and set it on the validator
MessageAwareErrorHandler errorHandler = errorHandlerFactory
.createMessageAwareErrorHandler();
validator.setErrorHandler(errorHandler);
DOMResult result = new DOMResult();
fault.setContent(null);
try {
// Only DOMSource and SAXSource are allowed for validating
// See
// http://java.sun.com/j2se/1.5.0/docs/api/javax/xml/validation/
// Validator.html#validate(javax.xml.transform.Source,%20javax.xml.transform.Result)
// As we expect a DOMResult as output, we must ensure that the input
// is a DOMSource
DOMSource src = sourceTransformer.toDOMSource(in.getContent());
// call the validation method
doValidation(validator, src, result);
// check if there were errors while validating
if (errorHandler.hasErrors()) {
/*
* check if this error handler supports the capturing of error
* messages.
*/
if (errorHandler.capturesMessages()) {
/*
* In descending order of preference select a format to use.
* If neither DOMSource, StringSource or String are
* supported throw a messaging exception.
*/
if (errorHandler.supportsMessageFormat(DOMSource.class)) {
fault.setContent((DOMSource) errorHandler
.getMessagesAs(DOMSource.class));
} else if (errorHandler
.supportsMessageFormat(StringSource.class)) {
fault.setContent(sourceTransformer
.toDOMSource((StringSource) errorHandler
.getMessagesAs(StringSource.class)));
} else if (errorHandler.supportsMessageFormat(String.class)) {
fault.setContent(sourceTransformer
.toDOMSource(new StringSource(
(String) errorHandler
.getMessagesAs(String.class))));
} else {
throw new MessagingException(
"MessageAwareErrorHandler implementation "
+ errorHandler.getClass().getName()
+ " does not support a compatible error message format.");
}
} else {
/*
* we can't do much here if the ErrorHandler implementation
* does not support capturing messages
*/
StringBuffer resultString = new StringBuffer();
resultString.append(TAG_RESULT_START);
resultString.append('\n');
resultString.append(TAG_WARNING_START);
resultString.append(errorHandler.getWarningCount());
resultString.append(TAG_WARNING_END);
resultString.append('\n');
resultString.append(TAG_ERROR_START);
resultString.append(errorHandler.getErrorCount());
resultString.append(TAG_ERROR_END);
resultString.append('\n');
resultString.append(TAG_FATAL_START);
resultString.append(errorHandler.getFatalErrorCount());
resultString.append(TAG_FATAL_END);
resultString.append('\n');
resultString.append(TAG_RESULT_END);
resultString.append('\n');
fault.setContent(new StringSource(resultString.toString()));
}
if (!handlingErrorMethod.equalsIgnoreCase(FAULT_FLOW)) {
// HANDLE AS JBI FAULT
throw new FaultException("Failed to validate against schema: " + schema, exchange, fault);
} else {
MessageUtil.transfer(fault, out);
}
} else {
// Retrieve the ouput of the validation
// as it may have been changed by the validator
out.setContent(new DOMSource(result.getNode(), result
.getSystemId()));
}
} catch (SAXException e) {
throw new MessagingException(e);
} catch (IOException e) {
throw new MessagingException(e);
} catch (ParserConfigurationException e) {
throw new MessagingException(e);
} catch (TransformerException e) {
throw new MessagingException(e);
}
}
/**
* does the validation
*
* @param validator
* @param src
* @param result
* @throws SAXException
* @throws IOException
*/
protected void doValidation(Validator validator, DOMSource src,
DOMResult result) throws SAXException, IOException {
validator.validate(src, result);
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.SimpleEndpoint#fail(javax.jbi.messaging.MessageExchange,
* java.lang.Exception)
*/
protected void fail(MessageExchange messageExchange, Exception e)
throws MessagingException {
super.fail(messageExchange, e);
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.SimpleEndpoint#send(javax.jbi.messaging.MessageExchange)
*/
protected void send(MessageExchange messageExchange)
throws MessagingException {
super.send(messageExchange);
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.SimpleEndpoint#sendSync(javax.jbi.messaging.MessageExchange)
*/
protected void sendSync(MessageExchange messageExchange)
throws MessagingException {
super.sendSync(messageExchange);
}
/*
* (non-Javadoc)
*
* @see org.apache.servicemix.common.endpoints.SimpleEndpoint#done(javax.jbi.messaging.MessageExchange)
*/
protected void done(MessageExchange messageExchange)
throws MessagingException {
super.done(messageExchange);
}
public String getHandlingErrorMethod() {
return handlingErrorMethod;
}
/**
* Configure how validation errors should be handled. Default value is FAULT_JBI
.
*
* FAULT_JBI
* - A jbi exception is thrown on validation errors (depending on used MEP)
* FAULT_FLOW
* - The validation result will be sent in out / fault message (depending on used MEP)
*
*
* @param handlingErrorMethod
*/
public void setHandlingErrorMethod(String handlingErrorMethod) {
this.handlingErrorMethod = handlingErrorMethod;
}
public Schema getSchema() {
return schema;
}
/**
* Set the validation schema instance.
*
* @param schema
*/
public void setSchema(Schema schema) {
this.schema = schema;
}
public String getSchemaLanguage() {
return schemaLanguage;
}
/**
* Set the validation schema language. Defaults to http://www.w3.org/2001/XMLSchema
*
* @param schemaLanguage
*/
public void setSchemaLanguage(String schemaLanguage) {
this.schemaLanguage = schemaLanguage;
}
public Source getSchemaSource() {
return schemaSource;
}
/**
* Set the validation schema as an XML Source.
*
* @param schemaSource
*/
public void setSchemaSource(Source schemaSource) {
this.schemaSource = schemaSource;
}
public Resource getSchemaResource() {
return schemaResource;
}
/**
* Set the validation schema as a Spring Resource.
*
* @param schemaResource
*/
public void setSchemaResource(Resource schemaResource) {
this.schemaResource = schemaResource;
}
public Resource getNoNamespaceSchemaResource() {
return noNamespaceSchemaResource;
}
/**
* Set the validation schema to be used when no namespace is specified.
*
* @param schemaResource
*/
public void setNoNamespaceSchemaResource(Resource schemaResource) {
this.noNamespaceSchemaResource = schemaResource;
}
public MessageAwareErrorHandlerFactory getErrorHandlerFactory() {
return errorHandlerFactory;
}
/**
* Set a custom error handler to deal with validation errors. Defaults to a CountingErrorHandlerFactory
.
*
* @param errorHandlerFactory
*/
public void setErrorHandlerFactory(
MessageAwareErrorHandlerFactory errorHandlerFactory) {
this.errorHandlerFactory = errorHandlerFactory;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy