日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Springboot集成Spring Security實現JWT認證的步驟詳解

瀏覽:25日期:2023-03-25 14:16:11
1 簡介

Spring Security作為成熟且強大的安全框架,得到許多大廠的青睞。而作為前后端分離的SSO方案,JWT也在許多項目中應用。本文將介紹如何通過Spring Security實現JWT認證。

用戶與服務器交互大概如下:

Springboot集成Spring Security實現JWT認證的步驟詳解

客戶端獲取JWT,一般通過POST方法把用戶名/密碼傳給server; 服務端接收到客戶端的請求后,會檢驗用戶名/密碼是否正確,如果正確則生成JWT并返回;不正確則返回錯誤; 客戶端拿到JWT后,在有效期內都可以通過JWT來訪問資源了,一般把JWT放在請求頭;一次獲取,多次使用; 服務端校驗JWT是否合法,合法則允許客戶端正常訪問,不合法則返回401。 2 項目整合

我們把要整合的Spring Security和JWT加入到項目的依賴中去:

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId></dependency><dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-security</artifactId></dependency><dependency> <groupId>io.jsonwebtoken</groupId> <artifactId>jjwt</artifactId> <version>0.9.1</version></dependency>2.1 JWT整合2.1.1 JWT工具類

JWT工具類起碼要具有以下功能:

根據用戶信息生成JWT; 校驗JWT是否合法,如是否被篡改、是否過期等; 從JWT中解析用戶信息,如用戶名、權限等;

具體代碼如下:

