sa-token/sa-token-doc/more/common-questions.md
2022-10-23 18:52:30 +08:00

23 KiB
Raw Blame History

常见问题排查

本篇整理大家在群聊里经常提问的一些问题如有补充欢迎提交pr

toc


一、常见报错

Q报错非Web上下文无法获取Request

报错原因Sa-Token 的部分 API 只能在 Web 上下文中调用,报这个错说明你调用 Sa-Token 的地方不在 Web 上下文中,请排查:

  1. 是否在 main 方法中调用了 Sa-Token 的API
  2. 是否在带有 @Async 注解的方法中调用了 Sa-Token 的API
  3. 是否在一些丢失web上下文的子线程中调用了 Sa-Token 的API例如 MyBatis-PlusinsertFill 自动填充
  4. 是否在一些非 Http 协议的 RPC 框架中(例如 Dubbo调用了 Sa-Token 的API
  5. 是否在 SpringBoot 启动初始化的方法中调用了 Sa-Token 的API例如@PostConstruct

解决方案:先获取你想要的值,再把这个值当做一个参数传递到这些方法中,而不是直接从方法内调用 Sa-Token 的API。

Q报错未初始化任何有效上下文处理器

报错原因Sa-Token底层不能确认最终运行的web容器所以抽象了 SaTokenContext 接口,对接不同容器时需要注入不同的实现,通常这个注入工作都是框架自动完成的, 你只需要按照文档开始部分集成相应的依赖即可

如果报了这个错误,说明框架没有注入正确的上下文实现,请排查:

  1. 如果你的项目是微服务项目,请直接参考:微服务-依赖引入说明,如果是单体项目,请往下看:
  2. 请判断你的项目是 SpringMVC 环境还是 WebFlux 环境
    • 如果是 SpringMVC 环境就引入 sa-token-spring-boot-starter 依赖,参考:在SpringBoot环境集成
    • 如果是 WebFlux 环境就引入 sa-token-reactor-spring-boot-starter 依赖,参考:在WebFlux环境集成
    • 引入错误的依赖会导致SaTokenContext初始化失败,抛出上述异常
    • 如果你还无法分辨你是哪个环境,就看你的 pom.xml 依赖,如果引入了spring-boot-starter-web就是SpringMVC环境如果引入了 spring-boot-starter-webflux 就是WebFlux环境。……什么你说你两个都引入了那你的项目能启动成功吗
  3. 如果是 WebFlux 环境而且正确引入了依赖,依然报错,请检查是否注册了全局过滤器,在 WebFlux 下这一步是必须的
  4. 如果以上步骤排除无误后依然报错,请直接提 issue 或者加入QQ群求助。

Q报错NotLoginExceptionxxx

这个错是说明调用接口的人没有通过登录校验,请注意通常异常提示语已经描述清楚了没有通过认证的具体原因:

如果是未能读取到有效Token

  • 可能1前端没有提交 Token最好从前端f12控制台看看请求参数里有 token 吗)。
  • 可能2前端提交了 Token但是参数名不对。默认参数名是 satoken,可通过配置文件 sa-token.token-name: satoken 来更改。
  • 可能3前端提交了 Token但是你配置了框架不读取比如说你配置了 is-read-header=false关闭header读取此时你再从 header 里提交token框架就无法读取到。
  • 可能4前端提交了 Token但是 Token前缀 不对,可参考:自定义 Token 前缀
  • 可能5你使用了 Nginx 反向代理,而且配置了 自定义Token名称而且自定义的名称还带有下划线比如 shop_token而且还是你的项目还是从 Header头提交Token的此时 Nginx 默认会吞掉你的下划线参数,可参考:nginx做转发时带下划线的header参数丢失

