Spring Security Config : HttpSecurity安全配置器 CorsConfigurer

本文深入解析了Spring Security中CorsConfigurer的配置逻辑与实现细节,包括如何通过不同的方式配置CorsFilter,以及与SpringMVC的集成关系。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

概述

介绍

作为一个配置HttpSecuritySecurityConfigurer,CorsConfigurer的配置任务如下 :

  • 配置如下安全过滤器Filter
    • CorsFilter
      1. 如果CorsConfigurer上设置了属性configurationSource,则基于它创建一个CorsFilter并使用;
      2. 否则如果名称为corsFilterbean存在,则使用该CorsFilter;
      3. 否则如果名称为corsConfigurationSourcebean存在,则基于它创建一个CorsFilter并使用;
      4. 否则如果名称为mvcHandlerMappingIntrospectorbean存在,则基于它创建一个CorsFilter并使用。

        名称为mvcHandlerMappingIntrospectorbean存在表明Spring MVC被使用,因为使用了Spring MVC的话,Spring MVC的配置机制(WebMvcConfigurationSupport)会注册一个类型为HandlerMappingIntrospector名称为mvcHandlerMappingIntrospectorbean到容器。

继承关系

CorsConfigurer

使用

	// HttpSecurity 代码片段
    public CorsConfigurer<HttpSecurity> cors() throws Exception {
		return getOrApply(new CorsConfigurer<>());
	}

源代码

源代码版本 Spring Security Config 5.1.4.RELEASE

package org.springframework.security.config.annotation.web.configurers;

// 省略 imports


public class CorsConfigurer<H extends HttpSecurityBuilder<H>>
		extends AbstractHttpConfigurer<CorsConfigurer<H>, H> {

	private static final String HANDLER_MAPPING_INTROSPECTOR = 
		"org.springframework.web.servlet.handler.HandlerMappingIntrospector";
	private static final String CORS_CONFIGURATION_SOURCE_BEAN_NAME = "corsConfigurationSource";
	private static final String CORS_FILTER_BEAN_NAME = "corsFilter";

	private CorsConfigurationSource configurationSource;

	/**
	 * Creates a new instance
	 *
	 * @see HttpSecurity#cors()
	 */
	public CorsConfigurer() {
	}

	public CorsConfigurer<H> configurationSource(
			CorsConfigurationSource configurationSource) {
		this.configurationSource = configurationSource;
		return this;
	}

	@Override
	public void configure(H http) throws Exception {
		ApplicationContext context = http.getSharedObject(ApplicationContext.class);

		CorsFilter corsFilter = getCorsFilter(context);
		if (corsFilter == null) {
			throw new IllegalStateException(
					"Please configure either a " + CORS_FILTER_BEAN_NAME + " bean or a "
							+ CORS_CONFIGURATION_SOURCE_BEAN_NAME + "bean.");
		}
		http.addFilter(corsFilter);
	}

	private CorsFilter getCorsFilter(ApplicationContext context) {
		if (this.configurationSource != null) {
			return new CorsFilter(this.configurationSource);
		}

		boolean containsCorsFilter = context
				.containsBeanDefinition(CORS_FILTER_BEAN_NAME);
		if (containsCorsFilter) {
			return context.getBean(CORS_FILTER_BEAN_NAME, CorsFilter.class);
		}

		boolean containsCorsSource = context
				.containsBean(CORS_CONFIGURATION_SOURCE_BEAN_NAME);
		if (containsCorsSource) {
			CorsConfigurationSource configurationSource = context.getBean(
					CORS_CONFIGURATION_SOURCE_BEAN_NAME, CorsConfigurationSource.class);
			return new CorsFilter(configurationSource);
		}

		boolean mvcPresent = ClassUtils.isPresent(HANDLER_MAPPING_INTROSPECTOR,
				context.getClassLoader());
		if (mvcPresent) {
			return MvcCorsFilter.getMvcCorsFilter(context);
		}
		return null;
	}


	static class MvcCorsFilter {
		private static final String HANDLER_MAPPING_INTROSPECTOR_BEAN_NAME = 
			"mvcHandlerMappingIntrospector";
		/**
		 * This needs to be isolated into a separate class as Spring MVC is an optional
		 * dependency and will potentially cause ClassLoading issues
		 * @param context
		 * @return
		 */
		private static CorsFilter getMvcCorsFilter(ApplicationContext context) {
			if (!context.containsBean(HANDLER_MAPPING_INTROSPECTOR_BEAN_NAME)) {
				throw new NoSuchBeanDefinitionException(HANDLER_MAPPING_INTROSPECTOR_BEAN_NAME, 
				"A Bean named " + HANDLER_MAPPING_INTROSPECTOR_BEAN_NAME +" of type " 
				+ HandlerMappingIntrospector.class.getName()
				+ " is required to use MvcRequestMatcher. Please ensure Spring Security & Spring MVC "
				+ "are configured in a shared ApplicationContext.");
			}
			HandlerMappingIntrospector mappingIntrospector = 
				context.getBean(HANDLER_MAPPING_INTROSPECTOR_BEAN_NAME, 
				HandlerMappingIntrospector.class);
			return new CorsFilter(mappingIntrospector);
		}
	}
}

