AspNetCore.Docs/aspnetcore/security/cors.md

300 lines
13 KiB
Markdown
Raw Normal View History

---
title: Enable Cross-Origin Requests (CORS) in ASP.NET Core
2016-10-29 01:35:15 +08:00
author: rick-anderson
description: Learn how CORS as a standard for allowing or rejecting cross-origin requests in an ASP.NET Core app.
2018-01-29 23:21:31 +08:00
ms.author: riande
ms.date: 05/17/2017
uid: security/cors
2016-10-29 01:35:15 +08:00
---
# Enable Cross-Origin Requests (CORS) in ASP.NET Core
2016-10-29 01:35:15 +08:00
By [Mike Wasson](https://github.com/mikewasson), [Shayne Boyer](https://twitter.com/spboyer), and [Tom Dykstra](https://github.com/tdykstra)
2016-10-29 01:35:15 +08:00
Browser security prevents a web page from making AJAX requests to another domain. This restriction is called the *same-origin policy*, and prevents a malicious site from reading sensitive data from another site. However, sometimes you might want to let other sites make cross-origin requests to your web API.
2016-10-29 01:35:15 +08:00
[Cross Origin Resource Sharing](http://www.w3.org/TR/cors/) (CORS) is a W3C standard that allows a server to relax the same-origin policy. Using CORS, a server can explicitly allow some cross-origin requests while rejecting others. CORS is safer and more flexible than earlier techniques such as [JSONP](https://wikipedia.org/wiki/JSONP). This topic shows how to enable CORS in an ASP.NET Core application.
2016-10-29 01:35:15 +08:00
## What is "same origin"?
Two URLs have the same origin if they have identical schemes, hosts, and ports. ([RFC 6454](http://tools.ietf.org/html/rfc6454))
These two URLs have the same origin:
* `http://example.com/foo.html`
2016-10-29 01:35:15 +08:00
* `http://example.com/bar.html`
2016-10-29 01:35:15 +08:00
These URLs have different origins than the previous two:
* `http://example.net` - Different domain
2016-10-29 01:35:15 +08:00
* `http://www.example.com/foo.html` - Different subdomain
2016-10-29 01:35:15 +08:00
* `https://example.com/foo.html` - Different scheme
2016-10-29 01:35:15 +08:00
* `http://example.com:9000/foo.html` - Different port
2016-10-29 01:35:15 +08:00
> [!NOTE]
> Internet Explorer doesn't consider the port when comparing origins.
2016-10-29 01:35:15 +08:00
## Setting up CORS
To set up CORS for your application add the `Microsoft.AspNetCore.Cors` package to your project.
2016-10-29 01:35:15 +08:00
Add the CORS services in Startup.cs:
[!code-csharp[](cors/sample/CorsExample1/Startup.cs?name=snippet_addcors)]
2016-10-29 01:35:15 +08:00
## Enabling CORS with middleware
To enable CORS for your entire application add the CORS middleware to your request pipeline using the `UseCors` extension method. Note that the CORS middleware must precede any defined endpoints in your app that you want to support cross-origin requests (ex. before any call to `UseMvc`).
You can specify a cross-origin policy when adding the CORS middleware using the `CorsPolicyBuilder` class. There are two ways to do this. The first is to call UseCors with a lambda:
[!code-csharp[](cors/sample/CorsExample1/Startup.cs?highlight=11,12&range=22-38)]
2016-10-29 01:35:15 +08:00
**Note:** The URL must be specified without a trailing slash (`/`). If the URL terminates with `/`, the comparison will return `false` and no header will be returned.
The lambda takes a `CorsPolicyBuilder` object. You'll find a list of the [configuration options](#cors-policy-options) later in this topic. In this example, the policy allows cross-origin requests from `http://example.com` and no other origins.
2016-10-29 01:35:15 +08:00
Note that CorsPolicyBuilder has a fluent API, so you can chain method calls:
[!code-csharp[](../security/cors/sample/CorsExample3/Startup.cs?highlight=3&range=29-32)]
2016-10-29 01:35:15 +08:00
The second approach is to define one or more named CORS policies, and then select the policy by name at run time.
[!code-csharp[](cors/sample/CorsExample2/Startup.cs?name=snippet_begin)]
2016-10-29 01:35:15 +08:00
This example adds a CORS policy named "AllowSpecificOrigin". To select the policy, pass the name to `UseCors`.
2016-10-29 01:35:15 +08:00
## Enabling CORS in MVC
You can alternatively use MVC to apply specific CORS per action, per controller, or globally for all controllers. When using MVC to enable CORS the same CORS services are used, but the CORS middleware isn't.
2016-10-29 01:35:15 +08:00
### Per action
To specify a CORS policy for a specific action add the `[EnableCors]` attribute to the action. Specify the policy name.
[!code-csharp[](cors/sample/CorsMVC/Controllers/ValuesController.cs?name=EnableOnAction)]
2016-10-29 01:35:15 +08:00
### Per controller
To specify the CORS policy for a specific controller add the `[EnableCors]` attribute to the controller class. Specify the policy name.
[!code-csharp[](cors/sample/CorsMVC/Controllers/ValuesController.cs?name=EnableOnController)]
2016-10-29 01:35:15 +08:00
### Globally
You can enable CORS globally for all controllers by adding the `CorsAuthorizationFilterFactory` filter to the global filter collection:
[!code-csharp[](cors/sample/CorsMVC/Startup2.cs?name=snippet_configureservices)]
2016-10-29 01:35:15 +08:00
The precedence order is: Action, controller, global. Action-level policies take precedence over controller-level policies, and controller-level policies take precedence over global policies.
### Disable CORS
To disable CORS for a controller or action, use the `[DisableCors]` attribute.
[!code-csharp[](cors/sample/CorsMVC/Controllers/ValuesController.cs?name=DisableOnAction)]
2016-10-29 01:35:15 +08:00
## CORS policy options
This section describes the various options that you can set in a CORS policy.
* [Set the allowed origins](#set-the-allowed-origins)
* [Set the allowed HTTP methods](#set-the-allowed-http-methods)
* [Set the allowed request headers](#set-the-allowed-request-headers)
* [Set the exposed response headers](#set-the-exposed-response-headers)
* [Credentials in cross-origin requests](#credentials-in-cross-origin-requests)
* [Set the preflight expiration time](#set-the-preflight-expiration-time)
For some options it may be helpful to read [How CORS works](#how-cors-works) first.
### Set the allowed origins
To allow one or more specific origins:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=19-23)]
2016-10-29 01:35:15 +08:00
To allow all origins:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs??range=27-31)]
2016-10-29 01:35:15 +08:00
Consider carefully before allowing requests from any origin. It means that literally any website can make AJAX calls to your API.
2016-10-29 01:35:15 +08:00
### Set the allowed HTTP methods
To allow all HTTP methods:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=44-49)]
2016-10-29 01:35:15 +08:00
This affects pre-flight requests and Access-Control-Allow-Methods header.
### Set the allowed request headers
A CORS preflight request might include an Access-Control-Request-Headers header, listing the HTTP headers set by the application (the so-called "author request headers").
To whitelist specific headers:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=53-58)]
2016-10-29 01:35:15 +08:00
To allow all author request headers:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=62-67)]
2016-10-29 01:35:15 +08:00
Browsers are not entirely consistent in how they set Access-Control-Request-Headers. If you set headers to anything other than "*", you should include at least "accept", "content-type", and "origin", plus any custom headers that you want to support.
### Set the exposed response headers
By default, the browser doesn't expose all of the response headers to the application. (See [http://www.w3.org/TR/cors/#simple-response-header](http://www.w3.org/TR/cors/#simple-response-header).) The response headers that are available by default are:
2016-10-29 01:35:15 +08:00
* Cache-Control
* Content-Language
* Content-Type
* Expires
* Last-Modified
* Pragma
The CORS spec calls these *simple response headers*. To make other headers available to the application:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=71-76)]
2016-10-29 01:35:15 +08:00
### Credentials in cross-origin requests
Credentials require special handling in a CORS request. By default, the browser doesn't send any credentials with a cross-origin request. Credentials include cookies as well as HTTP authentication schemes. To send credentials with a cross-origin request, the client must set XMLHttpRequest.withCredentials to true.
2016-10-29 01:35:15 +08:00
Using XMLHttpRequest directly:
```javascript
2016-10-29 01:35:15 +08:00
var xhr = new XMLHttpRequest();
xhr.open('get', 'http://www.example.com/api/test');
xhr.withCredentials = true;
```
2016-10-29 01:35:15 +08:00
In jQuery:
```jQuery
2016-10-29 01:35:15 +08:00
$.ajax({
type: 'get',
url: 'http://www.example.com/home',
xhrFields: {
withCredentials: true
}
```
2016-10-29 01:35:15 +08:00
In addition, the server must allow the credentials. To allow cross-origin credentials:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=80-85)]
2016-10-29 01:35:15 +08:00
Now the HTTP response will include an Access-Control-Allow-Credentials header, which tells the browser that the server allows credentials for a cross-origin request.
If the browser sends credentials, but the response doesn't include a valid Access-Control-Allow-Credentials header, the browser won't expose the response to the application, and the AJAX request fails.
2016-10-29 01:35:15 +08:00
2018-01-29 05:49:15 +08:00
Be careful when allowing cross-origin credentials. A website at another domain can send a logged-in user's credentials to the app on the user's behalf without the user's knowledge. The CORS specification also states that setting origins to "*" (all origins) is invalid if the `Access-Control-Allow-Credentials` header is present.
2016-10-29 01:35:15 +08:00
### Set the preflight expiration time
The Access-Control-Max-Age header specifies how long the response to the preflight request can be cached. To set this header:
[!code-csharp[](cors/sample/CorsExample4/Startup.cs?range=89-94)]
2016-10-29 01:35:15 +08:00
2017-10-14 04:50:30 +08:00
<a name="cors-how-cors-works"></a>
2016-10-29 01:35:15 +08:00
## How CORS works
2018-01-29 05:49:15 +08:00
This section describes what happens in a CORS request at the level of the HTTP messages. It's important to understand how CORS works so that the CORS policy can be configured correctly and troubleshooted when unexpected behaviors occur.
2016-10-29 01:35:15 +08:00
2018-01-29 05:49:15 +08:00
The CORS specification introduces several new HTTP headers that enable cross-origin requests. If a browser supports CORS, it sets these headers automatically for cross-origin requests. Custom JavaScript code isn't required to enable CORS.
2016-10-29 01:35:15 +08:00
2018-01-29 05:49:15 +08:00
Here is an example of a cross-origin request. The `Origin` header provides the domain of the site that's making the request:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
GET http://myservice.azurewebsites.net/api/test HTTP/1.1
Referer: http://myclient.azurewebsites.net/
Accept: */*
Accept-Language: en-US
Origin: http://myclient.azurewebsites.net
Accept-Encoding: gzip, deflate
User-Agent: Mozilla/5.0 (compatible; MSIE 10.0; Windows NT 6.2; WOW64; Trident/6.0)
Host: myservice.azurewebsites.net
```
2016-10-29 01:35:15 +08:00
If the server allows the request, it sets the Access-Control-Allow-Origin header in the response. The value of this header either matches the Origin header from the request, or is the wildcard value "*", meaning that any origin is allowed:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
HTTP/1.1 200 OK
Cache-Control: no-cache
Pragma: no-cache
Content-Type: text/plain; charset=utf-8
Access-Control-Allow-Origin: http://myclient.azurewebsites.net
Date: Wed, 20 May 2015 06:27:30 GMT
Content-Length: 12
Test message
```
2016-10-29 01:35:15 +08:00
If the response doesn't include the Access-Control-Allow-Origin header, the AJAX request fails. Specifically, the browser disallows the request. Even if the server returns a successful response, the browser doesn't make the response available to the client application.
2016-10-29 01:35:15 +08:00
### Preflight Requests
For some CORS requests, the browser sends an additional request, called a "preflight request", before it sends the actual request for the resource. The browser can skip the preflight request if the following conditions are true:
* The request method is GET, HEAD, or POST, and
* The application doesn't set any request headers other than Accept, Accept-Language, Content-Language, Content-Type, or Last-Event-ID, and
2016-10-29 01:35:15 +08:00
* The Content-Type header (if set) is one of the following:
* application/x-www-form-urlencoded
* multipart/form-data
* text/plain
The rule about request headers applies to headers that the application sets by calling setRequestHeader on the XMLHttpRequest object. (The CORS specification calls these "author request headers".) The rule doesn't apply to headers the browser can set, such as User-Agent, Host, or Content-Length.
2016-10-29 01:35:15 +08:00
Here is an example of a preflight request:
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
OPTIONS http://myservice.azurewebsites.net/api/test HTTP/1.1
Accept: */*
Origin: http://myclient.azurewebsites.net
Access-Control-Request-Method: PUT
Access-Control-Request-Headers: accept, x-my-custom-header
Accept-Encoding: gzip, deflate
User-Agent: Mozilla/5.0 (compatible; MSIE 10.0; Windows NT 6.2; WOW64; Trident/6.0)
Host: myservice.azurewebsites.net
Content-Length: 0
```
2016-10-29 01:35:15 +08:00
The pre-flight request uses the HTTP OPTIONS method. It includes two special headers:
* Access-Control-Request-Method: The HTTP method that will be used for the actual request.
* Access-Control-Request-Headers: A list of request headers that the application set on the actual request. (Again, this doesn't include headers that the browser sets.)
2016-10-29 01:35:15 +08:00
Here is an example response, assuming that the server allows the request:
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
HTTP/1.1 200 OK
Cache-Control: no-cache
Pragma: no-cache
Content-Length: 0
Access-Control-Allow-Origin: http://myclient.azurewebsites.net
Access-Control-Allow-Headers: x-my-custom-header
Access-Control-Allow-Methods: PUT
Date: Wed, 20 May 2015 06:33:22 GMT
```
2016-10-29 01:35:15 +08:00
The response includes an Access-Control-Allow-Methods header that lists the allowed methods, and optionally an Access-Control-Allow-Headers header, which lists the allowed headers. If the preflight request succeeds, the browser sends the actual request, as described earlier.