国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

個性化認證!Spring Security 輕松擴展登錄字段

開發 前端
Spring Security支持認證、授權、加密、會話管理等核心安全功能,并提供了與Spring MVC等Spring框架的無縫集成。通過簡單的配置和注解,開發者可以輕松地將其集成到應用程序中,保護應用程序免受各種安全威脅。

1. 簡介

Spring Security是一個功能強大且高度可定制的身份驗證和訪問控制框架,它是Spring家族中的核心一員。它基于Spring框架,為基于Java的企業應用程序提供全面的安全性解決方案。

Spring Security支持認證、授權、加密、會話管理等核心安全功能,并提供了與Spring MVC等Spring框架的無縫集成。通過簡單的配置和注解,開發者可以輕松地將其集成到應用程序中,保護應用程序免受各種安全威脅。

默認情況下,Spring Security 提供了基于用戶名/密碼的安全驗證,如下默認的登錄示例:

圖片圖片

該頁面是Spring Security內置的登錄頁面,它是基于用戶名和密碼的驗證,要完成該認證方式其實非常的簡單,我們只需要提供UserDetailsService和PasswordEncoder 兩個Bean,或者是提供AuthenticationProvider 一個Bean即可。

現在我們希望擴展登錄認證,添加域的登錄驗證,如下登錄頁面:

圖片圖片

該頁面的登錄認證,我們不僅局限于驗證用戶名和密碼,還額外要求驗證特定的域信息,以確保用戶身份的全面驗證與訪問控制的安全性。

本篇文章我們將詳細的介紹如何實現上面的認證要求。

2. 實戰案例

2.1 定義實體對象&Repository

@Entity
@Table(name = "s_user")
public class User implements UserDetails {
  @Id
  @GeneratedValue(strategy = GenerationType.IDENTITY)
  private Long id ;
  private String username ;
  private String domain ;
  private String password ;
  // getters, setters
}

該實體對象實現了UserDetails接口,其主要作用是,在后續自定義UserDetailsService時,能夠基于用戶名及域來查詢并返回相應的用戶詳情對象。

Repository接口定義

該接口提供一個根據用戶名及域的查詢方法

public interface UserRepository extends JpaRepository<User, Long> {


  User findByUsernameAndDomain(String username, String domain) ;
}

接下來,我們需要定義與安全認證相關的代碼了。

2.2 自定義過濾器

public class ExtraAuthenticationFilter extends OncePerRequestFilter {


  @Override
  protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
      throws ServletException, IOException {
    ExtraHttpRequest req = new ExtraHttpRequest(request) ;
    filterChain.doFilter(req, response) ;
  }
}

該過濾器的作用是用來,處理登錄頁面的請求參數,我們需要將用戶名及域兩個字段進行合并處理。

public class ExtraHttpRequest extends HttpServletRequestWrapper {


  public ExtraHttpRequest(HttpServletRequest request) {
    super(request) ;
  }
  @Override
  public String getParameter(String name) {
    // 判斷如果參數名是指定的名稱,則我們將用戶名與域兩個表單值進行拼接
    if (SecurityConfig.LOGIN_NAME_PARAMETER.equals(name)) {
      String username = super.getParameter(SecurityConfig.LOGIN_NAME_PARAMETER) ;
      String domain = super.getParameter(SecurityConfig.LOGIN_DOMAIN_PARAMETER) ;
      return username + Character.LINE_SEPARATOR + domain ;
    }
    return super.getParameter(name) ;
  }
}

這里拼接后,我們會在后續進行解析處理。

2.3 安全配置

@Configuration
public class SecurityConfig {


  public static final String LOGIN_NAME_PARAMETER = "username" ;
  public static final String LOGIN_DOMAIN_PARAMETER = "domain" ;
  
  @Bean
  SecurityFilterChain securityFilterChain(HttpSecurity http) throws Throwable {
    http.csrf(csrf -> csrf.disable()) ;
    http.authorizeHttpRequests(registry -> {
      registry.requestMatchers("*.html", "*.css", "*.js", "/login").permitAll() ;
      registry.requestMatchers("/**").authenticated() ;
    }) ;
    http.formLogin(form -> {
      // 自定義登錄頁面
      form.loginPage("/login").usernameParameter(LOGIN_NAME_PARAMETER) ;
    }) ;
    // 將我們自定義的過濾器,添加到安全過濾器鏈中,并且是在UsernamePasswordAuthenticationFilter
    // 過濾器之前執行
    http.addFilterBefore(extraAuthenticationFilter(), UsernamePasswordAuthenticationFilter.class) ;
    return http.build() ;
  }
  
  @Bean
  ExtraAuthenticationFilter extraAuthenticationFilter() {
    return new ExtraAuthenticationFilter() ;
  }
  
  @Bean
  PasswordEncoder noopPasswordEncoder() {
    return new PasswordEncoder() {
      @Override
      public boolean matches(CharSequence rawPassword, String encodedPassword) {
        return rawPassword != null && encodedPassword != null && rawPassword.equals(encodedPassword) ;
      }
      @Override
      public String encode(CharSequence rawPassword) {
        return rawPassword.toString() ;
      }
    };
  }
}

下面我們會配置自定義的UserDetailsService對象,所以我們還需要提供一個PasswordEncoder類型的bean,由于我們沒有對密碼進行加密處理,所以我們只是做了簡單的相等判斷。

@Component
public class PackUserDetailsService implements UserDetailsService {


  private final UserRepository userRepository ;
  public PackUserDetailsService(UserRepository userRepository) {
    this.userRepository = userRepository;
  }
  // 由于這里需要返回的是UserDetails對象,所以我們上面的User實體
  // 實現了該接口
  @Override
  public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
    // 上面的過濾器中我們將用戶與域進行了組裝,所以這里將用戶名進行解析處理
    String[] info = StringUtils.split(username, String.valueOf(Character.LINE_SEPARATOR)) ;
    return this.userRepository.findByUsernameAndDomain(info[0], info[1]) ;
  }
}

再次說明:在Spring Security中,要么你提供UserDetailsService和PasswordEncoder兩個Bean,要么提供一個AuthenticationProvider(通常我們可以定義DaoAuthenticationProvider即可)類型的Bean。這樣就能使用自定義的邏輯進行安全認證。

2.4 自定義登錄頁面

我們將使用thymeleaf來編寫登錄頁面,需要引入如下的依賴:

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<dependency>
  <groupId>org.thymeleaf.extras</groupId>
  <artifactId>thymeleaf-extras-springsecurity6</artifactId>
</dependency>

配置thymeleaf

spring:
  thymeleaf:
    prefix: classpath:/static/
    suffix: .html
    cache: false

下面是登錄頁面

<html lang="en">
 <head>
  <meta charset="UTF-8">
  <title>安全登錄</title>
  <link  rel="stylesheet">
  <style type="text/css">
    body {
      margin: 10px auto;
    }
    .form-signin {
      width: 50%; /* 根據需要調整寬度 */
        margin: 0 auto;
        padding: 20px; /* 可選,為表單添加內邊距 */
        border: 1px solid #ccc; /* 可選,為表單添加邊框 */
        border-radius: 6px; /* 可選,為表單邊框添加圓角 */
    }
</style>
 </head>
 <body>
   <form class="form-signin" th:action="@{/login}" method="post">
     <h2 class="form-signin-heading">安全登錄</h2>
     <p th:if="${param.error}" class="error">錯誤的用戶名/域, 密碼</p>
     <p>
       <label for="username" class="sr-only">帳號</label>
       <input type="text" id="username" name="username" class="form-control" placeholder="用戶名" required autofocus/>
     </p>
     <p>
       <label for="domain" class="sr-only">域</label>
       <input type="text" id="domain" name="domain" class="form-control" placeholder="登錄域" required autofocus/>
     </p>
     <p>
       <label for="password" class="sr-only">密碼</label>
       <input type="password" id="password" name="password" class="form-control" placeholder="密碼" required autofocus/>
     </p>
     <button class="btn btn-sm btn-primary btn-block" type="submit">登錄</button>
     <a href="/index" th:href="@{/index}">返回</a>
  </form> 
 </body>
</html>

該login.html頁面保存在classpath下的/static目錄中即可。

最后,我們還需要定義一個/login接口,用來跳轉到上面的登錄頁面

@Controller
public class LoginController {


  @GetMapping("/login")
  public String login() {
    return "login" ;
  }
}

以上我們就完成了所有的代碼編寫。

2.4 測試

@RestController
@RequestMapping("/api")
public class ApiController {


  @GetMapping("/query")
  public ResponseEntity<Object> query() {
    return ResponseEntity.ok("api query success") ;
  }
}

數據庫中的數據

圖片圖片

訪問/api/query接口將跳轉到登錄頁面

圖片圖片

成功登錄后,跳回之前的頁面/api/query

圖片圖片

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2020-06-28 07:00:00

推薦系統智能商務服務平臺

2023-09-25 15:54:28

Canvas國慶

2024-10-05 00:00:25

Cursor網站代碼

2022-11-01 07:19:45

推薦系統非個性化

2011-01-20 10:19:21

PowerShell個性化

2013-11-07 16:42:34

Windows 8.1個性化

2011-05-04 14:38:53

海爾江山帝景一體機

2023-03-21 12:46:30

智慧城市人工智能大數據

2020-08-31 12:00:17

Linux終端顏色命令

2009-07-13 15:33:24

桌面虛擬化虛擬化IT

2017-11-22 09:24:00

2013-01-04 09:41:11

云計算個性化精準促銷Me Marketin

2023-07-26 07:51:30

游戲中心個性化

2011-04-28 11:14:33

simpleframe

2021-10-19 08:00:00

Windows 11Windows微軟

2018-10-25 14:21:22

Oracle數字化助手

2024-01-16 15:51:55

個性化圖片3D

2010-04-30 17:07:03

組策略部署

2011-08-18 18:53:30

win7
點贊
收藏

51CTO技術棧公眾號

成人在线视频观看| 在线观看91av| 婷婷综合成人| 国产九色精品成人porny| 欧美影院一区二区三区| 性欧美xxxx交| 一区二区成人av| 中文字幕一区二区三区四区五区六区| 天堂中文视频在线| 亚洲裸色大胆大尺寸艺术写真| 国产一级久久| 3d动漫精品啪啪1区2区免费| 国产精品一区二区欧美黑人喷潮水| 日韩中文一区二区| 亚洲一区二区在线免费看| 最近的2019中文字幕免费一页| 日韩精品欧美一区二区三区| 在线三级电影| 老牛国产精品一区的观看方式| 欧美视频中文一区二区三区在线观看| 久久久久久久久久久免费视频| 国产精品亚洲成在人线| 蜜臀精品久久久久久蜜臀 | 丁香花在线电影小说观看| 成人情趣视频网站| 亚洲一区影音先锋| www黄色日本| 黄一区二区三区| 欧美v国产在线一区二区三区| 在线视频中文字幕久| 婷婷综合激情| 亚洲一区在线观看网站| 国产三区在线视频| 成人av免费在线观看| 中文字幕亚洲欧美日韩在线不卡| 欧美与动交zoz0z| 日韩一区视频在线| 北条麻妃一区二区三区在线| 91精品国产综合久久久久| 久久精品国产精品青草色艺| 日本在线www| 亚洲毛片网站| 日韩欧美一级在线播放| 一区二区日本| 极品视频在线| 国产精品18久久久久久久久 | 亚洲欧美日韩区| 日韩成人三级视频| 亚洲一区二区三区在线免费 | 免费观看一级特黄欧美大片| 亚洲人成77777| 黄网站免费久久| 欧美 亚洲 视频| 国产精品无码久久久久| 国产一区二区精品丝袜| 欧美女同在线观看| 91美女精品福利| 欧美亚洲国产另类| 在线免费观看黄色av| 轻轻草成人在线| 最近2019年日本中文免费字幕| 成人国产精品| 九九精品在线观看| 青青草在线免费观看| 狠狠躁夜夜躁人人爽天天天天97| 久久久久高清| 奇米四色…亚洲| 毛片在线播放视频| 亚洲伊人春色| 国产欧美中文字幕| 国产美女在线观看| 精品国内二区三区| 国产不卡在线观看| 性欧美videos高清hd4k| 高清av一区二区| 国产日韩欧美日韩| 海角社区69精品视频| 亚洲精选中文字幕| 亚洲精品中文字幕| 亚洲人吸女人奶水| 三级无遮挡在线观看| 欧美日韩一区小说| 成人国产二区| 57pao成人永久免费视频| 一区二区三区四区日韩| 欧美成人一区二区三区在线观看| 国产一二三在线| 欧美裸体xxxx极品少妇| 国产精品久久久久av蜜臀| 国产99在线|中文| 伊人久久综合一区二区| 亚洲午夜激情免费视频| 2020最新国产精品| 国产精品久久久久久久久久久久午夜片 | 亚洲伦理网站| 日韩欧美一区二区三区久久| 波多野结衣乳巨码无在线| 亚洲色欲色欲www| 毛片av在线| 伦伦影院午夜日韩欧美限制| 69久久精品| 日韩在线视频网| 999国产精品视频| 男人天堂新网址| 午夜精品福利在线| 日韩国产激情| 555www色欧美视频| crdy在线观看欧美| 国产精品一区二区三区不卡 | 国模一区二区| 亚洲影院污污.| 色婷婷综合久久久久久| 国产精品久久波多野结衣| 粉嫩嫩av羞羞动漫久久久| 久久久亚洲精华液精华液精华液| 亚洲成人av在线播放| 免费视频一区三区| 欧美成人午夜免费视在线看片| 大片免费在线看视频| 高清亚洲成在人网站天堂| 成人爽a毛片免费啪啪| 国产精品日韩av| av高清一区| 欧美一区二区三区四区在线| 黄页网站大全一区二区| 小草av在线播放| 欧美激情影音先锋| 国产一区二区三区免费在线| 久久久久久九九九九| 中文字幕佐山爱一区二区免费| vam成人资源在线观看| 蜜臀av国产精品久久久久| 爱情岛论坛成人| 亚洲精品资源美女情侣酒店| 中文字幕一区二区三区久久网站 | 日韩精品免费在线视频| 成年人网站大全| 国产精品一区二区视频| 激情小视频在线观看| 久久久日本电影| 美女视频黄 久久| 国产精品视频不卡| 久久综合一区二区| 毛片无码国产| 色一情一乱一伦一区二区三欧美| 日韩高清不卡在线| 在线观看日韩羞羞视频| 在线看日韩精品电影| 久久天堂电影| 国产精品高清在线| 一区二区日韩欧美| 丝袜国产免费观看| 日韩理论在线观看| 亚洲成人不卡| 波多野结衣激情| 国产精品亚洲成人| 欧美午夜大胆人体| 久久99精品久久久久久水蜜桃| 亚洲最大成人网4388xx| 国产精品久久久网站| 五月婷婷之综合激情| 欧美激情一区二区三区在线视频观看 | 欧美亚洲成人网| 精久久久久久久久久久| 国产一二三区在线观看| 亚洲xxxxx| 性做久久久久久| 欧美日韩高清| 久久久噜噜噜久久久| 成人小视频在线观看| 日本www高清视频| 91精品欧美一区二区三区综合在 | 这里只有精品66| 亚洲国产精品久久久久秋霞蜜臀| 成人性生交大片免费网站 | 亚洲91av视频| 国产精品超碰97尤物18| 欧美国产极品| 热国产热中文视频二区| 欧美一区二区三区免费视| 一区二区三区在线免费| 四虎成人精品永久免费av九九| 超碰在线公开免费| 99在线观看| 欧美一区二区成人6969| 美女网站一区二区| av成人亚洲| 国产偷人视频免费| 17婷婷久久www| 精品福利在线视频| 99成人在线| 老司机成人影院| 精品亚洲一区二区三区四区| 91精品久久久久久久| 欧美一区在线视频| www..com久久爱| 日韩毛片视频| 中文字幕欧美日韩一区二区三区| 精品丝袜一区二区三区| 日韩中文字幕高清在线观看|