新增插件:sa-token-hutool-timed-cache,用于整合 Hutool 缓存插件 TimedCache

This commit is contained in:
click33
2024-04-13 16:39:17 +08:00
parent 03e3925c4d
commit 3ff0bbdeaf
18 changed files with 930 additions and 0 deletions

View File

@@ -26,6 +26,7 @@
<module>sa-token-redisson-jackson2</module>
<module>sa-token-redisx</module>
<module>sa-token-alone-redis</module>
<module>sa-token-hutool-timed-cache</module>
<module>sa-token-dialect-thymeleaf</module>
<module>sa-token-sso</module>
<module>sa-token-oauth2</module>

View File

@@ -0,0 +1,35 @@
<?xml version='1.0' encoding='utf-8'?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>cn.dev33</groupId>
<artifactId>sa-token-plugin</artifactId>
<version>${revision}</version>
<relativePath>../pom.xml</relativePath>
</parent>
<packaging>jar</packaging>
<name>sa-token-hutool-timed-cache</name>
<artifactId>sa-token-hutool-timed-cache</artifactId>
<description>sa-token integrate hutool-TimedCache</description>
<dependencies>
<!-- sa-token-spring-boot-starter -->
<dependency>
<groupId>cn.dev33</groupId>
<artifactId>sa-token-core</artifactId>
</dependency>
<!-- Hutool Cache -->
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-cache</artifactId>
<version>5.8.27</version>
</dependency>
</dependencies>
</project>

View File

@@ -0,0 +1,200 @@
/*
* Copyright 2020-2099 sa-token.cc
*
* 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 cn.dev33.satoken.dao;
import cn.dev33.satoken.SaManager;
import cn.dev33.satoken.util.SaFoxUtil;
import cn.hutool.cache.CacheUtil;
import cn.hutool.cache.impl.CacheObj;
import cn.hutool.cache.impl.TimedCache;
import java.util.Iterator;
import java.util.List;
/**
* Sa-Token 持久层接口(基于 Hutool-TimedCache系统重启后数据丢失
*
* @author click33
* @since 1.38.0
*/
public class SaTokenDaoForHutoolTimedCache implements SaTokenDao {
//
/**
* 底层缓存对象:
* 参数填1000代表默认ttl为1000毫秒实际上此参数意义不大因为后续每个值都会单独设置自己的ttl值
*/
TimedCache<String, Object> timedCache = CacheUtil.newTimedCache(1000);
// ------------------------ String 读写操作
@Override
public String get(String key) {
return (String) getObject(key);
}
@Override
public void set(String key, String value, long timeout) {
setObject(key, value, timeout);
}
@Override
public void update(String key, String value) {
updateObject(key, value);
}
@Override
public void delete(String key) {
deleteObject(key);
}
@Override
public long getTimeout(String key) {
return getObjectTimeout(key);
}
@Override
public void updateTimeout(String key, long timeout) {
updateObjectTimeout(key, timeout);
}
// ------------------------ Object 读写操作
@Override
public Object getObject(String key) {
// 第二个参数代表:是否刷新最后访问时间
// 设置为false因为我们不需要刷新最后访问时间只需要取值即可
return timedCache.get(key, false);
}
@Override
public void setObject(String key, Object object, long timeout) {
if(timeout == 0 || timeout <= SaTokenDao.NOT_VALUE_EXPIRE) {
return;
}
// 如果为永不过期
// 在 sa-token 中,-1 代表永不过期
// 在 hutool-TimedCache 中0 代表永不过期
// 为了适应 hutool-TimedCache 规范,这里将 -1 转换为 0
if(timeout == SaTokenDao.NEVER_EXPIRE) {
timedCache.put(key, object, 0);
return;
}
// 正常情况
timedCache.put(key, object, timeout * 1000);
}
@Override
public void updateObject(String key, Object object) {
long expire = getObjectTimeout(key);
// -2 = 无此键
if(expire == SaTokenDao.NOT_VALUE_EXPIRE) {
return;
}
this.setObject(key, object, expire);
}
@Override
public void deleteObject(String key) {
timedCache.remove(key);
}
@Override
public long getObjectTimeout(String key) {
return getKeyTimeout(key);
}
@Override
public void updateObjectTimeout(String key, long timeout) {
// $$待优化对一个不存在的key进行修改timeout操作时可能会造成一些意外数据待进一步测试
this.setObject(key, this.getObject(key), timeout);
}
// ------------------------ Session 读写操作
// 使用接口默认实现
// --------- 会话管理
@Override
public List<String> searchData(String prefix, String keyword, int start, int size, boolean sortType) {
return SaFoxUtil.searchList(timedCache.keySet(), prefix, keyword, start, size, sortType);
}
// --------- 过期时间相关操作
/**
* 获取指定 key 的剩余存活时间 (单位:秒)
* @param key 指定 key
* @return 这个 key 的剩余存活时间,返回-1=永不过期,返回-2=无此键
*/
long getKeyTimeout(String key) {
final Iterator<CacheObj<String, Object>> values = timedCache.cacheObjIterator();
CacheObj<String, Object> co;
while (values.hasNext()) {
co = values.next();
if(co.getKey().equals(key)) {
long ttl = co.getTtl();
// 在 Hutool-TimedCache 中ttl=0 (或<0) 代表永不过期,统一返回 Sa-Token 可以理解的 -1
if(ttl <= 0) {
return NEVER_EXPIRE;
}
// 不为 0那就计算一下剩余有效期
// 单位:毫秒
long timeout = ttl - (System.currentTimeMillis() - co.getLastAccess());
if(timeout < 0) {
timeout = 0;
}
// 转秒返回
return timeout / 1000;
}
}
// 代码至此,说明缓存中没有这个值
return NOT_VALUE_EXPIRE;
}
// --------- 定时清理过期数据
/**
* 组件被安装时,开始刷新数据线程
*/
@Override
public void init() {
// 定时清理间隔
int dataRefreshPeriod = SaManager.getConfig().getDataRefreshPeriod();
// 配置为<=0代表不启用定时清理
if(dataRefreshPeriod <= 0) {
return;
}
// 启用定时清理(转毫秒)
timedCache.schedulePrune(dataRefreshPeriod * 1000L);
}
/**
* 组件被卸载时,结束定时任务,不再定时清理过期数据
*/
@Override
public void destroy() {
timedCache.cancelPruneSchedule();
}
}

View File

@@ -0,0 +1 @@
org.springframework.boot.autoconfigure.EnableAutoConfiguration=cn.dev33.satoken.dao.SaTokenDaoForHutoolTimedCache