Previous post introduced a new tree view of request parameters for invoking an endpoint. Here I'd like to show an example of a practical usage of such addition to Wise core: a customized SOAP message preview tool.
There're few scenarios in which a user needs to know how a SOAP message for invoking a given WSDL operation will look like. Computing that can be easy or difficult, depending on the complexity of the contract (number of schemas, wsdl style, etc.), that's why resorting to leveraging tools is actually a good idea ;-)
Using Wise, we start by building a WSDynamicClient instance consuming our target wsdl and selecting the WSMethod to use (here we already knew the service, port and operation names):
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
URL wsdlURL = new URL("http://localhost:8080/complex/RegistrationService?wsdl" ); | |
WSDynamicClientBuilder clientBuilder = WSDynamicClientFactory.getJAXWSClientBuilder(); | |
WSDynamicClient client = clientBuilder.tmpDir("target/temp/wise") | |
.verbose(true).keepSource(true).wsdlURL(wsdlURL.toString()).build(); | |
WSMethod method = client.getWSMethod("RegistrationServiceImplService", | |
"RegistrationServiceImplPort", "Register"); |
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
ElementBuilder builder = ElementBuilderFactory.getElementBuilder() | |
.client(client).request(true).useDefautValuesForNullLeaves(false); |
The builder is used to build up the request trees (one for each method parameter):
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
Map<String, Element> elementsMap = new HashMap<String, Element>(); | |
for (Entry<String, ? extends WebParameter> par : method.getWebParams().entrySet()) { | |
String parName = par.getKey(); | |
Element parElement = builder.buildTree(par.getValue().getType(), parName, null, true); | |
populateElement(parElement, 1); | |
elementsMap.put(parName, parElement); | |
} |
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
private void populateElement(Element element, int remainingLazyExpansions) { | |
if (element.isLazy()) { | |
if (!element.isResolved()) { | |
if (remainingLazyExpansions > 0) { //[D] | |
element.getChildrenCount(); // force resolution | |
populateElement(element, --remainingLazyExpansions); | |
} else { | |
return; | |
} | |
} | |
} | |
element.setNil(false); //[A] | |
if (element.isLeaf()) { | |
element.setValue(getDefaultValue((Class<?>) element.getClassType())); //[B] | |
} else { | |
if (element.isGroup()) { | |
element.incrementChildren(); //[C] | |
} | |
for (Iterator<? extends Element> it = element.getChildren(); it.hasNext();) { | |
populateElement(it.next(), remainingLazyExpansions); | |
} | |
} |
In this case, the default values are set in quite a dirty way in getDefaultValue(Class cl), basically we check provided class and return a "?" for String, zero for any numerical type, "1970-01-01T00:00:00.000Z" for XMLGregorianCalendar and an empty string otherwise.
Once the tree is available, getting the actual parameter objects to perform the invocation with is a matter of calling toObject() on each tree root element:
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
Map<String, Object> args = new java.util.HashMap<String, Object>(); | |
for (Entry<String, Element> elem : elementsMap.entrySet()) { | |
args.put(elem.getKey(), elem.getValue().toObject()); | |
} |
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
ByteArrayOutputStream bos = new ByteArrayOutputStream(); | |
method.writeRequestPreview(args, bos); |
You can have a look at the whole code in the MessagePreviewIntegrationTest I've recently added to the current Wise core 2.0.1-SNAPSHOT. Any feedback / comment is welcome!
And if you're wondering how this is currently implemented, well, you know Wise basically uses JAXWS tooling from JBossWS to generate a compliant ws client; the Wise model is built by parsing the generated client and the tree view is derived from the model. The message preview flow is pretty similar to the invocation one, except a special jaxws handler is installed in the client in order for writing down the SOAP message generated by the internal ws stack (Apache CXF here); the handler also stops the handler chain execution so that no message actually goes to wire and Wise early return the message preview.
As previously explained, delegating to a fully compliant WS stack here, ensure correctness of the messages, even of their preview ;-)
No comments:
Post a Comment