使用SpringSecurity下,发生重定向异常

news2025/4/12 15:04:01

使用SpringSecurity下,发生空转异常

image-20250408171305656

环境信息:

Spring Boot 3.4.4 , jdk 17 , springSecurity 6.4.4

问题背景:

没有自定义controller ,改写了login 页面,并且进行了成功后的跳转处理,发现无法进入到login.html 页面,并且报错。但是如果不进行改写,login 则会自动被跳转到默认登陆页面,完成正常拦截。http://localhost:9999/bootscoder/login 比起我的改写后的少了一个 后缀

问题复现:

核心代码:

配置文件:

// 自定义表单登录配置
        http.formLogin(form -> {
            form.loginPage("/login.html")  // 自定义登录页面
                    .usernameParameter("username")
                    .passwordParameter("password")
                    .loginProcessingUrl("/login")//登录路径,表单向该路径提交,提交后自动执行UserDetailsService的方法
                    .successForwardUrl("/main.html") // 使用重定向实现登录成功后的跳转
//                    .successHandler(new MyLoginSuccessHandler()) // 登录成功处理器
                    .failureForwardUrl("/fail.html");
        });

        // 设置放行的资源
        http.authorizeHttpRequests(authz -> {
            // 放行登录页、处理路径、失败页和静态资源
            authz.requestMatchers("/login.html", "/fail.html", "/main.html").permitAll();
            authz.requestMatchers("/css/*.css", "/js/*.js", "/img/**").permitAll();
            // 其余请求要求认证
            authz.anyRequest().authenticated();
        });
        // 关闭csrf防护(仅供测试使用,生产环境需谨慎关闭)
//        http.csrf(csrf -> csrf.disable());

登陆页面:

<div class="container">
    <h1>用户登录</h1>
    <form action="/bootscoder/login" method="post">
        <input type="hidden"  th:value = "${_csrf.token}" name="_csrf" th:if="${_csrf}">

        <div class="form-group">
            <label for="username">用户名</label>
            <input type="text" id="username" name="username" placeholder="请输入用户名" value="bootscoder" required>
        </div>
        <div class="form-group">
            <label for="password">密码</label>
            <input type="password" id="password" name="password" placeholder="请输入密码" value="root" required>
        </div>
        <input type="checkbox" name="remember-me" value="true"/>记住我</br>
        <button type="submit">登录</button>
    </form>
</div>

yml前缀配置:

server:
  port: 9999
  servlet:
    context-path: /bootscoder  # 所有接口前缀为 /bootscoder

错误信息:

This page isn’t working

localhost redirected you too many times.

Try deleting your cookies

ERR_TOO_MANY_REDIRECTS

分析原因

我的资源放到的是templates中, 也就是说 security 默认的登陆页面在static中。
所以 当我 访问templates下的路径,由于没有写contoller 接口,会发生意外跳转,被拦截后,跳转到static 的默认登陆,然后由于我改写了登陆page,所以又会跳回来,这样导致了 重定向异常。

Spring Security 在配置登录表单时,如果未能找到合适的页面映射(比如你没有正确配置自定义登录页面的访问路径),就可能会触发默认的登录页面(默认在 static 目录下)与自定义页面之间的冲突,从而产生重定向循环。

解决方案

你的分析是正确的。如果你的登录页面放在了 templates 目录下,而没有额外通过 Controller 将其渲染,那么 Spring Boot 默认配置下实际上并不会访问该模板,而是走了静态资源(static 目录)中的默认登录页面,从而产生了重定向循环。

下面有两种解决方案:


方案一:将登录页面移动到 static 目录

如果你不需要使用模板引擎渲染(例如 Thymeleaf 动态处理数据),可以直接把 login.html 文件放到 src/main/resources/static 目录下,这样资源可以直接被以静态资源的形式访问而不会依赖 Controller。

注意:这种方式下,你在 HTML 中写死的路径也需要注意避免重复拼接 context path。例如:

<form th:action="@{/login}" method="post">

<form action="/login" method="post">

(注意不要写成 /bootscoder/login,因为 Spring Boot 会自动处理 context path)


注意⚠️:

不能使用MVC模板引擎:静态资源文件夹中的HTML页面不会被模板引擎(如Thymeleaf)处理,所以不能使用th:action等Thymeleaf标签。

页面跳转方式变化

  • 在模板引擎处理的页面中,可以通过控制器方法返回视图名称来实现页面跳转
  • 在静态页面中,需要使用JavaScript进行页面重定向(window.location.href)或通过AJAX请求与后端交互

表单提交方式

  • 静态页面中的表单提交需要通过JavaScript拦截提交事件,使用fetchXMLHttpRequest等API发送请求到后端
  • 不能依赖Spring MVC的表单处理机制直接提交到控制器方法

方案二:为模板页面添加 Controller(推荐)

如果你需要利用模板引擎(如 Thymeleaf)进行页面渲染,则需要提供一个 Controller 映射 GET 请求去返回该模板。例如,你可以这样做:

package com.example.demo.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;

@Controller
public class LoginController {

