com.caucho.server.http.AsyncContextImpl Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.server.http;
import java.io.IOException;
import java.util.ArrayList;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.servlet.AsyncContext;
import javax.servlet.AsyncEvent;
import javax.servlet.AsyncListener;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import com.caucho.env.thread.ThreadPool;
import com.caucho.network.listen.AsyncController;
import com.caucho.network.listen.SocketLink;
import com.caucho.network.listen.SocketLinkCometListener;
import com.caucho.server.webapp.WebApp;
import com.caucho.util.L10N;
/**
* Public API to control a comet connection.
*/
public class AsyncContextImpl
implements AsyncContext, SocketLinkCometListener {
private static final L10N L = new L10N(ConnectionCometController.class);
private static final Logger log = Logger
.getLogger(ConnectionCometController.class.getName());
private SocketLink _tcpConnection;
private AsyncController _cometController;
private ServletRequest _request;
private ServletResponse _response;
private boolean _isOriginal;
private ArrayList _listeners;
private AtomicBoolean _isComplete = new AtomicBoolean();
private boolean _isTimeout;
private String _dispatchDefault;
private WebApp _dispatchWebApp;
private String _dispatchPath;
private boolean _isDispatch;
public AsyncContextImpl(AbstractHttpRequest httpConn)
{
_tcpConnection = httpConn.getConnection();
_cometController = _tcpConnection.toComet(this);
}
public void restart()
{
_cometController = _tcpConnection.toCometRestart(this);
}
public void clearListeners()
{
_listeners = null;
}
public void init(ServletRequest request,
ServletResponse response,
boolean isOriginal)
{
_request = request;
_response = response;
if (! (request instanceof HttpServletRequest)) {
throw new IllegalStateException(L.l("startAsync requires a HttpServletRequest"));
}
_dispatchWebApp = (WebApp) request.getServletContext();
HttpServletRequest req = (HttpServletRequest) request;
String servletPath = req.getServletPath();
String pathInfo = req.getPathInfo();
if (pathInfo == null)
_dispatchDefault = servletPath;
else if (servletPath == null)
_dispatchDefault = pathInfo;
else
_dispatchDefault = servletPath + pathInfo;
/* XXX: tck
if (_cometController == null)
throw new NullPointerException();
*/
_isOriginal = isOriginal;
onStart(req, response);
}
/**
* Returns the originating request for the async.
*/
@Override
public ServletRequest getRequest()
{
return _request;
}
/**
* Returns the originating request for the async.
*/
@Override
public ServletResponse getResponse()
{
return _response;
}
@Override
public boolean hasOriginalRequestAndResponse()
{
return _isOriginal;
}
/**
* Sets the suspend/idle timeout for the async request.
*/
@Override
public void setTimeout(long idleTimeout)
{
_cometController.setMaxIdleTime(idleTimeout);
}
/**
* Returns the suspend/idle timeout for the async request.
*/
@Override
public long getTimeout()
{
return _cometController.getMaxIdleTime();
}
/**
* Tests if the controller is active.
*/
private boolean isActive()
{
return _cometController != null;
}
public boolean isAsyncStarted()
{
AsyncController controller = _cometController;
return controller != null && controller.isAsyncStarted();
}
public boolean isAsyncComplete()
{
AsyncController controller = _cometController;
return controller != null && controller.isCometComplete();
}
//
// dispatch values
//
ServletContext getDispatchContext()
{
return _dispatchWebApp;
}
String getDispatchPath()
{
return _dispatchPath;
}
//
// AsyncListener
//
@Override
public void addListener(AsyncListener listener)
{
addListener(listener, _request, _response);
}
public void addListener(AsyncListener listener,
ServletRequest request,
ServletResponse response)
{
if (_listeners == null)
_listeners = new ArrayList();
_listeners.add(new AsyncListenerNode(listener, request, response, null));
}
@Override
public T createListener(Class cl)
throws ServletException
{
return _request.getServletContext().createListener(cl);
}
//
// Async dispatching
//
@Override
public void dispatch()
{
AsyncController cometController = _cometController;
if (cometController == null)
throw new IllegalStateException(L.l("dispatch is not valid when no AsyncContext is available"));
/*
else if (! cometController.isAsyncStarted())
throw new IllegalStateException(L.l("dispatch is not valid when async cycle has not started, i.e. before startAsync."));
*/
if (_dispatchPath == null) {
_dispatchPath = _dispatchDefault;
}
try {
cometController.wake();
} catch (RuntimeException e) {
if (isTimeout()) {
// server/1lda
log.log(Level.FINEST, e.toString(), e);
}
else {
throw e;
}
}
_isDispatch = true;
}
@Override
public void dispatch(String path)
{
_dispatchPath = path;
dispatch();
}
@Override
public void dispatch(ServletContext context, String path)
{
_dispatchWebApp = (WebApp) context;
_dispatchPath = path;
dispatch();
}
@Override
public void start(Runnable task)
{
if (isActive()) {
ThreadPool.getCurrent().schedule(task);
}
else
throw new IllegalStateException(
L.l("AsyncContext.start() is not allowed because the AsyncContext has been completed."));
}
/**
* Completes the comet connection
*/
@Override
public void complete()
{
AsyncController cometController = _cometController;
boolean isSuspend = _tcpConnection.isCometSuspend();
if (cometController != null) {
cometController.complete();
}
// TCK:
// XXX: should be on completing thread.
if (! isSuspend) {
onComplete();
}
}
//
// CometHandler callbacks
//
/**
* CometHandler callback when the connection starts.
*/
public void onStart(ServletRequest request,
ServletResponse response)
{
if (_listeners == null)
return;
AsyncEvent event = new AsyncEvent(this, request, response);
for (AsyncListenerNode node : _listeners) {
try {
node.onStart(event);
} catch (IOException e) {
log.log(Level.FINE, e.toString(), e);
}
}
if (_isDispatch) {
// server/1l9c
_isDispatch = false;
// _listeners = null;
}
}
public boolean isTimeout()
{
return _isTimeout;
}
/**
* CometHandler callback when the connection times out.
*/
@Override
public boolean onTimeout()
{
_isTimeout = true;
if (_listeners == null)
return true;
AsyncEvent event = new AsyncEvent(this, _request, _response);
for (AsyncListenerNode node : _listeners) {
try {
node.onTimeout(event);
} catch (IOException e) {
log.log(Level.FINE, e.toString(), e);
}
}
return _dispatchPath == null;
}
/**
* CometHandler callback when the connection times out.
*/
public void onError()
{
if (_listeners == null)
return;
AsyncEvent event = new AsyncEvent(this, _request, _response);
for (AsyncListenerNode node : _listeners) {
try {
node.onError(event);
} catch (IOException e) {
log.log(Level.FINE, e.toString(), e);
}
}
}
@Override
public void onComplete()
{
if (_isComplete.getAndSet(true))
return;
if (_listeners == null)
return;
AsyncEvent event = new AsyncEvent(this, _request, _response);
for (AsyncListenerNode node : _listeners) {
try {
node.onComplete(event);
} catch (IOException e) {
log.log(Level.FINE, e.toString(), e);
}
}
}
@Override
public String toString()
{
return getClass().getSimpleName() + "[" + _cometController + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy