C# HTTPWebRequest 与领域站点协商/基本

FES*_*EST 3 c# authentication httprequest basic-authentication negotiate

很抱歉就这个主题做另一个问题,但我一直在阅读,但无法真正让它发挥作用......

切入正题...我有一个可以下载 zip 文件的链接。我已获得访问该网站(我的用户)的权限,现在我正在尝试编写一个程序来每隔一段时间获取 zip.file,以便我可以对其数据进行处理。

当我刚刚尝试下载需要身份验证的东西的代码时(见下面的代码),我得到一个 HTTP 401。我想知道为什么我的凭据没问题,我可以使用浏览器获取文件。

HttpWebRequest request;
HttpWebResponse response;

request = (HttpWebRequest)WebRequest.Create(strURL);

request.Credentials = ((domain == null || domain == "") ?
                new System.Net.NetworkCredential(user, pass) :
                new System.Net.NetworkCredential(user, pass, domain)));

response = (HttpWebResponse)request.GetResponse();
//...do somthing
Run Code Online (Sandbox Code Playgroud)

在阅读和搜索后,我得出的结论是我必须明确告诉身份验证者。方法并找到了 CredentialCache 示例。我之后做的是打开 fiddler 并查看网站的响应并找到以下标题:

WWW-Authenticate: Negotiate
WWW-Authenticate: Basic realm="BACKLOG ::::::: AUTENTICACAO NECESSARIA - USAR APENAS USERNAME E PASSWORD DE DOMINIO"
Run Code Online (Sandbox Code Playgroud)

有了这个,我可以知道什么是 auth. 我需要输入,所以我将代码更改为以下内容:

HttpWebRequest request;
HttpWebResponse response;

request = (HttpWebRequest)WebRequest.Create(strURL);

CredentialCache cc = new CredentialCache();
cc.Add(new Uri(strURL), "Negotiate", (
                (domain == null || domain == "") ?
                new System.Net.NetworkCredential(user, pass) :
                new System.Net.NetworkCredential(user, pass, domain)));

request.Credentials = cc;

response = (HttpWebResponse)request.GetResponse();
//...do somthing
Run Code Online (Sandbox Code Playgroud)

尽管如此,我还是收到了 401 HTTP 错误。阅读更多内容后,我发现我可能需要添加 request.PreAuthenticate = true; 但结果是一样的。我也读过,我需要使用 request.Credentials = System.Net.CredentialCache.DefaultNetworkCredentials; 我停止了 401 HTTP 错误,但开始给我“尝试了太多自动重定向”错误,这让我放弃了这个想法,因为默认凭据不是我想要的,我真的希望能够更改用户凭据。

多次尝试后,我放弃并尝试了基本身份验证。方法但没有成功(我想我不知何故错过了“领域”部分)。

HttpWebRequest request;
HttpWebResponse response;

request = (HttpWebRequest)WebRequest.Create(strURL);

CredentialCache cc = new CredentialCache();
cc.Add(new Uri(strURL), "Basic", (
                (domain == null || domain == "") ?
                new System.Net.NetworkCredential(user, pass) :
                new System.Net.NetworkCredential(user, pass, domain)));

request.Credentials = cc;

response = (HttpWebResponse)request.GetResponse();
//...do somthing
Run Code Online (Sandbox Code Playgroud)

无论如何,我对此失去了一点希望,并对我所做的所有搜索感到困惑。我真的觉得请求和响应有点愚蠢......

任何人都可以对这个主题有所了解并帮助我吗?我更喜欢 Negotiate 方法,因为它更安全。我是否必须以某种方式编写第二个请求来模拟握手?

================================================== ================================================== ==== 更新 1

比较请求浏览器与应用程序后,我可以看到第一个请求中的一些差异。

浏览器

