在 swagger 中支持 asp.net core 可选路由参数

场景

  • 同一个 Controller 可能对不同环境有不同的行为,通过路由的 env 参数调整注入的参数

  • 每一次请求上下文可能会根据 env 的值修改 repo 的具体参数

  • 系统会提供默认的 env, 因此此处的路由参数为可选参数

  • swagger 根据 OpenApi v3.x 规范,即使使用 ? 标注为可选,ui 中仍然为 required

    C# 复制代码
    [Route("api/{env:env?}/[controller]")]
    [ApiController]
    public class DemoController(IDemoRepo repo) : ControllerBase
    {
        /* other codes */
    }

解决方法

  • 增加自定的方法修改已经生成的 swagger 文档
C# 复制代码
builder.Services.AddSwaggerGen(opt =>
{
   opt.OperationFilter<ApplyOptionalRouteParameterOperationFilter>();
   /* other codes */
}

public partial class ApplyOptionalRouteParameterOperationFilter : IOperationFilter
{
   public void Apply(OpenApiOperation operation, OperationFilterContext context)
   {
       if (operation.Parameters.Count == 0) return;
       // 获取方法以及类上面的 RouteAttribute
       var attrs = context.MethodInfo.GetCustomAttributes(true)
           .Concat(context.MethodInfo.DeclaringType?.GetCustomAttributes(true) ?? [])
           .OfType<RouteAttribute>().Where(x => x.Template.Contains('?')).ToList();
           
       // 遍历所有 RouteAttribute,如果包含可选参数,则对应修改 OpenAPI 参数定义
       foreach (var route in attrs)
       {
           var matches = RouteTemplateRegex().Matches(route.Template);
           foreach (Match match in matches)
           {
               var name = match.Groups["name"].Value;
               var parameter = operation.Parameters.FirstOrDefault(x => x.Name == name);
               if (parameter is not null)
               {
                   parameter.AllowEmptyValue = true;
                   parameter.Required = false;
                   parameter.Schema.Nullable = true;
               }
           }
       }
   }

   [GeneratedRegex(@"\{(?<name>\w+)(\:\w+)*\?\}")]
   private static partial Regex RouteTemplateRegex();
}
相关推荐
短剑重铸之日18 小时前
《7天学会Redis》特别篇: Redis分布式锁
java·redis·分布式·后端·缓存·redission·看门狗机制
小北方城市网18 小时前
SpringBoot 全局异常处理与接口规范实战:打造健壮可维护接口
java·spring boot·redis·后端·python·spring·缓存
hanqunfeng18 小时前
(三十三)Redisson 实战
java·spring boot·后端
小北方城市网19 小时前
SpringBoot 集成 MyBatis-Plus 实战(高效 CRUD 与复杂查询):简化数据库操作
java·数据库·人工智能·spring boot·后端·安全·mybatis
hanqunfeng20 小时前
(四十)SpringBoot 集成 Redis
spring boot·redis·后端
小北方城市网21 小时前
SpringBoot 集成 MinIO 实战(对象存储):实现高效文件管理
java·spring boot·redis·分布式·后端·python·缓存
程序员泠零澪回家种桔子21 小时前
RAG自查询:让AI精准检索的秘密武器
人工智能·后端·算法
曹轲恒21 小时前
SpringBoot配置文件(1)
java·spring boot·后端
小北方城市网1 天前
SpringBoot 安全认证实战(Spring Security + JWT):打造无状态安全接口体系
数据库·spring boot·后端·安全·spring·mybatis·restful
rannn_1111 天前
【Javaweb学习|Day7】事务管理、文件上传
后端·学习·javaweb