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

io.socket.engineio.server.EngineIoSocket Maven / Gradle / Ivy

package io.socket.engineio.server;

import io.socket.engineio.server.parser.Packet;
import io.socket.engineio.server.transport.Polling;
import io.socket.engineio.server.transport.WebSocket;
import io.socket.engineio.server.utils.JsonUtils;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;

import java.io.IOException;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicBoolean;

/**
 * An engine.io socket.
 *
 * Objects of this class represents connections to remote clients
 * one per object.
 */
@SuppressWarnings("unused")
public final class EngineIoSocket extends Emitter {

    public interface SocketedListener {

        void call(EngineIoSocket socket, Object... data);
    }

    private static final List> PAYLOAD_NOOP = new ArrayList>() {{
        add(new Packet<>(Packet.NOOP));
    }};

    private static final String EMPTY_UPGRADES = JsonUtils.toJson(new String[]{});
    private static final String WEBSOCKET_UPGRADES = JsonUtils.toJson(new String[] { WebSocket.NAME });
    private static final String HANDSHAKE_JSON = "{\"sid\": \"%s\", \"upgrades\": %s, \"pingInterval\": %d, \"pingTimeout\": %d}";

    private final String mSid;
    private final int mProtocolVersion;
    private final EngineIoServer mServer;
    private final LinkedList> mWriteBuffer = new LinkedList<>();
    private final ConcurrentHashMap> mCallbacks = new ConcurrentHashMap<>();

    private final Object mLockObject;
    private final ScheduledExecutorService mScheduledTaskHandler;
    private final Runnable mPingTask = this::sendPing;
    private final Runnable mPingTimeoutTask = () -> onClose("ping timeout", null);
    private ScheduledFuture mPingFuture = null;
    private ScheduledFuture mPingTimeoutFuture = null;

    private final AtomicBoolean mUpgrading = new AtomicBoolean(false);
    private Runnable mCleanupFunction = null;
    private ReadyState mReadyState;
    private Transport mTransport;
    private Map mInitialQuery;
    private Map> mInitialHeaders;

    EngineIoSocket(Object lockObject,
                   String sid,
                   int protocolVersion,
                   EngineIoServer server,
                   ScheduledExecutorService scheduledTaskHandler) {
        mLockObject = lockObject;

        mSid = sid;
        mProtocolVersion = protocolVersion;
        mServer = server;
        mScheduledTaskHandler = scheduledTaskHandler;

        mReadyState = ReadyState.OPENING;
    }

    /**
     * Gets the sid of this socket.
     */
    @SuppressWarnings("WeakerAccess")
    public String getId() {
        return mSid;
    }

    /**
     * Gets the protocol version of this socket.
     */
    public int getProtocolVersion() {
        return mProtocolVersion;
    }

    /**
     * Gets the ready state of this socket.
     */
    @SuppressWarnings("WeakerAccess")
    public ReadyState getReadyState() {
        return mReadyState;
    }

    /**
     * Get the query parameters of the initial HTTP connection.
     */
    public Map getInitialQuery() {
        return mInitialQuery;
    }

    /**
     * Get the headers of the initial HTTP connection.
     */
    public Map> getInitialHeaders() {
        return mInitialHeaders;
    }

    /**
     * Send a packet to the remote client.
     * Queuing of packets in case of polling transport are handled internally.
     * This method is thread safe.
     *
     * @param packet The packet to send.
     */
    public void send(Packet packet) {
        sendPacket(packet);
    }

    /**
     * Close this socket.
     */
    public void close() {
        if(mReadyState == ReadyState.OPEN) {
            mReadyState = ReadyState.CLOSING;

            if(mWriteBuffer.size() > 0) {
                mTransport.on("drain", args -> closeTransport());
            } else {
                closeTransport();
            }
        }
    }

    /**
     * Listen on the event.
     * NOTE: Unstable api. Might change in the future.
     *
     * @param event Event name
     * @param fn Event listener
     * @return A reference to this object
     */
    public EngineIoSocket on(String event, SocketedListener fn) {
        this.mCallbacks.computeIfAbsent(event, s -> new ConcurrentLinkedQueue<>());
        final ConcurrentLinkedQueue callbacks = this.mCallbacks.get(event);
        callbacks.add(fn);
        return this;
    }

