HttpWebRequest
我有一个使用/发出 GET 和 POST 请求的基本类HttpWebResponse
。
我使用我的班级登录 API,然后请求数据。在 Windows 8“Metro”应用程序中,它完全按预期工作。在 Windows Phone 8 应用程序上,登录似乎成功,但在随后的数据请求中,没有发送任何 cookie,并且服务器响应就像客户端未登录一样。
这是类,在 Windows 8 应用程序和 Windows Phone 应用程序中使用了完全相同的代码:
class Class1
{
CookieContainer cookieJar = new CookieContainer();
CookieCollection responseCookies = new CookieCollection();
public async Task<string> httpRequest(HttpWebRequest request)
{
string received;
using (var response = (HttpWebResponse)(await Task<WebResponse>.Factory.FromAsync(request.BeginGetResponse, request.EndGetResponse, null)))
{
using (var responseStream = response.GetResponseStream())
{
using (var sr = new StreamReader(responseStream))
{
cookieJar = request.CookieContainer;
responseCookies = response.Cookies;
received = await sr.ReadToEndAsync();
}
}
}
return received;
}
public async Task<string> get(string path)
{
var request = WebRequest.Create(new Uri(path)) as HttpWebRequest;
request.CookieContainer = cookieJar;
return await httpRequest(request);
}
public async Task<string> post(string path, string postdata)
{
var request = WebRequest.Create(new Uri(path)) as HttpWebRequest;
request.Method = "POST";
request.CookieContainer = cookieJar;
byte[] data = Encoding.UTF8.GetBytes(postdata);
using (var requestStream = await Task<Stream>.Factory.FromAsync(request.BeginGetRequestStream, request.EndGetRequestStream, null))
{
await requestStream.WriteAsync(data, 0, data.Length);
}
return await httpRequest(request);
}
}
以及发起请求的代码:
var n = new Class1();
await n.post("https://mydomain.com/api/login/", "username=myusername&password=mypassword");
await n.get("https://mydomain.com/reader/feeds/");
奇怪的是,如果我在域名前加上“www”。它适用于 Windows Phone 8 应用程序和 Windows 8 Metro 应用程序。
我认为这与域的处理方式有关。cookie 的域是“.mydomain.com”,如果没有前缀,它必须认为 cookie 不属于该域。经过一番搜索,我发现有人注意到类似问题的报告。
我不明白为什么在 Windows 8 应用程序中对它的处理方式与在 Windows Phone 应用程序中不同,因此逐行相同的代码在一个平台上工作但在另一个平台上失败。
我对此做了更多的研究。
我用于此的服务器代码是在 PHP 中:
<?php
if ($_REQUEST["what"] == "set")
{
setcookie("TestCookie",$_REQUEST["username"] . " " . $_REQUEST["password"], time()+3600*24, "/", "subd.mydomain.com");
}
if ($_GET["what"] == "get")
{
var_dump($_COOKIE);
C#中的客户端代码:
var n = new ClassLibrary1.Class1();
await n.get("http://subd.mydomain.com/?what=set&username=foo&password=bar");
await n.get("http://subd.mydomain.com/?what=get");
这是来自服务器的 cookie 响应示例
Set-Cookie: ARRAffinity=295612ca; Path=/;Domain=subd.mydomain.com
Set-Cookie: TestCookie=foo+bar; expires=Fri, 04-Jan-2013 17:19:25 GMT; path=/; domain=subd.mydomain.com
在 Windows 8 Store/Metro 应用程序中,结果如下:
array(2) {
["ARRAffinity"]=>
string(8) "295612ca"
["TestCookie"]=>
string(7) "foo bar"
}
在 Windows Phone 应用程序中,结果如下:
array(0){
}
以这种方式设置时,Windows Phone 不会看到任何 cookie。
我更改了TestCookie
设置方式,服务器的结果现在如下所示:
Set-Cookie: ARRAffinity=295612ca;Path=/;Domain=subd.mydomain.com
Set-Cookie: TestCookie=foo+bar; expires=Fri, 04-Jan-2013 17:29:59 GMT; path=/
TestCookie
现在没有明确设置域,ARRAffinity 不变。
Windows 8 Store/Metro 应用程序现在返回:
array(2) {
["TestCookie"]=>
string(7) "foo bar"
["ARRAffinity"]=>
string(8) "295612ca"
}
Windows Phone 8 应用程序返回以下内容:
array(1) {
["TestCookie"]=>
string(7) "foo bar"
}
ARRAffinity
cookie 未发送,因为它明确声明了一个域。
如果我分配一些断点并检查CookieContainer
请求,我在m_domainTable
+ [0] {[.subd.mydomain.com, System.Net.PathList]} System.Collections.Generic.KeyValuePair<string,System.Net.PathList>
+ [1] {[subd.mydomain.com, System.Net.PathList]} System.Collections.Generic.KeyValuePair<string,System.Net.PathList>
未发送的 cookie 在.subd.mydomain.com
容器中。这在 Windows 8 和 Windows Phone 8 上都是一样的。
但是,如果 cookie 像这样声明自己:
Set-Cookie: TestCookie=foo+bar; expires=Fri, 04-Jan-2013 17:19:25 GMT; path=/; domain=.mydomain.com
它在 Windows Phone 8 上正确发送。
在我原来的情况下,无论是通过 mydomain.com 还是 www.mydomain.com 访问,服务器都以相同的方式声明 cookie;作为“.mydomain.com”,但 Windows Phone 8 似乎不认为应该将“.mydomain.com”的 cookie 发送到“mydomain.com”。这是有问题的,因为即使服务器将“subd.mydomain.com”作为域,它也被视为具有前面的点,然后由于自身的故障而无法正常工作。似乎它必须不发送带有 cookie 的域信息才能正确处理它。