permitAll:詳解不同場景下的使用方法

一、permitAll方法

在Spring Security中,permitAll方法用於開放某些URL的訪問權限,即使沒有通過身份驗證也可以進行訪問。該方法在處理Spring Security的訪問控制時非常常見,使用方式如下:

@Configuration
@EnableWebSecurity
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {
 
  @Override
  protected void configure(HttpSecurity http) throws Exception {
    http.authorizeRequests()
      .antMatchers("/hello").permitAll()
      .anyRequest().authenticated()
      .and()
      .formLogin().loginPage("/login").defaultSuccessUrl("/home").permitAll()
      .and()
      .logout().permitAll();
  }
}

上面的示例代碼中,我們配置了「/hello」路徑為不需要身份認證的路徑,即使用戶沒有登錄也可以進行訪問。

二、permitAll方法原理

在上一個小節中,我們已經了解了permitAll方法的使用方法,那麼這個方法背後的原理是什麼呢?

在Spring Security中,訪問控制之所以能夠生效,是因為它將SecurityFilterChain對象添加到過濾器鏈中。在處理請求時,這個過濾器鏈將會按照順序執行一系列的過濾器,其中就包括了一個AnonymousAuthenticationFilter過濾器。

這個過濾器將在匿名用戶訪問時,自動為其創建一個AnonymousAuthenticationToken,這個Token將代表一個未認證的用戶,而permitAll方法的作用就是使這個Token獲得訪問控制內的訪問授權。

三、permitAll不生效

在實際開發工作中,我們可能會遇到這樣的情況——無論如何configure(HttpSecurity http)方法都無法禁用身份驗證,即使在特定的URL上使用了permitAll方法。

這個問題的原因是因為在配置的時候,我們的HttpSecurity對象可能設置了多個SecurityFilterChain,導致permitAll方法可能被其他SecurityFilterChain覆蓋。

解決這個問題的方法是,在configure(HttpSecurity http)方法中為每個SecurityFilterChain都設置permitAll方法,確保所有的SecurityFilterChain都生效。

@Configuration
@EnableWebSecurity
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {
 
  @Autowired
  private JwtAuthenticationEntryPoint unauthorizedHandler;
 
  @Autowired
  private JwtTokenProvider jwtTokenProvider;
 
  @Bean
  public JwtAuthenticationFilter jwtAuthenticationFilter() {
    return new JwtAuthenticationFilter(jwtTokenProvider);
  }
 
  @Override
  protected void configure(HttpSecurity http) throws Exception {
    http.csrf().disable()
      .authorizeRequests()
      .antMatchers("/api/auth/**").permitAll()
      .anyRequest().authenticated()
      .and()
      .exceptionHandling().authenticationEntryPoint(unauthorizedHandler).and()
      .sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS);
 
    http.addFilterBefore(jwtAuthenticationFilter(), UsernamePasswordAuthenticationFilter.class);
  }
 
  @Configuration
  @Order(1)
  public static class ApiWebSecurityConfigurationAdapter extends WebSecurityConfigurerAdapter {
    protected void configure(HttpSecurity http) throws Exception {
      http.antMatcher("/api/**")
        .authorizeRequests()
        .antMatchers(HttpMethod.OPTIONS).permitAll()
        .anyRequest().authenticated();
    }
  }
 
  @Configuration
  @Order(2)
  public static class FormLoginWebSecurityConfigurerAdapter extends WebSecurityConfigurerAdapter {
    protected void configure(HttpSecurity http) throws Exception {
      http.authorizeRequests()
        .antMatchers("/login**", "/signup**", "/h2-console/**").permitAll()
        .anyRequest().authenticated()
        .and()
        .formLogin()
        .loginPage("/login")
        .defaultSuccessUrl("/home")
        .failureUrl("/login?error=true")
        .permitAll()
        .and()
        .logout()
        .logoutSuccessUrl("/login")
        .permitAll()
        .and().csrf().disable();

      http.headers().frameOptions().disable();
    }
  }
}

四、permit和allow的區別

在Spring Security中,除了permitAll方法外,還有allowAll方法,那麼這兩者之間到底有什麼區別呢?