    /**
     * Removes the listener.
     * NOTE: Unstable api. Might change in the future.
     *
     * @param event an event name.
     * @param fn Event listener.
     * @return a reference to this object.
     */
    public EngineIoSocket off(String event, SocketedListener fn) {
        final ConcurrentLinkedQueue callbacks = this.mCallbacks.get(event);
        if (callbacks != null) {
            Iterator it = callbacks.iterator();
            while (it.hasNext()) {
                SocketedListener internal = it.next();
                if (fn.equals(internal)) {
                    it.remove();
                    break;
                }
            }
        }
        return this;
    }

    @Override
    public EngineIoSocket off(String event) {
        this.mCallbacks.remove(event);
        return (EngineIoSocket)super.off(event);
    }

    @Override
    public Emitter emit(String event, Object... args) {
        final ConcurrentLinkedQueue callbacks = this.mCallbacks.get(event);
        if (callbacks != null) {
            for (SocketedListener fn : callbacks) {
                try {
                    fn.call(this, args);
                } catch (Exception ignore) {
                }
            }
        }
        return super.emit(event, args);
    }

    /**
     * Called after instance creation to initialize transport.
     *
     * @param transport The opened transport.
     */
    void init(Transport transport) {
        setTransport(transport);
        mInitialQuery = transport.getInitialQuery();
        mInitialHeaders = transport.getInitialHeaders();
        onOpen();
    }

    void updateInitialHeadersFromActiveTransport() {
        mInitialQuery = mTransport.getInitialQuery();
        mInitialHeaders = mTransport.getInitialHeaders();
    }

    /**
     * Handle an HTTP request.
     *
     * @param request The HTTP request object.
     * @param response The HTTP response object.
     * @throws IOException On IO error.
     */
    void onRequest(HttpServletRequest request, HttpServletResponse response) throws IOException {
        mTransport.onRequest(request, response);

        if (mUpgrading.get() && mTransport.isWritable() && mWriteBuffer.isEmpty()) {
            mTransport.send(PAYLOAD_NOOP);
        }
    }

    /**
     * Checks whether the socket can be upgraded to another transport.
     *
     * @param transport The transport to upgrade to.
     * @return Boolean value indicating if upgrade is possible.
     */
    @SuppressWarnings("SameParameterValue")
    boolean canUpgrade(String transport) {
        return (!mUpgrading.get() && mTransport.getName().equals(Polling.NAME) && transport.equals(WebSocket.NAME));
    }

    /**
     * Perform upgrade to the specified transport.
     *
     * @param transport The transport to upgrade to.
     */
    void upgrade(final Transport transport) {
        mUpgrading.set(true);

        final Runnable cleanup = () -> {
            mUpgrading.set(false);
            transport.off("packet");
            transport.off("close");
            transport.off("error");
        };

        final Listener onError = args -> {
            cleanup.run();
            transport.close();
        };

        transport.on("packet", args -> {
            final Packet packet = (Packet) args[0];
            if(packet.type.equals(Packet.PING) && (packet.data != null) && packet.data.equals("probe")) {
                final Packet replyPacket = new Packet<>(Packet.PONG);
                replyPacket.data = "probe";

                transport.send(new ArrayList>() {{
                    add(replyPacket);
                }});

                if (mTransport.isWritable()) {
                    mTransport.send(PAYLOAD_NOOP);
                }

                emit("upgrading", transport);
            } else if(packet.type.equals(Packet.UPGRADE) && (mReadyState != ReadyState.CLOSED) && (mReadyState != ReadyState.CLOSING)) {
                cleanup.run();
                clearTransport();
                setTransport(transport);
                emit("upgrade", transport);
                flush();

                schedulePing();
            } else {
                cleanup.run();
                transport.close();
            }
        });
        transport.once("close", args -> onError.call("transport closed"));
        transport.once("error", onError);

        once("close", args -> onError.call("socket closed"));
    }

    /**
     * Get the name of the current transport.
     *
     * @return Name of current transport.
     */
    String getCurrentTransportName() {
        return mTransport.getName();
    }

