/* * Copyright 2005-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.api.WSBinding; import com.sun.xml.internal.ws.api.message.Packet; import com.sun.xml.internal.ws.api.message.Attachment; import com.sun.xml.internal.ws.api.message.AttachmentSet; import com.sun.xml.internal.ws.api.pipe.TubeCloner; import com.sun.xml.internal.ws.api.pipe.Tube; import com.sun.xml.internal.ws.api.pipe.helper.AbstractFilterTubeImpl; import com.sun.xml.internal.ws.api.model.wsdl.WSDLPort; import com.sun.xml.internal.ws.client.HandlerConfiguration; import com.sun.xml.internal.ws.binding.BindingImpl; import com.sun.xml.internal.ws.message.DataHandlerAttachment; import javax.xml.ws.handler.soap.SOAPHandler; import javax.xml.ws.handler.MessageContext; import javax.xml.ws.WebServiceException; import javax.activation.DataHandler; import java.util.*; /** * * @author WS Development Team */ public class ClientSOAPHandlerTube extends HandlerTube { private WSBinding binding; private List<SOAPHandler> soapHandlers; private Set<String> roles; /** * Creates a new instance of SOAPHandlerTube */ public ClientSOAPHandlerTube(WSBinding binding, WSDLPort port, Tube next) { super(next, port); if (binding.getSOAPVersion() != null) { // SOAPHandlerTube should n't be used for bindings other than SOAP. // TODO: throw Exception } this.binding = binding; } // Handle to LogicalHandlerTube means its used on SERVER-SIDE /** * This constructor is used on client-side where, LogicalHandlerTube is created * first and then a SOAPHandlerTube is created with a handler to that * LogicalHandlerTube. * With this handle, SOAPHandlerTube can call LogicalHandlerTube.closeHandlers() */ public ClientSOAPHandlerTube(WSBinding binding, Tube next, HandlerTube cousinTube) { super(next, cousinTube); this.binding = binding; } /** * Copy constructor for {@link com.sun.xml.internal.ws.api.pipe.Tube#copy(com.sun.xml.internal.ws.api.pipe.TubeCloner)}. */ private ClientSOAPHandlerTube(ClientSOAPHandlerTube that, TubeCloner cloner) { super(that, cloner); this.binding = that.binding; } boolean isHandlerChainEmpty() { return soapHandlers.isEmpty(); } /** * Close SOAPHandlers first and then LogicalHandlers on Client * Close LogicalHandlers first and then SOAPHandlers on Server */ public void close(MessageContext msgContext) { } /** * This is called from cousinTube. * Close this Tube's handlers. */ public void closeCall(MessageContext msgContext) { closeSOAPHandlers(msgContext); } //TODO: private void closeSOAPHandlers(MessageContext msgContext) { if (processor == null) return; if (remedyActionTaken) { //Close only invoked handlers in the chain //CLIENT-SIDE processor.closeHandlers(msgContext, processor.getIndex(), 0); processor.setIndex(-1); //reset remedyActionTaken remedyActionTaken = false; } else { //Close all handlers in the chain //CLIENT-SIDE processor.closeHandlers(msgContext, soapHandlers.size() - 1, 0); } } public AbstractFilterTubeImpl copy(TubeCloner cloner) { return new ClientSOAPHandlerTube(this, cloner); } void setUpProcessor() { // Take a snapshot, User may change chain after invocation, Same chain // should be used for the entire MEP soapHandlers = new ArrayList<SOAPHandler>(); HandlerConfiguration handlerConfig = ((BindingImpl) binding).getHandlerConfig(); List<SOAPHandler> soapSnapShot= handlerConfig.getSoapHandlers(); if (!soapSnapShot.isEmpty()) { soapHandlers.addAll(soapSnapShot); roles = new HashSet<String>(); roles.addAll(handlerConfig.getRoles()); processor = new SOAPHandlerProcessor(true, this, binding, soapHandlers); } } MessageUpdatableContext getContext(Packet packet) { SOAPMessageContextImpl context = new SOAPMessageContextImpl(binding, packet); context.setRoles(roles); return context; } boolean callHandlersOnRequest(MessageUpdatableContext context, boolean isOneWay) { boolean handlerResult; //Lets copy all the MessageContext.OUTBOUND_ATTACHMENT_PROPERTY to the message Map<String, DataHandler> atts = (Map<String, DataHandler>) context.get(MessageContext.OUTBOUND_MESSAGE_ATTACHMENTS); AttachmentSet attSet = packet.getMessage().getAttachments(); for(String cid : atts.keySet()){ Attachment att = new DataHandlerAttachment(cid, atts.get(cid)); attSet.add(att); } try { //CLIENT-SIDE handlerResult = processor.callHandlersRequest(HandlerProcessor.Direction.OUTBOUND, context, !isOneWay); } catch (WebServiceException wse) { remedyActionTaken = true; //no rewrapping throw wse; } catch (RuntimeException re) { remedyActionTaken = true; throw new WebServiceException(re); } if (!handlerResult) { remedyActionTaken = true; } return handlerResult; } void callHandlersOnResponse(MessageUpdatableContext context, boolean handleFault) { try { //CLIENT-SIDE processor.callHandlersResponse(HandlerProcessor.Direction.INBOUND, context, handleFault); } catch (WebServiceException wse) { //no rewrapping throw wse; } catch (RuntimeException re) { throw new WebServiceException(re); } } }