45  
aspnet
Поиск  
Always will be ready notify the world about expectations as easy as possible: job change page
Jun 3, 2021

Understand CORS and learn how to enable it for your ASP.NET API

Источник:
Просмотров:
3901

Browser security prevents a web page from making requests to a different domain than the one that served the web page. This restriction is called the same-origin policy. The same-origin policy prevents a malicious site from reading sensitive data from another site.

Sometimes, you might want to allow other sites to make cross-origin requests to your application. This is when you have an API hosted independently to your web applications talking to the API.

In such scenarios, we need to enable CORS support on the API so that the web application can call it.

What is CORS?

Cross Origin Resource Sharing or CORS, in short, is a W3C standard that allows a server to relax this same-origin policy.

Two URLs have the same origin if they have the same Scheme, Host, and Port. For the URL https://example.com:9000/foo.html,

  • Scheme https
  • Host example.com
  • Port 9000.

URL Structure of https://example.com:9000/foo.html 

So if any one of them (Scheme, Host, or Port) is different, it’s treated as a different origin.

CORS In Action In ASP Net Application

To understand CORS better, let’s first simulate an error to see it in action. Let’s use the ASP NET Single Page Application template (SPA).

  • Create a new ASP NET SPA application using dotnet new react. This will create an ASP NET SPA React app.
  • By default, the application is set to host the react application and the ASP NET API on the same server or origin.
  • To change this, open the client app folder and run npm start. This starts the react app on localhost:3000
  • Run the API, which by default runs it on port localhost:5001
  • The default SPA template talks to the weather forecast controller and renders the data on the page.
  • Since now we have changed the ports the application are running on, let’s update the fetchdata class to use the API running on the different server/origin.

Now when we hit the React application running on localhost:3000, it will throw an error in the Console as below.

CORS browser console error

This is because of CORS. We have an application running on one origin (http://localhost:3000) trying to access an application/API running on another origin (http://localhost:5001). Note in this case, the ports are different, which makes them different origins.

Configure Web API For CORS

Since the browser security prevents making cross-origin requests by default, we need to add in support for this.

This support needs to be added to the API server. The server then adds appropriate response headers/

To configure our ASP Net API for CORS, we need to add the below code to the Startup.cs class.

public void ConfigureServices(IServiceCollection services)
{
    ...
    services.AddCors(options =>
    {
        options.AddDefaultPolicy(builder =>
        {
            builder.WithOrigins("http://localhost:3000");
        });
    });
}

public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
    ...
    app.UseRouting();
    app.UseCors();
}

The AddCors method in the ConfigureServices of Startup.cs class registers a CORS Policy in the ASP NET Middleware. CORS is enabled in the Request pipeline via the ASP NET Middleware by calling the UseCors method in the Configure method. The AddCors call must be after the UseRouting call for this to work as expected.

The Origins URL specified must not have a trailing slash.

With this code added, the API now asks the browser to relax the Cross-Origin Policy. If we make another request again from the application and inspect the request headers, we can see the access-control-allow-origin headers added to the response. It

HTTP Request header has the Response Headers 'access-control-allow-origin: http://localhost:3000

Configurable Origins

Since we will need to run the API from different environments, it’s best not to hardcode the Origin URL in the source code. You can move this to the application settings.

{
  ...
  "AllowedOrigins": [
    "http://localhost:3000",
    "http://localhost:4000"
  ],
}

When registering the CORS policy, we can use the Origin URLs from the configuration.

options.AddDefaultPolicy(builder =>
{
    builder
        .WithOrigins(Configuration.GetSection("AllowedOrigins").Get<string[]>());
}

Since the WithOrigins method accepts an array of URL’s, we can pass the URL values from the Configuration object as above. In this case, the API is registered for Cross-Origin Request both from [localhost:3000](http://localhost:3000) and localhost:4000.

CORS And Pre-Flight Requests

Often you would have seen an additional request go to the API with the OPTIONS Method, called a Preflight request.

A Preflight request is a CORS request that checks for CORS support on the server. It is automatically issued by the browser and can be skipped under certain conditions.

The default code setup above meets those conditions and skips the Preflight request. To force a Preflight request, let’s update the FetchData.js file where the server request is made.

Add a dummy Authorization header to the API request. This forces the bowser to make a Preflight request to the API server before making an actual request.

async populateWeatherData() {
  const response = await fetch("https://localhost:5001/weatherforecast", {
    headers: {
      Authorization: "Bearer MYTOKEN",
    },
  });
  const data = await response.json();
  this.setState({ forecasts: data, loading: false });
}

For the Preflight request to be successful, we also need to update the server to indicate that the Authorization header is now supported. Use the WithHeaders method on the Policy Builder, as shown below.

options.AddDefaultPolicy(builder =>
{
    builder
        .WithOrigins(Configuration.GetSection("AllowedOrigins").Get<string[]>())
        .WithHeaders("Authorization");
});

Inspecting the Network Request from the application running on localhost:3000 we can see two requests - One OPTIONS and one GET call.

CORS ≠ SECURITY

One of CORS’s main misconceptions is that it is a security feature - But let’s get this straight.

CORS Is NOT A SECURITY Feature.

If anything it does, it relaxes security, it allows requests from one origin to be made to another. An API is not safer by allowing or adding CORS support. It’s up to the browsers to enforce CORS. Even with CORS enabled, it only restricts browsers from making Cross-Origin Requests. Nothing is stopping us from making a direct Request to the API or using tools like Fiddler or Postman.

However, if misconfigured, CORS can cause security vulnerabilities. One of the common mistakes is allowing all Origins to access your API. This is fine for API’s serving all public content. For most scenarios, it is better to be explicit about the Origins and the methods that it can access on the API.

There are different ways you can enable CORS. What we have seen above is using the Default Policy and middleware.

You can also enable using,

Similar to how we specified the Authentication header support on the CORS policy, we can also specify other Policy Options. This enables to handle different scenarios that we might run into. For example, if our API exposes DELETE functionality, this method support needs to be explicitly added by calling the WithMethods function.

I hope this gives you a good overview of what CORS is, and how you can configure your ASP NET API to support Cross-Origin Requests.

Похожее
Jun 26, 2021
SignalR is a popular software library that allows you send server code to client-side applications, like a React application. This tutorial will show you how to install SignalR on both your ASP.NET Core and React Typescript application. Creating your Applications...
Oct 24, 2022
Author: Chandan Das
For as long as web applications have been around, full-stack developers have had to work with different sets of technologies for the front and backend. For instance, a developer would use something like Angular for the frontend and Express.js for...
Jul 10, 2021
Author: Sam Walpole
I've recently gotten into using Docker in my development cycle and I'm really enjoying how much of a wonderful tool it is. One thing that always used to be a pain was setting up a development server to run SQL...
Aug 8
Author: Davit Asryan
The growth of the internet has made instant communication technology more important than ever, especially for the Internet of Things (IoT). With so many devices like smart home gadgets and industrial sensors needing to talk to each other smoothly, having...
Написать сообщение
Тип
Почта
Имя
*Сообщение