我有一个使用 .Net 4.7.1 的网络 API,在其中一个端点上执行获取时,我注意到一些奇怪的行为。当我执行到端点 http://localhost:52684/api/v1/some-resource?client_timezone=%2B0500
时,接收到的参数客户端时区为 "0500"
而不是 "+0500"
。我唯一能想到的是 Url 被解码两次,所以 "%2B"
变成 "+"
然后 ""
。有谁知道造成这种情况的任何常见原因或为什么会发生这种情况?
Controller 上有一个类似的端点
[HttpGet]
[Authorize]
[ApiRoute("some-resource", StartingVersion = 1)]
[EnableCors("*", "*", "GET", "*")]
public IHttpActionResult SomeResource([FromUri] string timezone)
{
if (!DataValidationUtililties.IsValidClientTimezone(timezone))
{
return BadRequest();
}
return Ok();
}
我想我找到了原因。我创建了一个 SnakeCaseActionSelector
,它基本上重写了所有请求 URI。在这样做时,它会解码 url 并且永远不会再次对查询字符串参数进行编码。
public class SnakeCaseActionSelector : ApiControllerActionSelector
{
public override HttpActionDescriptor SelectAction(HttpControllerContext controllerContext)
{
var requestUri = controllerContext.Request.RequestUri;
var queryPairs = controllerContext.Request.GetQueryNameValuePairs().ToList();
if (!queryPairs.Any())
{
return base.SelectAction(controllerContext);
}
queryPairs = queryPairs.Select(x =>
new KeyValuePair<string, string>(CamelCaseToSnakeCaseConverter.FromSnakeCase(x.Key), x.Value))
.ToList();
var newQueryParams = queryPairs.Select(x => $"{x.Key}={x.Value}").Aggregate((x, y) => x + "&" + y);
var builder = new UriBuilder(requestUri)
{
Query = newQueryParams
};
controllerContext.Request.RequestUri = builder.Uri;
return base.SelectAction(controllerContext);
}
}