實際上,這兩者之間的區別非常微妙,相信大部分開發者都無法分辨它們之間的差異。其實,在Spring Security中,permitAll就是allowAll的別名,這兩個方法完全等價。

五、permitAll和anonymous的區別

在Spring Security中,permitAll和anonymous都可以用來授權未認證的用戶,那麼這兩者之間有何區別呢?

簡而言之,permitAll是用於授權未認證用戶對系統內的URL資源進行訪問,而anonymous則是用於允許非匿名用戶訪問系統中的受保護資源。

在實際應用中,permitAll通常被用來允許未認證的用戶訪問一些公開資源,比如首頁、註冊頁面等。而anonymous則被用來授權已經被認證的用戶在系統內匿名訪問一些資源,比如用戶名片。

總結

本文詳細介紹了Spring Security中permitAll方法的使用方法和原理,並且討論了permitAll不生效、permit和allow的區別以及permitAll和anonymous的區別等相關問題。通過本文的閱讀,相信讀者們已經掌握了permitAll在不同場景下的使用方式和注意事項,並且可以在實際開發中熟練運用這個強大的功能。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/150856.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-11-09 02:14
下一篇 2024-11-09 02:14

相關推薦

  • Python中init方法的作用及使用方法

    Python中的init方法是一個類的構造函數,在創建對象時被調用。在本篇文章中,我們將從多個方面詳細討論init方法的作用,使用方法以及注意點。 一、定義init方法 在Pyth…

    編程 2025-04-29
  • 用不同的方法求素數

    素數是指只能被1和自身整除的正整數,如2、3、5、7、11、13等。素數在密碼學、計算機科學、數學、物理等領域都有着廣泛的應用。本文將介紹幾種常見的求素數的方法,包括暴力枚舉法、埃…

    編程 2025-04-29
  • Unity3D 創建沒有 Terrain Tile 的場景

    這篇文章將會介紹如何在 Unity3D 中創建一個沒有 Terrain Tile 的場景,同時也讓讀者了解如何通過編程實現這個功能。 一、基礎概念 在 Unity3D 中,Terr…

    編程 2025-04-29
  • Python符號定義和使用方法

    本文將從多個方面介紹Python符號的定義和使用方法,涉及注釋、變量、運算符、條件語句和循環等多個方面。 一、注釋 1、單行注釋 # 這是一條單行注釋 2、多行注釋 “”” 這是一…

    編程 2025-04-29
  • Python函數名稱相同參數不同:多態

    Python是一門面向對象的編程語言,它強烈支持多態性 一、什麼是多態多態是面向對象三大特性中的一種,它指的是:相同的函數名稱可以有不同的實現方式。也就是說,不同的對象調用同名方法…

    編程 2025-04-29
  • Python下載到桌面圖標使用方法用法介紹

    Python是一種高級編程語言,非常適合初學者,同時也深受老手喜愛。在Python中,如果我們想要將某個程序下載到桌面上,需要注意一些細節。本文將從多個方面對Python下載到桌面…

    編程 2025-04-29
  • Python匿名變量的使用方法

    Python中的匿名變量是指使用「_」來代替變量名的特殊變量。這篇文章將從多個方面介紹匿名變量的使用方法。 一、作為佔位符 匿名變量通常用作佔位符,用於代替一個不需要使用的變量。例…

    編程 2025-04-29
  • 百度地區熱力圖的介紹和使用方法

    本文將詳細介紹百度地區熱力圖的使用方法和相關知識。 一、什麼是百度地區熱力圖 百度地區熱力圖是一種用於展示區域內某種數據分佈情況的地圖呈現方式。它通過一張地圖上不同區域的顏色深淺,…

    編程 2025-04-29
  • Python強制轉型的實現方法和應用場景

    本文主要介紹Python強制轉型的實現方法和應用場景。Python強制轉型,也叫類型轉換,是指將一種數據類型轉換為另一種數據類型。在Python中,強制轉型主要通過類型構造函數、轉…

    編程 2025-04-29
  • Matlab中addpath的使用方法

    addpath函數是Matlab中的一個非常常用的函數,它可以在Matlab環境中增加一個或者多個文件夾的路徑,使得Matlab可以在需要時自動搜索到這些文件夾中的函數。因此,學會…

    編程 2025-04-29

發表回復

登錄後才能評論