| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| |
| package org.apache.axis2.jaxws.handler; |
| |
| import org.apache.axiom.soap.RolePlayer; |
| import org.apache.axis2.jaxws.ExceptionFactory; |
| import org.apache.axis2.jaxws.core.MessageContext; |
| import org.apache.axis2.jaxws.i18n.Messages; |
| import org.apache.axis2.jaxws.message.Block; |
| import org.apache.axis2.jaxws.message.Message; |
| import org.apache.axis2.jaxws.message.databinding.JAXBBlockContext; |
| import org.apache.axis2.jaxws.message.factory.BlockFactory; |
| import org.apache.axis2.jaxws.message.factory.JAXBBlockFactory; |
| import org.apache.axis2.jaxws.message.factory.MessageFactory; |
| import org.apache.axis2.jaxws.registry.FactoryRegistry; |
| import org.apache.axis2.util.JavaUtils; |
| import org.apache.commons.logging.Log; |
| import org.apache.commons.logging.LogFactory; |
| |
| import javax.xml.bind.JAXBContext; |
| import javax.xml.namespace.QName; |
| import javax.xml.soap.AttachmentPart; |
| import javax.xml.soap.SOAPConstants; |
| import javax.xml.soap.SOAPEnvelope; |
| import javax.xml.soap.SOAPMessage; |
| import javax.xml.soap.SOAPPart; |
| import javax.xml.stream.XMLStreamException; |
| import java.util.ArrayList; |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.List; |
| import java.util.Set; |
| |
| /** |
| * The SOAPMessageContext is the context handed to SOAP-based application handlers. It provides |
| * access to the SOAP message that represents the request or response via SAAJ. It also allows |
| * access to any properties that have been registered and set on the MessageContext. |
| */ |
| public class SoapMessageContext extends BaseMessageContext implements |
| javax.xml.ws.handler.soap.SOAPMessageContext { |
| private static final Log log = LogFactory.getLog(SoapMessageContext.class); |
| |
| // Cache the message object and SOAPMessage after transformation |
| Message cachedMessage = null; |
| SOAPMessage cachedSoapMessage = null; |
| |
| // Cache information about the SOAPMessage so that we can tell if it has changed |
| SOAPPart cachedSoapPart = null; |
| SOAPEnvelope cachedSoapEnvelope = null; |
| List<AttachmentPart> cachedAttachmentParts = new ArrayList<AttachmentPart>(); |
| |
| |
| |
| public SoapMessageContext(MessageContext messageCtx) { |
| super(messageCtx); |
| } |
| |
| public Object[] getHeaders(QName qname, JAXBContext jaxbcontext, boolean allRoles) { |
| if(log.isDebugEnabled()){ |
| log.debug("Getting all Headers for Qname: "+qname); |
| } |
| |
| if(qname == null){ |
| if(log.isDebugEnabled()){ |
| log.debug("Invalid QName, QName cannot be null"); |
| } |
| throw ExceptionFactory.makeWebServiceException(Messages.getMessage("soapMessageContextErr1")); |
| } |
| if(jaxbcontext == null){ |
| if(log.isDebugEnabled()){ |
| log.debug("Invalid JAXBContext, JAXBContext cannot be null"); |
| } |
| throw ExceptionFactory.makeWebServiceException(Messages.getMessage("soapMessageContextErr2")); |
| } |
| |
| // The header information is returned as a list of jaxb objects |
| List<Object> list = new ArrayList<Object>(); |
| String namespace = qname.getNamespaceURI(); |
| String localPart = qname.getLocalPart(); |
| BlockFactory blockFactory = (JAXBBlockFactory) |
| FactoryRegistry.getFactory(JAXBBlockFactory.class); |
| Message m = messageCtx.getMessage(); |
| JAXBBlockContext jbc = new JAXBBlockContext(jaxbcontext); |
| |
| // If allRoles is not specified, pass in a set of roles. |
| // The headers must support that role. |
| RolePlayer rolePlayer = null; |
| if (allRoles == false) { |
| rolePlayer = getRolePlayer(); |
| } |
| |
| if(m.getNumHeaderBlocks()>0){ |
| // Get the list of JAXB Blocks |
| List<Block> blockList = m.getHeaderBlocks(namespace, |
| localPart, |
| jbc, |
| blockFactory, |
| rolePlayer); |
| |
| // Create list of JAXB objects |
| if(blockList!=null && blockList.size() > 0){ |
| try{ |
| Iterator it = blockList.iterator(); |
| while (it.hasNext()) { |
| Block block = (Block) it.next(); |
| Object bo = block.getBusinessObject(false); |
| if(bo!=null){ |
| if(log.isDebugEnabled()){ |
| log.debug("Extracted BO from Header Block"); |
| } |
| list.add(bo); |
| } |
| } |
| |
| }catch(XMLStreamException e){ |
| throw ExceptionFactory.makeWebServiceException(e); |
| } |
| } |
| } |
| return list.toArray(new Object[0]); |
| |
| } |
| |
| public SOAPMessage getMessage() { |
| // set a property to indicate that we are accessing the message |
| if(log.isDebugEnabled()){ |
| log.debug("getMessage - accessing message."); |
| } |
| this.put("jaxws.isMessageAccessed", true); |
| Message msg = messageCtx.getMEPContext().getMessageObject(); |
| if (msg != cachedMessage) { |
| cachedMessage = msg; |
| cachedSoapMessage = msg.getAsSOAPMessage(); |
| cacheSOAPMessageInfo(cachedSoapMessage); |
| } |
| return cachedSoapMessage; |
| } |
| |
| /** |
| * Check the current (cached) SOAPMessage and make sure |
| * its internals are consistent with when it was created. |
| * If not, the Message is recreated. |
| */ |
| public void checkAndUpdate() { |
| if (log.isDebugEnabled()) { |
| log.debug("Start:checkAndUpdate"); |
| } |
| if (cachedSoapMessage != null) { |
| |
| boolean match = checkSOAPMessageInfo(cachedSoapMessage); |
| |
| if (!match) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkAndUpdate detected a mismatch.."); |
| } |
| setMessage(cachedSoapMessage); |
| } |
| } |
| if (log.isDebugEnabled()) { |
| log.debug("End:checkAndUpdate"); |
| } |
| } |
| |
| /** |
| * Updates information about the SOAPMessage so that |
| * we can determine later if it has changed |
| * @param sm SOAPMessage |
| */ |
| private void cacheSOAPMessageInfo(SOAPMessage sm) { |
| cachedSoapPart = null; |
| cachedSoapEnvelope = null; |
| cachedAttachmentParts.clear(); |
| try { |
| cachedSoapPart = sm.getSOAPPart(); |
| if (cachedSoapPart != null) { |
| cachedSoapEnvelope = cachedSoapPart.getEnvelope(); |
| } |
| if (sm.countAttachments() > 0) { |
| Iterator it = sm.getAttachments(); |
| while (it != null && it.hasNext()) { |
| AttachmentPart ap = (AttachmentPart) it.next(); |
| cachedAttachmentParts.add(ap); |
| } |
| } |
| } catch (Throwable t) { |
| if (log.isDebugEnabled()) { |
| log.debug("Ignoring ", t); |
| } |
| } |
| } |
| |
| /** |
| * Checks the information in SOAPMessage sm against |
| * the information previously cached. If an exception occurs |
| * @param sm SOAPMessage |
| * @return true if match , (exceptions are interpeted as false) |
| */ |
| private boolean checkSOAPMessageInfo(SOAPMessage sm) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo with " + JavaUtils.getObjectIdentity(sm)); |
| } |
| // Check SOAPPart and SOAPEnvelope identity |
| SOAPPart currentSoapPart = null; |
| SOAPEnvelope currentSoapEnvelope = null; |
| |
| try { |
| currentSoapPart = sm.getSOAPPart(); |
| if (currentSoapPart != null) { |
| currentSoapEnvelope = cachedSoapPart.getEnvelope(); |
| } |
| // Check object identity |
| if (cachedSoapPart != currentSoapPart) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: mismatched SOAPParts"); |
| } |
| return false; |
| } |
| if (cachedSoapEnvelope != currentSoapEnvelope) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: mismatched SOAPEnvelopes"); |
| } |
| return false; |
| } |
| } catch(Throwable t) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: ", t); |
| } |
| } |
| |
| // Check AttachmentParts |
| try { |
| int currentNumAttachmentParts = sm.countAttachments(); |
| if (currentNumAttachmentParts != cachedAttachmentParts.size()) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: " + |
| "current number of AttachmentParts is " + currentNumAttachmentParts + |
| " versus cached number is " + cachedAttachmentParts.size()); |
| } |
| return false; |
| } |
| if (currentNumAttachmentParts > 0) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo detected " + currentNumAttachmentParts + "AttachmentParts"); |
| } |
| Iterator cachedIT = cachedAttachmentParts.iterator(); |
| Iterator currentIT = sm.getAttachments(); |
| while (currentIT.hasNext() && cachedIT.hasNext()) { |
| AttachmentPart currentAP = (AttachmentPart) currentIT.next(); |
| AttachmentPart cachedAP = (AttachmentPart) cachedIT.next(); |
| if (currentAP != cachedAP) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: " + |
| "current AttachmentParts is " + JavaUtils.getObjectIdentity(currentAP) + |
| " and cached is " + JavaUtils.getObjectIdentity(cachedAP)); |
| } |
| return false; |
| } |
| } |
| } |
| } catch(Throwable t) { |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns false due to: ", t); |
| } |
| } |
| if (log.isDebugEnabled()) { |
| log.debug("checkSOAPMessageInfo returns true"); |
| } |
| return true; |
| } |
| |
| public Set<String> getRoles() { |
| // TODO implement better. We should be doing smarter checking of the header, |
| // especially for the Ultimate receiver actor/role |
| |
| /* |
| * JAVADOC to help get this implemented correctly: |
| * |
| * Gets the SOAP actor roles associated with an execution of the handler |
| * chain. Note that SOAP actor roles apply to the SOAP node and are |
| * managed using SOAPBinding.setRoles and SOAPBinding.getRoles. Handler |
| * instances in the handler chain use this information about the SOAP |
| * actor roles to process the SOAP header blocks. Note that the SOAP |
| * actor roles are invariant during the processing of SOAP message |
| * through the handler chain. |
| */ |
| |
| HashSet<String> roles = new HashSet<String>(3); |
| // JAX-WS 10.1.1.1 defaults: |
| // SOAP 1.1 |
| roles.add(SOAPConstants.URI_SOAP_ACTOR_NEXT); |
| // SOAP 1.2 |
| roles.add(SOAPConstants.URI_SOAP_1_2_ROLE_ULTIMATE_RECEIVER); |
| roles.add(SOAPConstants.URI_SOAP_1_2_ROLE_NEXT); |
| return roles; |
| } |
| |
| public void setMessage(SOAPMessage soapMessage) { |
| if(log.isDebugEnabled()){ |
| log.debug("setMessage new=" + JavaUtils.getObjectIdentity(soapMessage) + |
| " existing=" + JavaUtils.getObjectIdentity(cachedSoapMessage)); |
| } |
| try { |
| Message msg = |
| ((MessageFactory) FactoryRegistry.getFactory(MessageFactory.class)).createFrom(soapMessage); |
| messageCtx.getMEPContext().setMessage(msg); |
| cachedMessage = msg; |
| cachedSoapMessage = soapMessage; |
| cacheSOAPMessageInfo(cachedSoapMessage); |
| } catch (XMLStreamException e) { |
| if(log.isDebugEnabled()){ |
| log.debug("Ignoring exception " + e); |
| } |
| } |
| } |
| |
| private RolePlayer getRolePlayer() { |
| List roles = new ArrayList(getRoles()); |
| return new SMCRolePlayer(roles); |
| } |
| |
| class SMCRolePlayer implements RolePlayer { |
| private List roles; |
| SMCRolePlayer(List roles) { |
| this.roles = roles; |
| } |
| |
| public List getRoles() { |
| return roles; |
| } |
| public boolean isUltimateDestination() { |
| return true; |
| } |
| } |
| } |