blob: 44c99968162b9f5455267f3cb53f46fa5806115d [file] [log] [blame]
/*
* Portions Copyright 2006 Sun Microsystems, Inc. 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. Sun designates this
* particular file as subject to the "Classpath" exception as provided
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
* CA 95054 USA or visit www.sun.com if you need additional information or
* have any questions.
*/
package com.sun.xml.internal.ws.handler;
import com.sun.xml.internal.ws.pept.ept.MessageInfo;
import com.sun.xml.internal.ws.encoding.jaxb.JAXBBeanInfo;
import com.sun.xml.internal.ws.encoding.jaxb.JAXBTypeSerializer;
import com.sun.xml.internal.ws.encoding.soap.SOAPEPTFactory;
import com.sun.xml.internal.ws.encoding.soap.internal.InternalMessage;
import javax.xml.bind.JAXBContext;
import javax.xml.namespace.QName;
import javax.xml.soap.Name;
import javax.xml.soap.SOAPFactory;
import javax.xml.soap.SOAPHeader;
import javax.xml.soap.SOAPHeaderElement;
import javax.xml.soap.SOAPMessage;
import javax.xml.transform.Source;
import javax.xml.transform.dom.DOMSource;
import javax.xml.ws.WebServiceException;
import javax.xml.ws.handler.MessageContext;
import javax.xml.ws.handler.soap.SOAPMessageContext;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.xml.sax.InputSource;
/**
* Implementation of SOAPMessageContext. This class is used at runtime
* to pass to the handlers for processing soap messages.
*
* @see MessageContextImpl
*
* @author WS Development Team
*/
public class SOAPMessageContextImpl implements SOAPMessageContext {
private SOAPHandlerContext handlerCtxt;
private MessageContext ctxt;
private Set<String> roles;
private static Map<String, Class> allowedTypes = null;
private boolean failure;
public SOAPMessageContextImpl(SOAPHandlerContext handlerCtxt) {
this.handlerCtxt = handlerCtxt;
this.ctxt = handlerCtxt.getMessageContext();
if (allowedTypes == null) {
allowedTypes = new HashMap<String, Class>();
allowedTypes.put(MessageContext.INBOUND_MESSAGE_ATTACHMENTS, Map.class);
allowedTypes.put(MessageContext.OUTBOUND_MESSAGE_ATTACHMENTS, Map.class);
allowedTypes.put(MessageContext.WSDL_DESCRIPTION, InputSource.class);
allowedTypes.put(MessageContext.WSDL_SERVICE, QName.class);
allowedTypes.put(MessageContext.WSDL_PORT, QName.class);
allowedTypes.put(MessageContext.WSDL_INTERFACE, QName.class);
allowedTypes.put(MessageContext.WSDL_OPERATION, QName.class);
allowedTypes.put(MessageContext.MESSAGE_OUTBOUND_PROPERTY, Boolean.class);
}
}
public SOAPMessage getMessage() {
SOAPMessage soap = handlerCtxt.getSOAPMessage();
InternalMessage intr = handlerCtxt.getInternalMessage();
if (intr == null && soap != null) {
// Not much to do
} else if (intr != null && soap != null) {
// Overlay BodyBlock of InternalMessage on top of existing SOAPMessage
MessageInfo messageInfo = handlerCtxt.getMessageInfo();
SOAPEPTFactory eptf = (SOAPEPTFactory)messageInfo.getEPTFactory();
soap = eptf.getSOAPEncoder().toSOAPMessage(intr, soap);
setMessage(soap); // It also sets InernalMessage to null
} else if (intr != null && soap == null) {
// Convert InternalMessage to a SOAPMessage
MessageInfo messageInfo = handlerCtxt.getMessageInfo();
SOAPEPTFactory eptf = (SOAPEPTFactory)messageInfo.getEPTFactory();
soap = eptf.getSOAPEncoder().toSOAPMessage(intr, messageInfo);
setMessage(soap); // It also sets InernalMessage to null
} else {
throw new WebServiceException("Don't have SOAPMessage");
}
return soap;
}
public void setMessage(SOAPMessage soapMessage) {
handlerCtxt.setSOAPMessage(soapMessage);
// current InternalMessage is not valid anymore. So reset it.
handlerCtxt.setInternalMessage(null);
}
public Object[] getHeaders(QName header, JAXBContext jaxbContext, boolean allRoles) {
try {
List beanList = new ArrayList();
SOAPMessage msg = getMessage();
SOAPHeader sHeader = msg.getSOAPHeader();
if (sHeader == null) {
return new Object[0];
}
Iterator i = sHeader.getChildElements(header);
while(i.hasNext()) {
SOAPHeaderElement child = (SOAPHeaderElement)i.next();
if(allRoles) {
//If allRoles is true, add all headers
Source source = new DOMSource(child);
beanList.add(JAXBTypeSerializer.deserialize(source, jaxbContext));
} else {
//If allRoles is false, add only headers with matching roles and headers with no role
if( (child.getActor() == null)||
(getRoles().contains(child.getActor())) ) {
Source source = new DOMSource(child);
beanList.add(JAXBTypeSerializer.deserialize(source, jaxbContext));
}
}
}
return beanList.toArray();
} catch(Exception e) {
throw new WebServiceException(e);
}
}
public Set<String> getRoles() {
return roles;
}
void setRoles(Set<String> roles) {
this.roles = roles;
}
private boolean validateProperty(String name, Object value) {
if (allowedTypes.containsKey(name)) {
Class clazz = allowedTypes.get(name);
if (!(clazz.isInstance(value)))
throw new HandlerException("handler.messageContext.invalid.class",
new Object[] { value, name });
}
return true;
}
public void setScope(String name, Scope scope) {
ctxt.setScope(name, scope);
}
public Scope getScope(String name) {
return ctxt.getScope(name);
}
/* java.util.Map methods below here */
public void clear() {
ctxt.clear();
}
public boolean containsKey(Object obj) {
return ctxt.containsKey(obj);
}
public boolean containsValue(Object obj) {
return ctxt.containsValue(obj);
}
public Set<Entry<String, Object>> entrySet() {
return ctxt.entrySet();
}
public Object get(Object obj) {
return ctxt.get(obj);
}
public boolean isEmpty() {
return ctxt.isEmpty();
}
public Set<String> keySet() {
return ctxt.keySet();
}
public Object put(String str, Object obj) {
return ctxt.put(str, obj);
}
public void putAll(Map<? extends String, ? extends Object> map) {
ctxt.putAll(map);
}
public Object remove(Object obj) {
return ctxt.remove(obj);
}
public int size() {
return ctxt.size();
}
public Collection<Object> values() {
return ctxt.values();
}
}