HttpClient 忽略 AllowAutoRedirect 指令

本文关键字:指令 AllowAutoRedirect 忽略 HttpClient | 更新日期: 2023-09-27 17:55:54

我正在尝试使用System.Net.Http.HttpClient执行一些HEAD检查。它适用于除 301 重定向以外的所有内容。在这种情况下,HttpClient 说状态为 200,即使服务器说 301 并给出了一个新位置。

这是一个快速测试来验证我所看到的内容:

[Test]
public async void Test301Capture()
{
    const string url = "http://www.youtube.com/"; // http://www.youtube.com/ does a 301 to https://www.youtube.com/
    var httpClientHandler = new HttpClientHandler() { AllowAutoRedirect = false };
    using (var client = new HttpClient(httpClientHandler))
    using (var headRequest = new HttpRequestMessage(HttpMethod.Head, url))
    using (var headResponse = await client.SendAsync(headRequest))
    {
        Log.InfoFormat("Result of HEAD check on '"{0}'" is: {1}", url, headResponse.StatusCode); // *should* be a 301
        Assert.AreNotEqual(200, (int)headResponse.StatusCode); // *fails* because StatusCode *is* 200
    }
}

上面的输出是:

Result of HEAD check on "http://www.youtube.com/" is: OK

我知道服务器说 301,因为使用高级 REST 客户端测试 http://www.youtube.com/的快速 HEAD 检查说:

Redirect
To:https://www.youtube.com/ with status: 301 Show explanation HTTP/1.1 301 Moved Permanently
Redirection information has not been cached.
Date: Tue, 01 Apr 2014 20:08:09 GMT 
Server: gwiseguy/2.0 
Content-Length: 0 
Cache-Control: no-cache 
X-Content-Type-Options: nosniff
Content-Type: text/html; charset=utf-8 
Expires: Tue, 27 Apr 1971 19:44:06 EST 
X-XSS-Protection: 1; mode=block; report=https://www.google.com/appserve/security-bugs/log/youtube
Location: https://www.youtube.com/ 
Alternate-Protocol: 80:quic

我确实找到了这个答案,使用 HttpClient,我将如何防止自动重定向并获取原始状态代码并在 301 的情况下放弃 Url,但是我的结果不同。

HttpClient 忽略 AllowAutoRedirect 指令

我也尝试做一个win32 pinvoke版本,我仍然得到200 OK作为状态代码。但我确实找到了这个答案,它应该有效。http://mail-archives.apache.org/mod_mbox/hc-dev/200501.mbox/%3CA68B76BAB1C27346BE22FED8D74ACE2C0283BB@owm-exc-1.telecomsys.com%3E

如果你想看,下面是我用于研究的pinvoke代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
namespace HttpTest
{
    class Program
    {
        const int INTERNET_OPEN_TYPE_DIRECT = 1;     // direct to net
        const short INTERNET_DEFAULT_HTTP_PORT = 80;
        const int INTERNET_SERVICE_HTTP = 3;
        const uint INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTP = 0x00008000;
        const uint INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTPS = 0x00004000;
        const uint INTERNET_FLAG_KEEP_CONNECTION = 0x00400000;
        const uint INTERNET_FLAG_NO_AUTH = 0x00040000;
        const uint INTERNET_FLAG_NO_AUTO_REDIRECT = 0x00200000;
        const uint INTERNET_FLAG_NO_COOKIES = 0x00080000;
        const uint INTERNET_FLAG_NO_UI = 0x00000200;
        const uint INTERNET_FLAG_RELOAD = 0x80000000;
        const int HTTP_QUERY_STATUS_CODE = 19;
        [DllImport("wininet.dll", SetLastError = true, CharSet = CharSet.Auto)]
        static extern IntPtr InternetOpen(string lpszAgent, int dwAccessType, string lpszProxyName, string lpszProxyBypass, int dwFlags);
        [DllImport("wininet.dll", SetLastError = true, CharSet = CharSet.Auto)]
        static extern IntPtr InternetConnect(IntPtr hInternet, string lpszServerName, short nServerPort, string lpszUsername, string lpszPassword, int dwService, int dwFlags, IntPtr dwContext);
        [DllImport("wininet.dll", SetLastError = true)]
        static extern IntPtr HttpOpenRequest(
            IntPtr hConnect,
            string lpszVerb,
            string lpszObjectName,
            string lpszVersion,
            string lpszReferer,
            string[] lplpszAcceptTypes,
            uint dwFlags,
            IntPtr dwContext);
        [DllImport("wininet.dll", SetLastError = true, CharSet = CharSet.Auto)]
        static extern bool HttpSendRequest(IntPtr hRequest, IntPtr lpszHeaders, uint dwHeadersLength, IntPtr lpOptional, uint dwOptionalLength);
        [DllImport("wininet.dll", SetLastError = true)]
        static extern bool HttpQueryInfo(IntPtr hInternet, int dwInfoLevel, IntPtr lpBuffer, ref long lpdwBufferLength, ref long lpdwIndex);
        static void Main(string[] args)
        {
            IntPtr openPtr = InternetOpen("Kyle", INTERNET_OPEN_TYPE_DIRECT, null, null, 0);
            IntPtr context = IntPtr.Zero;
            IntPtr connectPtr = InternetConnect(openPtr, "www.youtube.com", INTERNET_DEFAULT_HTTP_PORT, null, null, INTERNET_SERVICE_HTTP, 0, context);
            IntPtr openRequestPtr = HttpOpenRequest(connectPtr, "GET", "/", null, null, null,
                INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTP |
                INTERNET_FLAG_IGNORE_REDIRECT_TO_HTTPS |
                INTERNET_FLAG_KEEP_CONNECTION |
                INTERNET_FLAG_NO_AUTH |
                INTERNET_FLAG_NO_AUTO_REDIRECT |
                INTERNET_FLAG_NO_COOKIES |
                INTERNET_FLAG_NO_UI |
                INTERNET_FLAG_RELOAD, IntPtr.Zero);
            if (HttpSendRequest(openRequestPtr, IntPtr.Zero, 0, IntPtr.Zero, 0))
            {
                IntPtr buffer = Marshal.AllocHGlobal(4); long lBufferLen = 4; long lHeaderIndex = 0;
                HttpQueryInfo(openRequestPtr, HTTP_QUERY_STATUS_CODE, buffer, ref lBufferLen, ref lHeaderIndex);
                string str2 = Marshal.PtrToStringAnsi(buffer);
            }
        }
    }
}