/** * GRANITE DATA SERVICES * Copyright (C) 2006-2015 GRANITE DATA SERVICES S.A.S. * * This file is part of the Granite Data Services Platform. * * Granite Data Services is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * Granite Data Services 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. See the GNU Lesser * General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, * USA, or see <http://www.gnu.org/licenses/>. */ package org.granite.gravity.jetty8; import flex.messaging.messages.AsyncMessage; import org.eclipse.jetty.continuation.Continuation; import org.granite.gravity.AbstractChannel; import org.granite.gravity.AsyncHttpContext; import org.granite.gravity.GravityConfig; import org.granite.gravity.GravityInternal; import org.granite.gravity.MessageReceivingException; import org.granite.logging.Logger; /** * @author William DRAI */ public class ContinuationChannel extends AbstractChannel { private static final Logger log = Logger.getLogger(ContinuationChannel.class); private Continuation continuation = null; public ContinuationChannel(GravityInternal gravity, String id, ContinuationChannelFactory factory, String clientType) { super(gravity, id, factory, clientType); } public void setContinuation(Continuation continuation) { log.debug("Setting continuation %s for client: %s", continuation, getId()); try { if (this.continuation != null && this.continuation.isSuspended()) { log.debug("Resuming previous continuation %s for client: %s", this.continuation, getId()); this.continuation.resume(); } } finally { this.continuation = continuation; } } public void close(boolean timeout) { this.continuation = null; // try { // if (this.continuation != null) // this.continuation. // } // finally { // this.continuation = null; // } } public void resume() { log.debug("Resuming pending continuation %s for client: %s", continuation, getId()); try { if (this.continuation != null && this.continuation.isSuspended()) this.continuation.resume(); } finally { this.continuation = null; } } @Override public void receive(AsyncMessage message) throws MessageReceivingException { if (message == null) throw new NullPointerException("message cannot be null"); GravityConfig gravityConfig = gravity.getGravityConfig(); receivedQueueLock.lock(); try { if (receivedQueue.size() + 1 > gravityConfig.getMaxMessagesQueuedPerChannel()) throw new MessageReceivingException(message, "Could not queue message (channel's queue is full) for channel: " + this); receivedQueue.add(message); } finally { receivedQueueLock.unlock(); } synchronized (this) { resume(); } } @Override protected boolean hasAsyncHttpContext() { return false; } @Override protected void releaseAsyncHttpContext(AsyncHttpContext context) { } @Override protected AsyncHttpContext acquireAsyncHttpContext() { return null; } public boolean isLocal() { return true; } @Override public void destroy(boolean timeout) { try { super.destroy(timeout); } finally { synchronized (this) { close(timeout); } } } }