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

io.femo.ws.WebSocketConnection Maven / Gradle / Ivy

package io.femo.ws;

import com.google.gson.JsonObject;
import io.femo.ws.lib.WebSocketLibraryConnection;
import org.jetbrains.annotations.NotNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.*;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.nio.ByteBuffer;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * Created by felix on 6/4/16.
 */
public class WebSocketConnection extends Thread {

    private static final Logger LOGGER = LoggerFactory.getLogger("HTTP-WS");

    private final InputStream inputStream;
    private final OutputStream outputStream;
    private WebSocketHandler webSocketHandler;
    private Socket socket;
    private AtomicBoolean open = new AtomicBoolean(true);
    private AtomicBoolean closing = new AtomicBoolean(false);
    private static AtomicInteger counter = new AtomicInteger(0);

    private ThreadLocal number = new ThreadLocal() {
        @NotNull
        @Override
        protected Integer initialValue() {
            return counter.getAndIncrement();
        }
    };

    public WebSocketConnection(WebSocketHandler webSocketHandler, Socket socket) throws IOException {
        this.webSocketHandler = webSocketHandler;
        this.socket = socket;
        this.inputStream = socket.getInputStream();
        this.outputStream = socket.getOutputStream();
    }

    @Override
    public void run() {
        setName(String.format("ws-handler-%04d", number.get()));
        ByteArrayOutputStream buffer = new ByteArrayOutputStream();
        boolean frag = false;
        int opcode = 0;
        while (open.get()) {
            try {
                WebSocketFrame webSocketFrame = WebSocketFrame.read(inputStream);
                if(!webSocketFrame.isFin()) {
                    if(!frag && webSocketFrame.getOpcode() == 0) {
                        LOGGER.warn("Protocol violation. Fragmentation must be initialized with an opcode other than 0. " +
                                "See RFC 6455 Section 5.4");
                    }
                    if(frag && webSocketFrame.getOpcode() == 0) {
                        buffer.write(webSocketFrame.getApplicationData());
                    }
                    if(frag && (webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.TEXT_FRAME ||
                            webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.BINARY_FRAME)) {
                        LOGGER.error("Not recoverable protocol violation. Fragments of different messages must no be interleaved." +
                                "See RFC 6455 Section 5.4. Closing connection.");
                        close();
                    } else if (webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.TEXT_FRAME ||
                            webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.BINARY_FRAME) {
                        frag = true;
                        opcode = webSocketFrame.getOpcode();
                        buffer.write(webSocketFrame.getApplicationData());
                    } else {
                        LOGGER.warn("Protocol violation. Fragmentation can not be applied to control frames." +
                                "See RFC 6455 Section 5.4.");
                    }
                } else {
                    if(frag && webSocketFrame.getOpcode() == 0) {
                        buffer.write(webSocketFrame.getApplicationData());
                        frag = false;
                        webSocketHandler.raise(opcode, buffer.toByteArray(), this);
                        buffer.reset();
                    } else if(frag && (webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.TEXT_FRAME ||
                            webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.BINARY_FRAME)) {
                        LOGGER.error("Not recoverable protocol violation. Fragments of different messages must no be interleaved." +
                                "See RFC 6455 Section 5.4. Closing connection.");
                        close();
                    } else if (frag) {
                        webSocketHandler.raise(webSocketFrame.getOpcode(), webSocketFrame.getApplicationData(), this);
                    } else if (webSocketFrame.getOpcode() == 0) {
                        LOGGER.warn("Protocol violation. Continuation frames can only be sent during active fragmentation." +
                                "See RFC 6455 Section 5.4.");
                    } else {
                        if(webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.CONNECTION_CLOSE) {
                            if(!closing.get()) {
                                close();
                            }
                            open.set(false);
                            inputStream.close();
                            outputStream.close();
                            socket.close();
                            LOGGER.debug("Connection closed.");
                            webSocketHandler.raise(webSocketFrame.getOpcode(), webSocketFrame.getApplicationData(), this);
                        } else if (webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.PING) {
                            WebSocketFrame pongFrame = new WebSocketFrame();
                            pongFrame.setOpcode(Constants.WEBSOCKET.FRAME.OPCODES.PONG);
                            pongFrame.setFin(true);
                            pongFrame.setPayloadLength(webSocketFrame.getPayloadLength());
                            pongFrame.setApplicationData(webSocketFrame.getApplicationData());
                            synchronized (outputStream) {
                                pongFrame.write(outputStream);
                            }
                        } else if (webSocketFrame.getOpcode() == Constants.WEBSOCKET.FRAME.OPCODES.PONG) {
                            LOGGER.debug("Received pong with message {} ", new String(webSocketFrame.getApplicationData()));
                        } else {
                            webSocketHandler.raise(webSocketFrame.getOpcode(), webSocketFrame.getApplicationData(), this);
                        }

                    }
                }
            } catch (WebSocketException e) {
                open.set(false);
            } catch (SocketTimeoutException e) {
                LOGGER.debug("Socket timed out. Pinging client!");
                ping("This is some really ridiculous ping message");
            } catch (IOException e) {
                LOGGER.warn("Error while handling WebSocket connection for " + socket.getInetAddress());
            }
        }
    }

