001    /*
002      GRANITE DATA SERVICES
003      Copyright (C) 2011 GRANITE DATA SERVICES S.A.S.
004    
005      This file is part of Granite Data Services.
006    
007      Granite Data Services is free software; you can redistribute it and/or modify
008      it under the terms of the GNU Library General Public License as published by
009      the Free Software Foundation; either version 2 of the License, or (at your
010      option) any later version.
011    
012      Granite Data Services is distributed in the hope that it will be useful, but
013      WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
014      FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License
015      for more details.
016    
017      You should have received a copy of the GNU Library General Public License
018      along with this library; if not, see <http://www.gnu.org/licenses/>.
019    */
020    
021    package org.granite.messaging.amf.process;
022    
023    import javax.servlet.http.HttpSession;
024    
025    import org.granite.config.GraniteConfig;
026    import org.granite.context.GraniteContext;
027    import org.granite.logging.Logger;
028    import org.granite.messaging.service.ServiceException;
029    import org.granite.messaging.service.ServiceFactory;
030    import org.granite.messaging.service.ServiceInvoker;
031    import org.granite.messaging.service.security.SecurityService;
032    import org.granite.messaging.service.security.SecurityServiceException;
033    import org.granite.messaging.webapp.HttpGraniteContext;
034    import org.granite.util.UUIDUtil;
035    
036    import flex.messaging.messages.AcknowledgeMessage;
037    import flex.messaging.messages.CommandMessage;
038    import flex.messaging.messages.ErrorMessage;
039    import flex.messaging.messages.Message;
040    import flex.messaging.messages.RemotingMessage;
041    
042    /**
043     * @author Franck WOLFF
044     */
045    public abstract class AMF3MessageProcessor {
046    
047        private static final Logger log = Logger.getLogger(AMF3MessageProcessor.class);
048    
049        public static Message process(Message request) {
050            GraniteContext context = GraniteContext.getCurrentInstance();
051            AMF3MessageInterceptor interceptor = context.getGraniteConfig().getAmf3MessageInterceptor();
052            
053            Message response = null;
054            try {
055                    if (interceptor != null)
056                        interceptor.before(request);
057                    
058                    if (request instanceof RemotingMessage)
059                        response = processRemotingMessage((RemotingMessage)request);
060                    else if (request instanceof CommandMessage)
061                        response = processCommandMessage((CommandMessage)request);
062                    else
063                        throw new IllegalArgumentException("Unknown request message type: " + request);
064            }
065            finally {
066                    if (interceptor != null)
067                        interceptor.after(request, response);
068            }
069            
070            if (context instanceof HttpGraniteContext) {
071                HttpSession session = ((HttpGraniteContext)context).getRequest().getSession(false);
072                if (session != null)
073                    response.setHeader("org.granite.sessionId", session.getId());
074            }
075            
076            return response;
077        }
078    
079        public static Message processCommandMessage(CommandMessage request) {
080    
081            log.debug(">> Processing AMF3 request:\n%s", request);
082    
083            Message response = null;
084            if (request.isSecurityOperation()) {
085                GraniteContext context = GraniteContext.getCurrentInstance();
086                GraniteConfig config = context.getGraniteConfig();
087    
088                if (!config.hasSecurityService())
089                    log.warn("Ignored security operation (no security settings in granite-config.xml): %s", request);
090                else {
091                    SecurityService securityService = config.getSecurityService();
092                    try {
093                        if (request.isLoginOperation())
094                            securityService.login(request.getBody());
095                        else if (request.isLogoutOperation())
096                            securityService.logout();
097                        else
098                            log.warn("Unknown security operation: %s", request);
099                    } catch (Exception e) {
100                        if (e instanceof SecurityServiceException) {
101                            securityService.handleSecurityException((SecurityServiceException)e);
102                            log.debug(e, "Could not process security operation: %s", request);
103                        }
104                        else
105                            log.error(e, "Could not process security operation: %s", request);
106                        response = new ErrorMessage(request, e);
107                    }
108                }
109            }
110    
111            if (response == null) {
112                response = new AcknowledgeMessage(request);
113                // For SDK 2.0.1_Hotfix2.
114                if (request.isSecurityOperation())
115                    response.setBody("success");
116            }
117    
118            // For SDK 2.0.1_Hotfix2.
119            if ("nil".equals(request.getHeader(Message.DS_ID_HEADER)))
120                response.getHeaders().put(Message.DS_ID_HEADER, UUIDUtil.randomUUID());
121    
122            log.debug("<< Returning AMF3 response:\n%s", response);
123    
124            return response;
125        }
126    
127        public static Message processRemotingMessage(RemotingMessage request) {
128    
129            log.debug(">> Processing AMF3 request:\n%s", request);
130            
131            Message response = null;
132            try {
133                // Execute method on service.
134                ServiceFactory factory = ServiceFactory.getFactoryInstance(request);
135                ServiceInvoker<?> service = factory.getServiceInstance(request);
136                Object result = service.invoke(request);
137    
138                response = new AcknowledgeMessage(request);
139                response.setBody(result);
140            } catch (ServiceException e) {
141                log.debug(e, "Could not process remoting message: %s", request);
142                response = new ErrorMessage(request, e);
143            }
144    
145            log.debug("<< Returning AMF3 response:\n%s", response);
146    
147            return response;
148        }
149    }