登录-10.Filter-登录校验过滤器

news/2025/2/25 14:43:38

一.登录校验过滤器的实现思路

我们要实现登录校验过滤器,就要首先明白登录校验过滤器的实现思路。登录校验过滤器是用来实现登录校验的。那么首先思考第一个问题,所有的请求都需要校验吗?

答案是否定的,因为login请求就不需要过滤器校验,因为登录请求本身就不携带JWT令牌,登录的目的就是为了获取JWT令牌用于后续的校验,如果login请求也需要校验的话,那么将没有用户可以登录成功进而访问服务器中的资源。因此登录请求"/login"是不需要校验的,要直接放行。

下面思考第二个问题,当我们拦截到请求后,什么情况下才可以放行?答案是当拦截到的请求中的请求头header所携带的JWT令牌存在且有效时才能够放行执行业务操作,只要不存在或者无效,那么就不会放行执行操作,从而跳转到登录页面。

基于以上分析,我们绘制出登录校验过滤器的流程分析图。

二.代码实现

package com.gjw.filter;


import com.alibaba.fastjson.JSONObject;
import com.gjw.pojo.Result;
import com.gjw.util.JwtUtils;
import jakarta.servlet.*;
import jakarta.servlet.annotation.WebFilter;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import lombok.extern.slf4j.Slf4j;
import org.springframework.util.StringUtils;

import java.io.IOException;

@Slf4j
@WebFilter(urlPatterns = "/*")
public class LoginCheckFilter implements Filter {


    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request = (HttpServletRequest) servletRequest;
        HttpServletResponse response = (HttpServletResponse) servletResponse;

        // 1.获取请求的url
        String requestURL = request.getRequestURL().toString();
        log.info("获取请求的URL:{}",requestURL);

        // 2.判断url中是否包含"login"
        if (requestURL.contains("login")) {
            log.info("登录操作,直接放行......");
            // 如果包含,直接放行
            filterChain.doFilter(servletRequest,servletResponse);
            return;
        }

        // 3.不包含则获取JWT令牌并检验
        String jwt = request.getHeader("token");

        // 4.检验JWT令牌是否存在,如果不存在,则返回错误结果(未登录)
        if(!StringUtils.hasLength(jwt)) {   // jwt不存在或者为null,返回false
            log.info("请求头token为空,返回登录页面......");
            Result error = Result.error("NOT_LOGIN");
            String notLogin = JSONObject.toJSONString(error);// 没有RestController注解,无法将直接return的数据以JSON格式返回,需要强转为JSON  手动转换:对象----->JSON格式的数据
            response.getWriter().write(notLogin);  // 通过response对象返回JSON格式的数据
            return;
        }

        // 5.如果存在,校验JWT令牌是否正确,如果解析失败,则返回错误结果(未登录)
        try {
            JwtUtils.parseJWT(jwt);     // 解析成功放行
        } catch (Exception e) {     // 失败捕获异常
            e.printStackTrace();
            log.info("解析失败,返回未登录的错误信息......");
            Result error = Result.error("NOT_LOGIN");
            String notLogin = JSONObject.toJSONString(error);// 没有RestController注解,无法将直接return的数据以JSON格式返回,需要强转为JSON  手动转换:对象----->JSON格式的数据
            response.getWriter().write(notLogin);  // 通过response对象返回JSON格式的数据
            return;
        }
        // 6.JWT令牌存在且解析成功,放行
        log.info("令牌合法,放行。");
        filterChain.doFilter(servletRequest,servletResponse);

    }
}

 1.获取请求的url

我们首先要获取到请求的url地址,为了获取请求的url地址,需要将ServletRequest和ServletResponse强转为HttpServletRequest和HttpServletResponse。强转后通过调用HttpServletRequest的getRequestURL()方法获取URL路径。

2.判断url中是否包含"login"

获取到URL路径后,接下来我们首先要判断路径中是否包含关键字"login",如果包含,那么证明该请求是一个登录请求,直接放行即可。使用doFilter方法将HttpServletRequest和HttpServletResponse对象传递进去。然后使用return直接结束该方法。

3.不包含则获取JWT令牌并检验

如果不包含关键字"login",那么证明该请求不是一个登录请求,那么就进行下面两部判断:

首先获取该请求中请求头header的token字段,从而获取到JWT令牌,然后判断JWT令牌是否存在。使用request.getHeader("token")来获取JWT令牌的字符串。

4.检验JWT令牌是否存在,如果不存在,则返回错误结果(未登录)

