In this post, we will see how to manage the versioning in a .net core Web API service, using three different approaches:
1) Query String-Based Versioning
2) URL-Based Versioning
3) HTTP Header-Based Versioning
First of all, we open Visual Studio 2019 and we create a Web API project:
data:image/s3,"s3://crabby-images/6ba0e/6ba0e773c8b9b32ef2f30b4002e50a181ca30693" alt=""
data:image/s3,"s3://crabby-images/d7d7c/d7d7c8e8413763a587502491f8c1e2779c94a995" alt=""
data:image/s3,"s3://crabby-images/c8662/c86625c5697d12870bc7a32576f0ca0c5c3928e0" alt=""
data:image/s3,"s3://crabby-images/a7da0/a7da0a137e86ae10987717706606312430151f45" alt=""
If we run the application, this will be the output:
data:image/s3,"s3://crabby-images/efd07/efd07821232e3e61860beccb7b7a6d0b05384f24" alt=""
Now, in order to manage the versioning, we have to install the library
Microsoft.AspNetCore.Mvc.Versioning
and then, we have to modify the method ConfigureServices in the startup file:
1 2 3 4 5 6 7 8 9 | public void ConfigureServices(IServiceCollection services) { services.AddControllers(); services.AddApiVersioning(o => { o.AssumeDefaultVersionWhenUnspecified = true ; o.DefaultApiVersion = new ApiVersion(1, 0); }); } |
QUERY STRING-BASED VERSIONING:
We open the file WeatherForecastController and we create two versions of the controller:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 | using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Logging; using System; using System.Collections.Generic; using System.Linq; namespace WebAPIVersioning.Controllers { [ApiController] [ApiVersion( "1.0" )] [Route( "api/Forecast" )] public class WeatherForecastController : ControllerBase { private static readonly string [] Summaries = new [] { "Freezing" , "Bracing" , "Chilly" , "Cool" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastController(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } [ApiController] [ApiVersion( "2.0" )] [Route( "api/Forecast" )] public class WeatherForecastControllerVersion2 : ControllerBase { private static readonly string [] Summaries = new [] { "Mild" , "Warm" , "Balmy" , "Hot" , "Sweltering" , "Scorching" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastControllerVersion2(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } } |
In order to call the different versions of the controller, we have to pass the version of controller in the query string:
data:image/s3,"s3://crabby-images/f375a/f375a95fde050c97cc84245ef6f9a64ff923a4aa" alt=""
data:image/s3,"s3://crabby-images/16551/1655126a77e0b2f9ae2925804c872afde9aa7891" alt=""
URL-BASED VERSIONING:
We open the file WeatherForecastController and we create two versions of the controller:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 | using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Logging; using System; using System.Collections.Generic; using System.Linq; namespace WebAPIVersioning.Controllers { [ApiController] [ApiVersion( "1.0" )] [Route( "api/v{v:apiVersion}/Forecast" )] public class WeatherForecastController : ControllerBase { private static readonly string [] Summaries = new [] { "Freezing" , "Bracing" , "Chilly" , "Cool" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastController(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } [ApiController] [ApiVersion( "2.0" )] [Route( "api/v{v:apiVersion}/Forecast" )] public class WeatherForecastControllerVersion2 : ControllerBase { private static readonly string [] Summaries = new [] { "Mild" , "Warm" , "Balmy" , "Hot" , "Sweltering" , "Scorching" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastControllerVersion2(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } } |
In order to call the different versions of the controller, we have to put the version of controller in an URL path segment:
data:image/s3,"s3://crabby-images/82e82/82e828158d243d97647f02cf7af0c087f6b3d594" alt=""
data:image/s3,"s3://crabby-images/2d26f/2d26ffc998d4ec302214d31aa07d6491e709d6bd" alt=""
HTTP HEADER-BASED VERSIONING:
We open the file WeatherForecastController and we create two versions of the controller:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 | using Microsoft.AspNetCore.Mvc; using Microsoft.Extensions.Logging; using System; using System.Collections.Generic; using System.Linq; namespace WebAPIVersioning.Controllers { [ApiController] [ApiVersion( "1.0" )] [Route( "api/Forecast" )] public class WeatherForecastController : ControllerBase { private static readonly string [] Summaries = new [] { "Freezing" , "Bracing" , "Chilly" , "Cool" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastController(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } [ApiController] [ApiVersion( "2.0" )] [Route( "api/Forecast" )] public class WeatherForecastControllerVersion2 : ControllerBase { private static readonly string [] Summaries = new [] { "Mild" , "Warm" , "Balmy" , "Hot" , "Sweltering" , "Scorching" }; private readonly ILogger<WeatherForecastController> _logger; public WeatherForecastControllerVersion2(ILogger<WeatherForecastController> logger) { _logger = logger; } [HttpGet] public IEnumerable<WeatherForecast> Get() { var rng = new Random(); return Enumerable.Range(1, 5).Select(index => new WeatherForecast { Date = DateTime.Now.AddDays(index), TemperatureC = rng.Next(-20, 55), Summary = Summaries[rng.Next(Summaries.Length)] }) .ToArray(); } } } |
Finally, we have to modify the method ConfigureServices, in the the startup file:
[STARTUP.CS]
1 2 3 4 5 6 7 8 9 10 | public void ConfigureServices(IServiceCollection services) { services.AddControllers(); services.AddApiVersioning(o => { o.AssumeDefaultVersionWhenUnspecified = true ; o.DefaultApiVersion = new ApiVersion(1, 0); o.ApiVersionReader = new HeaderApiVersionReader( "x-api-version" ); }); } |
Now, we run the application using Postman and we put in the header the controller’s version:
data:image/s3,"s3://crabby-images/9e66f/9e66fbb135a26fdd6666839504349ba0dae77b65" alt=""
data:image/s3,"s3://crabby-images/cebd1/cebd13826897e88113751d4deecb5952580cc763" alt=""