7

关于泛型类型响应对象的文档,我对 Swagger 的 ServiceStack 实现有疑问。强类型的响应对象被正确记录和显示,但是一旦将泛型类型的对象用作响应,则文档不准确且具有误导性。

请求 DTO

[Route("/users/{UserId}", "GET", Summary = "Get a specific User Profile")]
public class GetUser : IReturn<ServiceResponse<UserProfile>>
{
    [ApiMember(Description = "User Id", ParameterType = "path", IsRequired = true)]
    public int UserId { get; set; }
}

响应 DTO

public class ServiceResponse<T> : IServiceResponse<T>
{
    public IList<string> Errors { get; set; }
    public bool Successful { get; set; }
    public string Message { get; set; }
    public string StackTrace { get; set; }
    public T Data { get; set; }

    public ServiceResponse()
    {
        Errors = new List<string>();
    }
}

响应 DTO 类型

public class UserProfile : RavenDocument
{
    public UserProfile()
    {
        Races = new List<UserRace>();
        Workouts = new List<Workout>();
    }
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public string DisplayName { get; set; }
    public DateTime? BirthDate { get; set; }
    public Gender? Gender { get; set; }
    public string UltracartPassword { get; set; }
    public string UltracartCartId { get; set; }

    [UniqueConstraint]
    public string Email { get; set; }

    public string ImageUrl { get; set; }

    public FacebookUserInfo FacebookData { get; set; }
    public GoogleUserInfo GoogleData { get; set; }

    public DateTime CreatedOn { get; set; }
    public DateTime? LastUpdated { get; set; }
    public UserAddress ShippingAddress { get; set; }
    public UserAddress BillingAddress { get; set; }
    public IList<UserRace> Races { get; set; }
    public IList<Workout> Workouts { get; set; }
}

这些例子非常简单。没有什么真正的 hacky 或聪明的事情发生,但这是我从 Swagger 获得的开箱即用的示例文档:

大摇大摆的例子

如您所见,泛型类型没有正确记录,而是使用了其他类型。由于我对所有响应都使用相同的 ServiceResponse 包装器,因此这种情况正在全面发生。

4

1 回答 1

2

正如您所发现的,ServiceStack swagger 插件当前并未尝试干净地处理泛型类型。一个更好的简单替代方法是创建泛型类型的具体子类。例如:

public class UserProfileResponse : ServiceResponse<UserProfile> { ... }

public class GetUser : IReturn<UserProfileResponse> ...

这应该由 Swagger 妥善处理。

我发现泛型类型并不总是非常适合 ServiceStack DTO。您会在 StackOverflow 上找到许多讨论(例如这里这里这里),讨论这个问题,以及具体类型和通常避免继承对于 ServiceStack DTO 来说是一个好主意的原因。

需要努力克服将 DRY 原则应用于请求/响应 DTO 的诱惑。我的想法是泛型和继承是促进以泛型、可重用方式实现算法的语言特性,其中泛型方法或基类不需要了解具体类型的细节。虽然 DTO 表面上可能具有看起来像是继承或泛型机会的通用结构,但在这种情况下,每个 DTO 的实现和语义对于每个具体用法都是不同的,因此每个请求/响应消息的细节都值得明确定义。

于 2014-01-06T15:13:32.453 回答