如何在 Hyper 中正确处理多个 Set-Cookie headers?

How to correctly handle multiple Set-Cookie headers in Hyper?

我正在使用 Hyper 发送 HTTP 请求,但是当响应中包含多个 cookie 时,Hyper 会将它们合并为一个,然后解析过程失败。

例如,这里有一个简单的 PHP 脚本

<?php

setcookie("hello", "world");
setcookie("foo", "bar");

使用 curl 响应:

$ curl -sLD - http://local.example.com/test.php
HTTP/1.1 200 OK
Date: Sat, 24 Dec 2016 09:24:04 GMT
Server: Apache/2.4.25 (Unix) PHP/7.0.14
X-Powered-By: PHP/7.0.14
Set-Cookie: hello=world
Set-Cookie: foo=bar
Content-Length: 0
Content-Type: text/html; charset=UTF-8

但是对于以下 Rust 代码:

let client = Client::new();
let response = client.get("http://local.example.com/test.php")
    .send()
    .unwrap();
println!("{:?}", response);
for header in response.headers.iter() {
    println!("{}: {}", header.name(), header.value_string());
}

...输出将是:

Response { status: Ok, headers: Headers { Date: Sat, 24 Dec 2016 09:31:54 GMT, Server: Apache/2.4.25 (Unix) PHP/7.0.14, X-Powered-By: PHP/7.0.14, Set-Cookie: hello=worldfoo=bar, Content-Length: 0, Content-Type: text/html; charset=UTF-8, }, version: Http11, url: "http://local.example.com/test.php", status_raw: RawStatus(200, "OK"), message: Http11Message { is_proxied: false, method: None, stream: Wrapper { obj: Some(Reading(SizedReader(remaining=0))) } } }
Date: Sat, 24 Dec 2016 09:31:54 GMT
Server: Apache/2.4.25 (Unix) PHP/7.0.14
X-Powered-By: PHP/7.0.14
Set-Cookie: hello=worldfoo=bar
Content-Length: 0
Content-Type: text/html; charset=UTF-8

这对我来说似乎真的很奇怪。我使用 Wireshark 捕获响应,其中有 two Set-Cookie headers。我还检查了 Hyper 文档,但没有任何线索...

我注意到 Hyper 在内部使用 VecMap<HeaderName, Item> 来存储 headers。所以他们将它们连接成一个?那我以后应该怎么把它们分成一个个的饼干呢?

我认为 Hyper 更喜欢将 cookie 放在一起,以便更容易地使用它们做一些额外的事情,比如使用 CookieJar (cf. this implementation outline) 检查加密签名。

另一个原因可能是保持 API 简单。 Headers 在 Hyper 中按类型索引,您只能使用 Headers::get.

获得该类型的单个实例

在 Hyper 中,您通常会使用相应的类型来访问 header。在这种情况下,类型是 SetCookie。例如:

if let Some (&SetCookie (ref cookies)) = response.headers.get() {
    for cookie in cookies.iter() {
        println! ("Got a cookie. Name: {}. Value: {}.", cookie.name, cookie.value);
    }
}

访问 Set-Cookie 的原始 header 值意义不大,因为那样你就必须重新实现对引号和 cookie 属性的正确解析(参见 RFC 6265, 4.1)。


P.S。请注意,在 Hyper 10 中不再解析 cookie,because 用于解析的 crate 触发了 openssl 依赖地狱。