我们有100多个API,并且必须在顶部,200、500等处为所有API编写ProducesResponseType。是否有一种方法可以为我们所有的get函数设置此全局参数,所以我们不必继续重复代码?试图使API遵循Dry原则,并成为瘦控制器。
[HttpGet("[Action]/{id}")]
[ProducesResponseType(typeof(GetBookResponse), StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
[ProducesResponseType(typeof(GetBookResponse), StatusCodes.Status500InternalServerError)]
public async Task<ActionResult<GetBookResponse>> GetByBook(int id)
{
var book = await bookservice.GetBookById(id);
return Ok(book);
}
资源:
Set one ProducesResponseType typeof for several HttpStatusCodes
答案 0 :(得分:3)
您可以创建自定义IApplicationModelProvider
,并在OnProvidersExecuting
方法中添加所需的过滤器。
ProduceResponseTypeModelProvider.cs
public class ProduceResponseTypeModelProvider : IApplicationModelProvider
{
public int Order => 3;
public void OnProvidersExecuted(ApplicationModelProviderContext context)
{
}
public void OnProvidersExecuting(ApplicationModelProviderContext context)
{
foreach (ControllerModel controller in context.Result.Controllers)
{
foreach (ActionModel action in controller.Actions)
{
// I assume that all you actions type are Task<ActionResult<ReturnType>>
Type returnType = action.ActionMethod.ReturnType.GenericTypeArguments[0].GetGenericArguments()[0];
action.Filters.Add(new ProducesResponseTypeAttribute(StatusCodes.Status510NotExtended));
action.Filters.Add(new ProducesResponseTypeAttribute(returnType, StatusCodes.Status200OK));
action.Filters.Add(new ProducesResponseTypeAttribute(returnType, StatusCodes.Status500InternalServerError));
}
}
}
}
然后您需要将其注册到IServiceCollection
Startup.cs
public void ConfigureServices(IServiceCollection services)
{
...
services.TryAddEnumerable(ServiceDescriptor.Transient<IApplicationModelProvider, ProduceResponseTypeModelProvider>());
...
}
答案 1 :(得分:0)
Kahbazi的答案是正确的,我们也使用此行代码, 它提供错误处理,动词属性检查以及参数是否存在
public class ProduceResponseTypeModelProvider : IApplicationModelProvider
{
public int Order => 3;
public void OnProvidersExecuted(ApplicationModelProviderContext context)
{
}
public void OnProvidersExecuting(ApplicationModelProviderContext context)
{
foreach (ControllerModel controller in context.Result.Controllers)
{
foreach (ActionModel action in controller.Actions)
{
Type returnType = null;
if (action.ActionMethod.ReturnType.GenericTypeArguments.Any())
{
if (action.ActionMethod.ReturnType.GenericTypeArguments[0].GetGenericArguments().Any())
{
returnType = action.ActionMethod.ReturnType.GenericTypeArguments[0].GetGenericArguments()[0];
}
}
var methodVerbs = action.Attributes.OfType<HttpMethodAttribute>().SelectMany(x => x.HttpMethods).Distinct();
bool actionParametersExist = action.Parameters.Any();
AddUniversalStatusCodes(action, returnType);
if (actionParametersExist == true)
{
AddProducesResponseTypeAttribute(action, null, 404);
}
if (methodVerbs.Contains("POST"))
{
AddPostStatusCodes(action, returnType, actionParametersExist);
}
}
}
}
public void AddProducesResponseTypeAttribute(ActionModel action, Type returnType, int statusCodeResult)
{
if (returnType != null)
{
action.Filters.Add(new ProducesResponseTypeAttribute(returnType, statusCodeResult));
}
else if (returnType == null)
{
action.Filters.Add(new ProducesResponseTypeAttribute(statusCodeResult));
}
}
public void AddUniversalStatusCodes(ActionModel action, Type returnType)
{
AddProducesResponseTypeAttribute(action, returnType, 200);
AddProducesResponseTypeAttribute(action, null, 500);
}
public void AddPostStatusCodes(ActionModel action, Type returnType, bool actionParametersExist)
{
AddProducesResponseTypeAttribute(action, returnType, 201);
AddProducesResponseTypeAttribute(action, returnType, 400);
if (actionParametersExist == false)
{
AddProducesResponseTypeAttribute(action, null, 404);
}
}
}