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

org.voovan.network.HeartBeat Maven / Gradle / Ivy

There is a newer version: 4.3.8
Show newest version
package org.voovan.network;

import org.voovan.tools.buffer.ByteBufferChannel;
import org.voovan.tools.TEnv;

import java.nio.ByteBuffer;
import java.util.concurrent.LinkedBlockingDeque;

/**
 * 心跳类
 *
 * @author: helyho
 * Voovan Framework.
 * WebSite: https://github.com/helyho/Voovan
 * Licence: Apache v2 License
 */
public class HeartBeat {
    private byte[] ping;
    private byte[] pong;
    private ByteBuffer pingBuffer;
    private ByteBuffer pongBuffer;
    private boolean isFirstBeat = true;
    private LinkedBlockingDeque queue;
    private int failedCount = 0;


    /**
     * 构造方法
     * @param ping  ping 消息
     * @param pong  pong 消息
     * @return 心跳消息对象
     */
    private HeartBeat(String ping, String pong){
        this.ping = ping.getBytes();
        this.pong = pong.getBytes();
        this.pingBuffer = ByteBuffer.wrap(this.ping);
        this.pongBuffer = ByteBuffer.wrap(this.pong);
        queue = new LinkedBlockingDeque();
    }

    /**
     * 获取心跳包队列
     * @return 心跳包队列
     */
    private LinkedBlockingDeque getQueue() {
        return queue;
    }

    /**
     * 获取 ping 报文
     * @return ping 报文
     */
    public byte[] getPing() {
        return ping;
    }

    /**
     * 获取 pong 报文
     * @return pong 报文
     */
    public byte[] getPong() {
        return pong;
    }

    /**
     * 获取连续失败次数
     *      每次成功会被归零
     * @return 失败次数
     */
    public int getFailedCount() {
        return failedCount;
    }

    /**
     * 截断心跳消息
     * @param session 会话对象
     * @return true: 成功, false: 失败
     */
    public static boolean interceptHeartBeat(IoSession session){
        if(session==null || session.getHeartBeat()==null){
            return false;
        }

        HeartBeat heartBeat = session.getHeartBeat();
        ByteBufferChannel byteBufferChannel = session.getReadByteBufferChannel();

        if (heartBeat != null && byteBufferChannel.size() > 0) {
            //心跳处理
            if (heartBeat != null) {
                if (byteBufferChannel.startWith(heartBeat.getPing())) {
                    byteBufferChannel.shrink(0, heartBeat.getPing().length);
                    heartBeat.getQueue().addLast(1);
                    return true;
                }

                if (byteBufferChannel.startWith(heartBeat.getPong())) {
                    byteBufferChannel.shrink(0, heartBeat.getPong().length);
                    heartBeat.getQueue().addLast(2);
                    return true;
                }
            }
        }

        return false;
    }

    /**
     * 一次心跳动作
     * @param session 会话对象
     * @return true:心跳成功,false: 心跳失败
     */
    public static boolean beat(IoSession session) {
        HeartBeat heartBeat = session.getHeartBeat();

        if(!session.isConnected()){
            return false;
        }

        //收个心跳返回成功
        if (heartBeat.isFirstBeat) {
            heartBeat.isFirstBeat = false;
            if (session.socketContext().getConnectModel() == ConnectModel.CLIENT) {
                //等待这个时间的目的是为了等待客户端那边的心跳检测启动
                TEnv.sleep(session.getIdleInterval());
                session.send(heartBeat.pingBuffer);
                session.flush();
            }
            return true;
        }

        //弥补双方发送的时间差,等待心跳到来,如果超过空闲事件周期则认为是失败
        TEnv.wait(session.getIdleInterval() * 1000, false, ()->heartBeat.getQueue().size() == 0);

        if (heartBeat.getQueue().size() > 0) {
            int beatType = heartBeat.getQueue().pollFirst();

            if (beatType == 1) {
                session.send(heartBeat.pongBuffer);
                session.flush();
                heartBeat.failedCount = 0;
                return true;
            } else if (beatType == 2) {
                session.send(heartBeat.pingBuffer);
                session.flush();
                heartBeat.failedCount = 0;
                return true;
            } else {
                heartBeat.failedCount++;
                return false;
            }
        }

        heartBeat.failedCount++;
        return false;
    }
    /**
     * 将心跳绑定到 Session
     * @param session   会话
     * @param ping  ping 消息
     * @param pong  pong 消息
     * @return 心跳消息对象
     */
    public static HeartBeat attachSession(IoSession session, String ping, String pong){
        HeartBeat heartBeat = null;
        if(session.getHeartBeat()==null) {
            heartBeat = new HeartBeat(ping, pong);
            session.setHeartBeat(heartBeat);
        } else{
            heartBeat = session.getHeartBeat();
        }
        return heartBeat;

    }

    /**
     * 将心跳绑定到 Session
     *      默认使用 PING, PONG 作为心跳消息
     * @param session   Socket会话
     * @param connectModel  模式,指定发送第一个 PING 的模式,服务端先发,或者客户端先发
     * @return 心跳消息对象
     */
    public static HeartBeat attachSession(IoSession session, int connectModel){
        HeartBeat heartBeat = null;
        if(session.getHeartBeat()==null) {
            heartBeat = new HeartBeat("PING", "PONG");
            session.setHeartBeat(heartBeat);
        }else{
            heartBeat = session.getHeartBeat();
        }

        return heartBeat;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy