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.gravity.jbossweb;
022    
023    import java.io.IOException;
024    import java.io.InputStream;
025    
026    import javax.servlet.ServletException;
027    import javax.servlet.http.HttpServletRequest;
028    import javax.servlet.http.HttpServletResponse;
029    
030    import org.granite.gravity.AsyncHttpContext;
031    import org.granite.gravity.Gravity;
032    import org.granite.gravity.GravityManager;
033    import org.granite.gravity.tomcat.ByteArrayCometIO;
034    import org.granite.gravity.tomcat.CometIO;
035    import org.granite.logging.Logger;
036    import org.jboss.servlet.http.HttpEvent;
037    
038    import flex.messaging.messages.Message;
039    
040    /**
041     * @author Franck WOLFF
042     */
043    public class GravityJBossWebServlet extends AbstractHttpEventServlet {
044            
045        private static final long serialVersionUID = 1L;
046    
047        private static final Logger log = Logger.getLogger(GravityJBossWebServlet.class);
048    
049        @Override
050            public CometIO createCometIO() {
051                    return new ByteArrayCometIO();
052            }
053    
054            @Override
055        public boolean handleRequest(HttpEvent event, InputStream content) throws IOException, ServletException {
056    
057                    Gravity gravity = GravityManager.getGravity(getServletContext());
058    
059            HttpServletRequest request = event.getHttpServletRequest();
060            HttpServletResponse response = event.getHttpServletResponse();
061    
062            try {
063                initializeRequest(gravity, request, response);
064    
065                Message[] amf3Requests = deserialize(gravity, request, content);
066    
067                log.debug(">> [AMF3 REQUESTS] %s", (Object)amf3Requests);
068    
069                Message[] amf3Responses = null;
070                
071                boolean accessed = false;
072                for (int i = 0; i < amf3Requests.length; i++) {
073                    Message amf3Request = amf3Requests[i];
074    
075                    // Ask gravity to create a specific response (will be null for connect request from tunnel).
076                    Message amf3Response = gravity.handleMessage(amf3Request);
077                    String channelId = (String)amf3Request.getClientId();
078                    
079                    // Mark current channel (if any) as accessed.
080                    if (!accessed)
081                            accessed = gravity.access(channelId);
082                    
083                    // (Re)Connect message from tunnel...
084                    if (amf3Response == null) {
085                        if (amf3Requests.length > 1)
086                            throw new IllegalArgumentException("Only one request is allowed on tunnel.");
087    
088                        JBossWebChannel channel = (JBossWebChannel)gravity.getChannel(channelId);
089                            if (channel == null)
090                                    throw new NullPointerException("No channel on tunnel connect");
091                        
092                        // Try to send pending messages if any (using current container thread).
093                        if (channel.runReceived(new AsyncHttpContext(request, response, amf3Request)))
094                            return true; // Close http event.
095                        
096                        // No pending messages, wait for new ones or timeout.
097                        setConnectMessage(request, amf3Request);
098                        channel.setHttpEvent(event);
099                        return false; // Do not close http event.
100                    }
101    
102                    if (amf3Responses == null)
103                            amf3Responses = new Message[amf3Requests.length];
104                    amf3Responses[i] = amf3Response;
105                }
106    
107                log.debug("<< [AMF3 RESPONSES] %s", (Object)amf3Responses);
108    
109                serialize(gravity, response, amf3Responses);
110            }
111            catch (IOException e) {
112                log.error(e, "Gravity message error");
113                throw e;
114            }
115            catch (Exception e) {
116                log.error(e, "Gravity message error");
117                throw new ServletException(e);
118            }
119            finally {
120                    try {
121                            if (content != null)
122                                    content.close();
123                    }
124                    finally {
125                            cleanupRequest(request);
126                    }
127            }
128    
129            return true; // Close http event.
130        }
131    
132        @Override
133            public boolean handleEnd(HttpEvent event) throws IOException, ServletException {
134            return true; // Close http event.
135            }
136    
137            @Override
138        public boolean handleError(HttpEvent event) throws IOException {
139                    if (EventUtil.isErrorButNotTimeout(event))
140                            log.error("Got an error event: %s", EventUtil.toString(event));
141    
142            try {
143                    HttpServletRequest request = event.getHttpServletRequest();
144                    Message connect = getConnectMessage(request);
145                    if (connect != null) { // This should be a timeout.
146                            Gravity gravity = GravityManager.getGravity(getServletContext());
147                            String channelId = (String)connect.getClientId();
148                        JBossWebChannel channel = (JBossWebChannel)gravity.getChannel(channelId);
149                        
150                        // Cancel channel's execution (timeout or other errors).
151                        if (channel != null)
152                            channel.setHttpEvent(null);
153                    }
154            }
155            catch (Exception e) {
156                    log.error(e, "Error while processing event: %s", EventUtil.toString(event));
157            }
158                    
159                    return true; // Close http event.
160            }
161    }