通过查询参数处理 .NET Core 3.1 Web API 中的多个端点

Sai*_*men 4 .net c# asp.net asp.net-core asp.net-core-webapi

我正在将控制器从 .NET Framework 迁移到 .NET Core,并且希望与以前版本的 API 调用兼容。我在处理来自查询参数的多个路由时遇到问题。

我的示例控制器:

[Route("/api/[controller]")]
[Route("/api/[controller]/[action]")]
public class StaticFileController : ControllerBase
{
    [HttpGet("{name}")]
    public HttpResponseMessage GetByName(string name)
    {
    }

    [HttpGet]
    public IActionResult Get()
    {
    }
}
Run Code Online (Sandbox Code Playgroud)

打电话api/StaticFile?name=someFunnyName会让我采取Get()行动,而不是预期的 GetByName(string name)

我想要实现的目标:

  • 调用 GET api/StaticFile-> 执行Get()操作
  • 调用 GET api/StaticFile?name=someFunnyName-> 执行GetByName()操作

我的app.UseEndpoints()来自Startup.cs只有这些行:

endpoints.MapControllers();
endpoints.MapDefaultControllerRoute();
Run Code Online (Sandbox Code Playgroud)

如果我[HttpGet]到处使用并添加([FromQuery] string name)它会让我AmbiguousMatchException: The request matched multiple endpoints

感谢您花时间帮助我(也许还有其他人)

Fei*_*Han 5

我想要实现的目标:

  • 调用 GET api/StaticFile -> 转到 Get() 操作
  • 调用 GET api/StaticFile?name=someFunnyName -> 转到 GetByName() 操作

要实现上述基于查询字符串将请求与预期操作匹配的要求,您可以尝试实现自定义 ActionMethodSelectorAttribute并将其应用到您的操作,如下所示。

[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public class QueryStringConstraintAttribute : ActionMethodSelectorAttribute
{
    public string QueryStingName { get; set; }
    public bool CanPass { get; set; }
    public QueryStringConstraintAttribute(string qname, bool canpass)
    {
        QueryStingName = qname;
        CanPass = canpass;
    }
    public override bool IsValidForRequest(RouteContext routeContext, ActionDescriptor action)
    {
        StringValues value;

        routeContext.HttpContext.Request.Query.TryGetValue(QueryStingName, out value);

        if (QueryStingName == "" && CanPass)
        {
            return true;
        }
        else
        {
            if (CanPass)
            {
                return !StringValues.IsNullOrEmpty(value);
            }

            return StringValues.IsNullOrEmpty(value);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

应用于行动

[Route("api/[controller]")]
[ApiController]
public class StaticFileController : ControllerBase
{
    [HttpGet]
    [QueryStringConstraint("name", true)]
    [QueryStringConstraint("", false)]
    public IActionResult GetByName(string name)
    {
        return Ok("From `GetByName` Action");
    }

    [HttpGet]
    [QueryStringConstraint("name", false)]
    [QueryStringConstraint("", true)]
    public IActionResult Get()
    {
        return Ok("From `Get` Action");
    }
}
Run Code Online (Sandbox Code Playgroud)

测试结果

在此输入图像描述