如果不存在,即使用StringUtils工具类的hasLength方法(空串或者null返回false,有值返回true)判断结果为空串或者null,取反后(!)为true。

我们与前端约定好的

那么调用Result结果封装类中的error方法,传递一个"NOT_LOGIN"。但是我们要响应给前端的是一个JSON格式的数据,那么如何将这个Result对象转为JSON再响应给前端呢?在controller当中我们可以使用注解@RestController,会自动将方法的返回值转为JSON格式的数据返回回去,但是现在是在Filter过滤器当中,因此要手动转换。我们可以使用alibaba提供的fastJSON的工具包,首先引入依赖

<dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>2.0.52</version>
</dependency>

然后使用JSONObject将对象转为JSON格式的字符串,然后我们将该字符串相应给浏览器。通过响应对象response调用其中的getWriter()获取一个输出流,再调用里面的writer()方法将要响应未登陆的数据notLogin传递给前端。然后直接结束方法,跳转到登录页面。

5.如果存在,校验JWT令牌是否正确,如果解析失败,则返回错误结果(未登录)

如果令牌存在,那么首先校验令牌的合法性,如果合法,直接放行。如果不合法,那么和上面的代码逻辑一样。因此我们需要使用try/catch捕获异常信息,如果未捕获到异常,那么直接放行;如果捕获到异常,那么执行和上面一样的代码逻辑。

6.JWT令牌存在且解析成功,放行

令牌存在且解析成功,执行doFilter方法放行即可。


http://www.niftyadmin.cn/n/5865618.html

相关文章

计算机毕业设计SpringBoot+Vue.js明星周边产品销售网站(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

C++判断回文字符串

C判断回文字符串 1、使用reverse()函数反转2、使用循环&#xff0c;借助变量首尾比较3、双指针方法1&#xff1a;while循环方法2&#xff1a;for循环 1、使用reverse()函数反转 思路&#xff1a; 首先读取一个字符串并存储在变量 s 中 将字符串 s 复制到另一个字符串变量 s1 …

【Godot4.3】基于绘图函数的矢量蒙版效果与UV换算

概述 在设计圆角容器时突发奇想&#xff1a; 将圆角矩形的每个顶点坐标除以对应圆角矩形所在Rect2的size&#xff0c;就得到了顶点对应的UV坐标。然后使用draw_colored_polygon&#xff0c;便可以做到用图片填充圆角矩形的效果。而且这种计算的效果就是图片随着其填充的图像缩…

akka现有的分布式定时任务框架总结

根据你的需求&#xff0c;以下是一些基于 Akka 实现的分布式定时任务框架&#xff0c;以及相关的 GitHub 项目推荐&#xff1a; 1. Openjob Openjob 是一个基于 Akka 架构的新一代分布式任务调度框架&#xff0c;支持多种定时任务、延时任务、工作流设计&#xff0c;采用无中…

CES Asia 2025,技术创新成招商强引力

近日&#xff0c;2025年中国经济白皮书正式发布&#xff0c;为第七届亚洲消费电子技术贸易展&#xff08;CES Asia 2025&#xff09;的招商工作注入强大动力&#xff0c;其在技术创新方面展现出的显著优势&#xff0c;成为吸引全球科技企业的关键因素。 白皮书全面展示了中国在…

C#文件操作

文件类File 常用方法介绍Open()打开文件Create()创建文件Copy()复制文件Delete()删除文件Exists()判断文件是否存在Move()移动文件Replace()替换文件AppendAllText()新建并添加文本ReadAllText()打开并读取文本内容 1.读取文件 1.创建一个Filestream 对象 (引入System.IO命名…

Linux版本控制器Git【Ubuntu系统】

文章目录 **前言**一、版本控制器二、Git 简史三、安装 Git四、 在 Gitee/Github 创建项目五、三板斧1、git add 命令2、git commit 命令3、git push 命令 六、其他1、git pull 命令2、git log 命令3、git reflog 命令4、git stash 命令 七、.ignore 文件1、为什么使用 .gitign…

Deepseek引爆AI热潮 防静电地板如何守护数据中心安全

近期&#xff0c;Deepseek的爆火将人工智能推向了新的高度&#xff0c;也引发了人们对AI背后基础设施的关注。作为AI运行的“大脑”&#xff0c;数据中心承载着海量数据的存储、处理和传输&#xff0c;其安全稳定运行至关重要。而在这背后&#xff0c;防静电地板扮演着不可或缺…