    @GetMapping("/login.html")
    public String login() {
        // 返回 templates/login.html 渲染结果
        return "login";  // 注意:返回值 "login" 对应 src/main/resources/templates/login.html
    }
}
安全配置修改

同时,确保你的 Spring Security 配置中放行登录页面、登录处理 URL 以及失败页面,如下所示:

http.formLogin(form -> {
    form.loginPage("/login.html")  // 自定义登录页面,由 Controller 渲染
        .usernameParameter("username")
        .passwordParameter("password")
        .loginProcessingUrl("/login") // 处理登录请求,实际为 /bootscoder/login
        .defaultSuccessUrl("/main.html", true)
        .failureUrl("/fail.html");
});

http.authorizeHttpRequests(authz -> {
    // 放行登录页、登录处理 URL、失败页和静态资源
    authz.requestMatchers("/login.html", "/login", "/fail.html", "/main.html").permitAll();
    authz.requestMatchers("/css/*.css", "/js/*.js", "/img/**").permitAll();
    authz.anyRequest().authenticated();
});

// 如果需要调试,可以先关闭 CSRF(仅用于测试环境)
// http.csrf(csrf -> csrf.disable());

同时,为静态页面(如 fail.htmlmain.html)也建议确认它们能被正确访问,如果你没有为它们提供 Controller 渲染,那就把这类页面放入 static 目录中。

小结

  • 放在 static 目录:如果不依赖模板渲染,可以直接将 login.html 放至 static,避免因找不到自定义 Controller 而触发 Security 默认登录页。
  • 使用 Controller 渲染模板:如果需要把页面放在 templates 下利用 Thymeleaf 渲染,则必须提供映射该 GET 请求的 Controller,否则 Spring Security 可能依然会使用默认静态的登录页面,造成重定向循环。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2330952.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Elasticsearch | ES索引模板、索引和索引别名的创建与管理

关注&#xff1a;CodingTechWork 引言 在使用 Elasticsearch (ES) 和 Kibana 构建数据存储和分析系统时&#xff0c;索引模板、索引和索引别名的管理是关键步骤。本文将详细介绍如何通过 RESTful API 和 Kibana Dev Tools 创建索引模板、索引以及索引别名&#xff0c;并提供具…

力扣hot100_回溯(2)_python版本

一、39. 组合总和&#xff08;中等&#xff09; 代码&#xff1a; class Solution:def combinationSum(self, candidates: List[int], target: int) -> List[List[int]]:ans []path []def dfs(i: int, left: int) -> None:if left 0:# 找到一个合法组合ans.append(pa…

LPDDR4内存颗粒命名规则全解析:三星、镁光、海力士、南亚、长鑫等厂商型号解码与选型指南

由于之前DDR的系列选型文章有很好的反馈&#xff0c;所以补充LPDDR4低功耗内存的选型和命名规则&#xff0c;总结了目前市面上常用的内存&#xff0c;供硬件工程师及数码爱好者参考。 在智能手机、平板电脑和低功耗设备中&#xff0c;LPDDR4 SDRAM凭借其高带宽、低功耗特性成为…

【杂谈】Godot4.4导出到Android平台(正式导出)

学博而后可约&#xff0c;事历而后知要。 目录 一、准备二、Gradle构建三、配置Java SDK四、配置Android SDK五、配置密钥 一、准备 本文在前文【杂谈】Godot4.4导出到安卓平台&#xff08;调试导出&#xff09;的基础上&#xff0c;进行正式导出。调试导出并不是真正的编译导…

基于AI设计开发出来的业务系统是什么样的?没有菜单?没有表格?

基于AI设计开发出的业务系统仍然会包含菜单、表格等传统UI元素&#xff0c;但AI技术会显著改变它们的实现方式和交互逻辑。以下是具体分析&#xff1a; 一、传统元素的持续存在 功能刚需性 • 菜单承担着系统导航的核心功能&#xff0c;表格则是结构化数据展示的基础载体。根…

数字足迹管理(DFM):你的网络隐身指南

数字足迹管理&#xff08;DFM&#xff09;&#xff1a;你的网络隐身指南 ‌你可能不知道&#xff0c;你的姓名、电话、住址正在网上被“明码标价”‌ ——而这一切&#xff0c;可能只是因为你点过外卖、寄过快递&#xff0c;甚至注册过一个网站。 一、什么是数字足迹管理&#…

如何避免“过度承诺”导致的验收失败

如何避免“过度承诺”导致的验收失败&#xff1f;关键在于&#xff1a; 评估可行性、设置合理目标、高频沟通反馈、阶段性验收、做好风险管理。其中设置合理目标至关重要&#xff0c;很多团队往往在项目初期为迎合客户或领导而报出“最理想方案”&#xff0c;忽略了资源、技术及…

紧跟数字人热潮:123 数字人分身克隆系统源码部署与风口洞察

在当今数字化浪潮中&#xff0c;数字人技术无疑已成为最具活力与潜力的领域之一&#xff0c;正以迅猛之势席卷多个行业&#xff0c;重塑着人们的交互方式与商业运作模式。C 站作为技术交流的前沿阵地&#xff0c;汇聚了众多关注前沿科技的开发者与技术爱好者&#xff0c;今天来…

