【问题标题】:Web API versioning URLWeb API 版本控制 URL
【发布时间】:2017-08-03 09:20:38
【问题描述】:

我在尝试对我的 Web API 进行版本控制时遇到了一点问题:

[ApiVersion("1.0")]
[ApiVersion("2.0")]
[System.Web.Http.Route("api/v{version:apiVersion}/monitors")]
[ControllerName("Monitors")]
public sealed class MonitorsController : ApiController
{
   /* 
     /monitors/get?heartbeat=foo
   */
   [System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]
   [MapToApiVersion("1.0")]
   public JsonResult GetHeartbeatStatusV1(string heartbeat)
   {
      var x = new JsonResult {Data = "heartbeat v1"};
      return x;
   }
   [System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]
   [MapToApiVersion("2.0")]
   public JsonResult GetHeartbeatStatusV2(string heartbeat)
   {
      var x = new JsonResult { Data = "heartbeat v1" };
      return x;
   }
   /* 
      /monitors/get?alert=foo 
   */
   [System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]
   [MapToApiVersion("2.0")]
   public JsonResult GetAlertStatus(string alert)
   {
      var x = new JsonResult {Data = "alerts"};
      return x;
   }
   /* 
      /monitors/get?oDataQuery=foo 
   */
   [System.Web.Http.Route("monitors/get")]
   public JsonResult GetODataQuery(string oDataQuery)
   {
      var x = new JsonResult {Data = "oDataQuery"};
      return x;
   }
}

以上是我的控制器。我有一堆我希望能够根据版本调用的操作。

问题是我还需要它们都是 /get 然后根据参数名称决定调用哪个方法,这就是为什么我有:

[System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]

我想我可以用[MapToApiVersion("1.0")][MapToApiVersion("2.0")] 装饰方法来区分调用哪个方法。 但是,在发布的示例中,如果我调用 api/v1.0/monitors/get?heartbeat=foo,我会收到有关多个操作的错误。如果我调用 v2.0,它会说它根本找不到任何方法。

?

我希望能够做到

http://foo:blah/api/v1.0/monitors/get?heartbeat=foo

http://foo:blah/api/v2.0/monitors/get?heartbeat=foo

http://foo:blah/api/v2.0/monitors/get?alert=foo

这可能吗?

编辑


我尝试过的一件事是将它们分成 2 个不同的控制器,在不同的命名空间中。

[ApiVersion("2.0")]
[System.Web.Http.Route("api/v{version:apiVersion}/monitors")]
[ControllerName("Monitors")]
public sealed class MonitorsController : ApiController
{
    /* 
        /monitors/get?heartbeat=foo
    */
    [System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]
    [MapToApiVersion("2.0")]
    public JsonResult GetHeartbeatStatusV2(string heartbeat)
    {
        var x = new JsonResult {Data = "heartbeat v2"};
        return x;
    }
    /* 
        /monitors/get?alert=foo 
    */
    [System.Web.Http.Route("api/v{version:apiVersion}/monitors/get")]
    public JsonResult GetAlertStatus(string alert)
    {
        var x = new JsonResult {Data = "alerts"};
        return x;
    }

}

这与查找多个动作的错误相同。

我的 WebApiConfig.cs 如下所示:

public static class WebApiConfig
{
    public static void Register(HttpConfiguration config)
    {
        // Web API configuration and services
        // Configure Web API to use only bearer token authentication.
        config.SuppressDefaultHostAuthentication();
        config.Filters.Add(new HostAuthenticationFilter(OAuthDefaults.AuthenticationType));
        var constraintResolver = new DefaultInlineConstraintResolver
                                 {
                                     ConstraintMap =
                                     {
                                         ["apiVersion"] = typeof(ApiVersionRouteConstraint)
                                     }
                                 };
        config.MapHttpAttributeRoutes(constraintResolver);
        config.AddApiVersioning();
        config.Routes.MapHttpRoute(
            name: "DefaultApi",
            routeTemplate: "api/v{version:apiVersion}/{controller}/{id}",
            defaults: new { id = RouteParameter.Optional }
        );
    }
}

【问题讨论】:

    标签: c# asp.net-web-api


    【解决方案1】:

    您这样做的方式是正确的,但您可以使其更简单易读,而不是使用 ApiVersion 属性 只需将 MapHttpRoute 设为默认即可,您可以让您的控制器在同一个命名空间中,只需更改名称 MonitorsControllerV2

    [RoutePrefix("api/V2.0/monitors")] 
    public sealed class MonitorsControllerV2 : ApiController
    {
        /* 
            /monitors/get?heartbeat=foo
        */
    
         [Route("GetHeartbeat")]
        public JsonResult GetHeartbeatStatusV2(string heartbeat)
        {
            var x = new JsonResult {Data = "heartbeat v2"};
            return x;
        }
        /* 
            /monitors/get?alert=foo 
        */
          [Route(" GetAlertStatus")]
        public JsonResult GetAlertStatus(string alert)
        {
            var x = new JsonResult {Data = "alerts"};
            return x;
        }
    
    }
    

    【讨论】:

    • 我试过这个,当我转到 http://localhost:53713/api/v2.0/monitors/Getheartbeat?heartbeat=555 时,我现在收到一个错误,即需要版本但未指定。
    • 好的,把[RoutePrefix("api/V2.0/monitors")] 改成` [RoutePrefix("api/v{version:apiVersion}/monitors")]