如何让AutoRest通过控制器拆分API

时间:2016-10-06 18:47:01

标签: c# asp.net-core rest-client auto-generate autorest

现在我正在使用:

AutoRest\AutoRest.exe -Input %jsonUrl% -Namespace %projectName%ClientAutoGen -OutputDirectory %projectName%Client

生成我的ASP.NET Core Rest Client

令人烦恼的是AutoRestfile/class中的所有controllers创建了一个API。我已经使用pre-ASP.NET Core auto-generators将每个controller拆分为自己的file/class,有没有办法在AutoRest中强制执行此行为?

2 个答案:

答案 0 :(得分:3)

根据AutoRest团队在GitHub上的有用答案(此处:https://github.com/Azure/autorest/issues/1497),答案是在您希望拆分发生的OperationId中使用_。这是我Filter的版本:

public class SwashbuckleOperationFilter : IOperationFilter
{
    public void Apply(Operation operation, OperationFilterContext context)
    {
        try
        {
            var pathSegments = context.ApiDescription.RelativePath.Split(new[] { '/' }).ToList();

            var version = string.Empty;
            var controller = string.Empty;
            if (pathSegments.Count > 1)
            {
                version = pathSegments[0];
                controller = pathSegments[1] + "_";

                pathSegments = pathSegments.Skip(2).Where(x => !x.Contains("{")).ToList();
            }

            string httpMethod = FirstCharToUpper(context.ApiDescription.HttpMethod);
            var routeName = context.ApiDescription.ActionDescriptor?.AttributeRouteInfo?.Name ?? string.Empty;

            operation.OperationId = $"{version}{controller}{httpMethod}{string.Join("", pathSegments)}{routeName}";
        }
        catch (Exception ex)
        {
            throw new Exception("Are you missing the [Route(\"v1/[controller]/\")] on your Controller?", ex);
        }
    }

    private string FirstCharToUpper(string input)
    {
        if (String.IsNullOrEmpty(input))
            return string.Empty;

        input = input.Trim().ToLower();

        return input.First().ToString().ToUpper() + new string(input.Skip(1).ToArray());
    }
}

StartUp

中使用这样的内容
services.AddSwaggerGen(options =>
{
    options.OperationFilter<SwashbuckleOperationFilter>();
    //...
}

像这样转变API Controller Method

[Route("v1/[controller]/")]
public class ThingController : Controller
{
    [HttpGet("ById/{id}")]
    [Produces(typeof(ThingDTO))]
    public async Task<IActionResult> GetThing([FromRoute] long id)
    {
        // Your implementation
    }
}

进入这种生成的service method

public class ThingClient : IThingClient
{
    private readonly AppSettings appSettings;
    private readonly IMapper mapper;
    private IV1Thing api;

    public ThingClient(IOptions<AppSettings> appSettingsOptions, IMapper mapper)
    {
        appSettings = appSettingsOptions.Value;
        this.mapper = mapper;
    }

    private IV1Thing service => api ??
                (api = new V1Thing(new ThingService(new Uri(appSettings.URLs.ThingService))));

    public async Task<IThing> GetByIdAsync(long thingId)
    {
        return mapper.Map<IThing>(await service.GetByIdAsync(thingId));
    }
}

答案 1 :(得分:2)

这适用于.net core 2.1,在我的控制器中我定义了SwaggerOperationAttribute。

    public class SwashbuckleOperationFilter : IOperationFilter
{
    public void Apply(Operation operation, OperationFilterContext context)
    {
        if (context.ApiDescription.ActionDescriptor is ControllerActionDescriptor controllerActionDescriptor)
        {
            operation.OperationId = $"{controllerActionDescriptor.ControllerName}_{operation.OperationId}";
        }
    }

}