【发布时间】:2012-10-22 22:50:05
【问题描述】:
我的目标是使用 AuthorizationFilter 或 DelegatingHandler 对 Web API 请求进行身份验证。我想在几个地方查找客户端 ID 和身份验证令牌,包括请求正文。起初看起来这很容易,我可以做这样的事情
var task = _message.Content.ReadAsAsync<Credentials>();
task.Wait();
if (task.Result != null)
{
// check if credentials are valid
}
问题是 HttpContent 只能读取一次。如果我在处理程序或过滤器中执行此操作,那么我的操作方法中的内容对我不可用。我在 StackOverflow 上找到了一些答案,例如:Read HttpContent in WebApi controller 解释说这是故意的,但他们没有说为什么。这似乎是一个非常严重的限制,阻止我在过滤器或处理程序中使用任何很酷的 Web API 内容解析代码。
这是技术限制吗?它是否试图阻止我做我没有看到的非常糟糕的事情(tm)?
尸检:
我查看了 Filip 建议的来源。 ReadAsStreamAsync 返回内部流并且没有什么能阻止你调用 Seek 如果流支持它。在我的测试中,如果我调用 ReadAsAsync 然后这样做:
message.Content.ReadAsStreamAsync().ContinueWith(t => t.Result.Seek(0, SeekOrigin.Begin)).Wait();
自动模型绑定过程在遇到我的操作方法时可以正常工作。不过我没有用这个,我选择了更直接的东西:
var buffer = new MemoryStream(_message.Content.ReadAsByteArrayAsync().WaitFor());
var formatters = _message.GetConfiguration().Formatters;
var reader = formatters.FindReader(typeof(Credentials), _message.Content.Headers.ContentType);
var credentials = reader.ReadFromStreamAsync(typeof(Credentials), buffer, _message.Content, null).WaitFor() as Credentials;
使用扩展方法(我在 .NET 4.0 中没有 await 关键字)
public static class TaskExtensions
{
public static T WaitFor<T>(this Task<T> task)
{
task.Wait();
if (task.IsCanceled) { throw new ApplicationException(); }
if (task.IsFaulted) { throw task.Exception; }
return task.Result;
}
}
最后一个问题,HttpContent 有一个硬编码的最大缓冲区大小:
internal const int DefaultMaxBufferSize = 65536;
因此,如果您的内容要大于该内容,则需要在尝试调用 ReadAsByteArrayAsync 之前手动调用更大尺寸的 LoadIntoBufferAsync。
【问题讨论】:
-
虽然你找到了不止一次阅读正文的方法,但你们有没有发现为什么不能阅读两次?因为我真的很想弄清楚为什么首先会有这个限制。