GET http://example.host/maps/selector/download?map_name=workline_fornecedores&organization_id=1 HTTP/1.1 接受:application/x-ms-application, image/jpeg, application/xaml+xml, image/gif, image/pjpeg , application/x-ms-xbap, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, application/x-shockwave-flash, / 接受语言:PTSI 用户代理:Mozilla/4.0(兼容;MSIE 7.0;Windows NT 6.1;WOW64;Trident/4.0;SLCC2;.NET CLR 2.0.50727;.NET CLR 3.5.30729;.NET CLR 3.0 ; Media Center PC 6.0; InfoPath.2; MS-RTC LM 8; .NET4.0C; .NET4.0E; .NET CLR 1.1.4322) Accept-Encoding: gzip, deflate Connection: Keep-Alive Host: example.host cookie:WT_FPC=id=2edd0fbf206582555011375854627180:lv=1375854634144:ss=1375854627180;_swa_uv=527730711375858227;__utma=163226403.1634134224.1376900542.1376900542.1376900542.1

....

应用程序

GET http://example.host/maps/selector/download?map_name=workline_fornecedores&organization_id=1 HTTP/1.1 Host: example.host Cache-Control: no-store,no-cache Pragma: no-cache Connection: Keep-Alive

在两个示例中,服务器响应 401 HTTP 但有差异

...

浏览器

GET http://example.host/authorizer/check_remote_user?url=%2Fmaps%2Fselector%2Fdownload%3Fmap_name%3Dworkline_fornecedores%26organization_id%3D1 HTTP/1.1 接受:application/x-ms-application, image/jpeg, application/xaml+ xml、图像/gif、图像/pjpeg、应用程序/x-ms-xbap、应用程序/vnd.ms-excel、应用程序/vnd.ms-powerpoint、应用程序/msword、应用程序/x-shockwave-flash、/ 接受语言:PTSI 用户代理:Mozilla/4.0(兼容;MSIE 7.0;Windows NT 6.1;WOW64;Trident/4.0;SLCC2;.NET CLR 2.0.50727;.NET CLR 3.5.30729;.NET CLR 3.0 ; Media Center PC 6.0; InfoPath.2; MS-RTC LM 8; .NET4.0C; .NET4.0E; .NET CLR 1.1.4322) Accept-Encoding: gzip, deflate Connection: Keep-Alive Host: example.host cookie:WT_FPC=id=2edd0fbf206582555011375854627180:lv=1375854634144:ss=1375854627180;_swa_uv=527730711375858227;__utma=163226403.1634134224.1376900542.1376900542.1376900542.1;阿帕奇=10.188.178.249.1416485792985658

...

应用程序

获取http://example.host/authorizer/check_remote_user?url=%2Fmaps%2Fselector%2Fdownload%3Fmap_name%3Dworkline_fornecedores%26organization_id%3D1 HTTP/1.1 主机:example.host Cookie:Apache=10.188.178.249.1736Cache:无存储,无缓存 Pragma:无缓存

在此之后,浏览器请求不断发送凭据并获取文件,而我的应用程序不执行任何其他操作。

除了当前标头上的明显差异之外,我还注意到 cookie 标头差异可能是原因吗?我将在 Roder 中发出更多“浏览”请求以检查 cookie 值是否发生变化。

FES*_*EST 6

我终于做到了!

我发现,在使用我的代码发出请求时,我被重定向到第二个 url。这个 url 是固定的,如果我尝试向第二个链接发出请求,我可以下载文件。

还是不明白为什么...

我必须确保在我的 WebRequest 中有一个 CookieContainer,否则将抛出“尝试了太多自动重定向”异常(我相信这是因为目标尝试设置 cookie)。

最终代码 - 其中 strURL 是文本中的第二个链接(重定向链接)

HttpWebRequest request;
HttpWebResponse response;

request = (HttpWebRequest)WebRequest.Create(strURL);

CredentialCache cc = new CredentialCache();
cc.Add(new Uri(strURL), "Negotiate", (
            (domain == null || domain == "") ?
            new System.Net.NetworkCredential(user, pass) :
            new System.Net.NetworkCredential(user, pass, domain)));

request.Credentials = cc;
request.CookieContainer = new CookieContainer();

response = (HttpWebResponse)request.GetResponse();
Run Code Online (Sandbox Code Playgroud)