SpringSecurity認證授權完整流程

SpringSecurity認證流程:loadUserByUsername()方法內部實現。

實現步驟:

  1. 構建一個自定義的service接口,實現SpringSecurity的UserDetailService接口。
  2. 建一個service實現類,實現此loadUserByUsername方法。
  3. 調用登錄的login接口,會經過authenticationManager.authenticate(authenticationToken)方法。此方法會調用loadUserByUsername方法。
  4. 方法內部做用戶信息的查詢,判斷用戶名和密碼是否正確,這是第一道認證。
@Service
@RequiredArgsConstructor
public class UserDetailsServiceImpl implements UserDetailsService {private final SysUserMapper sysUserMapper;//用戶登錄請求/login,自動調用方法//根據用戶名獲取用戶信息//UserDetails 存儲用戶信息,包括用戶名,密碼,權限@Overridepublic UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {LambdaQueryWrapper<SysUser> wrapper = new LambdaQueryWrapper<>();wrapper.eq(SysUser::getUsername, username);SysUser sysUser = sysUserMapper.selectOne(wrapper);if (Objects.isNull(sysUser)){throw new UsernameNotFoundException("用戶名不存在");}//認證成功回UserDetails對象return new LoginUser(sysUser);}}

@ToString
public class LoginUser implements UserDetails {private SysUser sysUser;public LoginUser(SysUser sysUser) {this.sysUser = sysUser;}// 權限@Overridepublic Collection<? extends GrantedAuthority> getAuthorities() {return List.of();}@Overridepublic String getPassword() {return sysUser.getPassword();}@Overridepublic String getUsername() {return sysUser.getUsername();}// 賬號是否過期@Overridepublic boolean isAccountNonExpired() {return true;}// 賬號是否被鎖定@Overridepublic boolean isAccountNonLocked() {return true;}// 密碼是否過期@Overridepublic boolean isCredentialsNonExpired() {return true;}// 賬號是否可用@Overridepublic boolean isEnabled() {return true;}
}

5.如果沒有查到信息就拋出異常。

6.如果查到信息了再接著查用戶的權限信息,返回權限信息到loginUser實體。

7.此實體實現了SpringSecurity自帶的userDetail接口。實現了getAuthorities方法。

.8每次查詢權限都會調用此方法。

9.查詢到的權限,會被返回到login接口。進行后續操作。

10.如果認證通過,通過身份信息中的userid生產一個jwt。

11.把完整的用戶信息作為value,token作為key存入redis。

@RestController
@Tag(name = "認證模塊", description = "認證模塊")
@RequestMapping("/auth")
@RequiredArgsConstructor
public class AuthController {//注入AuthenticationManager(認證管理器)private final AuthenticationManager authenticationManager;private final JwtUtils jwtUtils;@PostMapping("/login")@Operation(summary = "登錄")public Result login(@RequestParam("username") String username, @RequestParam("password") String password) {System.out.println(username + password);//登錄邏輯//調用UserDetailsService.loadUserByUsername方法獲取//不能直接調用,需要通過AuthenticationManager進行認證Authentication authentication = new UsernamePasswordAuthenticationToken(username, password);Authentication authenticate = null;try {authenticate = authenticationManager.authenticate(authentication);} catch (BadCredentialsException e) {return Result.failed(ResultCode.USERNAME_OR_PASSWORD_ERROR);}//認證成功方法tokenString token=jwtUtils.generateToken(authenticate);return Result.success(token);}
}

@Configuration
@EnableWebSecurity // 開啟web安全
@EnableMethodSecurity
@RequiredArgsConstructor
public class SecurityConfig {private final IgnoredUrl ignoredUrl;/*** 配置認證管理器 AuthenticationManager* 作用:用于身份認證* 參數:UserDetailsService, PasswordEncoder*/@Beanpublic AuthenticationManager authenticationManager(UserDetailsService userDetailsService, PasswordEncoder passwordEncoder) {DaoAuthenticationProvider provider = new DaoAuthenticationProvider();provider.setUserDetailsService(userDetailsService);provider.setPasswordEncoder(passwordEncoder);return new ProviderManager(provider);}/*** 密碼編碼器** @return*/@Beanpublic PasswordEncoder passwordEncoder() {return new BCryptPasswordEncoder();}@Beanpublic SecurityFilterChain securityFilterChain(HttpSecurity http) throws Exception {//關閉csrf防護,否則回導致登錄失敗http.csrf(a -> a.disable()); //禁用SCRF//配置安全攔截規則http.authorizeHttpRequests(req ->req.requestMatchers(ignoredUrl.getUrls()).permitAll().anyRequest().authenticated());/*** 配置登錄頁*/http.formLogin(form -> form.loginPage("/").successForwardUrl("/index") //登錄成功跳轉頁面.loginProcessingUrl("/login")//登錄處理url.failureForwardUrl("/error") //登錄失敗跳轉頁面//                .usernameParameter("name")//自定義用戶名參數//                .passwordParameter("password")//自定義密碼參數);return http.build();}
}
@Component
public class JwtUtils {@Value("${jwt.secret}")private String secret;@Value("${jwt.expiration}")private Integer expiration;/****/public String generateToken(Authentication authentication) {Date now = new Date();Date expirationDate = DateUtil.offsetSecond(now, expiration);Map<String, Object> claims = new HashMap<>();claims.put("username", authentication.getName());//用戶名claims.put("exp", expirationDate);//        claims.put();return JWTUtil.createToken(claims, secret.getBytes());}
}
@Component
@Data
@ConfigurationProperties(prefix = "security.ignored")
@ToString
public class IgnoredUrl {private String[] urls;
}

登錄成功

登陸失敗

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/71685.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/71685.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/71685.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

本地部署DeepSeek R1大數據模型知識庫

DeepSeek-V3 的綜合能力 DeepSeek-V3 在推理速度上相較歷史模型有了大幅提升。在目前大模型主流榜單中&#xff0c;DeepSeek-V3 在開源模型中位列榜首&#xff0c;與世界上最先進OpenAI 閉源模型不分伯仲。 1、下載Ollama運行大數據庫 Ollama支持 Llama 3.3, DeepSeek-R1, Phi-…

云端秘境:EC2的奇幻之旅

在一個神秘的云端世界里&#xff0c;流傳著一個傳說——只要掌握了 EC2&#xff08;Elastic Compute Cloud&#xff09; 的奧秘&#xff0c;就能召喚出強大的騎士軍團&#xff0c;在云端之上建造屬于自己的帝國。年輕的程序法師 艾倫&#xff08;Allen&#xff09;&#xff0c;…

【javaEE】多線程(基礎)

1.????前言~&#x1f973;&#x1f389;&#x1f389;&#x1f389; Hello, Hello~ 親愛的朋友們&#x1f44b;&#x1f44b;&#xff0c;這里是E綿綿呀????。 如果你喜歡這篇文章&#xff0c;請別吝嗇你的點贊????和收藏&#x1f4d6;&#x1f4d6;。如果你對我的…

一周學會Flask3 Python Web開發-WTForms表單驗證

鋒哥原創的Flask3 Python Web開發 Flask3視頻教程&#xff1a; 2025版 Flask3 Python web開發 視頻教程(無廢話版) 玩命更新中~_嗶哩嗶哩_bilibili 我們可以通過WTForms表單類屬性的validators屬性來實現表單驗證。 常用的WTForms驗證器 驗證器說明DataRequired(messageNo…

STM32標準庫代碼詳解之GPIO

GPIO的初始化代碼如下&#xff1a; /*開啟時鐘*/ RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //開啟GPIOA的時鐘&#xff0c;使用外設必須開啟/*GPIO初始化*/ GPIO_InitTypeDef GPIO_InitStructure; //定義結構體變量 GPIO_InitStructure.GPIO_Mode GPIO_Mo…

華為歐拉系統 Tomcat 安裝詳解

1. 安裝或確認安裝 Java Tomcat 需要 Java 環境(JDK 或 JRE)才能運行。如果系統尚未安裝 Java,可以使用以下命令安裝 OpenJDK: # 更新軟件包索引 yum update -y# 安裝 OpenJDK 21(可根據需求安裝其他版本,如 8、11、17 等) yum install -y java-21-openjdk java-21-op…

第六課:數據庫集成:MongoDB與Mongoose技術應用

本文詳細介紹了如何在Node.js應用程序中集成MongoDB數據庫&#xff0c;并使用Mongoose庫進行數據操作。我們將涵蓋MongoDB在Ubuntu 20系統中的安裝、Bash命令的CRUD操作、Mongoose數據建模&#xff08;Schema/Model&#xff09;、關聯查詢與聚合管道&#xff0c;以及實戰案例—…

大數據學習(56)-Impala

&&大數據學習&& &#x1f525;系列專欄&#xff1a; &#x1f451;哲學語錄: 承認自己的無知&#xff0c;乃是開啟智慧的大門 &#x1f496;如果覺得博主的文章還不錯的話&#xff0c;請點贊&#x1f44d;收藏??留言&#x1f4dd;支持一下博主哦&#x1f91…

【C#】async與await介紹

1. 實例1 1.1 代碼 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace ConsoleApp1 {class Program{static void Main(string[] args){Method1();Method2();Console.ReadKey();}public static…

【大模型基礎_毛玉仁】1.1 基于統計方法的語言模型

【大模型基礎_毛玉仁】1.1 基于統計方法的語言模型 1.語言模型基礎1.1 基于統計方法的語言模型1.1.1 n-grams 語言模型1.1.2 n-grams 的統計學原理 1.語言模型基礎 語言是概率的。語言模型&#xff08;LanguageModels, LMs&#xff09;旨在準確預測語言符號的概率。 將按照語…

JavaScript(最后一個元素的索引就是數組的長度減 1)array.length - 1

在不同的編程語言中&#xff0c;表示數組中最后一個元素的方法略有不同&#xff0c;但基本思路都是利用數組的長度或索引來實現。 以下是一些常見編程語言中獲取數組最后一個元素的方法&#xff1a; 1. JavaScript: 使用 array.length - 1 索引: 這是最常見和傳統的方法。Java…

SpringCloud中使用服務名調用場景總結

一 API Gateway 在 Spring Cloud API Gateway 中&#xff0c;通過使用 lb:// 前綴可以實現基于服務名的負載均衡路由。以下是具體的配置和使用方法&#xff1a; 1. 配置服務注冊與發現 確保你的服務已經注冊到服務注冊中心&#xff08;如 Nacos 或 Eureka&#xff09;。API …

7V 至 30V 的超寬 VIN 輸入范圍,轉換效率高達 96%的WD5030

WD5030 具備 7V 至 30V 的超寬 VIN 輸入范圍&#xff0c;這一特性使其能夠適應多種不同電壓等級的供電環境&#xff0c;無論是在工業設備中常見的較高電壓輸入&#xff0c;還是在一些便攜式設備經過初步升壓后的電壓&#xff0c;WD5030 都能輕松應對&#xff0c;極大地拓展了應…

深度學習模型Transformer核心組件—自注意力機制

第一章&#xff1a;人工智能之不同數據類型及其特點梳理 第二章&#xff1a;自然語言處理(NLP)&#xff1a;文本向量化從文字到數字的原理 第三章&#xff1a;循環神經網絡RNN&#xff1a;理解 RNN的工作機制與應用場景(附代碼) 第四章&#xff1a;循環神經網絡RNN、LSTM以及GR…

利用Ollama+AnythingLLM+本地向量數據庫Milvus+本地DeepSeek大模型實現知識庫的搭建

1. Ollama的搭建 基本介紹 Ollama是一個支持在Windows、Linux和MacOS上本地運行大語言模型的工具。它允許用戶非常方便地運行和使用各種大語言模型,比如Qwen模型等。用戶只需一行命令就可以啟動模型。 Ollama 下載&#xff1a;https://ollama.com/download Ollama 官方主頁&a…

如何搭建個人靜態住宅IP:從零開始

你好&#xff01;今天我們將一起探索如何從頭開始搭建個人靜態住宅IP。無論您是為了遠程辦公、在線教育還是游戲加速&#xff0c;靜態住宅IP都能帶給您更穩定的網絡體驗。 一、準備階段 1. 明確需求 首先&#xff0c;您需要清楚自己為什么需要靜態住宅IP。可能是為了實現遠程…

國產編輯器EverEdit - 安裝擴展功能的方式

1 擴展管理 1.1 應用場景 由于各行各業工作場景的不同&#xff0c;編輯器不可能為所有行業都定制功能&#xff0c;因此&#xff0c;對于一些特殊的行業應用場景&#xff0c;某些資深工程師可能已經做了一些擴展&#xff0c;并分享到了共享平臺&#xff0c;普通工程師可以安裝這…

CUDA計時函數:精確測量GPU代碼執行時間

在GPU編程中&#xff0c;精確測量代碼執行時間是性能優化的關鍵步驟。CUDA提供了專門的計時工具來幫助開發者準確獲取核函數&#xff08;Kernel&#xff09;、內存拷貝等操作的耗時。本文將詳細介紹CUDA計時函數的使用方法&#xff0c;并通過實例代碼演示如何高效測量GPU代碼的…

Go語言集成DeepSeek API和GoFly框架文本編輯器實現流式輸出和對話(GoFly快速開發框架)

說明 本文是GoFly快速開發框架集成Go語言調用 DeepSeek API 插件&#xff0c;實現流式輸出和對話功能。為了方便實現更多業務功能我們在Go服務端調用AI即DeepSeek接口&#xff0c;處理好業務后再用Gin框架實現流失流式輸出到前端&#xff0c;前端使用fetch請求接收到流式的mar…

SAP服務器進程預警通知

在財務月結&#xff0c;HR薪資核算等系統用戶集中使用高峰時期。通過判斷判斷當前系統可用的并行對話框進程數&#xff0c;用戶使用過多給出提示&#xff0c;服務器進程預警通知。 1. 根據配置的進程最大可使用率80%&#xff0c;根據進程數判斷&#xff1a;當進程可用數少于20%…