All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.cometd.server.LocalSessionImpl Maven / Gradle / Ivy

/*
 * Copyright (c) 2008-2017 the original author or authors.
 *
 * 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.cometd.server;

import java.util.Map;
import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;

import org.cometd.bayeux.Channel;
import org.cometd.bayeux.ChannelId;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.client.ClientSession;
import org.cometd.bayeux.client.ClientSessionChannel;
import org.cometd.bayeux.server.LocalSession;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerSession;
import org.cometd.common.AbstractClientSession;

/**
 * 

A {@link LocalSession} implementation.

*

This {@link LocalSession} implementation communicates with its * {@link ServerSession} counterpart without any serialization.

*/ public class LocalSessionImpl extends AbstractClientSession implements LocalSession { private final Queue _queue = new ConcurrentLinkedQueue<>(); private final BayeuxServerImpl _bayeux; private final String _idHint; private ServerSessionImpl _session; protected LocalSessionImpl(BayeuxServerImpl bayeux, String idHint) { _bayeux = bayeux; _idHint = idHint; } @Override public void receive(Message.Mutable message) { super.receive(message); if (Channel.META_DISCONNECT.equals(message.getChannel()) && message.isSuccessful()) { _session = null; } } @Override protected AbstractSessionChannel newChannel(ChannelId channelId) { return new LocalChannel(channelId); } @Override protected ChannelId newChannelId(String channelId) { return _bayeux.newChannelId(channelId); } @Override protected void sendBatch() { int size = _queue.size(); while (size-- > 0) { ServerMessage.Mutable message = _queue.poll(); doSend(_session, message); } } @Override public ServerSession getServerSession() { if (_session == null) { throw new IllegalStateException("Method handshake() not invoked for local session " + this); } return _session; } @Override public void handshake() { handshake(null); } @Override public void handshake(Map template) { handshake(template, null); } @Override public void handshake(Map template, ClientSessionChannel.MessageListener callback) { if (_session != null) { throw new IllegalStateException(); } ServerSessionImpl session = new ServerSessionImpl(_bayeux, this, _idHint); ServerMessage.Mutable message = newMessage(); if (template != null) { message.putAll(template); } String messageId = newMessageId(); message.setId(messageId); message.setChannel(Channel.META_HANDSHAKE); registerCallback(messageId, callback); doSend(session, message); ServerMessage reply = message.getAssociated(); if (reply != null && reply.isSuccessful()) { _session = session; message = newMessage(); message.setId(newMessageId()); message.setChannel(Channel.META_CONNECT); message.getAdvice(true).put(Message.INTERVAL_FIELD, -1L); message.setClientId(session.getId()); doSend(session, message); } } @Override public void disconnect() { disconnect(null); } @Override public void disconnect(ClientSessionChannel.MessageListener callback) { if (_session != null) { ServerMessage.Mutable message = newMessage(); String messageId = newMessageId(); message.setId(messageId); message.setChannel(Channel.META_DISCONNECT); message.setClientId(_session.getId()); registerCallback(messageId, callback); send(message); while (isBatching()) { endBatch(); } } } @Override public String getId() { if (_session == null) { throw new IllegalStateException("Method handshake() not invoked for local session " + this); } return _session.getId(); } @Override public boolean isConnected() { return _session != null && _session.isConnected(); } @Override public boolean isHandshook() { return _session != null && _session.isHandshook(); } @Override public String toString() { return "L:" + (_session == null ? _idHint + "_" : _session.getId()); } @Override protected void send(Message.Mutable message) { if (message instanceof ServerMessage.Mutable) { send(_session, (ServerMessage.Mutable)message); } else { ServerMessage.Mutable mutable = newMessage(); mutable.putAll(message); send(_session, mutable); } } /** *

Enqueues or sends a message to the server.

*

This method will either enqueue the message, if this session {@link #isBatching() is batching}, * or perform the actual send by calling {@link #doSend(ServerSessionImpl, ServerMessage.Mutable)}.

* * @param session The ServerSession to send as. This normally the current server session, but during handshake it is a proposed server session. * @param message The message to send. */ protected void send(ServerSessionImpl session, ServerMessage.Mutable message) { if (isBatching()) { _queue.add(message); } else { doSend(session, message); } } /** *

Sends a message to the server.

* * @param from The ServerSession to send as. This normally the current server session, but during handshake it is a proposed server session. * @param message The message to send. */ protected void doSend(ServerSessionImpl from, ServerMessage.Mutable message) { String messageId = message.getId(); if (_session != null) { message.setClientId(_session.getId()); } if (!extendSend(message)) { return; } // Extensions may have changed the messageId. message.setId(messageId); ServerMessage.Mutable reply = _bayeux.handle(from, message); reply = _bayeux.extendReply(from, _session, reply); if (reply != null) { receive(reply); } } @Override protected ServerMessage.Mutable newMessage() { return _bayeux.newMessage(); } /** *

A channel scoped to this LocalSession.

*/ protected class LocalChannel extends AbstractSessionChannel { protected LocalChannel(ChannelId channelId) { super(channelId); } @Override public ClientSession getSession() { throwIfReleased(); return LocalSessionImpl.this; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy