What is and why do you need API Versioning?

Nov 06 2023

Many thanks to the sponsors who make it possible for this newsletter to be free for readers.

 

Today's issue is sponsored by

 

Ilovedotnet.org Master .NET with all the simple, practical examples and real-world practices using Ilovedotnet learning path. This open-source .NET learning platform with in browser demo's is founded by Abdul Rahman, Microsoft MVP from wider .NET community. Join their WhatsApp channel for free and start learning.

 
 

The Background

 
 

API versioning is a technique used to manage changes in an API while maintaining backward compatibility for existing clients. It allows developers to introduce new features, fix bugs, or make other modifications to the API without impacting the functionality of existing integrations.

 

API is software. Every piece of the software needs updates at some point. When you are creating some updates on your API, you need to ensure that the changes don’t affect your API consumers (users). To ensure that, you need to introduce API versioning.

 
 

Why do you need API versioning?

 
 

You need API versioning because it ensures compatibility, empowers flexible adoption of changes, provides controlled and documented modifications, enables coexistence of multiple versions, and grants granular control over the lifecycle of your API, resulting in successful and efficient software development. 

 

The 3 most important reasons:

 

Backward compatibility: It ensures that clients can continue using the older version while new clients can take advantage of the updated features.

 

API evolution: As your API evolves over time, versioning allows you to introduce new features, deprecate outdated functionality, and make improvements without disrupting existing clients.

 

Client flexibility: Different clients may have different requirements or may need specific features available in a particular version of the API.

 
 

ASP .NET Web API Versioning

 
 

There are multiple approaches to API versioning in .NET (> .NET 5):

 

• URI Versioning (Path Versioning)
• Query String Versioning
• Header Versioning
• Media Type Versioning (Accept header)
• Host Name Versioning

 

URI Versioning (Path Versioning)

 

• The version is included directly in the URL path (e.g., /api/v1/products ).
• This method is the most straightforward and easy to understand.
• However, it can lead to URL duplication and requires clients to change URLs to access different versions.

    [ApiVersion("1.0")]
    [Route("api/v{version:apiVersion}/products")]
    public class ProductsV1Controller : ApiController
    {
            public IHttpActionResult Get()
            {
                    // Implementation for version 1
            }
    }
    
    [ApiVersion("2.0")]
    [Route("api/v{version:apiVersion}/products")]
    public class ProductsV2Controller : ApiController
    {
            public IHttpActionResult Get()
            {
                    // Implementation for version 2
            }
    }

 

Query String Versioning

 

• The version is specified as a query parameter in the URL (e.g., /api/products?version=1 ).
• This method allows for an easy transition to new versions.
• It may not be as clean as path versioning and can be overlooked by users.

    public class QueryStringVersionController : ApiController
    {
            const string DefaultApiVersion = "1.0";
        
            [HttpGet]
            public IHttpActionResult Get()
            {
                    var apiVersion = HttpContext.Current.Request.QueryString["version"] ?? DefaultApiVersion;
                    if (apiVersion == "1.0")
                    {
                            // Implementation for version 1
                    }
                    else if (apiVersion == "2.0")
                    {
                            // Implementation for version 2
                    }
                    else
                    {
                            return BadRequest("Version not supported");
                    }
            }
    }

 

Header Versioning

 

• The version information is sent as a custom header in the HTTP request (e.g., X-API-Version: 1 ).
• It keeps the URL unchanged for different versions, which can be beneficial for SEO and caching.
• However, it's less discoverable and can be harder for testing as it requires setting HTTP headers.

    public class HeaderVersionController : ApiController
    {
            private const string HeaderName = "X-API-Version";
        
            [HttpGet]
            public IHttpActionResult Get()
            {
                    var apiVersion = HttpContext.Current.Request.Headers[HeaderName];
                    if (apiVersion == "1.0")
                    {
                            // Implementation for version 1
                    }
                    else if (apiVersion == "2.0")
                    {
                            // Implementation for version 2
                    }
                    else
                    {
                            return BadRequest("API version is required");
                    }
            }
    }

 

Media Type Versioning (Accept header)

 

• Version information is included in the Accept header of the HTTP request, often using a custom media type (e.g., application/vnd.myapi.v1+json ).
• This method follows the HTTP specification closely.
• It is more complex and can be difficult to handle for some API clients.

    [Produces("application/json")]
    public class MediaTypeVersionController : ApiController
    {
            [HttpGet]
            public IHttpActionResult Get()
            {
                    string apiVersion = GetApiVersionFromAcceptHeader();
                    if (apiVersion == "v1")
                    {
                            // Implementation for version 1
                    }
                    else if (apiVersion == "v2")
                    {
                            // Implementation for version 2
                    }
                    else
                    {
                            return BadRequest("API version is required");
                    }
            }
        
            private string GetApiVersionFromAcceptHeader()
            {
                    var acceptHeader = Request.Headers.Accept.FirstOrDefault();
                    if (acceptHeader != null)
                    {
                            var versionParameter = acceptHeader.Parameters
                                .FirstOrDefault(p => p.Name.Equals("version", StringComparison.OrdinalIgnoreCase));
                            return versionParameter?.Value;
                    }
                    return null;
            }
    }

 

Host Name Versioning

 

• Different versions of the API are hosted at different domain names (e.g., api-v1.example.com).
• This approach can be clear and intuitive for the end-users.
• It requires more infrastructure setup and can complicate SSL certificate management.

 

Note: Hostname versioning might be done through routing or with URL rewriting, depending on your setup. This isn't usually handled directly in the controller but rather in the IIS setup or with a reverse proxy configuration. However, you can simulate it with routing:

  // This would be set up in your routing configuration or Web API configuration.
  config.Routes.MapHttpRoute(
        name: "Version1",
        routeTemplate: "api-v1/{controller}/{id}",
        defaults: new { id = RouteParameter.Optional }
  );
  
  config.Routes.MapHttpRoute(
        name: "Version2",
        routeTemplate: "api-v2/{controller}/{id}",
        defaults: new { id = RouteParameter.Optional }
  );

 
 

Conslusion

 
 

Each method has its trade-offs, and the choice of which to use will depend on the specific requirements and constraints of your project. It's also possible to mix and match these strategies if needed. Ultimately, the goal is to maintain a reliable contract with the consumers of your API while allowing for the continued evolution and improvement of the API services you provide.

 

That's all from me today.

Join 13,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 13,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 13,250+ subscribers gaining practical tips and resources to enhance your .NET expertise.