当前位置: 首页 > news >正文

spring-security原理与应用系列:总体流程

目录

1.简单的使用示例

2.大致的流程

EnableWebSecurity

WebSecurityConfiguration

AutowiredWebSecurityConfigurersIgnoreParents

小结


1.简单的使用示例

        首先,新建一个config包用于存放spring-security通用配置;然后,新建一个WebSecurityConfig类,使其继承WebSecurityConfigurerAdapter。

        然后,给WebSecutiryConfig类中加上@EnableWebSecurity 注解后,这样便会自动被 Spring发现并注册。

@EnableWebSecurity
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {
   @Override
   protected void configure(HttpSecurity http) throws Exception {
      http.authorizeHttpRequests()

            .anyRequest().authenticated()

            .and()

            .formLogin(form -> form

                .loginPage("/myLogin.html")

                .permitAll())

            .logout(logout -> logout

                .permitAll())

            .csrf().disable();

        return http.build();

}

        现在重启服务,应用新的安全配置。在下次访问localhost:8080时,系统会要求我们进行表单认证。我们访问的地址会自动跳转到localhost:8080/myLogin.html,只要输入正确的用户名和密码便可跳转回原访问地址。

2.大致的流程

        一般在使用 Spring Security 框架时,我们会自定义一个配置类,配置类加上 @EnableWebSecurity 注解,重新启动程序,一个使用了安全框架的项目就起来了。

        点击自定义配置类上添加的 @EnableWebSecurity,如下所示:

EnableWebSecurity

@Retention(value = java.lang.annotation.RetentionPolicy.RUNTIME)
@Target(value = { java.lang.annotation.ElementType.TYPE })
@Documented
@Import({ WebSecurityConfiguration.class, SpringWebMvcImportSelector.class })
@EnableGlobalAuthentication
@Configuration
public @interface EnableWebSecurity {
   /**
    * Controls debugging support for Spring Security. Default is false.
    * @return if true, enables debug support with Spring Security
    */
   boolean debug() default false;
}

        重点关注 @Import 导入的类 WebSecurityConfiguration 。

        点击@Import 导入的类 WebSecurityConfiguration,如下所示:

WebSecurityConfiguration

    在这里,发现类中定义了很多Bean,其中就有跟安全管理相关的核心过滤器springSecurityFilterChain。

@Configuration
public class WebSecurityConfiguration implements ImportAware, BeanClassLoaderAware {

... ...

@Bean(name = AbstractSecurityWebApplicationInitializer.DEFAULT_FILTER_NAME)
public Filter springSecurityFilterChain() throws Exception {
   boolean hasConfigurers = webSecurityConfigurers != null
         && !webSecurityConfigurers.isEmpty();
   if (!hasConfigurers) {
      WebSecurityConfigurerAdapter adapter = objectObjectPostProcessor
            .postProcess(new WebSecurityConfigurerAdapter() {
            });
      webSecurity.apply(adapter);
   }
   return webSecurity.build();
}

        这个方法返回的是Filter,说明这个Bean是一个过滤器。

        webSecurity.build():这行代码说明这个Filter是通过webSecurity创建的,而且webSecurity是通过建造者模式来构建Filter对象的。后续再深入了解webSecurity建造者的构建过程。

        webSecurity.apply(adapter):这行代码是构建者webSecurity使用一个配置器。hasConfigurers为空时,new一个WebSecurityConfigurerAdapter对象,不为空时,查看如下的setFilterChainProxySecurityConfigurer方法。

@Autowired(required = false)
public void setFilterChainProxySecurityConfigurer(ObjectPostProcessor<Object> objectPostProcessor,   @Value("#{@autowiredWebSecurityConfigurersIgnoreParents.getWebSecurityConfigurers()}") List<SecurityConfigurer<Filter, WebSecurity>> webSecurityConfigurers)
      throws Exception {
   webSecurity = objectPostProcessor
         .postProcess(new WebSecurity(objectPostProcessor));
   if (debugEnabled != null) {
      webSecurity.debug(debugEnabled);
   }
   Collections.sort(webSecurityConfigurers, AnnotationAwareOrderComparator.INSTANCE);
   Integer previousOrder = null;
   Object previousConfig = null;
   for (SecurityConfigurer<Filter, WebSecurity> config : webSecurityConfigurers) {
      Integer order = AnnotationAwareOrderComparator.lookupOrder(config);
      if (previousOrder != null && previousOrder.equals(order)) {
         ... ...

      }
      previousOrder = order;
      previousConfig = config;
   }
   for (SecurityConfigurer<Filter, WebSecurity> webSecurityConfigurer : webSecurityConfigurers)   

   {
      webSecurity.apply(webSecurityConfigurer);
   }
   this.webSecurityConfigurers = webSecurityConfigurers;
}

       在这里,通过autowiredWebSecurityConfigurersIgnoreParents.getWebSecurityConfigurers()获取配置器,然后建造者webSecurity再使用这些配置器。

     点击autowiredWebSecurityConfigurersIgnoreParents.getWebSecurityConfigurers(),如下所示:

AutowiredWebSecurityConfigurersIgnoreParents

final class AutowiredWebSecurityConfigurersIgnoreParents {

... ...

public List<SecurityConfigurer<Filter, WebSecurity>> getWebSecurityConfigurers() {
   List<SecurityConfigurer<Filter, WebSecurity>> webSecurityConfigurers = new ArrayList<SecurityConfigurer<Filter, WebSecurity>>();
   Map<String, WebSecurityConfigurer> beansOfType = beanFactory
         .getBeansOfType(WebSecurityConfigurer.class);
   for (Entry<String, WebSecurityConfigurer> entry : beansOfType.entrySet()) {
      webSecurityConfigurers.add(entry.getValue());
   }
   return webSecurityConfigurers;
}

       在这里,通过beanFactory获取WebSecurityConfigurer实例,这些实例正是我们使用spring-security所自定义的配置类WebSecurityConfig。

小结

        整个流程主要由建造者和配置器构成,在服务启动时就是通过配置器对建造者所要构建的对象进行配置,配置完成再由建造者构建出核心过滤器springSecurityFilterChain。

相关文章:

  • Web Workers 教程
  • 前端知识点---innerHTML和innerText
  • Turtle基本操作(前进、后退、旋转)
  • QT零基础学习之路(十)--QDialog对话框的使用及信息传递
  • el-tree树多选,将选中的树对象中某个字段值改为true,并过滤出所有为true的对象,组成新的数组
  • 开源图生视频模型技术全景解析
  • QT学习笔记(对话框)
  • Next.Js 权限绕过漏洞复现(附脚本)(CVE-2025-29927)
  • Vue打包后如何在本地进行测试(附解决浏览器刷新无法访问的问题)
  • 【数据库-复试】sql语句综合练习
  • Mysql--日志(错误日志、二进制日志、查询日志、慢查询日志)
  • 使用 fn_dblog手动恢复误操作的 update(单列数值型数据恢复)
  • 用卡片笔记要改变写作习惯
  • (并查集 省份数量)leetcode 547
  • Sqladmin - FastAPI框架下一键生成管理后台
  • Git 钩子:特定操作脚本
  • 深入掌握Spring AOP:从原理到实战的完整指南
  • 在 Qt 中,不带参数或整形的参选的信号能够从 std::thread 发送成功,而带枚举离线的信号却发送失败
  • cocos creator 笔记-路边花草
  • java8循环解压zip文件---实现Excel文件数据追加
  • 复旦大学艺术馆开馆:以当代视角再看文科文脉
  • 广西鹿寨一水文站“倒刺扶手”存安全隐患,官方通报处理情况
  • 菲律宾选举委员会公布中期选举结果,马科斯阵营选情未达预期
  • 著名文博专家吴远明因交通事故离世,享年75岁
  • 上海国际珠宝时尚功能区未来三年如何建设?六大行动将开展
  • 马上评|重病老人取款身亡,如何避免类似悲剧?