10

我正在转换HttpContent为以下 dto:

public class ContentDto 
{
     public string ContentType {get; set;}
     public string Headers {get; set; }
     public object Data { get; set; }

     public ContentDto(HttpContent content)
     {
          Headers = content.Headers.Flatten();
          // rest of the setup
     }
}

并且正在对其进行一些单元测试:

[Fact]
public void CanBuild()
{
     var content = new StringContent("some json", Enconding.UTF8, "application/json");
     var dto = new ContentDto(content);

     var contentHeaders = content.Headers.Flatten();

     Assert.Equal(contentHeaders, dto.Headers);
}

并且该测试失败,因为Content-Length我的 dto 上没有捕获标头。但是,如果我这样做:

[Fact]
public void CanBuild()
{
     var content = new StringContent("some json", Enconding.UTF8, "application/json");

     var contentHeaders = content.Headers.Flatten();

     var dto = new ContentDto(content);

     Assert.Equal(contentHeaders, dto.Headers);
}

测试通过并捕获所有标头。甚至更多我也试过这个:

 [Fact]
 public void CanBuild()
 {
     var content = new StringContent("some json", Enconding.UTF8, "application/json");

     var dto = new ContentDto(content);

     var contentHeaders = content.Headers.Flatten();

     var dto1 = new ContentDto(content);

     Assert.Equal(contentHeaders, dto.Headers);                
     Assert.Equal(contentHeaders, dto1.Headers);
}

它失败dto了,因为没有Content-Length标题,但是dto1有。我什至尝试在类似工厂的方法中获取标题,如下所示:

 public static ContentDto FromContent<T>(T content) where T : HttpContent
 {
      // same as the constructor
 }

StringContent看看关于标题的类是否有什么特别之处Content-Length,但这没有区别,无论我使用构造函数(使用基类HttpContent)还是泛型方法FromContent(在这种情况下使用实际的 StringContent )结果是相同。

所以我的问题是:

这是 的预期行为HttpContent.Headers吗?
是否有一些特定于实际HttpContent类型的标题?
我在这里想念什么?

注意:这是Flatten扩展方法的代码:

 public static string Flatten(this HttpHeaders headers)
 {
      var data = headers.ToDictionary(h => h.Key, h => string.Join("; ", h.Value))
                        .Select(kvp => $"{kvp.Key}: {kvp.Value}");

      return string.Join(Environment.NewLine, data)
 }
4

2 回答 2

4

你的例子不完整。ContentLength只有在调用扩展方法之前访问该属性时,我才能重新创建您的问题。在您的代码中的某处(很可能//其余设置)您直接或间接调用该属性,该属性很可能遵循延迟加载模式,然后在您下次调用扩展方法时将其包含在标头中并包含在内在构造的字符串中。它们不匹配,因为您在访问内容长度属性之前生成了手动字符串。

HttpContentHeaders.ContentLength的源代码中

public long? ContentLength
{
    get
    {
        // 'Content-Length' can only hold one value. So either we get 'null' back or a boxed long value.
        object storedValue = GetParsedValues(HttpKnownHeaderNames.ContentLength);

        // Only try to calculate the length if the user didn't set the value explicitly using the setter.
        if (!_contentLengthSet && (storedValue == null))
        {
            // If we don't have a value for Content-Length in the store, try to let the content calculate
            // it's length. If the content object is able to calculate the length, we'll store it in the
            // store.
            long? calculatedLength = _calculateLengthFunc();

            if (calculatedLength != null)
            {
                SetParsedValue(HttpKnownHeaderNames.ContentLength, (object)calculatedLength.Value);
            }

            return calculatedLength;
        }

        if (storedValue == null)
        {
            return null;
        }
        else
        {
            return (long)storedValue;
        }
    }
    set
    {
        SetOrRemoveParsedValue(HttpKnownHeaderNames.ContentLength, value); // box long value
        _contentLengthSet = true;
    }
}

您可以看到,如果您没有明确设置内容长度,那么它会在您第一次尝试访问它时将其添加(延迟加载)到标题中。

这证明了我关于在生成/展平字符串然后访问ContentLength属性并解释不一致的枚举之后添加它的原始理论。

于 2016-08-03T01:39:52.750 回答
0

似乎 HttpContent 类的 headers 属性有一个非常奇怪的行为。不知何故,内容长度似乎是按照此处所述计算的。它没有具体解决您的问题,但您可以使用类似于初始对象的新 httpContent 对象进行测试。我很确定您将能够毫无问题地获得内容长度。

于 2016-07-28T08:05:15.883 回答