当存在多种内容类型时,读取HttpContent字节在DelegatingHandler内部失败

时间:2016-08-16 05:05:34

标签: c# httprequest hmac

我正在尝试为API实施HMAC安全性。一切正常,直到我尝试将数据值与MultipartFormDataContent中的文件一起发布。

当遇到要读取字节的异步代码行时,HttpClient DelegatingHandler会无声地失败。

以下是构建请求的代码:

private FileOutputViewModel GetApiOutput(Uri apiResource, string filename, byte[] file, IDictionary<string, string> extraParameters)
{
    FileOutputViewModel result = new FileOutputViewModel();

    if (file != null)
    {
        using (var content = new MultipartFormDataContent())
        {
            if (extraParameters != null)
            {
                foreach (var param in extraParameters)
                {
                    content.Add(new StringContent(param.Value), param.Key); // <- If I don't have this, everything works fine
                }
            }

            var fileContent = new ByteArrayContent(file);
            fileContent.Headers.ContentDisposition = new ContentDispositionHeaderValue("attachment")
            {
                FileName = filename
            };
            content.Add(fileContent);

            var response = HttpClient.PostAsync(apiResource.ToString(), content).Result;

            result.Output = JsonConvert.DeserializeObject(response.Content.ReadAsStringAsync().Result);

            result.Filename = Path.GetFileName(filename);
        }
    }

    return result;
}

如果我不使用DelegatingHandler,一切正常,但HMAC安全性并未针对请求实施,因此在API结束时会被拒绝。

如果我不使用文件旁边的StringContent项添加数据值,那么读取字节就没问题了。但是我留下了一个不完整的请求,因为我需要传递更多信息和文件。

DelegatingHandler中失败的代码行如下所示:

private static async Task<byte[]> ComputeHash(HttpContent httpContent)
{
    using (var md5 = MD5.Create())
    {
        byte[] hash = null;
        if (httpContent != null)
        {
            var ms = new MemoryStream();
            await httpContent.CopyToAsync(ms); // <- Fails here
            ms.Seek(0, SeekOrigin.Begin);

            var content = ms.ToArray();
            if (content.Length != 0)
            {
                hash = md5.ComputeHash(content);
            }
        }
        return hash;
    }
}

最初的失败路线是:

var content = await httpContent.ReadAsByteArrayAsync();

但即使只是文件本身(先前的Stackoverflow question)也失败了。使用MemoryStream向前迈出了一步,但并没有让我一路走来。

我有什么想法可以解决这个问题吗?

1 个答案:

答案 0 :(得分:1)

似乎这是由System.Net.Http.DelegatingHandler.SendAsync方法的异步签名引起的。委托覆盖最初是:

protected async override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)

当我调整代码以便我可以将其更改为:

protected override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
一切都开始按预期工作了。

似乎.NET框架的这一部分必然存在线程问题。如果您需要尝试其他解决方法,请参阅此处介绍的其他解决方法:https://social.msdn.microsoft.com/Forums/vstudio/en-US/55f5571d-fe94-4b68-b1d4-bfb91fd721dd/reading-httpcontent-bytes-fails-inside-delegatinghandler-when-multiple-content-types-present?forum=wcf