来自查询字符串

时间:2019-11-22 15:06:06

标签: c# asp.net-core json-deserialization

我正在ASP.NET Core(3.0)中构建RESTful API。每个控制器都有一种用于搜索域对象的方法,该方法接受 List > 作为如下定义的querystring参数:

    [HttpGet]
    [ProducesResponseType(200)]
    public ActionResult<IEnumerable<Service.Models.Asset>> Get(
        [FromQuery(Name = "criteria")] List<KeyValuePair<string, string>> criteria,
        [FromQuery] int? page = 0, 
        [FromQuery] int? size = null)
    {
        var dtoList = _assetService.SearchPaged(criteria, page, size);
        return Ok(dtoList);
    }

我将条件作为JSON序列化字符串传递,请求URI为:

http://api/asset?criteria=[{"key":"uprn","value":"h1"},{"key":"uprn","value":"h2"}]&page=0&size=100

我无法让控制器接受标准参数的值,<标准>标准列表始终为零长度。我尝试用URL编码文本,没有区别。

如果将参数更改为字符串,则可以使用JsonConvert成功地反序列化参数(相同的URI)。

    [HttpGet]
    [ProducesResponseType(200)]
    public ActionResult<IEnumerable<Service.Models.Asset>> Get(
        [FromQuery(Name = "criteria")] string criteriaString,
        [FromQuery] int? page = 0, 
        [FromQuery] int? size = null)
    {
        var criteria = Newtonsoft.Json.JsonConvert.DeserializeObject<IEnumerable<KeyValuePair<string, string>>>(criteriaString);
        var dtoList = _assetService.SearchPaged(criteria, page, size);
        return Ok(dtoList);
    }

我不想将模型绑定与强类型参数对象一起使用,因为我有一个通用参数转换器,该转换器从条件列表中构建一个表达式以应用于我的EF DbSet。这适用于许多不同的控制器。

这是可能的,还是我应该放弃并坚持使用我的字符串参数并自己管理反序列化?

1 个答案:

答案 0 :(得分:2)

基于SO How to bind Json Query string in asp.net core web api上的答案,您不能以json格式绑定查询参数。您需要编写自己的模型联编程序以反序列化为List<KeyValuePair<string, string>>

我想,在没有进行测试的情况下,根据链接的答案,应该可以执行以下操作:

创建模型联编程序实现:

public class KeyValueListModelBinder : IModelBinder
{
    public Task BindModelAsync(ModelBindingContext bindingContext)
    {
        var key = bindingContext.ModelName;
        var jsonString = bindingContext.ValueProvider.GetValue(key).FirstValue;
        MyCustomModel result = JsonConvert.DeserializeObject<IEnumerable<KeyValuePair<string, string>>>(jsonString);

        bindingContext.Result = ModelBindingResult.Success(result);
        return Task.CompletedTask;
    }
}

创建提供者:

public class KeyValueListModelBinderProvider : IModelBinderProvider
{
    public IModelBinder GetBinder(ModelBinderProviderContext context)
    {
        if (context.Metadata.ModelType == typeof(List<KeyValuePair<string, string>>))
            return new KeyValueListModelBinder();

        return null;
    }
}

在启动时注册提供商:

public void ConfigureServices(IServiceCollection services)
{
    services.AddMvc(config => config.ModelBinderProviders.Insert(0, new KeyValueListModelBinderProvider()));
}