But on the other hand side, SOAP has some advantages over REST. It allows us to define a contract and specify the data it will process. Once you have such WSLD, you can throw it over the hedge, and you do not have to answer questions like how do I delete a created resource, what 403 means in this case, or is this call idempotent?
WSLD contains everything needed to define an RPC interface and document it. I would choose REST for most cases, but still, there are some good use cases for SOAP.
This one thing bothers me about SOAP: how do I get proper WSLD? Should I use a tool to generate it or write it? Honestly... I really do not want to dig into SOAP to just write WSLD, and I do not want to spend hours playing around with some tools generating WSLD, importing it into my project, and binding generated stuff to actual implementation only to find out about the end that it does not operate as expected.
The perfect solution would be to write Java code, annotate it, and generate WSDL out of it. Those annotations should provide enough flexibility to influence WSLD generation so that we archive something that can be delivered to the customers.
I've tried a few Java frameworks, and all provide a way to generate WSLD out of Java code, but they all need to include one fundamental feature: you cannot offer XSD for data types. So it's impossible to specify the content of an email field or provide a format for a date. We get only half of a possible functionality of a WSDL: method calls, exception handling, documentation, and security, but there is no way to specify the data format.
I would like to have WSLD that is as precise as possible so that the client can call the particular method and know precisely what is possible and is not allowed.
Let's start with the standard SOAP Service generated out of Java code. We will use Apache CXF and Spring Boot for it. The source code is here: https://github.com/maciejmiklas/apache-cxf-soap.
It's a simple application where users can submit registration as a SOAP Request on http://localhost:8080/soap/Registration:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// http://localhost:8080/soap/Registration?wsdl | |
@WebService(targetNamespace = NAMESPACE, serviceName = SERVICE_NAME) | |
@SchemaValidation(type = BOTH) | |
@SOAPBinding | |
public class RegistrationService { | |
private final Set<Registration> registered = new HashSet<>(); | |
@WebMethod | |
public void register(Registration registration) throws AlreadyRegisteredException { | |
if (registered.add(registration)) { | |
throw new AlreadyRegisteredException(); | |
} | |
} | |
} | |
@XmlRootElement | |
class Registration { | |
@XmlElement(required = true) | |
private String email; | |
@XmlElement(required = true) | |
private String name; | |
private String phone; | |
private Integer age; | |
// getters and setter omitted | |
} | |
@SpringBootApplication | |
@Import(CxfConfiguration.class) | |
public class ApacheCxfSoapApplication { | |
@Bean | |
public RegistrationService registrationService() { | |
return new RegistrationService(); | |
} | |
@Bean | |
public JaxWsServiceFactoryBean jaxWsServiceFactoryBean() { | |
JaxWsServiceFactoryBean serviceFactoryBean = new JaxWsServiceFactoryBean(); | |
serviceFactoryBean.setValidate(true); | |
return serviceFactoryBean; | |
} | |
@Bean | |
public JaxWsServerFactoryBean jaxWsServerFactoryBean() { | |
return new JaxWsServerFactoryBean(jaxWsServiceFactoryBean()); | |
} | |
@Bean | |
public Endpoint registrationEndpoint(SpringBus springBus) { | |
EndpointImpl endpoint = new EndpointImpl(springBus, registrationService(), jaxWsServerFactoryBean()); | |
endpoint.publish(SERVICE_ADDRESS); | |
return endpoint; | |
} | |
public static void main(String[] args) { | |
SpringApplication.run(ApacheCxfSoapApplication.class, args); | |
} | |
} |
This code outpost following WSDL:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!-- http://localhost:8080/soap/Registration?wsdl --> | |
<?xml version='1.0' encoding='UTF-8'?><wsdl:definitions xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:tns="http://orginal.apachecxfsoap.ast.org/" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:ns1="http://schemas.xmlsoap.org/soap/http" name="registration" targetNamespace="http://orginal.apachecxfsoap.ast.org/"> | |
<wsdl:types> | |
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:tns="http://orginal.apachecxfsoap.ast.org/" attributeFormDefault="unqualified" elementFormDefault="unqualified" targetNamespace="http://orginal.apachecxfsoap.ast.org/"> | |
<xs:element name="register" type="tns:register"/> | |
<xs:element name="registerResponse" type="tns:registerResponse"/> | |
<xs:element name="registration" type="tns:registration"/> | |
<xs:complexType name="register"> | |
<xs:sequence> | |
<xs:element minOccurs="0" name="arg0" type="tns:registration"/> | |
</xs:sequence> | |
</xs:complexType> | |
<xs:complexType name="registration"> | |
<xs:sequence> | |
<xs:element name="email" type="xs:string"/> | |
<xs:element name="name" type="xs:string"/> | |
</xs:sequence> | |
</xs:complexType> | |
<xs:complexType name="registerResponse"> | |
<xs:sequence/> | |
</xs:complexType> | |
<xs:element name="AlreadyRegisteredException" type="tns:AlreadyRegisteredException"/> | |
<xs:complexType name="AlreadyRegisteredException"> | |
<xs:sequence> | |
<xs:element minOccurs="0" name="message" type="xs:string"/> | |
</xs:sequence> | |
</xs:complexType> | |
</xs:schema> | |
</wsdl:types> | |
<wsdl:message name="registerResponse"> | |
<wsdl:part element="tns:registerResponse" name="parameters"> | |
</wsdl:part> | |
</wsdl:message> | |
<wsdl:message name="AlreadyRegisteredException"> | |
<wsdl:part element="tns:AlreadyRegisteredException" name="AlreadyRegisteredException"> | |
</wsdl:part> | |
</wsdl:message> | |
<wsdl:message name="register"> | |
<wsdl:part element="tns:register" name="parameters"> | |
</wsdl:part> | |
</wsdl:message> | |
<wsdl:portType name="RegistrationService"> | |
<wsdl:operation name="register"> | |
<wsdl:input message="tns:register" name="register"> | |
</wsdl:input> | |
<wsdl:output message="tns:registerResponse" name="registerResponse"> | |
</wsdl:output> | |
<wsdl:fault message="tns:AlreadyRegisteredException" name="AlreadyRegisteredException"> | |
</wsdl:fault> | |
</wsdl:operation> | |
</wsdl:portType> | |
<wsdl:binding name="registrationSoapBinding" type="tns:RegistrationService"> | |
<soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"/> | |
<wsdl:operation name="register"> | |
<soap:operation soapAction="" style="document"/> | |
<wsdl:input name="register"> | |
<soap:body use="literal"/> | |
</wsdl:input> | |
<wsdl:output name="registerResponse"> | |
<soap:body use="literal"/> | |
</wsdl:output> | |
<wsdl:fault name="AlreadyRegisteredException"> | |
<soap:fault name="AlreadyRegisteredException" use="literal"/> | |
</wsdl:fault> | |
</wsdl:operation> | |
</wsdl:binding> | |
<wsdl:service name="registration"> | |
<wsdl:port binding="tns:registrationSoapBinding" name="RegistrationServicePort"> | |
<soap:address location="http://localhost:8080/soap/Registration"/> | |
</wsdl:port> | |
</wsdl:service> | |
</wsdl:definitions> |
The problem lies in JAXB. It ignores annotations to influence generated XML schema. There is a pull request (jaxb-facets) that would solve this issue, but it's already a few years old and looks like it will only be integrated for a while.
The limitation is not caused by SOAP frameworks but by the fact that they are based on JAXB.
This does not change the fact that I will not write WSLD myself. There has to be a better way!
We will implement a simple extension for Apache CXF to combine generated WSLD with custom XSD. We have to write XSD in this case, but CXF will generate WSLD for us and include the given schema.
XSD is mighty, so I prefer writing it by hand because it's possible to specify the data format precisely. Using annotations to generate XSD would be convenient, but it would still cover only some common areas.
The implementation below has its limitations. You might run into some issues. It might stop working after the next CXF update. But! I use it, and it does what I need, so it might be something for you as well ;)
Now we are going to modify the first example. The idea is to write XSD that defines simple types, references those types in transfer classes, and generates WSLD, combining them all.
In the beginning, we have to write a schema that defines types for transfer objects:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" version="1.0"> | |
<xs:simpleType name="email"> | |
<xs:restriction base="xs:string"> | |
<xs:maxLength value="256"/> | |
<xs:minLength value="5"/> | |
<xs:pattern | |
value="([0-9a-zA-Z]([-.\w]*[0-9a-zA-Z])*@([0-9a-zA-Z][-\w]*[0-9a-zA-Z]\.)+[a-zA-Z]{2,9})"></xs:pattern> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:simpleType name="phone"> | |
<xs:restriction base="xs:string"> | |
<xs:maxLength value="25" /> | |
<xs:minLength value="4" /> | |
<xs:pattern value="[0-9 ]+"></xs:pattern> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:simpleType name="age"> | |
<xs:restriction base="xs:integer"> | |
<xs:minInclusive value="0"></xs:minInclusive> | |
<xs:maxInclusive value="120"></xs:maxInclusive> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:simpleType name="code"> | |
<xs:restriction base="xs:integer"> | |
<xs:minInclusive value="0"></xs:minInclusive> | |
<xs:maxInclusive value="99999"></xs:maxInclusive> | |
</xs:restriction> | |
</xs:simpleType> | |
</xs:schema> |
- We will replace the default Apache CXF data binding with a custom implementation. It reads Schama from the file and integrates it into generated WSLD,
- Some fields in transfer objects are annotated with @XmlSchemaType - this annotation provides a connection between Java types and types defined in XSD. For example, the ExRegistration#email is annotated with @XmlSchemaType(name = "email"), and XSD contains the email type. The email generated in WSDL references the type from a provided schema,
- the classes following the pattern *Registration* have been renamed to *ExRegistration*
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// http://localhost:8080/soap/ExRegistration?wsdl | |
@WebService(targetNamespace = NAMESPACE, serviceName = SERVICE_NAME) | |
@SchemaValidation(type = BOTH) | |
@SOAPBinding | |
public class ExRegistrationService { | |
private final Set<ExRegistration> registered = new HashSet<>(); | |
@WebMethod | |
public void register(ExRegistration registration) throws AlreadyRegisteredException { | |
if (registered.add(registration)) { | |
throw new AlreadyRegisteredException(); | |
} | |
} | |
} | |
@XmlRootElement | |
class ExRegistration { | |
@XmlSchemaType(name = "email", namespace = NAMESPACE) | |
@XmlElement(required = true) | |
private String email; | |
@XmlElement(required = true) | |
private String name; | |
@XmlSchemaType(name = "phone", namespace = NAMESPACE) | |
private String phone; | |
@XmlSchemaType(name = "age", namespace = NAMESPACE) | |
private Integer age; | |
// getters and setter omitted | |
} | |
@SpringBootApplication | |
@Import(CxfConfiguration.class) | |
public class ApacheCxfSoapExApplication { | |
@Bean | |
public ExRegistrationService registrationService() { | |
return new ExRegistrationService(); | |
} | |
@Bean | |
public JaxWsServiceFactoryBean jaxWsServiceFactoryBean() { | |
JaxWsServiceFactoryBean serviceFactoryBean = new JaxWsServiceFactoryBean(); | |
serviceFactoryBean.setValidate(true); | |
return serviceFactoryBean; | |
} | |
@Bean | |
public JaxWsServerFactoryBean jaxWsServerFactoryBean() { | |
return new JaxWsServerFactoryBean(jaxWsServiceFactoryBean()); | |
} | |
@Bean | |
public Endpoint registrationEndpoint(SpringBus springBus) { | |
EndpointImpl endpoint = new EndpointImpl(springBus, registrationService(), jaxWsServerFactoryBean()); | |
// THIS IS NEW | |
SchemaJAXBDataBinding dataBinding = new SchemaJAXBDataBinding("types.xsd"); | |
endpoint.setDataBinding(dataBinding); | |
// THIS IS NEW | |
endpoint.publish(SERVICE_ADDRESS); | |
return endpoint; | |
} | |
public static void main(String[] args) { | |
SpringApplication.run(ApacheCxfSoapExApplication.class, args); | |
} | |
} | |
class SchemaJAXBDataBinding extends JAXBDataBinding { | |
private final String[] locations; | |
private final static String FILTER_PATH = "//*[@type='tns:{}']"; | |
public SchemaJAXBDataBinding(String... locations) { | |
super(true); | |
this.locations = locations; | |
} | |
@Override | |
public XmlSchema addSchemaDocument(ServiceInfo serviceInfo, SchemaCollection col, Document document, String systemId) { | |
Arrays.stream(locations).map(this::load).forEach(xsd -> append(document, xsd)); | |
return super.addSchemaDocument(serviceInfo, col, document, systemId); | |
} | |
private void append(Document destination, Document extra) { | |
Element root = destination.getDocumentElement(); | |
NodeList childNodes = extra.getDocumentElement().getChildNodes(); | |
IntStream.range(0, childNodes.getLength()).mapToObj(childNodes::item). | |
filter(inNode -> contains(destination, inNode)).forEach(inNode -> { | |
root.appendChild(destination.importNode(inNode, true)); | |
}); | |
} | |
/** | |
* filter out XSD types that are not referenced in WSDL. | |
*/ | |
private boolean contains(Document doc, Node inNode) { | |
if (inNode.getAttributes() == null || inNode.getAttributes().getNamedItem("name") == null) { | |
return false; | |
} | |
Node node = inNode.getAttributes().getNamedItem("name"); | |
String exprStr = FILTER_PATH.replace("{}", node.getNodeValue()); | |
try { | |
XPathExpression expr = XPathFactory.newInstance().newXPath().compile(exprStr); | |
NodeList found = (NodeList) expr.evaluate(doc.getDocumentElement(), XPathConstants.NODESET); | |
return found != null && found.getLength() > 0; | |
} catch (XPathExpressionException e) { | |
throw new InvalidXmlSchemaReferenceException("Error parsing document", e); | |
} | |
} | |
private Document load(String location) { | |
ResourceManager resourceManager = getBus().getExtension(ResourceManager.class); | |
URL url = resourceManager.resolveResource(location, URL.class); | |
if (url == null) { | |
throw new InvalidXmlSchemaReferenceException("XSD not found: " + location); | |
} | |
try { | |
Document document = StaxUtils.read(url.openStream()); | |
return document; | |
} catch (IOException | XMLStreamException e) { | |
throw new InvalidXmlSchemaReferenceException("Error reading XSD from: " + location, e); | |
} | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?xml version='1.0' encoding='UTF-8'?><wsdl:definitions xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:tns="http://extended.apachecxfsoap.ast.org/" xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:ns1="http://schemas.xmlsoap.org/soap/http" name="exRegistration" targetNamespace="http://extended.apachecxfsoap.ast.org/"> | |
<wsdl:types> | |
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:tns="http://extended.apachecxfsoap.ast.org/" attributeFormDefault="unqualified" elementFormDefault="unqualified" targetNamespace="http://extended.apachecxfsoap.ast.org/"> | |
<xs:element name="exRegistration" type="tns:exRegistration"/> | |
<xs:element name="register" type="tns:register"/> | |
<xs:element name="registerResponse" type="tns:registerResponse"/> | |
<xs:complexType name="register"> | |
<xs:sequence> | |
<xs:element minOccurs="0" name="arg0" type="tns:exRegistration"/> | |
</xs:sequence> | |
</xs:complexType> | |
<xs:complexType name="exRegistration"> | |
<xs:sequence> | |
<xs:element name="email" type="tns:email"/> | |
<xs:element name="name" type="xs:string"/> | |
<xs:element minOccurs="0" name="phone" type="tns:phone"/> | |
<xs:element minOccurs="0" name="age" type="tns:age"/> | |
</xs:sequence> | |
</xs:complexType> | |
<xs:complexType name="registerResponse"> | |
<xs:sequence/> | |
</xs:complexType> | |
<xs:simpleType name="email"> | |
<xs:restriction base="xs:string"> | |
<xs:maxLength value="256"/> | |
<xs:minLength value="5"/> | |
<xs:pattern value="([0-9a-zA-Z]([-.\w]*[0-9a-zA-Z])*@([0-9a-zA-Z][-\w]*[0-9a-zA-Z]\.)+[a-zA-Z]{2,9})"/> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:simpleType name="phone"> | |
<xs:restriction base="xs:string"> | |
<xs:maxLength value="25"/> | |
<xs:minLength value="4"/> | |
<xs:pattern value="[0-9 ]+"/> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:simpleType name="age"> | |
<xs:restriction base="xs:integer"> | |
<xs:minInclusive value="0"/> | |
<xs:maxInclusive value="120"/> | |
</xs:restriction> | |
</xs:simpleType> | |
<xs:element name="AlreadyRegisteredException" type="tns:AlreadyRegisteredException"/> | |
<xs:complexType name="AlreadyRegisteredException"> | |
<xs:sequence> | |
<xs:element minOccurs="0" name="message" type="xs:string"/> | |
</xs:sequence> | |
</xs:complexType> | |
</xs:schema> | |
<!-- binding are identical to first WSDL--> |