/** * 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.openejb.server.axis; import org.apache.openejb.server.ServerRuntimeException; import javax.xml.rpc.JAXRPCException; import javax.xml.rpc.handler.Handler; import javax.xml.rpc.handler.HandlerInfo; import javax.xml.rpc.handler.MessageContext; import javax.xml.rpc.handler.soap.SOAPMessageContext; import javax.xml.rpc.soap.SOAPFaultException; import javax.xml.soap.SOAPBody; import javax.xml.soap.SOAPElement; import javax.xml.soap.SOAPException; import javax.xml.soap.SOAPMessage; import java.util.ArrayList; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import java.util.Map; /** * Implementation of HandlerChain */ public class HandlerChainImpl extends ArrayList implements javax.xml.rpc.handler.HandlerChain { private String[] roles; private LinkedList<Handler> invokedHandlers = new LinkedList<Handler>(); public HandlerChainImpl(List<HandlerInfo> handlerInfos) { this(handlerInfos, null); } @SuppressWarnings({"unchecked"}) public HandlerChainImpl(List<HandlerInfo> handlerInfos, String[] roles) { this.roles = roles; for (int i = 0; i < handlerInfos.size(); i++) { HandlerInfo handlerInfo = handlerInfos.get(i); try { Handler handler = (Handler) handlerInfo.getHandlerClass().newInstance(); handler.init(handlerInfo); add(handler); } catch (Exception e) { throw new JAXRPCException("Unable to initialize handler class: " + handlerInfo.getHandlerClass().getName(), e); } } } public String[] getRoles() { return roles; } public void setRoles(String[] roles) { if (roles == null) { this.roles = new String[0]; } else { this.roles = roles; } } public void init(Map map) { } public void destroy() { for (Iterator iterator = invokedHandlers.iterator(); iterator.hasNext(); ) { Handler handler = (Handler) iterator.next(); handler.destroy(); } invokedHandlers.clear(); clear(); } public boolean handleRequest(MessageContext context) { MessageSnapshot snapshot = new MessageSnapshot(context); try { for (int i = 0; i < size(); i++) { Handler currentHandler = (Handler) get(i); invokedHandlers.addFirst(currentHandler); try { if (!currentHandler.handleRequest(context)) { return false; } } catch (SOAPFaultException e) { throw e; } } } finally { saveChanges(context); } if (!snapshot.equals(context)) { throw new IllegalStateException("The soap message operation or arguments were illegally modified by the HandlerChain"); } return true; } public boolean handleResponse(MessageContext context) { MessageSnapshot snapshot = new MessageSnapshot(context); try { for (Iterator iterator = invokedHandlers.iterator(); iterator.hasNext(); ) { Handler handler = (Handler) iterator.next(); if (!handler.handleResponse(context)) { return false; } } } finally { saveChanges(context); } if (!snapshot.equals(context)) { throw new IllegalStateException("The soap message operation or arguments were illegally modified by the HandlerChain"); } return true; } public boolean handleFault(MessageContext context) { MessageSnapshot snapshot = new MessageSnapshot(context); try { for (Iterator iterator = invokedHandlers.iterator(); iterator.hasNext(); ) { Handler handler = (Handler) iterator.next(); if (!handler.handleFault(context)) { return false; } } } finally { saveChanges(context); } if (!snapshot.equals(context)) { throw new IllegalStateException("The soap message operation or arguments were illegally modified by the HandlerChain"); } return true; } private void saveChanges(MessageContext context) { try { SOAPMessage message = ((SOAPMessageContext) context).getMessage(); if (message != null) { message.saveChanges(); } } catch (SOAPException e) { throw new ServerRuntimeException("Unable to save changes to SOAPMessage : " + e.toString()); } } /** * Handlers cannot: * <p/> * - re-target a request to a different component. * - change the operation * - change the message part types * - change the number of message parts. */ static class MessageSnapshot { private final String operationName; private final List<String> parameterNames; public MessageSnapshot(MessageContext soapMessage) { SOAPMessage message = ((SOAPMessageContext) soapMessage).getMessage(); if (message == null || message.getSOAPPart() == null) { operationName = null; parameterNames = null; } else { SOAPBody body = getBody(message); SOAPElement operation = ((SOAPElement) body.getChildElements().next()); this.operationName = operation.getElementName().toString(); this.parameterNames = new ArrayList<String>(); for (Iterator i = operation.getChildElements(); i.hasNext(); ) { SOAPElement parameter = (SOAPElement) i.next(); String element = parameter.getElementName().toString(); parameterNames.add(element); } } } private SOAPBody getBody(SOAPMessage message) { try { return message.getSOAPPart().getEnvelope().getBody(); } catch (SOAPException e) { throw new ServerRuntimeException(e); } } public boolean equals(Object obj) { return (obj instanceof SOAPMessageContext) && equals((SOAPMessageContext) obj); } private boolean equals(SOAPMessageContext soapMessage) { SOAPMessage message = soapMessage.getMessage(); if (operationName == null) { return message == null || message.getSOAPPart() == null; } SOAPBody body = getBody(message); // Handlers can't change the operation SOAPElement operation = ((SOAPElement) body.getChildElements().next()); if (!this.operationName.equals(operation.getElementName().toString())) { return false; } Iterator parameters = operation.getChildElements(); for (Iterator i = parameterNames.iterator(); i.hasNext(); ) { // Handlers can't remove parameters if (!parameters.hasNext()) { return false; } String original = (String) i.next(); SOAPElement parameter = (SOAPElement) parameters.next(); // Handlers can't change parameter types if (parameter == null || !original.equals(parameter.getElementName().toString())) { return false; } } // Handlers can't add parameters return !parameters.hasNext(); } } }