org.atmosphere.gwt.server.AtmosphereGwtHandler Maven / Gradle / Ivy
/*
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright 2007-2008 Sun Microsystems, Inc. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can obtain
* a copy of the License at https://glassfish.dev.java.net/public/CDDL+GPL.html
* or glassfish/bootstrap/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at glassfish/bootstrap/legal/LICENSE.txt.
* Sun designates this particular file as subject to the "Classpath" exception
* as provided by Sun in the GPL Version 2 section of the License file that
* accompanied this code. If applicable, add the following below the License
* Header, with the fields enclosed by brackets [] replaced by your own
* identifying information: "Portions Copyrighted [year]
* [name of copyright owner]"
*
* Contributor(s):
*
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*
*/
package org.atmosphere.gwt.server;
import com.google.gwt.user.client.rpc.SerializationException;
import com.google.gwt.user.server.rpc.SerializationPolicy;
import com.google.gwt.user.server.rpc.SerializationPolicyProvider;
import com.google.gwt.user.server.rpc.impl.ServerSerializationStreamReader;
import java.io.Serializable;
import java.util.List;
import java.util.logging.Level;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import org.atmosphere.gwt.server.impl.GwtAtmosphereResourceImpl;
import org.atmosphere.gwt.server.impl.RPCUtil;
import java.io.BufferedReader;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import javax.servlet.ServletContext;
import org.atmosphere.cpr.AtmosphereResource;
import org.atmosphere.cpr.AtmosphereServletProcessor;
import org.atmosphere.cpr.Broadcaster;
import org.atmosphere.cpr.BroadcasterFactory;
import org.atmosphere.cpr.DefaultBroadcaster;
import org.atmosphere.handler.AbstractReflectorAtmosphereHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
*
* @author p.havelaar
*/
public class AtmosphereGwtHandler extends AbstractReflectorAtmosphereHandler
implements Executor, AtmosphereServletProcessor {
public static final int NO_TIMEOUT = -1;
public static final String GWT_BROADCASTER_ID = "GWT_BROADCASTER";
private static final int DEFAULT_HEARTBEAT = 15 * 1000; // 15 seconds by default
private ExecutorService executorService;
private int heartbeat = DEFAULT_HEARTBEAT;
protected final Logger logger = LoggerFactory.getLogger(getClass());
protected SerializationPolicyProvider cometSerializationPolicyProvider = new SerializationPolicyProvider() {
@Override
public SerializationPolicy getSerializationPolicy(String moduleBaseURL, String serializationPolicyStrongName) {
return RPCUtil.createSimpleSerializationPolicy();
}
};
public int doComet(GwtAtmosphereResource resource) throws ServletException, IOException {
Broadcaster broadcaster = BroadcasterFactory.getDefault().lookup(Broadcaster.class, GWT_BROADCASTER_ID);
try {
broadcaster = BroadcasterFactory.getDefault().get(DefaultBroadcaster.class, GWT_BROADCASTER_ID);
} catch (IllegalAccessException ex) {
logger.error("Failed to get broadcaster", ex);
} catch (InstantiationException ex) {
logger.error("Failed to get broadcaster", ex);
}
resource.getAtmosphereResource().setBroadcaster(broadcaster);
return NO_TIMEOUT;
}
public void cometTerminated(GwtAtmosphereResource cometResponse, boolean serverInitiated) {
resources.remove(cometResponse.getConnectionID());
}
/**
* Default implementation echo's the message back to the client
* @param messages
* @param r
*/
public void doPost(List messages, GwtAtmosphereResource r) {
if (messages.size() == 1) {
r.broadcast(messages.get(0));
} else {
r.broadcast((List)messages);
}
}
protected Broadcaster getBroadcaster(GwtAtmosphereResource resource) {
if (resource == null) {
return BroadcasterFactory.getDefault().lookup(Broadcaster.class, GWT_BROADCASTER_ID);
} else {
return resource.getBroadcaster();
}
}
/**
* This can be used to lookup a resource for instance if you are implementing a remote service call
* You will need to pass the connectionID, which you can pass as an url parameter {getConnectionID()} or
* directly in your remote call
*
* @param connectionId
* @return
*/
protected GwtAtmosphereResource lookupResource(int connectionId) {
GwtAtmosphereResource r = resources.get(connectionId);
if (r != null) {
return r;
} else {
logger.info("Failed to find resource for [" + connectionId + "]");
}
return null;
}
// -------------- you most likely don't need to override the functions below -----------------
private Map resources;
private ServletContext context;
@Override
public void init(ServletConfig servletConfig) throws ServletException {
executorService = Executors.newCachedThreadPool();
String heartbeat = servletConfig.getInitParameter("heartbeat");
context = servletConfig.getServletContext();
if (heartbeat != null) {
this.heartbeat = Integer.parseInt(heartbeat);
}
}
@Override
public void destroy() {
if (executorService != null) {
executorService.shutdown();
}
}
public int getHeartbeat() {
return heartbeat;
}
public void setHeartbeat(int heartbeat) {
this.heartbeat = heartbeat;
}
public ServletContext getServletContext() {
return context;
}
protected void reapResources() {
for (GwtAtmosphereResource resource : resources.values()) {
if (!resource.isAlive()) {
resources.remove(resource.getConnectionID());
}
}
}
@Override
public void onRequest(AtmosphereResource resource) throws IOException {
HttpServletRequest request = resource.getRequest();
String servertransport = request.getParameter("servertransport");
if ("rpcprotocol".equals(servertransport)){
Integer connectionID = Integer.parseInt(request.getParameter("connectionID"));
doServerMessage(request.getReader(), connectionID);
return;
}
try {
int requestHeartbeat = heartbeat;
String requestedHeartbeat = request.getParameter("heartbeat");
if (requestedHeartbeat != null) {
try {
requestHeartbeat = Integer.parseInt(requestedHeartbeat);
if (requestHeartbeat <= 0) {
throw new IOException("invalid heartbeat parameter");
}
requestHeartbeat = computeHeartbeat(requestHeartbeat);
}
catch (NumberFormatException e) {
throw new IOException("invalid heartbeat parameter");
}
}
GwtAtmosphereResourceImpl resourceWrapper = new GwtAtmosphereResourceImpl(resource, this, requestHeartbeat);
doCometImpl(resourceWrapper);
}
catch (IOException e) {
// GwtAtmosphereResourceImpl resource = new GwtAtmosphereResourceImpl(atm, this, -1);
logger.error("Unable to initiated comet" + e.getMessage(), e);
// resource.getResponseWriter().sendError(HttpServletResponse.SC_BAD_REQUEST, e.getMessage());
}
}
/// --- server message handlers
protected void doServerMessage(BufferedReader data, int connectionID) {
List postMessages = new ArrayList();
GwtAtmosphereResource resource = lookupResource(connectionID);
if (resource == null) {
return;
}
try {
while (true) {
String event = data.readLine();
if (event == null) {
break;
}
String messageData = data.readLine();
if (messageData == null) {
break;
}
data.readLine();
if (logger.isTraceEnabled()) {
logger.trace("["+connectionID+"] Server message received: " +event + ";" + messageData.charAt(0));
}
if (event.equals("o")) {
if (messageData.charAt(0) == 'p') {
Serializable message = deserialize(messageData.substring(1));
if (message != null) {
postMessages.add(message);
}
} else if (messageData.charAt(0) == 'b') {
Serializable message = deserialize(messageData.substring(1));
broadcast(message, resource);
}
} else if (event.equals("s")) {
if (messageData.charAt(0) == 'p') {
String message = messageData.substring(1);
postMessages.add(message);
} else if (messageData.charAt(0) == 'b') {
Serializable message = messageData.substring(1);
broadcast(message, resource);
}
} else if (event.equals("c")) {
if (messageData.equals("d")) {
disconnect(resource);
}
}
}
} catch (IOException ex) {
logger.error("["+connectionID+"] Failed to read", ex);
}
if (postMessages.size() > 0) {
post(postMessages, resource);
}
}
// protected void writePostResponse(HttpServletRequest request,
// HttpServletResponse response, ServletContext context, String responsePayload) throws IOException {
// boolean gzipEncode = RPCServletUtils.acceptsGzipEncoding(request)
// && shouldCompressResponse(request, response, responsePayload);
//
// RPCServletUtils.writeResponse(context, response,
// responsePayload, gzipEncode);
// }
protected Serializable deserialize(String data) {
try {
ServerSerializationStreamReader reader = new ServerSerializationStreamReader(getClass().getClassLoader(), cometSerializationPolicyProvider);
reader.prepareToRead(data);
return (Serializable) reader.readObject();
} catch (SerializationException ex) {
logger.error("Failed to deserialize message", ex);
return null;
}
}
//
// protected String serialize(Serializable message) throws SerializationException {
// ServerSerializationStreamWriter streamWriter = new ServerSerializationStreamWriter(RPCUtil.createSimpleSerializationPolicy());
// streamWriter.prepareToWrite();
// streamWriter.writeObject(message);
// return streamWriter.toString();
// }
final public void post(List messages, GwtAtmosphereResource resource) {
if (messages == null) {
return;
}
if (resource != null) {
doPost(messages, resource);
}
}
public void broadcast(Serializable message, GwtAtmosphereResource resource) {
if (message == null) {
return;
}
getBroadcaster(resource).broadcast(message);
}
public void broadcast(List messages, GwtAtmosphereResource resource) {
if (messages == null) {
return;
}
getBroadcaster(resource).broadcast(messages);
}
public void disconnect(GwtAtmosphereResource resource) {
if (resource != null) {
logger.debug("Resuming connection["+resource.getConnectionID()+"] after client disconnect message");
resource.getAtmosphereResource().resume();
}
}
/// --- end server message handlers
/**
* Execute a task in a seperate thread, the thread pool will grow and shrink depending on demand
* @param command
*/
@Override
public void execute(Runnable command) {
executorService.execute(command);
}
protected int computeHeartbeat(int requestedHeartbeat) {
return requestedHeartbeat < heartbeat ? heartbeat : requestedHeartbeat;
}
private void doCometImpl(GwtAtmosphereResourceImpl resource) throws IOException {
try {
// setup the request
resource.getWriterImpl().initiate();
if (resources == null) {
resources = new ConcurrentHashMap(5);
resource.getBroadcaster().getBroadcasterConfig().getScheduledExecutorService().scheduleWithFixedDelay(new Runnable() {
@Override
public void run() {
reapResources();
}
}, 30, 10, TimeUnit.SECONDS);
}
resources.put(resource.getConnectionID(), resource);
} catch (IOException e) {
logger.error("Error initiating GwtComet", e);
return;
}
int timeout;
try {
// call the application code
timeout = doComet(resource);
if (timeout == -1) {
logger.info("You have set an infinite timeout for your comet connection this is not recommended");
}
}
catch (ServletException e) {
logger.error("Error calling doComet()", e);
// resource.getResponseWriter().sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e.getMessage());
return;
}
catch (IOException e) {
logger.error("Error calling doComet()", e);
// resource.getResponseWriter().sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR, e.getMessage());
return;
}
// at this point the application may have spawned threads to process this response
// so we have to be careful about concurrency from here on
resource.suspend(timeout);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy