Как сделать управление версиями в ASP.NET Boilerplate Web API?
Я пытаюсь сделать управление версиями в рамках ASP.NET Boilerplate.
Я создал две версии в Swagger Gen ("v1.0" и "v2.0") и установил версию API для Web API, но каждый раз я получаю все API в обеих версиях от Swagger.
Startup.cs:
AddSwaggerGen
в ConfigureServices()
:
services.AddSwaggerGen(options =>
{
options.SwaggerDoc("v1.0", new Info { Title = "My API", Version = "v1.0" });
options.SwaggerDoc("v2.0", new Info { Title = "My API", Version = "v2.0" });
options.DocInclusionPredicate((docName, description) => true);
// Define the BearerAuth scheme that's in use
options.AddSecurityDefinition("bearerAuth", new ApiKeyScheme()
{
Description = "JWT Authorization header using the Bearer scheme. Example: \"Authorization: Bearer {token}\"",
Name = "Authorization",
In = "header",
Type = "apiKey"
});
// Assign scope requirements to operations based on AuthorizeAttribute
options.OperationFilter<SecurityRequirementsOperationFilter>();
});
UseSwaggerUI
в Configure():
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
app.UseAbp(options => { options.UseAbpRequestLocalization = false; });
app.UseCors(_defaultCorsPolicyName);
app.UseStaticFiles();
app.UseAuthentication();
app.UseAbpRequestLocalization();
app.UseSignalR(routes =>
{
routes.MapHub<AbpCommonHub>("/signalr");
});
app.UseMvc(routes =>
{
routes.MapRoute(
name: "defaultWithArea",
template: "{area}/{controller=Home}/{action=Index}/{id?}");
routes.MapRoute(
name: "default",
template: "{controller=Home}/{action=Index}/{id?}");
});
app.UseSwagger();
app.UseSwaggerUI(options =>
{
options.InjectOnCompleteJavaScript("/swagger/ui/abp.js");
options.InjectOnCompleteJavaScript("/swagger/ui/on-complete.js");
options.SwaggerEndpoint("/swagger/v1.0/swagger.json", "My API V1.0");
options.SwaggerEndpoint("/swagger/v2.0/swagger.json", "My API V2.0");
});
}
Контроллер API - v1.0:
[ApiVersion("v1.0")]
[Route("/api/invoicemodule/1.0/[controller]")]
public class InvoiceController : MyControllerBase
{
[HttpGet, MapToApiVersion("v1.0")]
public IActionResult GetInvoiceById(string invoiceid)
{
//BusinessService.SparePartHistoryService sas = new BusinessService.SparePartHistoryService(_logger, _localizer, _configuration);
if (string.IsNullOrEmpty(invoiceid)) return BadRequest("'Id' cannot be null or empty.");
try
{
BusinessModels.Invoice sp = new BusinessModels.Invoice
{
Id = ""
};
if (sp != null)
{
return Ok(sp);
}
else
{
return NotFound();
}
}
catch (Exception e)
{
return BadRequest(e.Message);
}
}
}
Контроллер API - v2.0:
[ApiVersion("v2.0")]
[Route("/api/invoicemodule/2.0/[controller]")]
public class InvoiceController : MyControllerBase
{
[HttpGet, MapToApiVersion("v2.0")]
public IActionResult GetInvoiceById(string invoiceid)
{
//BusinessService.SparePartHistoryService sas = new BusinessService.SparePartHistoryService(_logger, _localizer, _configuration);
if (string.IsNullOrEmpty(invoiceid)) return BadRequest("'Id' cannot be null or empty.");
try
{
BusinessModels.Invoice sp = new BusinessModels.Invoice
{
Id = ""
};
if (sp != null)
{
return Ok(sp);
}
else
{
return NotFound();
}
}
catch (Exception e)
{
return BadRequest(e.Message);
}
}
}
Выход
2 ответа
Да, я получил решение
Ссылка: https://github.com/domaindrivendev/Swashbuckle.AspNetCore
Я немного изменил код Service.AddSwaggerGen() в методе ConfigurationService() файла Startup.cs
Комментарии добавлены в код, где я изменился
services.AddSwaggerGen(options =>
{
options.SwaggerDoc("v1", new Info { Version = "1" }); // Version must require in integer. can't set Version = "v1"
options.SwaggerDoc("v2", new Info { Version = "2" });
options.DocInclusionPredicate((docName, description) =>
{ // insert this {} code instead of "true"
var versions = description.ControllerAttributes()
.OfType<ApiVersionAttribute>()
.SelectMany(attr => attr.Versions);
return versions.Any(v => $"v{v.ToString()}" == docName);
});
});
Шаг 1. Создайте класс ApiVersion1RoutePrefixAttribute.cs
public class ApiVersion1RoutePrefixAttribute: RoutePrefixAttribute {
private const string RouteBase = "api/{apiVersion:apiVersionConstraint(v1)}";
private const string PrefixRouteBase = RouteBase + "/";
public ApiVersion1RoutePrefixAttribute(string routePrefix):base (string.IsNullOrWhiteSpace(routePrefix)?RouteBase: PrefixRouteBase+routePrefix) {
}
}
Шаг 2. Создайте класс ApiVersion2RoutePrefixAttribute.cs
public class ApiVersion1RoutePrefixAttribute: RoutePrefixAttribute {
private const string RouteBase = "api/{apiVersion:apiVersionConstraint(v1)}";
private const string PrefixRouteBase = RouteBase + "/";
public ApiVersion1RoutePrefixAttribute(string routePrefix):base (string.IsNullOrWhiteSpace(routePrefix)?RouteBase: PrefixRouteBase+routePrefix) {
}
}
Шаг 3. Создайте класс ApiVersionConstraint
public class ApiVersionConstraint : IHttpRouteConstraint {
public string AllowedVersion { get; private set; }
public ApiVersionConstraint(string allowedVersion) {
this.AllowedVersion = allowedVersion.ToLowerInvariant();
}
public bool Match(HttpRequestMessage request, IHttpRoute route, string parameterName, IDictionary<string, object> values, HttpRouteDirection routeDirection) {
object value;
if(values.TryGetValue(parameterName,out value)&& value != null) {
return AllowedVersion.Equals(value.ToString().ToLowerInvariant());
}
return false;
}
}
Шаг 4 Создайте класс NamespaceHttpControllerSelector.cs
public class NamespaceHttpControllerSelector : IHttpControllerSelector {
private readonly HttpConfiguration _configuration;
private readonly Lazy<Dictionary<string, HttpControllerDescriptor>> _controllers;
public NamespaceHttpControllerSelector(HttpConfiguration config) {
_configuration = config;
_controllers = new Lazy<Dictionary<string, HttpControllerDescriptor>>(InitializeControllerDictionary);
}
public HttpControllerDescriptor SelectController(HttpRequestMessage request) {
var routeData = request.GetRouteData();
if (routeData == null) {
throw new HttpResponseException(HttpStatusCode.NotFound);
}
var controllerName = GetControllerName(routeData);
if (controllerName == null) {
throw new HttpResponseException(HttpStatusCode.NotFound);
}
var namespaceName = GetVersion(routeData);
if (namespaceName == null) {
throw new HttpResponseException(HttpStatusCode.NotFound);
}
var controllerKey = String.Format(CultureInfo.InvariantCulture, "{0}.{1}", namespaceName, controllerName);
HttpControllerDescriptor controllerDescriptor;
if (_controllers.Value.TryGetValue(controllerKey, out controllerDescriptor)) {
return controllerDescriptor;
}
throw new HttpResponseException(HttpStatusCode.NotFound);
}
public IDictionary<string, HttpControllerDescriptor> GetControllerMapping() {
return _controllers.Value;
}
private Dictionary<string, HttpControllerDescriptor> InitializeControllerDictionary() {
var dictionary = new Dictionary<string, HttpControllerDescriptor>(StringComparer.OrdinalIgnoreCase);
var assembliesResolver = _configuration.Services.GetAssembliesResolver();
var controllersResolver = _configuration.Services.GetHttpControllerTypeResolver();
var controllerTypes = controllersResolver.GetControllerTypes(assembliesResolver);
foreach (var controllerType in controllerTypes) {
var segments = controllerType.Namespace.Split(Type.Delimiter);
var controllerName = controllerType.Name.Remove(controllerType.Name.Length - DefaultHttpControllerSelector.ControllerSuffix.Length);
var controllerKey = String.Format(CultureInfo.InvariantCulture, "{0}.{1}", segments[segments.Length - 1], controllerName);
if (!dictionary.Keys.Contains(controllerKey)) {
dictionary[controllerKey] = new HttpControllerDescriptor(_configuration,
controllerType.Name,
controllerType);
}
}
return dictionary;
}
private T GetRouteVariable<T>(IHttpRouteData routeData, string name) {
object result;
if (routeData.Values.TryGetValue(name, out result)) {
return (T)result;
}
return default(T);
}
private string GetControllerName(IHttpRouteData routeData) {
var subroute = routeData.GetSubRoutes().FirstOrDefault();
if (subroute == null) return null;
//((HttpActionDescriptor[])subroute.Route.DataTokens["actions"]).First()
var dataTokenValue = subroute.Route.DataTokens["actions"];
if (dataTokenValue == null)
return null;
var controllerName = ((HttpActionDescriptor[])dataTokenValue).First().ControllerDescriptor.ControllerName.Replace("Controller", string.Empty);
return controllerName;
}
private string GetVersion(IHttpRouteData routeData) {
var subRouteData = routeData.GetSubRoutes().FirstOrDefault();
if (subRouteData == null) return null;
return GetRouteVariable<string>(subRouteData, "apiVersion");
}
}
Шаг 5. Ваш файл WebApiConfig.cs должен выглядеть следующим образом
public static void Register(HttpConfiguration config) {
var constraintResolver = new DefaultInlineConstraintResolver();
constraintResolver.ConstraintMap.Add("apiVersionConstraint", typeof(ApiVersionConstraint));
config.MapHttpAttributeRoutes(constraintResolver);
config.Services.Replace(typeof(IHttpControllerSelector), new NamespaceHttpControllerSelector(config));
}
Шаг 6. Ваш контроллер должен выглядеть следующим образом
ваш контроллер версии 1
namespace AgentExperienceAPI.Controllers.v1 {
[Route("GetStatus")]
[AcceptVerbs("GET")]
public HttpResponseMessage GetStatus() {
return Request.CreateResponse(HttpStatusCode.OK, new Dictionary<string, object> {
{ "Status", "OK" }
});
}
}
ваш контроллер версии 2
namespace AgentExperienceAPI.Controllers.v1 {
[Route("GetStatus")]
[AcceptVerbs("GET")]
public HttpResponseMessage GetStatus() {
return Request.CreateResponse(HttpStatusCode.OK, new Dictionary<string, object> {
{ "Status", "OK" }
});
}
}
ваш контроллер должен быть в двух отдельных папках см. изображение
Swagger позаботится об остальном
Вы пробовали настроить Swashbuckle следующим образом?
options.DocInclusionPredicate((docName, apiDesc) =>
{
if (!apiDesc.ActionDescriptor.IsControllerAction())
{
return false;
}
switch (docName)
{
case "v1":
return apiDesc.GroupName == null || apiDesc.GroupName == "v1";
case "v2":
return apiDesc.GroupName == null || apiDesc.GroupName == "v2";
default:
return false;
}
});
и, наконец, на вашем контроллере:
[ApiExplorerSettings(GroupName = "v2")]
[Route("api/[controller]/[action]")]
public class PublicAppService : ASAAppServiceBase, IPublicAppService
{
public PublicAppService()
{
}
[HttpGet]
public string Test()
{
return "test";
}
}
это обсуждалось на
https://github.com/aspnetboilerplate/aspnetboilerplate/issues/3814