参考文章

### Spring Web 框架使用教程及常见问题 #### Spring Web 概述 Spring Web 是 Spring Framework 的一个重要模块,专注于构建基于 HTTP 协议的应用程序。它提供了处理请求响应的核心功能以及与其他组件集成的能力[^1]。 以下是关于 Spring Web 的一些核心概念和常见问题解答: --- #### 1. **创建一个简单的 Spring Web 应用** 要快速启动并运行一个 Spring Web 应用,可以通过以下方式实现: - 使用 Spring Initializr 工具初始化项目。 - 添加 `spring-boot-starter-web` 依赖项到项目的 Maven 或 Gradle 文件中。 Maven 示例配置如下所示: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> ``` Gradle 示例配置如下所示: ```gradle implementation 'org.springframework.boot:spring-boot-starter-web' ``` 通过上述依赖引入后,即可利用 Spring MVC 提供的功能开发 RESTful API 和其他 Web 功能。 --- #### 2. **如何测试 Spring Web 控制** 为了确保控制逻辑正常工作,可以借助 JUnit 进行单元测试。推荐使用 `@WebMvcTest` 注解来加载特定的 Controller 并模拟 HTTP 请求。 示例代码如下: ```java import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; import org.springframework.http.MediaType; import org.springframework.test.web.servlet.MockMvc; import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; @WebMvcTest(MyController.class) public class MyControllerTests { @Autowired private MockMvc mockMvc; @Test public void testGetEndpoint() throws Exception { this.mockMvc.perform(get("/api/resource").accept(MediaType.APPLICATION_JSON)) .andExpect(status().isOk()); } } ``` 此方法适用于轻量级的接口测试场景。 --- #### 3. **Spring Web 中的安全性支持** 对于现代应用程序而言,安全性至关重要。Spring Security 可以为 Spring Web 提供全面的身份认证与授权机制。其典型应用场景包括但不限于传统 Web 应用、前后端分离架构下的 Token 认证以及微服务中的网关安全控制[^2]。 例如,在 URL 身份验证方面,可通过定义权限规则保护资源访问路径。具体配置可参考以下代码片段: ```java @Configuration @EnableWebSecurity public class SecurityConfig extends WebSecurityConfigurerAdapter { @Override protected void configure(HttpSecurity http) throws Exception { http.authorizeRequests() .antMatchers("/admin/**").hasRole("ADMIN") // 仅允许管理员角色访问 /admin/* .anyRequest().authenticated(); // 所有其他请求需登录才能访问 } } ``` --- #### 4. **常见的性能优化建议** 当面对高并发流量时,可以从以下几个角度提升 Spring Web 性能: - 合理设置线程池大小以匹配服务硬件能力; - 利用缓存技术减少数据库查询次数; - 对静态文件启用浏览缓存策略; 这些措施能够显著改善用户体验并降低系统负载。 --- #### 5. **跨域资源共享 (CORS)** 处理方案 如果前端应用部署于不同域名下,则需要正确配置 CORS 政策以便顺利调用后端接口。一种简单的方式是在全局范围内开启 CORS 支持: ```java @Bean public WebMvcConfigurer corsConfigurer() { return new WebMvcConfigurer() { @Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping("/**") .allowedOrigins("*") .allowedMethods("GET", "POST", "PUT", "DELETE"); } }; } ``` 注意生产环境中应避免使用通配符 "*" 来指定允许来源地址列表。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值