@Componentpublic class JwtTokenProvider { @Autowired JwtProperties jwtProperties; @Autowired private CustomUserDetailsService userDetailsService; private String secretKey; @PostConstruct protected void init() { secretKey = Base64.getEncoder().encodeToString(jwtProperties.getSecretKey().getBytes()); } public String createToken(String username, List<String> roles) { Claims claims = Jwts.claims().setSubject(username); claims.put('roles', roles); Date now = new Date(); Date validity = new Date(now.getTime() + jwtProperties.getValidityInMs()); return Jwts.builder()// .setClaims(claims)// .setIssuedAt(now)// .setExpiration(validity)// .signWith(SignatureAlgorithm.HS256, secretKey)// .compact(); } public Authentication getAuthentication(String token) { UserDetails userDetails = this.userDetailsService.loadUserByUsername(getUsername(token)); return new UsernamePasswordAuthenticationToken(userDetails, '', userDetails.getAuthorities()); } public String getUsername(String token) { return Jwts.parser().setSigningKey(secretKey).parseClaimsJws(token).getBody().getSubject(); } public String resolveToken(HttpServletRequest req) { String bearerToken = req.getHeader('Authorization'); if (bearerToken != null && bearerToken.startsWith('Bearer ')) { return bearerToken.substring(7); } return null; } public boolean validateToken(String token) { try { Jws<Claims> claims = Jwts.parser().setSigningKey(secretKey).parseClaimsJws(token); if (claims.getBody().getExpiration().before(new Date())) { return false; } return true; } catch (JwtException | IllegalArgumentException e) { throw new InvalidJwtAuthenticationException('Expired or invalid JWT token'); } }}

工具類還實現了另一個功能:從HTTP請求頭中獲取JWT。

2.1.2 Token處理的Filter

Filter是Security處理的關鍵,基本上都是通過Filter來攔截請求的。首先從請求頭取出JWT,然后校驗JWT是否合法,如果合法則取出Authentication保存在SecurityContextHolder里。如果不合法,則做異常處理。

public class JwtTokenAuthenticationFilter extends GenericFilterBean { private JwtTokenProvider jwtTokenProvider; public JwtTokenAuthenticationFilter(JwtTokenProvider jwtTokenProvider) { this.jwtTokenProvider = jwtTokenProvider; } @Override public void doFilter(ServletRequest req, ServletResponse res, FilterChain filterChain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) req; HttpServletResponse response = (HttpServletResponse) res; try { String token = jwtTokenProvider.resolveToken(request); if (token != null && jwtTokenProvider.validateToken(token)) { Authentication auth = jwtTokenProvider.getAuthentication(token); if (auth != null) { SecurityContextHolder.getContext().setAuthentication(auth); } } } catch (InvalidJwtAuthenticationException e) { response.setStatus(HttpStatus.UNAUTHORIZED.value()); response.getWriter().write('Invalid token'); response.getWriter().flush(); return; } filterChain.doFilter(req, res); }}

對于異常處理,使用@ControllerAdvice是不行的,應該這個是Filter,在這里拋的異常還沒有到DispatcherServlet,無法處理。所以Filter要自己做異常處理:

catch (InvalidJwtAuthenticationException e) { response.setStatus(HttpStatus.UNAUTHORIZED.value()); response.getWriter().write('Invalid token'); response.getWriter().flush(); return;}

最后的return不能省略,因為已經要把輸出的內容給Response了,沒有必要再往后傳遞,否則報錯

java.lang.IllegalStateException: getWriter() has already been called2.1.3 JWT屬性

JWT需要配置一個密鑰來加密,同時還要配置JWT令牌的有效期。

@Configuration@ConfigurationProperties(prefix = 'pkslow.jwt')public class JwtProperties { private String secretKey = 'pkslow.key'; private long validityInMs = 3600_000;//getter and setter}2.2 Spring Security整合

Spring Security的整個框架還是比較復雜的,簡化后大概如下圖所示:

Springboot集成Spring Security實現JWT認證的步驟詳解

它是通過一連串的Filter來進行安全管理。細節這里先不展開講。

2.2.1 WebSecurityConfigurerAdapter配置

這個配置也可以理解為是FilterChain的配置,可以不用理解,代碼很好懂它做了什么:

@Configurationpublic class SecurityConfig extends WebSecurityConfigurerAdapter { @Autowired JwtTokenProvider jwtTokenProvider; @Bean @Override public AuthenticationManager authenticationManagerBean() throws Exception { return super.authenticationManagerBean(); } @Bean public PasswordEncoder passwordEncoder() { return NoOpPasswordEncoder.getInstance(); } @Override protected void configure(HttpSecurity http) throws Exception { http .httpBasic().disable() .csrf().disable() .sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS) .and() .authorizeRequests() .antMatchers('/auth/login').permitAll() .antMatchers(HttpMethod.GET, '/admin').hasRole('ADMIN') .antMatchers(HttpMethod.GET, '/user').hasRole('USER') .anyRequest().authenticated() .and() .apply(new JwtSecurityConfigurer(jwtTokenProvider)); }}

這里通過HttpSecurity配置了哪些請求需要什么權限才可以訪問。

/auth/login用于登陸獲取JWT,所以都能訪問; /admin只有ADMIN用戶才可以訪問; /user只有USER用戶才可以訪問。

而之前實現的Filter則在下面配置使用:

public class JwtSecurityConfigurer extends SecurityConfigurerAdapter<DefaultSecurityFilterChain, HttpSecurity> { private JwtTokenProvider jwtTokenProvider; public JwtSecurityConfigurer(JwtTokenProvider jwtTokenProvider) { this.jwtTokenProvider = jwtTokenProvider; } @Override public void configure(HttpSecurity http) throws Exception { JwtTokenAuthenticationFilter customFilter = new JwtTokenAuthenticationFilter(jwtTokenProvider); http.exceptionHandling() .authenticationEntryPoint(new JwtAuthenticationEntryPoint()) .and() .addFilterBefore(customFilter, UsernamePasswordAuthenticationFilter.class); }}2.2.2 用戶從哪來

通常在Spring Security的世界里,都是通過實現UserDetailsService來獲取UserDetails的。

@Componentpublic class CustomUserDetailsService implements UserDetailsService { private UserRepository users; public CustomUserDetailsService(UserRepository users) { this.users = users; } @Override public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException { return this.users.findByUsername(username) .orElseThrow(() -> new UsernameNotFoundException('Username: ' + username + ' not found')); }}

對于UserRepository,可以從數據庫中讀取,或者其它用戶管理中心。為了方便,我使用Map放了兩個用戶:

@Repositorypublic class UserRepository { private static final Map<String, User> allUsers = new HashMap<>(); @Autowired private PasswordEncoder passwordEncoder; @PostConstruct protected void init() { allUsers.put('pkslow', new User('pkslow', passwordEncoder.encode('123456'), Collections.singletonList('ROLE_ADMIN'))); allUsers.put('user', new User('user', passwordEncoder.encode('123456'), Collections.singletonList('ROLE_USER'))); } public Optional<User> findByUsername(String username) { return Optional.ofNullable(allUsers.get(username)); }}3 測試

完成代碼編寫后,我們來測試一下:

(1)無JWT訪問,失敗

curl http://localhost:8080/admin{'timestamp':'2021-02-06T05:45:06.385+0000','status':403,'error':'Forbidden','message':'Access Denied','path':'/admin'}$ curl http://localhost:8080/user{'timestamp':'2021-02-06T05:45:16.438+0000','status':403,'error':'Forbidden','message':'Access Denied','path':'/user'}

(2)admin獲取JWT,密碼錯誤則失敗,密碼正確則成功

$ curl http://localhost:8080/auth/login -X POST -d ’{'username':'pkslow','password':'xxxxxx'}’ -H ’Content-Type: application/json’{'timestamp':'2021-02-06T05:47:16.254+0000','status':403,'error':'Forbidden','message':'Access Denied','path':'/auth/login'}$ curl http://localhost:8080/auth/login -X POST -d ’{'username':'pkslow','password':'123456'}’ -H ’Content-Type: application/json’eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJwa3Nsb3ciLCJyb2xlcyI6WyJST0xFX0FETUlOIl0sImlhdCI6MTYxMjU5MDYxNCwiZXhwIjoxNjEyNTkxMjE0fQ.d4Gi50aaOsHHqpM0d8Mh1960otnZf7rlE3x6xSfakVo

(3)admin帶JWT訪問/admin,成功;訪問/user失敗

$ curl http://localhost:8080/admin -H ’Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJwa3Nsb3ciLCJyb2xlcyI6WyJST0xFX0FETUlOIl0sImlhdCI6MTYxMjU5MDYxNCwiZXhwIjoxNjEyNTkxMjE0fQ.d4Gi50aaOsHHqpM0d8Mh1960otnZf7rlE3x6xSfakVo’you are admin$ curl http://localhost:8080/user -H ’Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJwa3Nsb3ciLCJyb2xlcyI6WyJST0xFX0FETUlOIl0sImlhdCI6MTYxMjU5MDYxNCwiZXhwIjoxNjEyNTkxMjE0fQ.d4Gi50aaOsHHqpM0d8Mh1960otnZf7rlE3x6xSfakVo’{'timestamp':'2021-02-06T05:51:23.099+0000','status':403,'error':'Forbidden','message':'Forbidden','path':'/user'}

(4)使用過期的JWT訪問,失敗

$ curl http://localhost:8080/admin -H ’Authorization: Bearer eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJwa3Nsb3ciLCJyb2xlcyI6WyJST0xFX0FETUlOIl0sImlhdCI6MTYxMjU5MDQ0OSwiZXhwIjoxNjEyNTkwNTA5fQ.CSaubE4iJcYATbLmbb59aNFU1jNCwDFHUV3zIakPU64’Invalid token4 總結

代碼請查看:https://github.com/LarryDpk/pkslow-samples

以上就是Springboot集成Spring Security實現JWT認證的步驟詳解的詳細內容,更多關于Springboot集成Spring Security的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
欧美日韩激情在线一区二区三区| 丝袜av一区| 亚洲制服一区| 红桃视频国产精品| 91精品福利| 国产精品腿扒开做爽爽爽挤奶网站| 午夜久久黄色| 亚洲欧洲日本mm| 噜噜噜躁狠狠躁狠狠精品视频| 国产婷婷精品| 亚洲麻豆一区| 日韩高清不卡在线| 国产精品亲子伦av一区二区三区| 国产精品2023| 日本精品黄色| 日本精品影院| 午夜欧美视频| 亚洲精品三级| 国产亚洲一区二区三区啪| 久久av影视| 热三久草你在线| 激情综合自拍| 亚洲精品精选| 久久只有精品| 精品国模一区二区三区| 欧美日韩国产免费观看视频| 亚洲视频电影在线| 国产精品手机在线播放| 国产suv精品一区| 久久亚洲成人| 蜜桃久久久久久| 国产精品久久久免费| а√天堂8资源在线| 蜜桃精品在线| 三级欧美在线一区| 国产精品极品国产中出| 久久国产日本精品| 亚洲精选av| 欧美精品二区| 好看的亚洲午夜视频在线| 日韩精品三区四区| 黑人精品一区| 亚洲我射av| 国产精品精品| 夜夜嗨网站十八久久| 日韩1区2区日韩1区2区| 中文字幕成在线观看| 美女国产一区| 久久亚洲人体| 国产美女精品| 久久精品系列| 亚洲国产日韩欧美在线| 日韩精品国产欧美| 日韩在线二区| 日本中文字幕一区二区视频| 日韩av有码| 亚州精品视频| 韩国精品主播一区二区在线观看| 一区二区电影在线观看| 精品亚洲成人| 午夜国产一区二区| 国产激情久久| 国产精品免费看| 麻豆精品99| 亚洲人成高清| 激情欧美亚洲| 日韩成人a**站| 日本91福利区| jiujiure精品视频播放| 国产精品2023| 亚洲欧洲专区| 精品一区三区| 国产一区二区三区免费在线| 蜜桃av一区二区三区电影| 国产成人精品一区二区三区免费| 国产亚洲一区在线| 亚洲欧洲高清| 国产欧美欧美| 香蕉久久久久久久av网站| 国产精品毛片久久| 欧美一级全黄| 日韩在线观看一区二区| 欧美sss在线视频| 精品国产日韩欧美精品国产欧美日韩一区二区三区 | 亚洲精品国产嫩草在线观看| 欧美日韩a区| 在线亚洲观看| 成人小电影网站| 国产精品一区二区精品| 免费看精品久久片| 久久久一二三| 亚洲精品在线国产| 亚洲制服少妇| 免费不卡中文字幕在线| 成人污污视频| 日本少妇一区二区| 三级欧美韩日大片在线看| 国产 日韩 欧美一区| 国产精品免费99久久久| 日本免费在线视频不卡一不卡二| 999在线观看精品免费不卡网站| 美女福利一区二区三区| 国产精品主播| 日韩一区二区三区精品| 在线综合亚洲| 亚洲不卡av不卡一区二区| 久久99影视| 国产精品激情| 青青国产精品| 亚洲精品国模| 只有精品亚洲| 黄色亚洲大片免费在线观看| 丁香六月综合| 福利片在线一区二区| 国产精品天堂蜜av在线播放| 日本欧美大码aⅴ在线播放| 中文字幕日韩高清在线| 日韩中文字幕亚洲一区二区va在线| 99久久久久| 日韩在线中文| 美女免费视频一区| 国产精品一区二区免费福利视频| 日韩成人午夜精品| 亚州精品视频| 日韩av资源网| 91精品丝袜国产高跟在线| 日韩一区二区三免费高清在线观看| 国产视频一区免费看| av亚洲在线观看| 亚洲国产日韩欧美在线| 免费成人网www| 婷婷成人在线| 久久蜜桃av| 红桃视频国产一区| 国产亚洲综合精品| 国产精品呻吟| 免费在线看一区| 久久午夜影视| 亚洲精一区二区三区| 天海翼亚洲一区二区三区| 亚洲精品成人一区| 日本欧美一区二区在线观看| 久久精品99国产精品日本| 久久国产精品免费精品3p | 免费人成在线不卡| 亚洲三级国产| 日韩国产成人精品| 国产亚洲欧美日韩在线观看一区二区| 日本强好片久久久久久aaa| 日本va欧美va精品发布| 国产一卡不卡| 国产色播av在线| 久久精品国产99久久| 99日韩精品| 综合亚洲自拍| 欧美国产三级| 不卡一二三区| 亚洲一区免费| 日韩欧美中文字幕在线视频| 久久爱www成人| 成人va天堂| 国产亚洲毛片在线| 国产精品一在线观看| 国产一区二区三区亚洲| 欧洲精品一区二区三区| 黑丝一区二区三区| 欧美日韩1区| 国产成人精品一区二区免费看京| 在线一区视频观看| 亚洲最新av| 欧美国产另类| 亚洲手机视频| 日韩高清不卡一区二区| 精品丝袜久久| 亚洲成人一区| 奇米亚洲欧美| 亚洲美女久久精品| 亚洲五月综合| 国产自产自拍视频在线观看| 久久人人99| 久久精品av麻豆的观看方式| 亚洲性色av| 日韩欧美中文字幕电影| 日韩伦理一区| 天堂av一区| 亚洲深夜视频| 日韩成人在线看| 精品成人免费一区二区在线播放| 午夜一级久久| 欧美91在线| 不卡中文一二三区| 国产日产精品一区二区三区四区的观看方式 | 91av一区| 久久免费大视频| 国产欧美日韩精品高清二区综合区| 欧洲在线一区| 国产精品久久久久久久久久久久久久久| 婷婷综合六月| 777久久精品| 欧美日韩国产欧|