2

我正在尝试编写自己的输入格式化程序,它将读取请求正文,按行拆分并将其传递给控制器​​操作中的字符串数组参数。


这有效(将整个主体作为字符串传递):

启动.cs

public void ConfigureServices(IServiceCollection services)
{
    services.AddMvcCore(options =>
    {
        options.InputFormatters.Add(new MyInputFormatter());
    }
}


MyInputFormatter.cs

public override async Task<InputFormatterResult> ReadRequestBodyAsync(InputFormatterContext context)
{
    using (StreamReader reader = new StreamReader(context.HttpContext.Request.Body))
    {
        return InputFormatterResult.Success(await reader.ReadToEndAsync());
    }
}

我的控制器.cs

[HttpPost("/foo", Name = "Foo")]
public IActionResult Bar([FromBody] string foo)
{
    return Ok(foo);
}

这不起作用(参数 foo 为空):

MyInputFormatter.cs

public override async Task<InputFormatterResult> ReadRequestBodyAsync(InputFormatterContext context)
{
    List<string> input = new List<string>();

    using (StreamReader reader = new StreamReader(context.HttpContext.Request.Body))
    {
        while (!reader.EndOfStream)
        {
            string line = (await reader.ReadLineAsync()).Trim();
            input.Add(line);
        }
    }

    return InputFormatterResult.Success(input.ToArray());
}

我的控制器.cs

[HttpPost("/foo", Name = "Foo")]
public IActionResult Bar([FromBody] string[] foo)
{
    return Ok(string.Join(" ", foo));
}

不同之处在于,在控制器中,我现在接受的是字符串数组而不是字符串,而在格式化程序中,我正在逐行读取输入,最后将其作为数组返回。


我错过了什么?:/


编辑:我的格式化程序实际上看起来如何,或多或少(如果有什么不同的话):

    public class MyInputFormatter : InputFormatter
    {
        public MyInputFormatter()
        {
            this.SupportedMediaTypes.Add(new MediaTypeHeaderValue(MimeType.URI_LIST)); // "text/uri-list"
        }

        public override bool CanRead(InputFormatterContext context)
        {
            if (context == null) throw new ArgumentNullException(nameof(context)); // breakpoint here not reached

            if (context.HttpContext.Request.ContentType == MimeType.URI_LIST)
                return true;

            return false;
        }

        protected override bool CanReadType(Type dataType)
        {
            return typeof(string[]).IsAssignableFrom(dataType); // breakpoint here not reached
        }

        public override async Task<InputFormatterResult> ReadRequestBodyAsync(InputFormatterContext context)
        {
            List<string> input = new List<string>(); // breakpoint here not reached

            using (StreamReader reader = new StreamReader(context.HttpContext.Request.Body))
            {
                while (!reader.EndOfStream)
                {
                    string line = (await reader.ReadLineAsync()).Trim();

                    if (string.IsNullOrEmpty(line))
                    {
                        continue;
                    }

                    if (!line.StartsWith("foo-"))
                    {
                        return InputFormatterResult.Failure();
                    }

                    input.Add(line.Substring("foo-".Length));
                }
            }

            return InputFormatterResult.Success(input.ToArray());
        }
4

2 回答 2

0

我弄清楚到底是什么问题。我有一个自定义的 ModelBinder 进行干扰,捕获任何不是字符串的内容和自定义接口的实现(用于其他发布数据)。这就是为什么它适用于字符串和其他输入有效负载(接口的实现),但不适用于字符串数组。该绑定器应该用于查询参数(以便能够处理自定义类型),但最终也触发了这个 POST 有效负载。

于 2019-08-06T12:54:23.830 回答
0

我已经在请求处理程序方法中使用您的代码创建了一个测试输入格式化程序,它工作正常,这就是它的样子:

public class TestInputFormatter : IInputFormatter
{
    public bool CanRead(InputFormatterContext context) => true;

    public async Task<InputFormatterResult> ReadAsync(InputFormatterContext context)
    {
        List<string> input = new List<string>();

        using (StreamReader reader = new StreamReader(context.HttpContext.Request.Body))
        {
            while (!reader.EndOfStream)
            {
                string line = (await reader.ReadLineAsync()).Trim();
                input.Add(line);
            }
        }

        return InputFormatterResult.Success(input.ToArray());
    }
}

我在您的代码中只看到一点可能是错误的 - 您的输入格式化程序的注册。文档说:格式化程序按照您插入它们的顺序进行评估。第一个优先。 尝试像这样注册它:

options.InputFormatters.Insert(0, new TestInputFormatter());

它在我的测试项目中工作,正是这样的注册。因为当您调用options.InputFormatters.Add它时,它将被添加到输入格式化程序集合的末尾,并且您的请求可能会由位于该集合中的第一个其他输入格式化程序处理。

于 2019-08-05T21:50:38.540 回答