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

org.yx.http.user.RemoteUserSession Maven / Gradle / Ivy

There is a newer version: 4.0.2
Show newest version
/**
 * Copyright (C) 2016 - 2030 youtongluan.
 *
 * 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.yx.http.user;

import java.util.concurrent.TimeUnit;

import org.yx.conf.AppInfo;
import org.yx.http.kit.HttpSettings;
import org.yx.redis.Redis;
import org.yx.util.S;
import org.yx.util.StringUtil;
import org.yx.util.Task;

public class RemoteUserSession extends AbstractUserSession {
	private static final byte[] NX = { 'N', 'X' };
	private static final byte[] PX = { 'P', 'X' };

	protected long noFreshTime = AppInfo.getLong("sumk.http.session.cache.nofreshtime", 1000L * 2);

	protected final int maxSize;

	private final Redis redis;

	public RemoteUserSession(Redis redis) {
		this.redis = redis.isMuted() ? redis : redis.mute();
		this.maxSize = Math.max(AppInfo.getInt("sumk.http.session.cache.maxsize", 5000), 10);
		long seconds = AppInfo.getLong("sumk.http.session.period", 30L);
		Task.scheduleAtFixedRate(() -> {

			long duration = AppInfo.getLong("sumk.http.session.remote.duration", 1000L * 60);
			if (duration > HttpSettings.httpSessionTimeoutInMs()) {
				duration = HttpSettings.httpSessionTimeoutInMs();
			}
			CacheHelper.expire(cache, duration);
		}, seconds, seconds, TimeUnit.SECONDS);
	}

	protected final String singleKey(String userId) {
		return "_SINGLE_SES_".concat(userId);
	}

	protected final byte[] redisSessionKey(String sessionId) {
		if (sessionId == null || sessionId.isEmpty()) {
			return null;
		}
		return ("_SES_#" + sessionId).getBytes(AppInfo.UTF8);
	}

	private void removeOne() {
		try {
			String key = cache.keySet().iterator().next();
			log.debug("remove session {} from local cache", key);
			cache.remove(key);
		} catch (Exception e) {
			log.error(e.getMessage(), e);
		}
	}

	@Override
	protected TimedCachedObject loadTimedCachedObject(String sid, boolean needRefresh) {
		if (sid == null) {
			return null;
		}
		byte[] bigKey = this.redisSessionKey(sid);
		TimedCachedObject to = cache.get(sid);
		long now = System.currentTimeMillis();
		if (to == null) {
			byte[] bv = redis.get(bigKey);
			to = TimedCachedObject.deserialize(bv);
			if (to == null) {
				if (log.isTraceEnabled()) {
					log.trace("{} cannot found from redis", sid);
				}
				return null;
			}
			if (needRefresh) {
				if (log.isTraceEnabled()) {
					log.trace("{} add to local cache", sid);
				}
				if (cache.size() >= this.maxSize) {
					removeOne();
				}
				cache.put(sid, to);
			}
		}

		if (needRefresh && to.refreshTime + this.noFreshTime < now) {
			long durationInMS = HttpSettings.httpSessionTimeoutInMs();
			Long v = redis.pexpire(bigKey, durationInMS);
			if (v != null && v.longValue() == 0) {
				cache.remove(sid);
				log.trace("{} was pexpire by redis,and remove from local cache", sid);
				return null;
			}
			if (log.isTraceEnabled()) {
				log.trace("{} was refresh in redis", sid);
			}
			to.refreshTime = now;
		}
		return to;
	}

	@Override
	public void removeSession(String sessionId) {
		byte[] bigKey = this.redisSessionKey(sessionId);
		if (bigKey == null) {
			return;
		}
		if (HttpSettings.isSingleLogin()) {
			SessionObject t = getUserObjectBySessionId(sessionId);
			if (t != null) {
				redis.del(this.singleKey(t.getUserId()));
			}
		}
		this.cache.remove(sessionId);
		redis.del(bigKey);
	}

	@Override
	public boolean setSession(String sessionId, SessionObject sessionObj, byte[] key, boolean singleLogin) {
		long sessionTimeout = HttpSettings.httpSessionTimeoutInMs();
		byte[] bigKey = this.redisSessionKey(sessionId);
		String json = S.json().toJson(sessionObj);
		byte[] data = TimedCachedObject.toBytes(json, key);
		String ret = redis.set(bigKey, data, NX, PX, sessionTimeout);
		if (!"OK".equalsIgnoreCase(ret) && !"1".equals(ret)) {
			return false;
		}
		if (!singleLogin) {
			return true;
		}

		String singleUserKey = singleKey(sessionObj.userId);
		String oldSessionId = redis.get(singleUserKey);
		if (StringUtil.isNotEmpty(oldSessionId) && !oldSessionId.equals(sessionId)) {
			redis.del(redisSessionKey(oldSessionId));
			cache.remove(oldSessionId);
		}
		long expireTime = sessionObj.getExpiredTime() != null ? sessionObj.getExpiredTime() - System.currentTimeMillis()
				: -1;
		if (expireTime < 1) {

			expireTime = AppInfo.getLong("sumk.http.session.single.maxTime", 1000L * 3600 * 20);
		}
		redis.psetex(singleUserKey, expireTime, sessionId);
		return true;
	}

	@Override
	public String getSessionIdByUserFlag(String userId) {
		return redis.get(this.singleKey(userId));
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy