Using CORS in your applications

Oct 9 2023

 
 

The Background

 
 

Imagine we are visiting a foreign country. We want to send a package to someone in another country. However, the customs officers at the border won't allow us to send the package unless we have the proper documentation and the destination country has agreed to accept it.

 

• A web page served from one domain (e.g., example.com) is like a person in one country.

 

• A server at a different domain (e.g., another-example.com) is like a person in another country.

 

• Like how customs officers at the border check for proper documentation and ensure that the destination country will accept the package, CORS checks for the appropriate headers in HTTP requests and responses to ensure that the web page is allowed to make requests to the other domain.

 

• If the server at another-example.com wants to allow requests from example.com, it can set the Access-Control-Allow-Origin header in its responses to example.com. The browser will then allow the web page from the example.com to make requests to another example.com, just like how the customs officers at the border will allow the package to be sent if the documentation is in order and the destination country has agreed to accept it.

 

Cross-Origin Resource Sharing (CORS) is a security feature implemented by web browsers that controls how web pages in one domain can request resources, like APIs, from another domain.

 

By default, web pages are not allowed to make requests to a different domain than the one that served the web page. This is a security measure to prevent a malicious website from making unwanted requests to a legitimate one.

 

In the context of .NET, if you're building an API using ASP.NET Core, you might want to enable CORS to allow web pages from certain domains to access your API.

 

Let's see how to do that...

 
 

Setting up CORS in .NET

 
 

1. Install the necessary package - because we are displaying the raw identifiers to the user, especially sequential ones.

  dotnet add package Microsoft.AspNetCore.Cors

 

2. Configure Services in Program.cs - Add CORS services to the DI container and set up a CORS policy.

  builder.Services.AddCors(opt =>
  {
      opt.AddPolicy("CorsPolicy", policy =>
      {
          policy.AllowAnyMethod()
         .AllowAnyHeader()
         .WithOrigins("https://localhost:7235");
      }
});

 

3. Apply the CORS policy in the middleware pipeline: - Add CORS services to the DI container and set up a CORS policy.

  app.UseCors("CorsPolicy");

 
 

Adjusting the CORS Policy

 
 

The policy we created is rather strict. You can adjust it as per your requirements. Some of the things you can configure are:

 

• Allowing multiple specific origins:

  policy.WithOrigins("https://myapp.com", "https://anotherapp.com");

 

• Allowing all origins:

 policy.AllowAnyOrigin();

 

• Allowing specific headers:

 policy.WithHeaders("header1", "header2");

 

• Allowing specific HTTP methods:

 policy.WithMethods("GET", "POST");
Note: Be cautious with allowing all origins (AllowAnyOrigin) as it can expose your API to potential security threats from malicious websites. Always ensure that you understand the security implications of the CORS settings you choose.

 
 

Pre-Flight Requests

 
 

When you make a request (e.g., a POST request) that could potentially have side effects, browsers will first send a "pre-flight" request using the OPTIONS method to check if the actual request is allowed. This is done to make sure that the server accepts the request and knows about the origin making it.

 

For instance, if your client sends a request with a custom header or uses an HTTP method other than GET or HEAD, a pre-flight request is triggered.

 

In our previous .NET Core CORS setup, pre-flight requests are automatically handled by the middleware because we used .AllowAnyHeader() and .AllowAnyMethod().

 
 

Tightening Security

 
 

While it might be tempting to use .AllowAnyOrigin() and .AllowAnyMethod() for simplicity, it's more secure to only enable what's needed:

 

Specify allowed methods: If your API endpoint only needs to handle GET and POST, then only allow those:

 policy.WithOrigins("https://myapp.com")
       .WithMethods("GET", "POST");

 

Specify allowed headers : If your application requires specific headers, specify them:

 policy.WithOrigins("https://myapp.com")
       .WithHeaders("X-Custom-Header", "Authorization");
Note: If you use .AllowCredentials(), you can't use .AllowAnyOrigin() due to security concerns. You need to explicitly specify the origins.

 
 

Exposing Headers

 
 

There might be situations where you want your client-side application to have access to certain headers in the CORS response. You can specify which headers should be exposed:

 policy.WithOrigins("https://myapp.com")
       .WithExposedHeaders("X-My-Custom-Header");

 
 

Wrapping up

 
 

Understanding and managing CORS in .NET Core is crucial when building APIs that are accessed from different origins. By configuring the CORS middleware correctly and securely, you ensure both functionality and security for your web applications. Remember to always review your CORS configuration to minimize potential security risks.

 

That's all from me today.

Join 14,250+ subscribers to improve your .NET Knowledge.

There are 3 ways I can help you:

Design Patterns Simplified ebook

Go-to resource for understanding the core concepts of design patterns without the overwhelming complexity. In this concise and affordable ebook, I've distilled the essence of design patterns into an easy-to-digest format. It is a Beginner level. Check out it here.


Sponsorship

Promote yourself to 14,250+ subscribers by sponsoring this newsletter.


Join TheCodeMan.net Newsletter

Every Monday morning, I share 1 actionable tip on C#, .NET & Arcitecture topic, that you can use right away.


Subscribe to
TheCodeMan.net

Subscribe to the TheCodeMan.net and be among the 14,250+ subscribers gaining practical tips and resources to enhance your .NET expertise.