diff -r 000000000000 -r 373ffda63c9a src/share/jaxws_classes/com/sun/xml/internal/ws/wsdl/parser/W3CAddressingWSDLParserExtension.java --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/share/jaxws_classes/com/sun/xml/internal/ws/wsdl/parser/W3CAddressingWSDLParserExtension.java Wed Apr 27 01:27:09 2016 +0800 @@ -0,0 +1,261 @@ +/* + * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +package com.sun.xml.internal.ws.wsdl.parser; + +import com.sun.xml.internal.ws.api.addressing.AddressingVersion; +import com.sun.xml.internal.ws.api.model.wsdl.WSDLFeaturedObject; +import static com.sun.xml.internal.ws.api.model.wsdl.WSDLBoundOperation.ANONYMOUS; +import com.sun.xml.internal.ws.api.model.wsdl.editable.*; +import com.sun.xml.internal.ws.api.wsdl.parser.WSDLParserExtension; +import com.sun.xml.internal.ws.api.wsdl.parser.WSDLParserExtensionContext; +import com.sun.xml.internal.ws.streaming.XMLStreamReaderUtil; + +import javax.xml.namespace.QName; +import javax.xml.stream.XMLStreamException; +import javax.xml.stream.XMLStreamReader; +import javax.xml.ws.WebServiceException; +import javax.xml.ws.soap.AddressingFeature; + +/** + * W3C WS-Addressing Runtime WSDL parser extension + * + * @author Arun Gupta + */ +public class W3CAddressingWSDLParserExtension extends WSDLParserExtension { + @Override + public boolean bindingElements(EditableWSDLBoundPortType binding, XMLStreamReader reader) { + return addressibleElement(reader, binding); + } + + @Override + public boolean portElements(EditableWSDLPort port, XMLStreamReader reader) { + return addressibleElement(reader, port); + } + + private boolean addressibleElement(XMLStreamReader reader, WSDLFeaturedObject binding) { + QName ua = reader.getName(); + if (ua.equals(AddressingVersion.W3C.wsdlExtensionTag)) { + String required = reader.getAttributeValue(WSDLConstants.NS_WSDL, "required"); + binding.addFeature(new AddressingFeature(true, Boolean.parseBoolean(required))); + XMLStreamReaderUtil.skipElement(reader); + return true; // UsingAddressing is consumed + } + + return false; + } + + @Override + public boolean bindingOperationElements(EditableWSDLBoundOperation operation, XMLStreamReader reader) { + EditableWSDLBoundOperation edit = (EditableWSDLBoundOperation) operation; + + QName anon = reader.getName(); + if (anon.equals(AddressingVersion.W3C.wsdlAnonymousTag)) { + try { + String value = reader.getElementText(); + if (value == null || value.trim().equals("")) { + throw new WebServiceException("Null values not permitted in wsaw:Anonymous."); + // TODO: throw exception only if wsdl:required=true + // TODO: is this the right exception ? + } else if (value.equals("optional")) { + edit.setAnonymous(ANONYMOUS.optional); + } else if (value.equals("required")) { + edit.setAnonymous(ANONYMOUS.required); + } else if (value.equals("prohibited")) { + edit.setAnonymous(ANONYMOUS.prohibited); + } else { + throw new WebServiceException("wsaw:Anonymous value \"" + value + "\" not understood."); + // TODO: throw exception only if wsdl:required=true + // TODO: is this the right exception ? + } + } catch (XMLStreamException e) { + throw new WebServiceException(e); // TODO: is this the correct behavior ? + } + + return true; // consumed the element + } + + return false; + } + + public void portTypeOperationInputAttributes(EditableWSDLInput input, XMLStreamReader reader) { + String action = ParserUtil.getAttribute(reader, getWsdlActionTag()); + if (action != null) { + input.setAction(action); + input.setDefaultAction(false); + } + } + + + public void portTypeOperationOutputAttributes(EditableWSDLOutput output, XMLStreamReader reader) { + String action = ParserUtil.getAttribute(reader, getWsdlActionTag()); + if (action != null) { + output.setAction(action); + output.setDefaultAction(false); + } + } + + + public void portTypeOperationFaultAttributes(EditableWSDLFault fault, XMLStreamReader reader) { + String action = ParserUtil.getAttribute(reader, getWsdlActionTag()); + if (action != null) { + fault.setAction(action); + fault.setDefaultAction(false); + } + } + + /** + * Process wsdl:portType operation after the entire WSDL model has been populated. + * The task list includes:

+ *

+ * @param context + */ + @Override + public void finished(WSDLParserExtensionContext context) { + EditableWSDLModel model = context.getWSDLModel(); + for (EditableWSDLService service : model.getServices().values()) { + for (EditableWSDLPort port : service.getPorts()) { + EditableWSDLBoundPortType binding = port.getBinding(); + + // populate actions for the messages that do not have an explicit wsaw:Action + populateActions(binding); + + // patch the default value of wsaw:Anonymous=optional if none is specified + patchAnonymousDefault(binding); + } + } + } + + protected String getNamespaceURI() { + return AddressingVersion.W3C.wsdlNsUri; + } + + protected QName getWsdlActionTag() { + return AddressingVersion.W3C.wsdlActionTag; + } + /** + * Populate all the Actions + * + * @param binding soapbinding:operation + */ + private void populateActions(EditableWSDLBoundPortType binding) { + EditableWSDLPortType porttype = binding.getPortType(); + for (EditableWSDLOperation o : porttype.getOperations()) { + // TODO: this may be performance intensive. Alternatively default action + // TODO: can be calculated when the operation is actually invoked. + EditableWSDLBoundOperation wboi = binding.get(o.getName()); + + if (wboi == null) { + //If this operation is unbound set the action to default + o.getInput().setAction(defaultInputAction(o)); + continue; + } + String soapAction = wboi.getSOAPAction(); + if (o.getInput().getAction() == null || o.getInput().getAction().equals("")) { + // explicit wsaw:Action is not specified + + if (soapAction != null && !soapAction.equals("")) { + // if soapAction is non-empty, use that + o.getInput().setAction(soapAction); + } else { + // otherwise generate default Action + o.getInput().setAction(defaultInputAction(o)); + } + } + + // skip output and fault processing for one-way methods + if (o.getOutput() == null) + continue; + + if (o.getOutput().getAction() == null || o.getOutput().getAction().equals("")) { + o.getOutput().setAction(defaultOutputAction(o)); + } + + if (o.getFaults() == null || !o.getFaults().iterator().hasNext()) + continue; + + for (EditableWSDLFault f : o.getFaults()) { + if (f.getAction() == null || f.getAction().equals("")) { + f.setAction(defaultFaultAction(f.getName(), o)); + } + + } + } + } + + /** + * Patch the default value of wsaw:Anonymous=optional if none is specified + * + * @param binding WSDLBoundPortTypeImpl + */ + protected void patchAnonymousDefault(EditableWSDLBoundPortType binding) { + for (EditableWSDLBoundOperation wbo : binding.getBindingOperations()) { + if (wbo.getAnonymous() == null) + wbo.setAnonymous(ANONYMOUS.optional); + } + } + + private String defaultInputAction(EditableWSDLOperation o) { + return buildAction(o.getInput().getName(), o, false); + } + + private String defaultOutputAction(EditableWSDLOperation o) { + return buildAction(o.getOutput().getName(), o, false); + } + + private String defaultFaultAction(String name, EditableWSDLOperation o) { + return buildAction(name, o, true); + } + + protected static final String buildAction(String name, EditableWSDLOperation o, boolean isFault) { + String tns = o.getName().getNamespaceURI(); + + String delim = SLASH_DELIMITER; + + // TODO: is this the correct way to find the separator ? + if (!tns.startsWith("http")) + delim = COLON_DELIMITER; + + if (tns.endsWith(delim)) + tns = tns.substring(0, tns.length()-1); + + if (o.getPortTypeName() == null) + throw new WebServiceException("\"" + o.getName() + "\" operation's owning portType name is null."); + + return tns + + delim + + o.getPortTypeName().getLocalPart() + + delim + + (isFault ? o.getName().getLocalPart() + delim + "Fault" + delim : "") + + name; + } + + protected static final String COLON_DELIMITER = ":"; + protected static final String SLASH_DELIMITER = "/"; +}