Package org.jboss.errai.bus.server.servlet

Source Code of org.jboss.errai.bus.server.servlet.GrizzlyCometServlet

/*
* Copyright 2010 JBoss, a divison Red Hat, Inc
*
* Licensed 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.jboss.errai.bus.server.servlet;

import com.google.inject.Singleton;
import com.sun.grizzly.comet.CometContext;
import com.sun.grizzly.comet.CometEngine;
import com.sun.grizzly.comet.CometEvent;
import com.sun.grizzly.comet.handlers.ReflectorCometHandler;
import org.jboss.errai.bus.client.api.Message;
import org.jboss.errai.bus.client.framework.ClientMessageBus;
import org.jboss.errai.bus.client.framework.MarshalledMessage;
import org.jboss.errai.bus.server.api.MessageQueue;
import org.jboss.errai.bus.server.api.QueueActivationCallback;
import org.jboss.errai.bus.server.api.QueueSession;
import org.mvel2.util.StringAppender;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.CharBuffer;

import static org.jboss.errai.bus.server.io.MessageFactory.createCommandMessage;

/**
* The <tt>GrizzlyCometServlet</tt> provides the HTTP-protocol gateway between the server bus and the client buses,
* using Glassfish.
*/
@Singleton
public class GrizzlyCometServlet extends AbstractErraiServlet {

    private static CometContext context = null;

    public void init(ServletConfig config) throws ServletException {
        super.init(config);
    }

    private static CometContext createCometContext(String id) {
        CometEngine cometEngine = CometEngine.getEngine();
        CometContext ctx = cometEngine.register(id);
        ctx.setExpirationDelay(45 * 1000);
        ctx.setBlockingNotification(false);
        return ctx;
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        pollForMessages(sessionProvider.getSession(request.getSession(),
                request.getHeader(ClientMessageBus.REMOTE_QUEUE_ID_HEADER)), request, response);
    }


    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        final QueueSession session = sessionProvider.getSession(request.getSession(),
                request.getHeader(ClientMessageBus.REMOTE_QUEUE_ID_HEADER));

        BufferedReader reader = request.getReader();
        StringAppender sb = new StringAppender(request.getContentLength());
        CharBuffer buffer = CharBuffer.allocate(10);

        int read;
        while ((read = reader.read(buffer)) > 0) {
            buffer.rewind();
            for (; read > 0; read--) {
                sb.append(buffer.get());
            }
            buffer.rewind();
        }

        Message msg = createCommandMessage(session, sb.toString());
        if (msg != null) {
            service.store(msg);
        }

        pollQueue(service.getBus().getQueue(session), request, response);
    }

    private void pollForMessages(QueueSession session, HttpServletRequest httpServletRequest,
                                 HttpServletResponse httpServletResponse) throws IOException {

        try {
            final MessageQueue queue = service.getBus().getQueue(session);

            if (queue == null) {
                sendDisconnectWithReason(httpServletResponse.getOutputStream(),
                        "There is no queue associated with this session.");
            }

            synchronized (queue) {
                if (context == null)
                    context = createCometContext(httpServletRequest.getSession().getId());

                final ReflectorCometHandler handler = new ReflectorCometHandler(true);
                context.addCometHandler(handler);

                if (!queue.messagesWaiting()) {
                    queue.setActivationCallback(new QueueActivationCallback() {
                        public void activate(MessageQueue queue) {
                            queue.setActivationCallback(null);
                            context.resumeCometHandler(handler);
                            try {
                                context.notify(null, CometEvent.NOTIFY, handler);
                            } catch (IOException e) {
                                // Should never get here
                            }
                        }
                    });

                    if (!queue.messagesWaiting()) {
                        context.setExpirationDelay(45 * 1000);
                    }
                } else {
                    queue.setActivationCallback(null);
                }

                pollQueue(queue, httpServletRequest, httpServletResponse);
            }
        }
        catch (final Throwable t) {
            t.printStackTrace();

            httpServletResponse.setHeader("Cache-Control", "no-cache");
            httpServletResponse.addHeader("Payload-Size", "1");
            httpServletResponse.setContentType("application/json");
            OutputStream stream = httpServletResponse.getOutputStream();

            stream.write('[');

            writeToOutputStream(stream, new MarshalledMessage() {
                public String getSubject() {
                    return "ClientBusErrors";
                }

                public Object getMessage() {
                    StringBuilder b = new StringBuilder("{ErrorMessage:\"").append(t.getMessage()).append("\",AdditionalDetails:\"");
                    for (StackTraceElement e : t.getStackTrace()) {
                        b.append(e.toString()).append("<br/>");
                    }

                    return b.append("\"}").toString();
                }
            });

            stream.write(']');
        }
    }

    private static void pollQueue(MessageQueue queue, HttpServletRequest httpServletRequest,
                                  HttpServletResponse httpServletResponse) throws IOException {

        queue.heartBeat();

        httpServletResponse.setHeader("Cache-Control", "no-cache");
        httpServletResponse.setContentType("application/json");

        queue.poll(false, httpServletResponse.getOutputStream());

//        OutputStream stream = httpServletResponse.getOutputStream();
//
//        Iterator<MarshalledMessage> iter = messages.iterator();
//
//        stream.write('[');
//        while (iter.hasNext()) {
//            writeToOutputStream(stream, iter.next());
//            if (iter.hasNext()) {
//                stream.write(',');
//            }
//        }
//        stream.write(']');
//        stream.flush();
    }
}
TOP

Related Classes of org.jboss.errai.bus.server.servlet.GrizzlyCometServlet

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.