如果是Token无效6ad93254-b286-4ec9-9997-4430b0341ca0

  • 可能1前端提交的 token 是乱填的,或者从别的项目拷过来的,或者多个项目一起开发时彼此的 Token 串项目了。
  • 可能2前端提交的 token 已过期timeout超时了
  • 可能3在不集成 Redis 的情况下:颁发 token 后,项目重启了,导致 token 无效。
  • 可能4在集成 Redis 的情况下:颁发 token 后Redis重启了导致 token 无效。
  • 可能5你提交的 token 和框架读取到的 token 不一致:
    • 可能5.1:比如说你配置了is-read-header=false关闭header读取然后你从header提交token-A而框架从Cookie里读取token-B,导致鉴权不通过(框架读取顺序为body->header->cookie
    • 可能5.2:比如说你配置了token-name=x-token自定义token名称此时你从header提交satoken:token-A参数名没对上然后框架从header里读取不到你提交的token转而继续从Cookie读取到了token-B
  • 可能6在集成 jwt 插件的情况下:
    • 如果使用的是 Simple 模式情况和不集成jwt一样。
    • 如果使用的是 Mixin 和 Stateless 模式:查看这个 token 颁发后是否更改了 jwtSecretKey 配置项。
  • 可能7同一账号登录数量超过12个导致最先登录的被强制注销掉这个值可以通过 maxLoginCount 来配置默认值12-1代表不做限制。
  • 可能8在配置了 is-concurrent=true, is-share=true的情况下你和别人共同登录了同一账号此时对方注销了登录由于你们使用的是同一个token导致你这边的会话也失效了。
  • 可能9可能是多账号鉴权的关系在多账号模式下如果是 StpUserUtil.login() 颁发的token你从 StpUtil.checkLogin() 进行校验永远都是无效token因为账号体系没对上。

如果是Token已过期6ad93254-b286-4ec9-9997-4430b0341ca0

  • 可能1前端提交的 token 临时过期activity-timeout超时了比如配置了 activity-timeout=120但是超过了120秒没有访问接口
  • 可能2集成jwt而且使用的是 Mixin 或 Stateless 模式而且token过期了timeout超时了

如果是Token已被顶下线6ad93254-b286-4ec9-9997-4430b0341ca0

  • 可能1在项目配置了 is-concurrent=false 的前提下,这个账号又被别人登录了,导致旧登录被挤了下去。
  • 可能2这个账号被 StpUtil.replaced(loginId, device) 方法强制顶下线了。

如果是Token已被踢下线6ad93254-b286-4ec9-9997-4430b0341ca0

  • 可能1这个账号被 StpUtil.kickout(loginId) 方法强制踢下线了。

Q集成 Redis 后,明明 Redis 中有值却还是提示无效Token

根据以往的处理经验,发生这种情况 90% 的概率是因为你找错了Redis代码连接的Redis和你用管理工具看到的Redis并不是同一个。

你可能会问:我看配置文件明明是同一个啊?

我的回答是:别光看配置文件,不一定准确,在启动时直接执行 SaManager.getSaTokenDao().set("name", "value", 100000); 随便写入一个值看看能不能根据你的预期写进这个Redis如果能的话才能证明代码连接的Reids你用管理工具看到的Redis 是同一个,再进行下一步排查。

Q加了注解进行鉴权认证不生效

  1. 注解鉴权功能默认关闭两种方式任选其一进行打开注册注解拦截器、集成AOP模块参考注解式鉴权
  2. 在Spring环境中, 如果同时配置了WebMvcConfigurerWebMvcConfigurationSupport时, 也会导致拦截器失效.
    • 常见场景: 很多项目中会在WebMvcConfigurationSupport中配置addResourceHandlers方法开放Swagger等相关静态资源映射, 同时基于Sa-Token添加了WebMvcConfigurer配置addInterceptors方法注册注解拦截器, 这样会导致注解拦截器失效.
    • 解决方案: WebMvcConfigurerWebMvcConfigurationSupport只选一个配置, 建议统一通过实现WebMvcConfigurer接口进行配置.
  3. 如果以上步骤处理后仍然没有效果,加群说明一下复现步骤

Q我加了拦截器鉴权但是好像没有什么效果请求没有被拦截住

  • 可能1这个拦截器可能没有注册成功。
  • 可能2你访问的请求没有进入这个拦截器。

尝试按照下面的代码测试一下看看:

// 注册拦截器 
@Configuration
public class SaTokenConfigure implements WebMvcConfigurer {
	@Override
	public void addInterceptors(InterceptorRegistry registry) {
		System.out.println("--------- flag 1");
		registry.addInterceptor(new SaInterceptor(handle -> {
			System.out.println("--------- flag 2");
			StpUtil.checkLogin();  // 登录校验,只有会话登录后才能通过这句代码 
		}))
		.addPathPatterns("/user/**")
		.excludePathPatterns("/user/doLogin");
	}
}

在启动时 flag 1 被打印出来,才证明拦截器注册成功了,在访问请求时 flag 2 被打印出来,才证明请求进入了拦截器。

如果拦截器没有注册成功,则:

  • 可能1SpringBoot 版本较高(>= 2.6.0),请尝试在启动类加上 @EnableWebMvc 注解再重新启动。
  • 可能2SaTokenConfigure 配置类不在启动类的同包或者子包下,导致没有被 SpringBoot 扫描到。
  • 可能3SaTokenConfigure 配置类在启动类的同包或者子包下,但启动类上加了 @ComponentScan("com.xxx") 注解,导致包扫描范围不正确,请将此注解删除或移动到其它配置类上。
  • 可能4项目属于 Maven 多模块项目,SaTokenConfigure 和启动类没有在一个模块,且启动类模块没有引入配置类的模块,导致加载不到。

如果拦截器已经注册成功,但请求没有进入拦截器:

  • 可能1你访问的 path没有被 .addPathPatterns("/user/**") 拦截住。
  • 可能2你访问的 path.excludePathPatterns("/xxx/xx") 排除掉了。
  • 可能3你访问的是另一个项目请把当前项目停掉看看你的请求还能不能访问成功。

注:以上的排查步骤,对过滤器不生效的情形一样适用。

Q我使用拦截器鉴权时明明排除了某个路径却仍然被拦截了

  • 可能1你的项目可能是跨域了先把跨域问题解决掉参考解决跨域问题
  • 可能2你访问的接口可能是404了SpringBoot环境下如果访问接口404后会被转发到/error,然后被再次拦截。请确保你访问的 path 有对应的 Controller 承接!
  • 可能3可能这里并没有拦截但是又被其他地方拦截了。请先把这个拦截器给注释掉看看还会不会拦截如果依然拦截那说明不是这个拦截器的锅请仔细查看一下控制台抛出的堆栈信息定位一下到底是哪行代码拦截住这个请求的。
  • 可能4后端拦截的 path 未必是你前端访问的这个path建议先打印一下 path 信息,看看和你预想的是否一致,再做分析。
  • 可能5你写了多个匹配规则请求只越过了第一个规则被其它规则拦下了例如以下代码
registry.addInterceptor(new SaInterceptor(handler -> {
	SaRouter.match("/**").notMatch("/user/doLogin").check(r -> StpUtil.checkLogin());  // 第1个规则 
	SaRouter.match("/**").notMatch("/article/getList").check(r -> StpUtil.checkLogin());  // 第2个规则 
	SaRouter.match("/**").notMatch("/goods/getList").check(r -> StpUtil.checkLogin());  // 第3个规则 
})).addPathPatterns("/**");

以上代码,当你未登录访问 /user/doLogin会被第1条规则越过然后被第2条拦下校验登录然后抛出异常NotLoginExceptionxxx

Q我在配置文件中加了一些关于 Sa-Token 的配置,但是没有生效。

首先有没有生效的最佳判断方式是在main方法中加一个打印看看打印出来的和你配置文件的一致吗

@SpringBootApplication
public class SaTokenApplication {
	public static void main(String[] args) {
		SpringApplication.run(SaTokenApplication.class, args); 
		System.out.println("\n启动成功Sa-Token配置如下" + SaManager.getConfig());
	}
}

如果不一致,请排查:

  • 可能1项目中还存在代码配置而代码配置会覆盖 application.yml 中配置,详细参考:框架配置
  • 可能2你的配置文件名字错误SpringBoot 项目正常情况下配置文件名称应该是:application.ymlapplication.properties
  • 可能3可能是你的配置前缀不对或者配置缩进不对
# 错误示例,多加了 spring 前缀
spring:
	sa-token: 
		token-name: xxx-token
# 错误示例,缩进不对
sa-token: 
token-name: xxx-token
# 正确的应该是以 sa-token 开头
sa-token: 
	token-name: xxx-token

Q我自定义了组件但是好像没有生效

1、情况1是可能组件没有注入成功排查方法为在 main 里打印这个组件是否为自定义的class限定名

@SpringBootApplication
public class SaTokenApplication {
	public static void main(String[] args) {
		SpringApplication.run(SaTokenApplication.class, args); 
		System.out.println(SaManager.getStpInterface());  // 打印全局的 StpInterface 实现类 
	}
}

如果打印出的是你的自定义实现类,则证明注入成功,如果不是,则证明没有注入成功,请排查:

  • 自定义的组件实现类上是否加上了 @Component 注解,只有加上这个注解,组件才会被 Spring 自动实例化并注入。
  • 自定义的组件实现类是否在启动类的同目录或者子目录上,如果不在则无法被 springboot 启动时扫描,扫描不到也就无法注入。
  • 启动类上是否加了 @ComponentScan 注解,导致包扫描范围不正确,请将此注解删除或移动到其它配置类上。

2、情况2是这个组件注入成功了但是还没到执行时机比如 StpInterface 组件,只有在鉴权时才会触发,如果你的代码仅仅是登录校验,就不会执行到这个组件。

Q有时候我不加 Token 也可以通过鉴权,请问是怎么回事?

  • 可能1你访问的这个接口根本就没有鉴权的代码所以可以安全的访问通过。
  • 可能2可能是 Cookie 帮你自动提交了 Token在浏览器或 Postman 中会自动维护Cookie模式如不需要可以在配置文件is-read-cookie: false,然后重启项目再测试一下。

Q一个 User 对象存进 Session 后,再取出来时报错:无法从 User 类型转换成 User 类型?

  • 可能1你的 User 类中途换了包名,导致存进去时和取出来时对不上,无法成功创建实例。
  • 可能2你打开了代码热刷新模式先存进去的对象热刷新后再取出会报错关闭热刷新即可解决。

Q在 SaServletFilter 中调用 SpringMVCUtil.getRequest() 报错非Web上下文无法获取Request

  • 可能1项目中有配置类继承了 extends WebMvcConfigurationSupport
  • 可能2项目中有配置类添加了注解 @EnableWebMvc

解决方案:不要加 @EnableWebMvc,不要 extends WebMvcConfigurationSupport,要 implements WebMvcConfigurer

如果不是以上原因可以加群提供复现demo。

Q我配置了 active-timeout 值,但是当我每次续签时 Redis 中的 ttl 并没有更新,是不是 bug 了?

不更新是正常现象,active-timeout不是根据 ttl 计算的是根据value值计算的value 记录的是该 Token 最后访问系统的时间戳, 每次验签时用:当前时间 - 时间戳 > active-timeout来判断这个 Token 是否已经超时。

Q整合 Redis 时先选择了默认jdk序列化后又改成 jackson 序列化程序开始报错SerializationException

两者的序列化算法不一致导致的反序列化失败,如果要更改序列化方式,则需要先将 Redis 中历史数据清除,再做更新。

Q调用 StpUtil.getExtra("name") 报错:this api is disabled

StpUtil.getExtra(key) 是给 sa-token-jwt 插件提供的不集成这个插件就不能调用这个API如果是普通模式需要存储自定义参数请在 SaSession 上存储

// 在登录时缓存参数
StpUtil.getSession().set("name", "zhangsan");

// 然后我们就可以在任意处获取这个参数 
String name = StpUtil.getSession().getString("name");

Q我加了 Sa-Token 的全局过滤器,浏览器报错跨域了怎么办?

参考:https://blog.csdn.net/shengzhang_/article/details/119928794

Q集成redis后对象模型序列化异常

假设执行如下代码:

@Data
public class User implements Serializable {
    private Long userId;
    private String username;
    private String password;
}

User user = new User();
user.setUserId(10000L);
user.setUsername("oneName");
user.setPassword("onePass");        
StpUtil.getSession().set("userObjKey", user); // 这里报错

报错信息如下:

SerializationException: Could not read JSON: 
Cannot deserialize value of type `java.lang.Long` from Array value (token `JsonToken.START_ARRAY`)

springboot 集成 satoken redis 后, 一旦 springboot 切换版本就有可能出现此问题

原因是redis里面有之前的 satoken 会话数据, 清空 Redis 即可

二、常见疑问

Q登录方法需要我自己实现吗

是的,不同于shiro等框架,Sa-Token不会在登录流程中强插一脚,开发者比对完用户的账号和密码之后,只需要调用StpUtil.login(id)通知一下框架即可

Q框架抛出的权限不足异常我想根据自定义提示信息可以吗

可以,在全局异常拦截器里捕获NotPermissionException,可以通过getPermission()获取没有通过认证的权限码,可以据此自定义返回信息

Q我的项目权限模型不是RBAC模型很复杂可以集成吗

无论什么模型只要能把一个用户具有的所有权限塞到一个List里返回给框架就能集成

QStpInterface 接口的 方法,在什么时候执行?

每次鉴权时执行,例如你调用了 StpUtil.checkgetPermission("xxx") 方法,框架就会调用底层的 StpInterface#getPermissionList 方法来获取权限数据。

如果你的 getPermissionList 里有读数据库的代码,那么你每鉴一次权,系统将访问一次数据库。如果要减小性能消耗,可以把权限数据放在缓存中,参考:把权限放在缓存里

Q当我配置不并发登录时每次登陆都会产生一个新的 Token旧 Token 依然被保存在 Redis 中,框架为什么不删除呢?

首先,不删除旧 Token 的原因是为了在旧 Token 再次访问系统时提示他:已被顶下线。

而且这个 Token 不会永远留在 Redis 里,在其 TTL 到期后就会自动清除,如果你想让它立即消失,可以:

  • 方法一:配置文件把 is-concurrentis-share 都打开,这样每次登陆都会复用以前的旧 Token就不会有废弃 Token 产生了。
  • 方法二:每次登录前把先调用注销方法 StpUtil.logout(10001) ,把这个账号的旧登录都给清除了。
  • 方法三:写一个定时任务查询 Redis 值进行删除。

Q我使用过滤器鉴权 or 全局拦截器鉴权,结果 Swagger 不能访问了,我应该排除哪些地址?

尝试加上排除 "/swagger-resources/**", "/webjars/**", "/v2/**", "/swagger-ui.html/**" ,"/doc.html/**","/error","/favicon.ico"

不同版本可能会有所不同,其实在前端摁一下 F12 看看哪个 url 报错排除哪个就行了(另附:注解鉴权是不需要排除的,因为 Swagger 本身也没有使用 Sa-Token 的注解)

QSaRouter.match 有多个路径需要排除怎么办?

可以点进去源码看一下,SaRouter.match方法有多个重载,可以放一个集合, 例如:

SaRouter.match("/**").notMatch("/login", "/reg").check(r -> StpUtil.checkLogin());

Q为什么StpUtil.login() 不能直接写入一个User对象

StpUtil.login()只是为了给当前会话做个唯一标记,通常写入UserId即可如果要存储User对象可以使用StpUtil.getSession()获取Session对象进行存储。

Q前后台分离模式下和普通模式有何不同

主要是失去了Cookie无法自动化保存和提交token秘钥,可以参考章节:前后台分离

Q前后台分离时前端提交的 header 参数是叫 token 还是 satoken 还是 tokenName

默认是satoken如果想换一个名字更改一下配置文件的tokenName即可。

Q一个账号拥有哪些权限可以做成动态的吗

权限本来就是动态的,框架预留的 StpInterface 接口,就是为了让你可以写任意代码来获取数据

Q路由拦截鉴权可以做成动态的吗

参考:把路由拦截鉴权动态化

Q我不想让框架自动操作Cookie怎么办

在配置文件将isReadCookie值配置为false

Q怎么关掉每次启动时的字符画打印

在配置文件将isPrint值配置为false

QStpUtil.getSession()必须登录后才能调用吗?如果我想在用户未登录之前存储一些数据应该怎么办?

StpUtil.getSession()获取的是User-Session必须登录后才能使用如果需要在未登录状态下也使用Session功能请使用Token-Session
步骤:先在配置文件里将tokenSessionCheckLogin配置为false,然后通过StpUtil.getTokenSession()获取Session

Q我只使用header来传输token还需要打开Cookie模式吗

不需要如果只使用header来传输token可以在配置文件关闭Cookie模式isReadCookie=false

Q我想让用户修改密码后立即掉线重新登录应该怎么做

框架内置 [强制指定账号下线] 的APi在执行修改密码逻辑之后调用此API即可: StpUtil.logout()

Q代码鉴权、注解鉴权、路由拦截鉴权我该如何选择

这个问题没有标准答案,这里只能给你提供一些建议,从鉴权粒度的角度来看:

  1. 路由拦截鉴权:粒度最粗,只能粗略的拦截一个模块进行权限认证
  2. 注解鉴权:粒度较细,可以详细到方法级,比较灵活
  3. 代码鉴权粒度最细不光可以控制到方法级甚至可以if语句决定是否鉴权

So从鉴权粒度的角度来看需要针对一个模块鉴权的时候就用路由拦截鉴权需要控制到方法级的时候就用注解鉴权需要根据条件判断是否鉴权的时候就用代码鉴权

QSa-Token的全局过滤器我应该怎么指定它的优先级呢

为了保证相关组件能够及时初始化,框架默认给过滤器注册的优先级为-100如果你想更改优先级直接在注册过滤器的方法上加上 @Order(xxx) 即可覆盖框架的默认配置

Q还是有不明白到的地方?

请在giteegithub 提交 issues或者加入qq群交流群链接在首页