    public void close () {
        close((short) 1000);
    }

    public void ping(String message) {
        WebSocketFrame pingFrame = new WebSocketFrame();
        pingFrame.setOpcode(Constants.WEBSOCKET.FRAME.OPCODES.PING);
        pingFrame.setFin(true);
        byte[] payload = message.getBytes();
        pingFrame.setPayloadLength(payload.length);
        pingFrame.setApplicationData(payload);
        synchronized (outputStream) {
            try {
                pingFrame.write(outputStream);
            } catch (IOException e) {
                LOGGER.warn("Error while sending ping!", e);
            }
        }
    }

    public void close(short statuscode) {
        WebSocketFrame webSocketFrame = new WebSocketFrame();
        webSocketFrame.setFin(true);
        webSocketFrame.setOpcode(Constants.WEBSOCKET.FRAME.OPCODES.CONNECTION_CLOSE);
        webSocketFrame.setPayloadLength(2);
        webSocketFrame.setApplicationData(ByteBuffer.allocate(2).putShort(statuscode).array());
        closing.set(true);
        synchronized (outputStream) {
            try {
                webSocketFrame.write(outputStream);
            } catch (IOException e) {
                LOGGER.warn("Exception while closing the connection with status " + statuscode + ". Connection probably already closed!");
            }
        }
    }

    public void send(Constants.WEBSOCKET.FRAME.DataType dataType, byte[] data) throws IOException {
        WebSocketFrame webSocketFrame = new WebSocketFrame();
        webSocketFrame.setFin(true);
        if(dataType == Constants.WEBSOCKET.FRAME.DataType.BINARY) {
            webSocketFrame.setOpcode(Constants.WEBSOCKET.FRAME.OPCODES.BINARY_FRAME);
        } else if (dataType == Constants.WEBSOCKET.FRAME.DataType.TEXT) {
            webSocketFrame.setOpcode(Constants.WEBSOCKET.FRAME.OPCODES.TEXT_FRAME);
        }
        webSocketFrame.setPayloadLength(data.length);
        webSocketFrame.setApplicationData(data);
        synchronized (outputStream) {
            webSocketFrame.write(outputStream);
        }
    }

    public void send(String data) throws IOException {
        send(Constants.WEBSOCKET.FRAME.DataType.TEXT, data.getBytes("UTF-8"));
    }

    public WebSocketHandler handler() {
        return webSocketHandler;
    }

    public boolean isOpen() {
        return open.get();
    }

    public boolean isClosed() {
        return !isOpen();
    }

    public void broadcastExcept(String data, List but) {
        broadcastExcept(Constants.WEBSOCKET.FRAME.DataType.TEXT, data.getBytes(), but);
    }

    public void broadcastExcept(Constants.WEBSOCKET.FRAME.DataType dataType, byte[] data, List but) {
        webSocketHandler.sendToAllExcept(dataType, data, but);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy