自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

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

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

1. 簡介

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

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

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

圖片圖片

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

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

圖片圖片

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

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

2. 實戰(zhàn)案例

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
}

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

Repository接口定義

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

public interface UserRepository extends JpaRepository<User, Long> {


  User findByUsernameAndDomain(String username, String domain) ;
}

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

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) ;
  }
}

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

public class ExtraHttpRequest extends HttpServletRequestWrapper {


  public ExtraHttpRequest(HttpServletRequest request) {
    super(request) ;
  }
  @Override
  public String getParameter(String name) {
    // 判斷如果參數(shù)名是指定的名稱,則我們將用戶名與域兩個表單值進行拼接
    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) ;
  }
}

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

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
    // 過濾器之前執(zhí)行
    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實體
  // 實現(xiàn)了該接口
  @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%; /* 根據(jù)需要調(diào)整寬度 */
        margin: 0 auto;
        padding: 20px; /* 可選,為表單添加內(nèi)邊距 */
        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接口,用來跳轉(zhuǎn)到上面的登錄頁面

@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") ;
  }
}

數(shù)據(jù)庫中的數(shù)據(jù)

圖片圖片

訪問/api/query接口將跳轉(zhuǎn)到登錄頁面

圖片圖片

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

圖片圖片

責任編輯:武曉燕 來源: Spring全家桶實戰(zhàn)案例源碼
相關(guān)推薦

2020-06-28 07:00:00

推薦系統(tǒng)智能商務服務平臺

2023-09-25 15:54:28

Canvas國慶

2024-10-05 00:00:25

Cursor網(wǎng)站代碼

2022-11-01 07:19:45

推薦系統(tǒng)非個性化

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

智慧城市人工智能大數(shù)據(jù)

2020-08-31 12:00:17

Linux終端顏色命令

2009-07-13 15:33:24

桌面虛擬化虛擬化IT

2011-04-28 11:14:33

simpleframe

2017-11-22 09:24:00

2013-01-04 09:41:11

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

2023-07-26 07:51:30

游戲中心個性化

2021-10-19 08:00:00

Windows 11Windows微軟

2018-10-25 14:21:22

Oracle數(shù)字化助手

2024-01-16 15:51:55

個性化圖片3D

2011-08-18 18:53:30

win7

2010-04-30 17:07:03

組策略部署
點贊
收藏

51CTO技術(shù)棧公眾號