    private void setTransport(final Transport transport) {
        mTransport = transport;
        transport.once("error", args -> onError());
        transport.once("close", args -> {
            String description = (args.length > 0)? ((String) args[0]) : null;
            onClose("transport close", description);
        });
        transport.on("packet", args -> onPacket((Packet) args[0]));
        transport.on("drain", args -> flush());

        mCleanupFunction = () -> {
            transport.off("error");
            transport.off("close");
            transport.off("packet");
            transport.off("drain");
        };
    }

    private void closeTransport() {
        mTransport.close();
    }

    private void clearTransport() {
        if(mCleanupFunction != null) {
            mCleanupFunction.run();
        }

        mTransport.close();
    }

    private void onOpen() {
        mReadyState = ReadyState.OPEN;

        final String upgrades;
        if (mTransport.getName().equals(Polling.NAME)) {
            upgrades = WEBSOCKET_UPGRADES;
        } else {
            upgrades = EMPTY_UPGRADES;
        }

        Packet openPacket = new Packet<>(Packet.OPEN);
        openPacket.data = String.format(
                HANDSHAKE_JSON,
                JsonUtils.escape(mSid),
                upgrades,
                mServer.getOptions().getPingInterval(),
                mServer.getOptions().getPingTimeout());

        sendPacket(openPacket);

        if (mServer.getOptions().getInitialPacket() != null) {
            sendPacket(mServer.getOptions().getInitialPacket());
        }

        emit("open");

        switch (mProtocolVersion) {
            case 3:
                resetPingTimeout(mServer.getOptions().getPingTimeout() + mServer.getOptions().getPingInterval());
                break;
            case 4:
                schedulePing();
                break;
            default:
                throw new RuntimeException("Invalid protocol version");
        }
    }

    private void onClose(String reason, String description) {
        if(mReadyState != ReadyState.CLOSED) {
            mReadyState = ReadyState.CLOSED;
            if(mPingFuture != null) {
                mPingFuture.cancel(false);
            }

            clearTransport();
            emit("close", reason, description);
        }
    }

    private void onError() {
        onClose("transport error", null);
    }

    private void onPacket(Packet packet) {
        if(mReadyState == ReadyState.OPEN) {
            emit("packet", packet);

            resetPingTimeout(mServer.getOptions().getPingTimeout() + mServer.getOptions().getPingInterval());

            switch (packet.type) {
                case Packet.PING:
                    if (mProtocolVersion != 3) {
                        onError();
                    } else {
                        sendPacket(new Packet<>(Packet.PONG));
                        emit("heartbeat");
                    }
                    break;
                case Packet.PONG:
                    schedulePing();
                    emit("heartbeat");
                    break;
                case Packet.ERROR:
                    onClose("parse error", null);
                    break;
                case Packet.MESSAGE:
                    emit("data", packet.data);
                    emit("message", packet.data);
                    break;
            }
        }
    }

    private void sendPacket(Packet packet) {
        synchronized (mLockObject) {
            if ((mReadyState != ReadyState.CLOSING) && (mReadyState != ReadyState.CLOSED)) {
                mWriteBuffer.add(packet);

                flush();
            }
        }
    }

    private void flush() {
        synchronized (mLockObject) {
            if ((mReadyState != ReadyState.CLOSED) && (mTransport.isWritable()) && (mWriteBuffer.size() > 0)) {
                emit("flush", Collections.unmodifiableCollection(mWriteBuffer));

                mTransport.send(mWriteBuffer);
                mWriteBuffer.clear();

                emit("drain");
            }
        }
    }

    private void sendPing() {
        synchronized (mLockObject) {
            sendPacket(new Packet<>(Packet.PING));
            resetPingTimeout(mServer.getOptions().getPingTimeout());
        }
    }

    private void schedulePing() {
        synchronized (mLockObject) {
            if(mPingFuture != null) {
                mPingFuture.cancel(false);
            }

            mPingFuture = mScheduledTaskHandler.schedule(
                    mPingTask,
                    mServer.getOptions().getPingInterval(),
                    TimeUnit.MILLISECONDS);
        }
    }

    private void resetPingTimeout(long timeout) {
        synchronized (mLockObject) {
            if(mPingTimeoutFuture != null) {
                mPingTimeoutFuture.cancel(false);
            }

            mPingTimeoutFuture = mScheduledTaskHandler.schedule(
                    mPingTimeoutTask,
                    timeout,
                    TimeUnit.MILLISECONDS);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy