/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 2006-2013 Oracle and/or its affiliates. All rights reserved. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License"). You * may not use this file except in compliance with the License. You can * obtain a copy of the License at * http://glassfish.java.net/public/CDDL+GPL_1_1.html * or packager/legal/LICENSE.txt. See the License for the specific * language governing permissions and limitations under the License. * * When distributing the software, include this License Header Notice in each * file and include the License file at packager/legal/LICENSE.txt. * * GPL Classpath Exception: * Oracle designates this particular file as subject to the "Classpath" * exception as provided by Oracle in the GPL Version 2 section of the License * file that accompanied this code. * * Modifications: * If applicable, add the following below the License Header, with the fields * enclosed by brackets [] replaced by your own identifying information: * "Portions Copyright [year] [name of copyright owner]" * * Contributor(s): * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license." If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above. However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */ package wsa.fromwsdl.custom.client; import com.sun.xml.ws.addressing.W3CAddressingConstants; import com.sun.xml.ws.api.addressing.AddressingVersion; import com.sun.xml.ws.api.addressing.OneWayFeature; import com.sun.xml.ws.api.addressing.WSEndpointReference; import com.sun.xml.ws.api.WSService; import com.sun.xml.ws.util.DOMUtil; import com.sun.xml.ws.streaming.XMLStreamReaderException; import com.sun.xml.ws.developer.MemberSubmissionAddressingFeature; import org.custommonkey.xmlunit.XMLTestCase; import org.w3c.dom.Element; import testutil.ClientServerTestUtil; import testutil.WsaUtils; import javax.xml.namespace.QName; import javax.xml.soap.SOAPMessage; import javax.xml.soap.SOAPFault; import javax.xml.soap.SOAPFactory; import javax.xml.soap.SOAPConstants; import javax.xml.ws.*; import javax.xml.ws.soap.AddressingFeature; import javax.xml.ws.soap.SOAPBinding; import javax.xml.ws.soap.SOAPFaultException; import javax.xml.transform.Source; import javax.xml.transform.dom.DOMSource; import java.io.File; import java.io.InputStream; import java.util.ArrayList; /** * @author Arun Gupta */ public class AddNumbersClient extends XMLTestCase { private static final QName SERVICE_QNAME = new QName("http://example.com/", "AddNumbersService"); private static final QName PORT_QNAME = new QName("http://example.com/", "AddNumbersPort"); private static final QName PORT_TYPE_QNAME = new QName("http://example.com/", "AddNumbersPortType"); private static final String ENDPOINT_ADDRESS = "http://localhost:8080/jaxrpc-wsa_fromwsdl_custom/hello"; private static final String CORRECT_ACTION = "http://example.com/AddNumbersPortType/addNumbersRequest"; public AddNumbersClient(String name) { super(name); } private String getAddress() { if (ClientServerTestUtil.useLocal()) return ClientServerTestUtil.getLocalAddress(PORT_QNAME); else return ENDPOINT_ADDRESS; } private String getWsdlAddress() { // we'll fix the test harness correctly later, // so that test code won't have to hard code any endpoint address nor transport, // but for now let's just support local and HTTP to make unit tests happier. // this is not a good code, but it's just a bandaid solutino that works for now. if (ClientServerTestUtil.useLocal()) return "local://" + new File(System.getProperty("tempdir")).getAbsolutePath().replace('\\', '/') + '?' + "hello_literal.wsdl"; else return "http://localhost:8080/jaxrpc-wsa_fromwsdl_custom/hello?wsdl"; } private Dispatch<SOAPMessage> createDispatchWithoutWSDL() throws Exception { Service service = Service.create(SERVICE_QNAME); service.addPort(PORT_QNAME, SOAPBinding.SOAP11HTTP_BINDING, getAddress()); Dispatch<SOAPMessage> dispatch = service.createDispatch(PORT_QNAME, SOAPMessage.class, Service.Mode.MESSAGE, new AddressingFeature(false, false)); return dispatch; } private Dispatch<SOAPMessage> createDispatchWithWSDL() throws Exception { AddNumbersService service = new AddNumbersService(); return service.createDispatch(PORT_QNAME, SOAPMessage.class, Service.Mode.MESSAGE, new AddressingFeature()); } private AddNumbersPortType createStub(WebServiceFeature... features) throws Exception { return new AddNumbersService().getAddNumbersPort(features); } /** * This test is added to test Addressing with only requitred headers in the message. * In 2.1.3, wsa:To is checked for presence although it is not requried to be present * as per W3C Spec. * This test sends only wsa:Action and wsa:MessageId headers in the request. * @throws Exception */ public void testAddressingWithOnlyRequiredHeaders () throws Exception{ WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.SIMPLE_ADDRESSING_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, CORRECT_ACTION); } public void testBadAction() throws Exception { try { WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.BAD_ACTION_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.ACTION_NOT_SUPPORTED_QNAME); } } public void testMissingAddressingHeaders() throws Exception { try { WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.NO_ADDRESSING_MESSAGE, WsaUtils.S11_NS); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.MAP_REQUIRED_QNAME); } } public void testMissingAction() throws Exception { try { WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.MISSING_ACTION_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.MAP_REQUIRED_QNAME); } } public void testMissingAddressInReplyTo() throws Exception { try { WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.MISSING_ADDRESS_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), CORRECT_ACTION); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.MISSING_ADDRESS_IN_EPR); } } public void testReplyToRefps() throws Exception { SOAPMessage response = WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.REPLY_TO_REFPS_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, CORRECT_ACTION); // ByteArrayOutputStream baos = new ByteArrayOutputStream(); // response.writeTo(baos); // assertXpathExists(REPLY_TO_REFPS, baos.toString()); // assertXpathEvaluatesTo("Key#123456789", REPLY_TO_REFPS_VALUE, baos.toString()); // assertXpathExists(REPLY_TO_REFPS_ISREFP, baos.toString()); } public void testInvalidReplyTo() throws Exception { try { SOAPMessage response = WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.INVALID_REPLY_TO_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), "WRONG", W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, CORRECT_ACTION); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertTrue("Got SOAPFaultException", true); } } public void testFaultToRefps() throws Exception { try { WsaUtils.invoke(createDispatchWithoutWSDL(), WsaUtils.FAULT_TO_REFPS_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, CORRECT_ACTION); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertTrue("Got SOAPFaultException", true); } } public void testDuplicateToHeader() throws Exception { try { WsaUtils.invoke(createDispatchWithWSDL(), WsaUtils.DUPLICATE_TO_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress(), getAddress()); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.INVALID_CARDINALITY); } } public void testDuplicateReplyToHeader() throws Exception { try { WsaUtils.invoke(createDispatchWithWSDL(), WsaUtils.DUPLICATE_REPLY_TO_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.INVALID_CARDINALITY); } } public void testDuplicateFaultToHeader() throws Exception { try { WsaUtils.invoke(createDispatchWithWSDL(), WsaUtils.DUPLICATE_FAULT_TO_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS, W3CAddressingConstants.WSA_ANONYMOUS_ADDRESS); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.INVALID_CARDINALITY); } } public void testDuplicateActionHeader() throws Exception { try { WsaUtils.invoke(createDispatchWithWSDL(), WsaUtils.DUPLICATE_ACTION_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS,getAddress(),CORRECT_ACTION, CORRECT_ACTION); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.INVALID_CARDINALITY); } } public void testDuplicateMessageIDHeader() throws Exception { try { WsaUtils.invoke(createDispatchWithWSDL(), WsaUtils.DUPLICATE_MESSAGE_ID_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress()); fail("SOAPFaultException must be thrown"); } catch (SOAPFaultException sfe) { assertFault(sfe, W3CAddressingConstants.INVALID_CARDINALITY); } } public void testDuplicateMessageIDHeaderOneway() throws Exception { WsaUtils.invokeOneWay(createDispatchWithWSDL(), WsaUtils.DUPLICATE_MESSAGE_ID_MESSAGE_ONEWAY, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, getAddress()); } public void testOnewayWithReplyTo() throws Exception { AddNumbersPortType stub = createStub(new OneWayFeature(true, new WSEndpointReference(getAddress(), AddressingVersion.W3C))); stub.addNumbers5(10, 10); } /** * This test tests the functionality of OnewayFeature and AddressingFeature the way it being used in WS-AT implementation * In WS-AT impl, Server-side has to send fault messages to predefined coordinator and this test replicates that usage. * * @throws Exception */ public void testCustomFault() throws Exception { if(ClientServerTestUtil.useLocal()){ System.out.println("Only Testable in HTTP transport!"); return; } SOAPFault fault = SOAPFactory.newInstance().createFault("custom fault from client", SOAPConstants.SOAP_SENDER_FAULT); InputStream is = getClass().getClassLoader().getResourceAsStream("wsa/fromwsdl/custom/config/AddNumbers.wsdl"); assertNotNull("WSDL cannot be read", is); ArrayList<Element> metadata = new ArrayList<Element>(); metadata.add((Element)DOMUtil.createDOMNode(is).getFirstChild()); // WSEndpointReference to = new WSEndpointReference(AddressingVersion.W3C, // getAddress(), // SERVICE_QNAME, // PORT_QNAME, // PORT_TYPE_QNAME, // metadata, // getWsdlAddress(), // null); OneWayFeature owf = new OneWayFeature(); owf.setRelatesToID("uuid:foobar"); Service service = Service.create(SERVICE_QNAME); service.addPort(PORT_QNAME, SOAPBinding.SOAP11HTTP_BINDING, getAddress()); // Dispatch<Source> dispatch = service.createDispatch(to.toSpec(), Source.class, // Service.Mode.PAYLOAD, // new AddressingFeature(true, true), // owf); Dispatch<Source> dispatch = service.createDispatch(PORT_QNAME,Source.class, Service.Mode.PAYLOAD, new MemberSubmissionAddressingFeature(true), owf); //Since this fault is not a wsdl operation, we need to set SOAPAction for correct wsa:Action dispatch.getRequestContext().put(BindingProvider.SOAPACTION_USE_PROPERTY,true); dispatch.getRequestContext().put(BindingProvider.SOAPACTION_URI_PROPERTY,"http://example.com/myfault"); try { dispatch.invokeOneWay(new DOMSource(fault)); } catch (WebServiceException e) { // since the server-side is not provider based for this test. // it does n't know from the fault message request that it is oneway and throws 500 code. // so, expect a WebServcieException here. } // System.out.println(dispatch.getRequestContext().get(BindingProvider.ENDPOINT_ADDRESS_PROPERTY)); } /** * This test tests the functionality of OnewayFeature and AddressingFeature the way it being used in WS-AT implementation * In WS-AT impl, Server-side has to send fault messages to predefined coordinator and this test replicates that usage. * * @throws Exception */ public void testCustomFault1() throws Exception { SOAPFault fault = SOAPFactory.newInstance().createFault("custom fault from client", SOAPConstants.SOAP_SENDER_FAULT); WSEndpointReference to = new WSEndpointReference( getAddress(), AddressingVersion.MEMBER); OneWayFeature owf = new OneWayFeature(); owf.setRelatesToID("uuid:foobar"); WSService service = WSService.create(); service.addPort(PORT_QNAME, SOAPBinding.SOAP11HTTP_BINDING, getAddress()); Dispatch<Source> dispatch = service.createDispatch(PORT_QNAME, to, Source.class, Service.Mode.PAYLOAD, new MemberSubmissionAddressingFeature(true), owf); //Since this fault is not a wsdl operation, we need to set SOAPAction for correct wsa:Action dispatch.getRequestContext().put(BindingProvider.SOAPACTION_USE_PROPERTY, true); dispatch.getRequestContext().put(BindingProvider.SOAPACTION_URI_PROPERTY, "http://example.com/myfault"); try { dispatch.invokeOneWay(new DOMSource(fault)); } catch (WebServiceException e) { // since the server-side is not provider based for this test. // it does n't know from the fault message request that it is oneway and throws 500 code. // so, expect a WebServcieException here. } } public void testIncorrectNonAnonymousURI() throws Exception { WsaUtils.invokeOneWay(createDispatchWithoutWSDL(), WsaUtils.INVALID_NON_ANONYMOUS_URI_MESSAGE, WsaUtils.S11_NS, WsaUtils.W3C_WSA_NS, CORRECT_ACTION, getAddress()); } private void assertFault(SOAPFaultException sfe, QName expected) { assertNotNull("SOAPFaultException is null", sfe); assertNotNull("SOAPFault is null", sfe.getFault()); assertEquals(expected, sfe.getFault().getFaultCodeAsQName()); } private static final String REPLY_TO_REFPS = "//[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Envelope']/" + "[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Header']/" + "[namespace-uri()='http://example.org/customer' and local-name()='CustomerKey']"; private static final String REPLY_TO_REFPS_VALUE = "string(//[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Envelope']/" + "[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Header']/" + "[namespace-uri()='http://example.org/customer' and local-name()='CustomerKey'])"; private static final String REPLY_TO_REFPS_ISREFP = "//[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Envelope']/" + "[namespace-uri()='http://schemas.xmlsoap.org/soap/envelope/' and local-name()='Header']/" + "[namespace-uri()='http://example.org/customer' and local-name()='CustomerKey']" + "[@[namespace-uri()='" + W3CAddressingConstants.WSA_NAMESPACE_NAME + "' and local-name()='IsReferenceParameter']]"; }