QT控件 修改QtTreePropertyBrowser自定义属性编辑器源码,添加第一列标题勾选,按钮,右键菜单事件等功能

头阵子遇到一个需要修改QtTreePropertyBrowser控件的需求&#xff0c;QT开发做这么久了&#xff0c;这个控件倒是第一次用&#xff0c;费了点时间研究&#xff0c;在这里做个简单的总结。 QtTreePropertyBrowser控件 是 Qt 解决方案 (Qt Solutions) 中的一个组件&#xff0c;用…

开源模型应用落地-模型上下文协议(MCP)-从数据孤岛到万物互联(一)

一、前言 当开发者还在为每个AI工具编写臃肿的API适配器时&#xff0c;一场关于「连接」的技术革命已悄然降临。模型上下文协议&#xff08;MCP&#xff09;正在用一套全新的交互语法&#xff0c;重新定义人工智能与物理世界的对话方式。MCP协议如同为AI系统装上了“万能接口”…

【区块链安全 | 第三十八篇】合约审计之获取私有数据(二)

文章目录 前言漏洞代码代码审计攻击步骤修复建议审计思路 前言 在【区块链安全 | 第三十七篇】合约审计之获取私有数据&#xff08;一&#xff09;中&#xff0c;介绍了私有数据、访问私有数据实例、Solidity 中的数据存储方式等知识&#xff0c;本文通过分析具体合约代码进行…

mac 苍穹外卖 后端初始 SkyApplication 报错

报错内容 java: java.lang.NoSuchFieldError: Class com.sun.tools.javac.tree.JCTree$JCImport does not have member field com.sun.tools.javac.tree.JCTree qualid deepseek 解决 打开 File > Project Structure > Project SDK, 选择 JDK17。我没有 JDK17就下载了一…

Proximal Policy Optimization (PPO)

2.1 策略梯度方法 策略梯度方法计算策略梯度的估计值并将其插入到随机梯度上升算法中。最常用的梯度估计器的形式如下&#xff1a; g ^ E t [ ∇ θ log ⁡ π θ ( a t ∣ s t ) A ^ t ] (1) \hat{g} \mathbb{E}_t \left[ \nabla_{\theta} \log \pi_{\theta}(a_t | s_t) \h…

微信小程序:动态表格实现,表头单元格数据完全从data中获取,宽度自定义,自定义文本框,行勾选,样式效果,横向滚动表格(解决背景色不足的问题)等

一、样式效果 二、代码 1、wxml <view class"line flex flex-center"><view class"none" wx:if"{{info.length 0}}">暂无料号</view><view wx:else class"table-container"><!-- 动态生成表头 -->&…

python-Leetcode 65.搜索旋转排序数组

题目&#xff1a; 整数数组nums按升序排列&#xff0c;数组中的值互不相同 在传递给函数之前&#xff0c;nums在预先未知的某个小标K上进行了旋转&#xff0c;使数组变为[nums[k], nums[k1], ..., nums[n-1], nums[0], nums[1], ..., nums[k-1]]&#xff0c;小标从0开始计数。…

Django学习记录-1

Django学习记录-1 虽然网上教程都很多&#xff0c;但是感觉自己记录一下才属于自己&#xff0c;之后想找也方面一点&#xff0c;文采不佳看的不爽可绕道。 参考贴 从零开始的Django框架入门到实战教程(内含实战实例) - 01 创建项目与app、加入静态文件、模板语法介绍&#xff…

K8s私有仓库拉取镜像报错解决:x509 certificate signed by unknown authority

前言 在Kubernetes环境中使用自签名证书的私有Harbor镜像仓库时&#xff0c;常会遇到证书验证失败的问题。本文将详细讲解如何解决这个常见的证书问题。 环境信息&#xff1a; Kubernetes版本&#xff1a;1.28.2容器运行时&#xff1a;containerd 1.6.20私有仓库&#xff1a…

LabVIEW 长期项目开发

LabVIEW 凭借其图形化编程的独特优势&#xff0c;在工业自动化、测试测量等领域得到了广泛应用。对于长期运行、持续迭代的 LabVIEW 项目而言&#xff0c;其开发过程涵盖架构设计、代码管理、性能优化等多个关键环节&#xff0c;每个环节都对项目的成功起着至关重要的作用。下面…

蓝桥杯真题——好数、R格式

目录 蓝桥杯2024年第十五届省赛真题-好数 【模拟题】 题目描述 输入格式 输出格式 样例输入 样例输出 提示 代码1&#xff1a;有两个案例过不了&#xff0c;超时 蓝桥杯2024年第十五届省赛真题-R 格式 【vector容器的使用】 题目描述 输入格式 输出格式 样例输入…

Windows版-RabbitMQ自动化部署

一键完成Erlang环境变量配置&#xff08;ERLANG_HOME系统变量&#xff09;‌ 一键完成RabbitMQ环境变量配置&#xff08;RabbitMQ系统变量&#xff09;‌ 实现快速安装部署RabbitMQ PS&#xff1a; 需提前下载安装&#xff1a; - otp_win64_25.0.exe (Erlang) - rabbit…