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

net.ymate.platform.serv.AbstractSessionManager Maven / Gradle / Ivy

/*
 * Copyright 2007-2018 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 net.ymate.platform.serv;

import net.ymate.platform.core.support.Speedometer;
import net.ymate.platform.core.support.impl.DefaultSpeedListener;
import net.ymate.platform.core.util.DateTimeUtils;
import net.ymate.platform.core.util.ThreadUtils;
import net.ymate.platform.serv.impl.DefaultSessionIdleChecker;
import net.ymate.platform.serv.nio.INioCodec;
import net.ymate.platform.serv.nio.INioSession;
import org.apache.commons.lang.NullArgumentException;

import java.net.InetSocketAddress;
import java.util.Collection;
import java.util.Collections;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

/**
 * @author 刘镇 ([email protected]) on 2018/11/14 11:35 AM
 * @version 1.0
 */
public abstract class AbstractSessionManager implements ISessionManager {

    private Map __sessions;

    private IServer __server;

    private IServerCfg __serverCfg;

    private INioCodec __codec;

    private long __idleTimeInMillis;

    private ISessionIdleChecker __idleChecker;

    private ScheduledExecutorService __idleCheckExecutorService;

    private Speedometer __speedometer;

    private final Object __locker = new Object();

    /**
     * 构造器
     *
     * @param serverCfg        服务端配置接口实现
     * @param codec            编解码器接口实现
     * @param idleTimeInMillis 会话空闲时间毫秒值, 小于等于0表示不开启空闲检查
     */
    public AbstractSessionManager(IServerCfg serverCfg, INioCodec codec, long idleTimeInMillis) {
        __sessions = new ConcurrentHashMap();
        __serverCfg = serverCfg;
        __codec = codec;
        __idleTimeInMillis = idleTimeInMillis;
    }

    @Override
    public SESSION_WRAPPER sessionWrapper(SESSION_ID sessionId) {
        return sessionId == null ? null : __sessions.get(sessionId);
    }

    @Override
    public Collection sessionWrappers() {
        return Collections.unmodifiableCollection(__sessions.values());
    }

    @Override
    public boolean contains(SESSION_ID sessionId) {
        return __sessions.containsKey(sessionId);
    }

    @Override
    @SuppressWarnings("unchecked")
    public void closeSessionWrapper(SESSION_WRAPPER sessionWrapper) {
        if (sessionWrapper != null && sessionWrapper.getId() != null) {
            removeSessionWrapper((SESSION_ID) sessionWrapper.getId());
        }
    }

    @Override
    public long sessionCount() {
        return __sessions.size();
    }

    @Override
    public void speedometer(Speedometer speedometer) {
        if (__server == null) {
            __speedometer = speedometer;
        }
    }

    @Override
    public void idleChecker(ISessionIdleChecker sessionIdleChecker) {
        if (__server == null) {
            __idleChecker = sessionIdleChecker;
        }
    }

    /**
     * 触发速度计数
     */
    public void speedTouch() {
        if (__speedometer != null) {
            __speedometer.touch();
        }
    }

    /**
     * 注册客户端会话
     *
     * @param session       会话对象
     * @param socketAddress 目标来源套接字地址
     * @return 返回注册的客户端会话包装器对象
     */
    @SuppressWarnings("unchecked")
    protected SESSION_WRAPPER __doRegisterSession(INioSession session, InetSocketAddress socketAddress) {
        SESSION_WRAPPER _wrapper = doBuildSessionWrapper(session, socketAddress);
        if (doRegister(_wrapper)) {
            putSessionWrapper((SESSION_ID) _wrapper.getId(), _wrapper);
            return _wrapper;
        }
        return null;
    }

    /**
     * 执行会话注册逻辑
     *
     * @param session 会话包装器对象
     * @return 返回值为false表示不向管理器注册当前会话
     */
    protected boolean doRegister(SESSION_WRAPPER session) {
        return true;
    }

    /**
     * 将会话包装器对象放入管理器
     *
     * @param sessionId      会话标识符
     * @param sessionWrapper 会话包装器对象
     */
    protected void putSessionWrapper(SESSION_ID sessionId, SESSION_WRAPPER sessionWrapper) {
        __sessions.put(sessionId, sessionWrapper);
    }

    /**
     * 移除会话
     *
     * @param sessionId 会话标识符
     * @return 返回被移除的会话对象, 若不存在则返回null
     */
    protected SESSION_WRAPPER removeSessionWrapper(SESSION_ID sessionId) {
        return __sessions.remove(sessionId);
    }

    /**
     * 根据会话对象构建包装器
     *
     * @param session       会话对象
     * @param socketAddress 目标来源套接字地址
     * @return 返回包装器对象
     */
    protected abstract SESSION_WRAPPER doBuildSessionWrapper(INioSession session, InetSocketAddress socketAddress);

    protected abstract IServer doBuildServer(IServ owner, IServerCfg serverCfg, INioCodec codec);

    @Override
    public void init(IServ owner) throws Exception {
        synchronized (__locker) {
            if (__server == null) {
                __server = doBuildServer(owner, __serverCfg, __codec);
                if (__server == null) {
                    throw new NullArgumentException("server");
                }
            }
        }
        __server.start();
        if (__speedometer != null && !__speedometer.isStarted()) {
            __speedometer.start(new DefaultSpeedListener(__speedometer));
        }
        //
        if (__idleTimeInMillis > 0) {
            if (__idleChecker == null) {
                __idleChecker = new DefaultSessionIdleChecker();
            }
            if (!__idleChecker.isInited()) {
                __idleChecker.init(this);
            }
            //
            __idleCheckExecutorService = ThreadUtils.newScheduledThreadPool(1, ThreadUtils.createFactory("SessionIdleChecker-"));
            __idleCheckExecutorService.scheduleWithFixedDelay(new Runnable() {
                @Override
                public void run() {
                    __idleChecker.processIdleSession(__sessions, __idleTimeInMillis);
                }
            }, DateTimeUtils.SECOND, DateTimeUtils.SECOND, TimeUnit.MILLISECONDS);
        }
    }

    @Override
    public void destroy() throws Exception {
        if (__speedometer != null && __speedometer.isStarted()) {
            __speedometer.close();
        }
        __idleCheckExecutorService.shutdownNow();
        __server.close();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy