c# - 为同一端点使用任何类型的内容类型

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

我有一个公开此功能的 asp.net 核心 (v2.1) webapi 项目:

[HttpPost]
[Route("v1/do-something")]
public async Task<IActionResult> PostDoSomething(ModelData model)
{
    //...
}

和这个模型:

public class ModelData
{
    [Required]
    public string Email { get; set; }
}

从内容类型的角度来看,我想让这个端点更灵活。因此,应该可以在正文中发送此端点不同的内容类型。

例如,那些“BODY”参数将被允许:

// application/x-www-form-urlencoded
email="abc123@gmail.com"

// application/json
{
    "email": "abc123@gmail.com"
}

与旧的 .net 框架相比,在 dotnet 核心中这是不允许开箱即用的。我发现我需要添加 Consume 属性和 [FormForm] 属性。但是,如果我将 [FormForm] 属性添加到模型参数,它将不再适用于 JSON(例如)- 因为那时它应该是 [FromBody]

我认为可以使用这样的代码:

[HttpPost]
[Route("v1/do-something")]
public async Task<IActionResult> PostDoSomething([FromBody] [FromForm] ModelData model)
{
    //...
}

但如您所料,此代码无法正常工作。

因此,为了实现这种灵 active ,我必须复制我的所有端点——这听起来是一个非常非常糟糕的主意。

[HttpPost]
[Route("v1/do-something")]
[Consume ("application/json")]
public async Task<IActionResult> PostDoSomething([FromBody] ModelData model)
{
    //...
}

[HttpPost]
[Route("v1/do-something")]
[Consume ("application/x-www-form-urlencoded")]
public async Task<IActionResult> PostDoSomething([FromForm] ModelData model)
{
    //...
}

// ... Other content types here ...

这听起来很容易。但似乎更复杂。

我错过了什么?如何使端点在任何内容类型中工作?

最佳答案

这里是 a custom model binder基于内容类型的绑定(bind)。

public class BodyOrForm : IModelBinder
{
    private readonly IModelBinderFactory factory;

    public BodyOrForm(IModelBinderFactory factory) => this.factory = factory;

    public async Task BindModelAsync(ModelBindingContext bindingContext)
    {
        var contentType = 
            bindingContext.ActionContext.HttpContext.Request.ContentType;

        BindingInfo bindingInfo = new BindingInfo();
        if (contentType == "application/json")
        {
            bindingInfo.BindingSource = BindingSource.Body;
        }
        else if (contentType == "application/x-www-form-urlencoded")
        {
            bindingInfo.BindingSource = BindingSource.Form;
        }
        else
        {
            bindingContext.Result = ModelBindingResult.Failed();
        }

        var binder = factory.CreateBinder(new ModelBinderFactoryContext
        {
            Metadata = bindingContext.ModelMetadata,
            BindingInfo = bindingInfo,
        });

        await binder.BindModelAsync(bindingContext);
    }
}

通过以下操作进行测试。

[HttpPost]
[Route("api/body-or-form")]
public IActionResult PostDoSomething([ModelBinder(typeof(BodyOrForm))] ModelData model)
{
    return new OkObjectResult(model);
}

这是一个demo is on GitHub .

关于c# - 为同一端点使用任何类型的内容类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52135408/

相关文章:

c# - connection.Close() 后对 SQLite 的更改丢失

c# - web.config>configuration>runtime>assemblyBinding 中生成的条目的含义/原因是什么?

jquery - Angular JS API 身份验证

c# - 登录后覆盖.Net Core "CultureInfo"

c# - 为什么这个字符在 Visual Studio 中不可见?

c# - 多线程多个生产者和使用者线程不会同步BlockingCollection争用条件

asp.net-core - 在身份服务器 4 中成功验证后如何在应用程序级别授权用户

javascript - 如何从 API JSON 数据创建下拉菜单

c# - 什么是 HttpContext.Response.HasStarted?

c# - 通过 Serilog 在 Azure 上记录 ASP.NET Core