/*
 * 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.api.message;

import com.sun.istack.internal.NotNull;
import com.sun.istack.internal.Nullable;
import com.sun.xml.internal.stream.buffer.XMLStreamBuffer;
import com.sun.xml.internal.ws.api.SOAPVersion;
import com.sun.xml.internal.ws.api.WSBinding;
import com.sun.xml.internal.ws.api.addressing.AddressingVersion;
import com.sun.xml.internal.ws.api.message.saaj.SAAJFactory;
import com.sun.xml.internal.ws.api.pipe.Tube;
import com.sun.xml.internal.ws.api.pipe.Codecs;
import com.sun.xml.internal.ws.fault.SOAPFaultBuilder;
import com.sun.xml.internal.ws.message.AttachmentSetImpl;
import com.sun.xml.internal.ws.message.DOMMessage;
import com.sun.xml.internal.ws.message.EmptyMessageImpl;
import com.sun.xml.internal.ws.message.ProblemActionHeader;
import com.sun.xml.internal.ws.message.stream.PayloadStreamReaderMessage;
import com.sun.xml.internal.ws.message.jaxb.JAXBMessage;
import com.sun.xml.internal.ws.message.source.PayloadSourceMessage;
import com.sun.xml.internal.ws.message.source.ProtocolSourceMessage;
import com.sun.xml.internal.ws.spi.db.BindingContextFactory;
import com.sun.xml.internal.ws.streaming.XMLStreamReaderException;
import com.sun.xml.internal.ws.streaming.XMLStreamReaderUtil;
import com.sun.xml.internal.ws.util.DOMUtil;
import com.sun.xml.internal.ws.addressing.WsaTubeHelper;
import com.sun.xml.internal.ws.addressing.model.MissingAddressingHeaderException;
import com.sun.xml.internal.ws.resources.AddressingMessages;
import org.w3c.dom.Element;
import org.w3c.dom.Node;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBElement;
import javax.xml.bind.Marshaller;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.namespace.QName;
import javax.xml.soap.*;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamReader;
import javax.xml.transform.Source;
import javax.xml.transform.sax.SAXSource;
import javax.xml.transform.stream.StreamSource;
import javax.xml.transform.dom.DOMSource;
import javax.xml.ws.ProtocolException;
import javax.xml.ws.WebServiceException;

Factory methods for various Message implementations.

This class provides various methods to create different flavors of Message classes that store data in different formats.

This is a part of the JAX-WS RI internal API so that Tube implementations can reuse the implementations done inside the JAX-WS.

If you find some of the useful convenience methods missing from this class, please talk to us.

Author:Kohsuke Kawaguchi
/** * Factory methods for various {@link Message} implementations. * * <p> * This class provides various methods to create different * flavors of {@link Message} classes that store data * in different formats. * * <p> * This is a part of the JAX-WS RI internal API so that * {@link Tube} implementations can reuse the implementations * done inside the JAX-WS. * * <p> * If you find some of the useful convenience methods missing * from this class, please talk to us. * * * @author Kohsuke Kawaguchi */
public abstract class Messages { private Messages() {}
Creates a Message backed by a JAXB bean.
Params:
  • context – The context to be used to produce infoset from the object. Must not be null.
  • jaxbObject – The JAXB object that represents the payload. must not be null. This object must be bound to an element (which means it either is a JAXBElement or an instanceof a class with XmlRootElement).
  • soapVersion – The SOAP version of the message. Must not be null.
Deprecated:
/** * Creates a {@link Message} backed by a JAXB bean. * @deprecated * @param context * The context to be used to produce infoset from the object. Must not be null. * @param jaxbObject * The JAXB object that represents the payload. must not be null. This object * must be bound to an element (which means it either is a {@link JAXBElement} or * an instanceof a class with {@link XmlRootElement}). * @param soapVersion * The SOAP version of the message. Must not be null. */
public static Message create(JAXBContext context, Object jaxbObject, SOAPVersion soapVersion) { return JAXBMessage.create(context,jaxbObject,soapVersion); }
Deprecated: For use when creating a Dispatch object with an unknown JAXB implementation for he JAXBContext parameter.
/** * @deprecated * For use when creating a Dispatch object with an unknown JAXB implementation * for he JAXBContext parameter. * */
public static Message createRaw(JAXBContext context, Object jaxbObject, SOAPVersion soapVersion) { return JAXBMessage.createRaw(context,jaxbObject,soapVersion); }
Deprecated: Use create(JAXBRIContext, Object, SOAPVersion)
/** * @deprecated * Use {@link #create(JAXBRIContext, Object, SOAPVersion)} */
public static Message create(Marshaller marshaller, Object jaxbObject, SOAPVersion soapVersion) { return create(BindingContextFactory.getBindingContext(marshaller).getJAXBContext(),jaxbObject,soapVersion); }
Creates a Message backed by a SAAJ SOAPMessage object.

If the SOAPMessage contains headers and attachments, this method does the right thing.

Params:
  • saaj – The SOAP message to be represented as a Message. Must not be null. Once this method is invoked, the created Message will own the SOAPMessage, so it shall never be touched directly.
/** * Creates a {@link Message} backed by a SAAJ {@link SOAPMessage} object. * * <p> * If the {@link SOAPMessage} contains headers and attachments, this method * does the right thing. * * @param saaj * The SOAP message to be represented as a {@link Message}. * Must not be null. Once this method is invoked, the created * {@link Message} will own the {@link SOAPMessage}, so it shall * never be touched directly. */
public static Message create(SOAPMessage saaj) { return SAAJFactory.create(saaj); }
Creates a Message using Source as payload.
Params:
  • payload – Source payload is Message's payload Must not be null. Once this method is invoked, the created Message will own the Source, so it shall never be touched directly.
  • ver – The SOAP version of the message. Must not be null.
/** * Creates a {@link Message} using {@link Source} as payload. * * @param payload * Source payload is {@link Message}'s payload * Must not be null. Once this method is invoked, the created * {@link Message} will own the {@link Source}, so it shall * never be touched directly. * * @param ver * The SOAP version of the message. Must not be null. */
public static Message createUsingPayload(Source payload, SOAPVersion ver) { if (payload instanceof DOMSource) { if (((DOMSource)payload).getNode() == null) { return new EmptyMessageImpl(ver); } } else if (payload instanceof StreamSource) { StreamSource ss = (StreamSource)payload; if (ss.getInputStream() == null && ss.getReader() == null && ss.getSystemId() == null) { return new EmptyMessageImpl(ver); } } else if (payload instanceof SAXSource) { SAXSource ss = (SAXSource)payload; if (ss.getInputSource() == null && ss.getXMLReader() == null) { return new EmptyMessageImpl(ver); } } return new PayloadSourceMessage(payload, ver); }
Creates a Message using XMLStreamReader as payload.
Params:
  • payload – XMLStreamReader payload is Message's payload Must not be null. Once this method is invoked, the created Message will own the XMLStreamReader, so it shall never be touched directly.
  • ver – The SOAP version of the message. Must not be null.
/** * Creates a {@link Message} using {@link XMLStreamReader} as payload. * * @param payload * XMLStreamReader payload is {@link Message}'s payload * Must not be null. Once this method is invoked, the created * {@link Message} will own the {@link XMLStreamReader}, so it shall * never be touched directly. * * @param ver * The SOAP version of the message. Must not be null. */
public static Message createUsingPayload(XMLStreamReader payload, SOAPVersion ver) { return new PayloadStreamReaderMessage(payload, ver); }
Creates a Message from an Element that represents a payload.
Params:
  • payload – The element that becomes the child element of the SOAP body. Must not be null.
  • ver – The SOAP version of the message. Must not be null.
/** * Creates a {@link Message} from an {@link Element} that represents * a payload. * * @param payload * The element that becomes the child element of the SOAP body. * Must not be null. * * @param ver * The SOAP version of the message. Must not be null. */
public static Message createUsingPayload(Element payload, SOAPVersion ver) { return new DOMMessage(ver,payload); }
Creates a Message from an Element that represents the whole SOAP message.
Params:
  • soapEnvelope – The SOAP envelope element.
/** * Creates a {@link Message} from an {@link Element} that represents * the whole SOAP message. * * @param soapEnvelope * The SOAP envelope element. */
public static Message create(Element soapEnvelope) { SOAPVersion ver = SOAPVersion.fromNsUri(soapEnvelope.getNamespaceURI()); // find the headers Element header = DOMUtil.getFirstChild(soapEnvelope, ver.nsUri, "Header"); HeaderList headers = null; if(header!=null) { for( Node n=header.getFirstChild(); n!=null; n=n.getNextSibling() ) { if(n.getNodeType()==Node.ELEMENT_NODE) { if(headers==null) headers = new HeaderList(ver); headers.add(Headers.create((Element)n)); } } } // find the payload Element body = DOMUtil.getFirstChild(soapEnvelope, ver.nsUri, "Body"); if(body==null) throw new WebServiceException("Message doesn't have <S:Body> "+soapEnvelope); Element payload = DOMUtil.getFirstChild(soapEnvelope, ver.nsUri, "Body"); if(payload==null) { return new EmptyMessageImpl(headers, new AttachmentSetImpl(), ver); } else { return new DOMMessage(ver,headers,payload); } }
Creates a Message using Source as entire envelope.
Params:
  • envelope – Source envelope is used to create Message Must not be null. Once this method is invoked, the created Message will own the Source, so it shall never be touched directly.
/** * Creates a {@link Message} using Source as entire envelope. * * @param envelope * Source envelope is used to create {@link Message} * Must not be null. Once this method is invoked, the created * {@link Message} will own the {@link Source}, so it shall * never be touched directly. * */
public static Message create(Source envelope, SOAPVersion soapVersion) { return new ProtocolSourceMessage(envelope, soapVersion); }
Creates a Message that doesn't have any payload.
/** * Creates a {@link Message} that doesn't have any payload. */
public static Message createEmpty(SOAPVersion soapVersion) { return new EmptyMessageImpl(soapVersion); }
Creates a Message from XMLStreamReader that points to the start of the envelope.
Params:
  • reader – can point to the start document or the start element (of <s:Envelope>)
/** * Creates a {@link Message} from {@link XMLStreamReader} that points to * the start of the envelope. * * @param reader * can point to the start document or the start element (of &lt;s:Envelope>) */
public static @NotNull Message create(@NotNull XMLStreamReader reader) { // skip until the root element if(reader.getEventType()!=XMLStreamConstants.START_ELEMENT) XMLStreamReaderUtil.nextElementContent(reader); assert reader.getEventType()== XMLStreamConstants.START_ELEMENT :reader.getEventType(); SOAPVersion ver = SOAPVersion.fromNsUri(reader.getNamespaceURI()); return Codecs.createSOAPEnvelopeXmlCodec(ver).decode(reader); }
Creates a Message from XMLStreamBuffer that retains the whole envelope infoset.
Params:
  • xsb – This buffer must contain the infoset of the whole envelope.
/** * Creates a {@link Message} from {@link XMLStreamBuffer} that retains the * whole envelope infoset. * * @param xsb * This buffer must contain the infoset of the whole envelope. */
public static @NotNull Message create(@NotNull XMLStreamBuffer xsb) { // TODO: we should be able to let Messae know that it's working off from a buffer, // to make some of the operations more efficient. // meanwhile, adding this as an API so that our users can take advantage of it // when we get around to such an implementation later. try { return create(xsb.readAsXMLStreamReader()); } catch (XMLStreamException e) { throw new XMLStreamReaderException(e); } }
Creates a Message that represents an exception as a fault. The created message reflects if t or t.getCause() is SOAPFaultException. creates a fault message with default faultCode env:Server if t or t.getCause() is not SOAPFaultException. Otherwise, it use SOAPFaultException's faultCode
Returns: Always non-null. A message that wraps this Throwable.
/** * Creates a {@link Message} that represents an exception as a fault. The * created message reflects if t or t.getCause() is SOAPFaultException. * * creates a fault message with default faultCode env:Server if t or t.getCause() * is not SOAPFaultException. Otherwise, it use SOAPFaultException's faultCode * * @return * Always non-null. A message that wraps this {@link Throwable}. * */
public static Message create(Throwable t, SOAPVersion soapVersion) { return SOAPFaultBuilder.createSOAPFaultMessage(soapVersion, null, t); }
Creates a fault Message.

This method is not designed for efficiency, and we don't expect to be used for the performance critical codepath.

Params:
  • fault – The populated SAAJ data structure that represents a fault in detail.
Returns: Always non-null. A message that wraps this SOAPFault.
/** * Creates a fault {@link Message}. * * <p> * This method is not designed for efficiency, and we don't expect * to be used for the performance critical codepath. * * @param fault * The populated SAAJ data structure that represents a fault * in detail. * * @return * Always non-null. A message that wraps this {@link SOAPFault}. */
public static Message create(SOAPFault fault) { SOAPVersion ver = SOAPVersion.fromNsUri(fault.getNamespaceURI()); return new DOMMessage(ver,fault); }
Deprecated: Use createAddressingFaultMessage(WSBinding, Packet, QName)
/** * @deprecated * Use {@link #createAddressingFaultMessage(WSBinding, Packet, QName)} */
public static Message createAddressingFaultMessage(WSBinding binding, QName missingHeader) { return createAddressingFaultMessage(binding,null,missingHeader); }
Creates a fault Message that captures the code/subcode/subsubcode defined by WS-Addressing if one of the expected WS-Addressing headers is missing in the message
Params:
  • binding – WSBinding
  • p – Packet that was missing a WS-Addressing header.
  • missingHeader – The missing WS-Addressing Header
Returns: A message representing SOAPFault that contains the WS-Addressing code/subcode/subsubcode.
/** * Creates a fault {@link Message} that captures the code/subcode/subsubcode * defined by WS-Addressing if one of the expected WS-Addressing headers is * missing in the message * * @param binding WSBinding * @param p * {@link Packet} that was missing a WS-Addressing header. * @param missingHeader The missing WS-Addressing Header * @return * A message representing SOAPFault that contains the WS-Addressing code/subcode/subsubcode. */
public static Message createAddressingFaultMessage(WSBinding binding, Packet p, QName missingHeader) { AddressingVersion av = binding.getAddressingVersion(); if(av == null) { // Addressing is not enabled. throw new WebServiceException(AddressingMessages.ADDRESSING_SHOULD_BE_ENABLED()); } WsaTubeHelper helper = av.getWsaHelper(null,null,binding); return create(helper.newMapRequiredFault(new MissingAddressingHeaderException(missingHeader,p))); }
Creates a fault Message that captures the code/subcode/subsubcode defined by WS-Addressing if wsa:Action is not supported.
Params:
  • unsupportedAction – The unsupported Action. Must not be null.
  • av – The WS-Addressing version of the message. Must not be null.
  • sv – The SOAP Version of the message. Must not be null.
Returns: A message representing SOAPFault that contains the WS-Addressing code/subcode/subsubcode.
/** * Creates a fault {@link Message} that captures the code/subcode/subsubcode * defined by WS-Addressing if wsa:Action is not supported. * * @param unsupportedAction The unsupported Action. Must not be null. * @param av The WS-Addressing version of the message. Must not be null. * @param sv The SOAP Version of the message. Must not be null. * * @return * A message representing SOAPFault that contains the WS-Addressing code/subcode/subsubcode. */
public static Message create(@NotNull String unsupportedAction, @NotNull AddressingVersion av, @NotNull SOAPVersion sv) { QName subcode = av.actionNotSupportedTag; String faultstring = String.format(av.actionNotSupportedText, unsupportedAction); Message faultMessage; SOAPFault fault; try { if (sv == SOAPVersion.SOAP_12) { fault = SOAPVersion.SOAP_12.getSOAPFactory().createFault(); fault.setFaultCode(SOAPConstants.SOAP_SENDER_FAULT); fault.appendFaultSubcode(subcode); Detail detail = fault.addDetail(); SOAPElement se = detail.addChildElement(av.problemActionTag); se = se.addChildElement(av.actionTag); se.addTextNode(unsupportedAction); } else { fault = SOAPVersion.SOAP_11.getSOAPFactory().createFault(); fault.setFaultCode(subcode); } fault.setFaultString(faultstring); faultMessage = SOAPFaultBuilder.createSOAPFaultMessage(sv, fault); if (sv == SOAPVersion.SOAP_11) { faultMessage.getHeaders().add(new ProblemActionHeader(unsupportedAction, av)); } } catch (SOAPException e) { throw new WebServiceException(e); } return faultMessage; }
To be called to convert a ProtocolException and faultcode for a given SOAPVersion in to a Message.
Params:
Returns:Message representing SOAP fault
/** * To be called to convert a {@link ProtocolException} and faultcode for a given {@link SOAPVersion} in to a {@link Message}. * * @param soapVersion {@link SOAPVersion#SOAP_11} or {@link SOAPVersion#SOAP_12} * @param pex a ProtocolException * @param faultcode soap faultcode. Its ignored if the {@link ProtocolException} instance is {@link javax.xml.ws.soap.SOAPFaultException} and it has a * faultcode present in the underlying {@link SOAPFault}. * @return {@link Message} representing SOAP fault */
public static @NotNull Message create(@NotNull SOAPVersion soapVersion, @NotNull ProtocolException pex, @Nullable QName faultcode){ return SOAPFaultBuilder.createSOAPFaultMessage(soapVersion, pex, faultcode); } }