Spring 裏那麼多種 CORS 的配置方式,到底有什麼區別

做爲一個後端開發,咱們常常遇到的一個問題就是須要配置 CORS,好讓咱們的前端可以訪問到咱們的 API,而且不讓其餘人訪問。而在 Spring 中,咱們見過不少種 CORS 的配置,不少資料都只是告訴咱們能夠這樣配置、能夠那樣配置,可是這些配置有什麼區別?前端

CORS 是什麼

首先咱們要明確,CORS 是什麼,以及規範是如何要求的。這裏只是梳理一下流程,具體的規範請看 這裏java

CORS 全稱是 Cross-Origin Resource Sharing,直譯過來就是跨域資源共享。要理解這個概念就須要知道資源同源策略這三個概念。git

  • 域,指的是一個站點,由 protocalhostport 三部分組成,其中 host 能夠是域名,也能夠是 ipport 若是沒有指明,則是使用 protocal 的默認端口
  • 資源,是指一個 URL 對應的內容,能夠是一張圖片、一種字體、一段 HTML 代碼、一份 JSON 數據等等任何形式的任何內容
  • 同源策略,指的是爲了防止 XSS,瀏覽器、客戶端應該僅請求與當前頁面來自同一個域的資源,請求其餘域的資源須要經過驗證。

瞭解了這三個概念,咱們就能理解爲何有 CORS 規範了:從站點 A 請求站點 B 的資源的時候,因爲瀏覽器的同源策略的影響,這樣的跨域請求將被禁止發送;爲了讓跨域請求可以正常發送,咱們須要一套機制在不破壞同源策略的安全性的狀況下、容許跨域請求正常發送,這樣的機制就是 CORSgithub

預檢請求

CORS 中,定義了一種預檢請求,即 preflight request,當實際請求不是一個 簡單請求 時,會發起一次預檢請求。預檢請求是針對實際請求的 URL 發起一次 OPTIONS 請求,並帶上下面三個 headersspring

  • Origin:值爲當前頁面所在的域,用於告訴服務器當前請求的域。若是沒有這個 header,服務器將不會進行 CORS 驗證。
  • Access-Control-Request-Method:值爲實際請求將會使用的方法
  • Access-Control-Request-Headers:值爲實際請求將會使用的 header 集合

若是服務器端 CORS 驗證失敗,則會返回客戶端錯誤,即 4xx 的狀態碼。後端

不然,將會請求成功,返回 200 的狀態碼,並帶上下面這些 headers跨域

  • Access-Control-Allow-Origin:容許請求的域,多數狀況下,就是預檢請求中的 Origin 的值
  • Access-Control-Allow-Credentials:一個布爾值,表示服務器是否容許使用 cookies
  • Access-Control-Expose-Headers:實際請求中能夠出如今響應中的 headers 集合
  • Access-Control-Max-Age:預檢請求返回的規則能夠被緩存的最長時間,超過這個時間,須要再次發起預檢請求
  • Access-Control-Allow-Methods:實際請求中可使用到的方法集合

瀏覽器會根據預檢請求的響應,來決定是否發起實際請求。瀏覽器

小結

到這裏, 咱們就知道了跨域請求會經歷的故事:緩存

  1. 訪問另外一個域的資源
  2. 有可能會發起一次預檢請求(非簡單請求,或超過了 Max-Age
  3. 發起實際請求

接下來,咱們看看在 Spring 中,咱們是如何讓 CORS 機制在咱們的應用中生效的。安全

幾種配置的方式

Spring 提供了多種配置 CORS 的方式,有的方式針對單個 API,有的方式能夠針對整個應用;有的方式在一些狀況下是等效的,而在另外一些狀況下卻又出現不一樣。咱們這裏例舉幾種典型的方式來看看應該如何配置。

假設咱們有一個 API:

@RestController
class HelloController {
    @GetMapping("hello")
    fun hello(): String {
        return "Hello, CORS!"
    }
}

@CrossOrigin 註解

使用@CorssOrigin 註解須要引入 Spring Web 的依賴,該註解能夠做用於方法或者類,能夠針對這個方法或類對應的一個或多個 API 配置 CORS 規則:

@RestController
class HelloController {
    @GetMapping("hello")
    @CrossOrigin(origins = ["http://localhost:8080"])
    fun hello(): String {
        return "Hello, CORS!"
    }
}

實現 WebMvcConfigurer.addCorsMappings 方法

WebMvcConfigurer 是一個接口,它一樣來自於 Spring Web。咱們能夠經過實現它的 addCorsMappings 方法來針對全局 API 配置 CORS 規則:

@Configuration
@EnableWebMvc
class MvcConfig: WebMvcConfigurer {
    override fun addCorsMappings(registry: CorsRegistry) {
        registry.addMapping("/hello")
                .allowedOrigins("http://localhost:8080")
    }
}

注入 CorsFilter

CorsFilter 一樣來自於 Spring Web,可是實現 WebMvcConfigurer.addCorsMappings 方法並不會使用到這個類,具體緣由咱們後面來分析。咱們能夠經過注入一個 CorsFilter 來使用它:

@Configuration
class CORSConfiguration {
    @Bean
    fun corsFilter(): CorsFilter {
        val configuration = CorsConfiguration()
        configuration.allowedOrigins = listOf("http://localhost:8080")
        val source = UrlBasedCorsConfigurationSource()
        source.registerCorsConfiguration("/hello", configuration)
        return CorsFilter(source)
    }
}

注入 CorsFilter 不止這一種方式,咱們還能夠經過注入一個 FilterRegistrationBean 來實現,這裏就不給例子了。

在僅僅引入 Spring Web 的狀況下,實現 WebMvcConfigurer.addCorsMappings 方法和注入 CorsFilter 這兩種方式能夠達到一樣的效果,二選一便可。它們的區別會在引入 Spring Security 以後會展示出來,咱們後面再來分析。

Spring Security 中的配置

在引入了 Spring Security 以後,咱們會發現前面的方法都不能正確的配置 CORS,每次 preflight request 都會獲得一個 401 的狀態碼,表示請求沒有被受權。這時,咱們須要增長一點配置才能讓 CORS 正常工做:

@Configuration
class SecurityConfig : WebSecurityConfigurerAdapter() {
    override fun configure(http: HttpSecurity?) {
        http?.cors()
    }
}

或者,乾脆不實現 WebMvcConfigurer.addCorsMappings 方法或者注入 CorsFilter ,而是注入一個 CorsConfigurationSource ,一樣能與上面的代碼配合,正確的配置 CORS

@Bean
fun corsConfigurationSource(): CorsConfigurationSource {
    val configuration = CorsConfiguration()
    configuration.allowedOrigins = listOf("http://localhost:8080")
    val source = UrlBasedCorsConfigurationSource()
    source.registerCorsConfiguration("/hello", configuration)
    return source
}

到此,咱們已經看過了幾種典型的例子了,完整的內容能夠在 Demo 中查看,咱們接下來看看 Spring 究竟是如何實現 CORS 驗證的。

這些配置有什麼區別

咱們會主要分析實現 WebMvcConfigurer.addCorsMappings 方法和調用 HttpSecurity.cors 方法這兩種方式是如何實現 CORS 的,但在進行以前,咱們要先複習一下 FilterInterceptor 的概念。

Filter 與 Interceptor

Spring Web 調用關係

上圖很形象的說明了 FilterInterceptor 的區別,一個做用在 DispatcherServlet 調用前,一個做用在調用後。

但實際上,它們自己並無任何關係,是徹底獨立的概念。

FilterServlet 標準定義,要求 Filter 須要在 Servlet 被調用以前調用,做用顧名思義,就是用來過濾請求。在 Spring Web 應用中,DispatcherServlet 就是惟一的 Servlet 實現。

Interceptor 由 Spring 本身定義,由 DispatcherServlet 調用,能夠定義在 Handler 調用先後的行爲。這裏的 Handler ,在多數狀況下,就是咱們的 Controller 中對應的方法。

對於 FilterInterceptor 的複習就到這裏,咱們只須要知道它們會在何時被調用到,就能理解後面的內容了。

WebMvcConfigurer.addCorsMappings 方法作了什麼

咱們從 WebMvcConfigurer.addCorsMappings 方法的參數開始,先看看 CORS 配置是如何保存到 Spring 上下文中的,而後在瞭解一下 Spring 是如何使用的它們。

注入 CORS 配置

CorsRegistry 和 CorsRegistration

WebMvcConfigurer.addCorsMappings 方法的參數 CorsRegistry 用於註冊 CORS 配置,它的源碼以下:

public class CorsRegistry {
    private final List<CorsRegistration> registrations = new ArrayList<>();

    public CorsRegistration addMapping(String pathPattern) {
        CorsRegistration registration = new CorsRegistration(pathPattern);
        this.registrations.add(registration);
        return registration;
    }

    protected Map<String, CorsConfiguration> getCorsConfigurations() {
        Map<String, CorsConfiguration> configs = new LinkedHashMap<>(this.registrations.size());
        for (CorsRegistration registration : this.registrations) {
            configs.put(registration.getPathPattern(), registration.getCorsConfiguration());
        }
        return configs;
    }
}

咱們發現這個類僅僅有兩個方法:

  • addMapping 接收一個 pathPattern,建立一個 CorsRegistration 實例,保存到列表後將其返回。在咱們的代碼中,這裏的 pathPattern 就是 /hello
  • getCorsConfigurations 方法將保存的 CORS 規則轉換成 Map 後返回

CorsRegistration 這個類,一樣很簡單,咱們看看它的部分源碼:

public class CorsRegistration {
    private final String pathPattern;
    private final CorsConfiguration config;


    public CorsRegistration(String pathPattern) {
        this.pathPattern = pathPattern;
        this.config = new CorsConfiguration().applyPermitDefaultValues();
    }

    public CorsRegistration allowedOrigins(String... origins) {
        this.config.setAllowedOrigins(Arrays.asList(origins));
        return this;
    }
}

不難發現,這個類僅僅保存了一個 pathPattern 字符串和 CorsConfiguration,很好理解,它保存的是一個 pathPattern 對應的 CORS 規則。

在它的構造函數中,調用的 CorsConfiguration.applyPermitDefaultValues 方法則用於配置默認的 CORS 規則:

  • allowedOrigins 默認爲全部域
  • allowedMethods 默認爲 GETHEADPOST
  • allowedHeaders 默認爲全部
  • maxAge 默認爲 30 分鐘
  • exposedHeaders 默認爲 null,也就是不暴露任何 header
  • credentials 默認爲 null

建立 CorsRegistration 後,咱們能夠經過它的 allowedOriginsallowedMethods 等方法修改它的 CorsConfiguration,覆蓋掉上面的默認值。

如今,咱們已經經過 WebMvcConfigurer.addCorsMappings 方法配置好 CorsRegistry 了,接下來看看這些配置會在什麼地方被注入到 Spring 上下文中。

WebMvcConfigurationSupport

CorsRegistry.getCorsConfigurations 方法,會被 WebMvcConfigurationSupport.getConfigurations 方法調用,這個方法以下:

protected final Map<String, CorsConfiguration> getCorsConfigurations() {
    if (this.corsConfigurations == null) {
        CorsRegistry registry = new CorsRegistry();
        addCorsMappings(registry);
        this.corsConfigurations = registry.getCorsConfigurations();
    }
    return this.corsConfigurations;
}
addCorsMappings(registry) 調用的是本身的方法,由子類 DelegatingWebMvcConfiguration 經過委託的方式調用到 WebMvcConfigurer.addCorsMappings 方法,咱們的配置也由此被讀取到。

getCorsConfigurations 是一個 protected 方法,是爲了在擴展該類時,仍然可以直接獲取到 CORS 配置。而這個方法在這個類裏被四個地方調用到,這四個調用的地方,都是爲了註冊一個 HandlerMapping 到 Spring 容器中。每個地方都會調用 mapping.setCorsConfigurations 方法來接收 CORS 配置,而這個 setCorsConfigurations 方法,則由 AbstractHandlerMapping 提供,CorsConfigurations 也被保存在這個抽象類中。

到此,咱們的 CORS 配置藉由 AbstractHandlerMapping 被注入到了多個 HandlerMapping 中,而這些 HandlerMapping 以 Spring 組件的形式被註冊到了 Spring 容器中,當請求來臨時,將會被調用。

獲取 CORS 配置

還記得前面關於 FilterInterceptor 那張圖嗎?當請求來到 Spring Web 時,必定會到達 DispatcherServlet 這個惟一的 Servlet

DispatcherServlet.doDispatch 方法中,會調用全部 HandlerMapping.getHandler 方法。好巧不巧,這個方法又是由 AbstractHandlerMapping 實現的:

@Override
@Nullable
public final HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
    // 省略代碼
    if (CorsUtils.isCorsRequest(request)) {
        CorsConfiguration globalConfig = this.corsConfigurationSource.getCorsConfiguration(request);
        CorsConfiguration handlerConfig = getCorsConfiguration(handler, request);
        CorsConfiguration config = (globalConfig != null ? globalConfig.combine(handlerConfig) : handlerConfig);
        executionChain = getCorsHandlerExecutionChain(request, executionChain, config);
    }
    return executionChain;
}

在這個方法中,關於 CORS 的部分都在這個 if 中。咱們來看看最後這個 getCorsHandlerExecutionChain 作了什麼:

protected HandlerExecutionChain getCorsHandlerExecutionChain(HttpServletRequest request,
        HandlerExecutionChain chain, @Nullable CorsConfiguration config) {
    if (CorsUtils.isPreFlightRequest(request)) {
        HandlerInterceptor[] interceptors = chain.getInterceptors();
        chain = new HandlerExecutionChain(new PreFlightHandler(config), interceptors);
    }
    else {
        chain.addInterceptor(new CorsInterceptor(config));
    }
    return chain;
}

能夠看到:

  • 針對 preflight request,因爲不會有對應的 Handler 來處理,因此這裏就建立了一個 PreFlightHandler 來做爲此次請求的 handler
  • 對於其餘的跨域請求,由於會有對應的 handler,因此就在 handlerExecutionChain 中加入一個 CorsInterceptor 來進行 CORS 驗證

這裏的 PreFlightHandlerCorsInterceptor 都是 AbstractHandlerMapping 的內部類,實現幾乎一致,區別僅僅在於一個是 HttpRequestHandler,一個是 HandlerInterceptor;它們對 CORS 規則的驗證都交由 CorsProcessor 接口完成,這裏採用了默認實現 DefaultCorsProcessor

DefaultCorsProcessor 則是依照 CORS 標準來實現,並在驗證失敗的時候打印 debug 日誌並拒絕請求。咱們只須要關注一下標準中沒有定義的驗證失敗時的狀態碼:

protected void rejectRequest(ServerHttpResponse response) throws IOException {
    response.setStatusCode(HttpStatus.FORBIDDEN);
    response.getBody().write("Invalid CORS request".getBytes(StandardCharsets.UTF_8));
}

CORS 驗證失敗時調用這個方法,並設置狀態碼爲 403


小結

經過對源碼的研究,咱們發現實現 WebMvcConfigurer.addCorsMappings 方法的方式配置 CORS,會在 Interceptor 或者 Handler 層進行 CORS 驗證。

HtttpSecurity.cors 方法作了什麼

在研究這個方法的行爲以前,咱們先來回想一下,咱們調用這個方法解決的是什麼問題。

前面咱們經過某種方式配置好 CORS 後,引入 Spring SecurityCORS 就失效了,直到調用這個方法後,CORS 規則才從新生效。

下面這些緣由,致使了 preflight request 沒法經過身份驗證,從而致使 CORS 失效:

  1. preflight request 不會攜帶認證信息
  2. Spring Security 經過 Filter 來進行身份驗證
  3. InterceptorHttpRequestHanlderDispatcherServlet 以後被調用
  4. Spring Security 中的 Filter 優先級比咱們注入的 CorsFilter 優先級高

接下來咱們就來看看 HttpSecurity.cors 方法是如何解決這個問題的。

CorsConfigurer 如何配置 CORS 規則

HttpSecurity.cors 方法中其實只有一行代碼:

public CorsConfigurer<HttpSecurity> cors() throws Exception {
    return getOrApply(new CorsConfigurer<>());
}

這裏調用的 getOrApply 方法會將 SecurityConfigurerAdapter 的子類實例加入到它的父類 AbstractConfiguredSecurityBuilder 維護的一個 Map 中,而後一個個的調用 configure 方法。因此,咱們來關注一下 CorsConfigurer.configure 方法就行了。

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

這段代碼很好理解,就是在當前的 Spring Context 中找到一個 CorsFilter,而後將它加入到 http 對象的 filters 中。由上面的 HttpSecurity.cors 方法可知,這裏的 http 對象實際類型就是 HttpSecurity

getCorsFilter 方法作了什麼

也許你會好奇,HttpSecurity 要如何保證 CorsFilter 必定在 Spring SecurityFilters 以前調用。可是在研究這個以前,咱們先來看看一樣重要的 getCorsFilter 方法,這裏能夠解答咱們前面的一些疑問。

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

這是 CorsConfigurer 尋找 CorsFilter 的所有邏輯,咱們用人話來講就是:

  1. CorsConfigurer 本身是否有配置 CorsConfigurationSource,若是有的話,就用它建立一個 CorsFilter
  2. 在當前的上下文中,是否存在一個名爲 corsFilter 的實例,若是有的話,就把他看成一個 CorsFilter 來用。
  3. 在當前的上下文中,是否存在一個名爲 corsConfigurationSourceCorsConfigurationSource 實例,若是有的話,就用它建立一個 CorsFilter
  4. 在當前上下文的類加載器中,是否存在類 HandlerMappingIntrospector,若是有的話,則經過 MvcCorsFilter 這個內部類建立一個 CorsFilter
  5. 若是沒有找到,那就返回一個 null,調用的地方最後會拋出異常,阻止 Spring 初始化。

上面的第 二、三、4 步能解答咱們前面的配置爲何生效,以及它們的區別。

註冊 CorsFilter 的方式,這個 Filter 最終會被直接註冊到 Servlet container 中被使用到。

註冊 CorsConfigurationSource 的方式,會用這個 source 建立一個 CorsFiltet 而後註冊到 Servlet container 中被使用到。

而第四步的狀況比較複雜。HandlerMappingIntrospectorSpring Web 提供的一個類,實現了 CorsConfigurationSource 接口,因此在 MvcCorsFilter 中,它被直接用於建立 CorsFilter。它實現的 getCorsConfiguration 方法,會經歷:

  1. 遍歷 HandlerMapping
  2. 調用 getHandler 方法獲得 HandlerExecutionChain
  3. 從中找到 CorsConfigurationSource 的實例
  4. 調用這個實例的 getCorsConfiguration 方法,返回獲得的 CorsConfiguration

因此獲得的 CorsConfigurationSource 實例,實際上就是前面講到的 CorsInterceptor 或者 PreFlightHandler

因此第四步實際上匹配的是實現 WebMvcConfigurer.addCorsMappings 方法的方式。

因爲在 CorsFilter 中每次處理請求時都會調用 CorsConfigurationSource.getCorsConfiguration 方法,而 DispatcherServlet 中也會每次調用 HandlerMapping.getHandler 方法,再加上這時的 HandlerExecutionChain 中還有 CorsInterceptor,因此使用這個方式相對於其餘方式,作了不少重複的工做。因此 WebMvcConfigurer.addCorsMappings + HttpSecurity.cors 的方式下降了咱們代碼的效率,也許微乎其微,但能避免的狀況下,仍是不要使用。

HttpSecurity 中的 filters 屬性

CorsConfigurer.configure 方法中調用的 HttpSecurity.addFilter 方法,由它的父類 HttpSecurityBuilder 聲明,並約定了不少 Filter 的順序。然而 CorsFilter 並不在其中。不過在 Spring Security 中,目前還只有 HttpSecurity 這一個實現,因此咱們來看看這裏的代碼實現就知道 CorsFilter 會排在什麼地方了。

public HttpSecurity addFilter(Filter filter) {
    Class<? extends Filter> filterClass = filter.getClass();
    if (!comparator.isRegistered(filterClass)) {
        throw new IllegalArgumentException("...");
    }
    this.filters.add(filter);
    return this;
}

咱們能夠看到,Filter 會被直接加到 List 中,而不是按照必定的順序來加入的。但同時,咱們也發現了一個 comparator 對象,而且只有被註冊到了該類的 Filter 才能被加入到 filters 屬性中。這個 comparator 又是用來作什麼的呢?

在 Spring Security 建立過程當中,會調用到 HttpSeciryt.performBuild 方法,在這裏咱們能夠看到 filterscomparator 是如何被使用到的。

protected DefaultSecurityFilterChain performBuild() throws Exception {
    Collections.sort(filters, comparator);
    return new DefaultSecurityFilterChain(requestMatcher, filters);
}

能夠看到,Spring Security 使用了這個 comparator 在獲取 SecurityFilterChain 的時候來保證 filters 的順序,因此,研究這個 comparator 就能知道在 SecurityFilterChain 中的那些 Filter 的順序是如何的了。

這個 comparator 的類型是 FilterComparator ,從名字就能看出來是專用於 Filter 比較的類,它的實現也並不神祕,從構造函數就能猜到是如何實現的:

FilterComparator() {
    Step order = new Step(INITIAL_ORDER, ORDER_STEP);
    put(ChannelProcessingFilter.class, order.next());
    put(ConcurrentSessionFilter.class, order.next());
    put(WebAsyncManagerIntegrationFilter.class, order.next());
    put(SecurityContextPersistenceFilter.class, order.next());
    put(HeaderWriterFilter.class, order.next());
    put(CorsFilter.class, order.next());
  // 省略代碼
}

能夠看到 CorsFilter 排在了第六位,在全部的 Security Filter 以前,由此便解決了 preflight request 沒有攜帶認證信息的問題。

小結

引入 Spring Security 以後,咱們的 CORS 驗證明際上是依然運行着的,只是由於 preflight request 不會攜帶認證信息,因此沒法經過身份驗證。使用 HttpSecurity.cors 方法會幫助咱們在當前的 Spring Context 中找到或建立一個 CorsFilter 並安排在身份驗證的 Filter 以前,以保證能對 preflight request 正確處理。

總結

研究了 Spring 中 CORS 的代碼,咱們瞭解到了這樣一些知識:

  • 實現 WebMvcConfigurer.addCorsMappings 方法來進行的 CORS 配置,最後會在 Spring 的 InterceptorHandler 中生效
  • 注入 CorsFilter 的方式會讓 CORS 驗證在 Filter 中生效
  • 引入 Spring Security 後,須要調用 HttpSecurity.cors 方法以保證 CorsFilter 會在身份驗證相關的 Filter 以前執行
  • HttpSecurity.cors + WebMvcConfigurer.addCorsMappings 是一種相對低效的方式,會致使跨域請求分別在 FilterInterceptor 層各經歷一次 CORS 驗證
  • HttpSecurity.cors + 註冊 CorsFilterHttpSecurity.cors + 註冊 CorsConfigurationSource 在運行的時候是等效的
  • 在 Spring 中,沒有經過 CORS 驗證的請求會獲得狀態碼爲 403 的響應
相關文章
相關標籤/搜索