我试图使用C#(WebClient / HttpWebRequest)登录我的WordPress管理面板 .

我发送POST请求到 /wp-login.php . 它响应这样的cookie:

Set-Cookie: wordpress_26...cf9e; path=/wordpress/wp-content/plugins; httponly
Set-Cookie: wordpress_26...cf9e; path=/wordpress/wp-admin; httponly
Set-Cookie: wordpress_logged_in_26...43b3; path=/wordpress/; httponly

并且它还重定向到 /wp-admin/Location: http://109.120.169.99/wordpress/wp-admin/

问题是第二个cookie(带有 path=/wp-admin )未添加到CookieContainer,因此登录失败 .

我查看了HttpWebRequest源代码和http://referencesource.microsoft.com/#System/net/System/Net/_CookieModule.cs(OnReceiveHeaders),发现它使用 ResponseUri 表示所有cookie .

我尝试使用响应uri手动添加cookie,并获得路径无效的异常 . 之后我发现了这个答案CookieException with CookieContainer: The 'Path' part of the cookie is invalid:它说Uri必须在添加到CookieContainer时匹配cookie中的路径 .

我最终禁用了AllowAutoRedict(手动处理)并手动添加这些cookie .

这是我的代码:

public static void Auth(string url, string username, string password)
{
    var webClient = new CookieAwareWebClient();

    // load page to get some cookies
    string loginPageHtml = webClient.DownloadString(url);

    webClient.Headers.Add(HttpRequestHeader.ContentType, "application/x-www-form-urlencoded");
    webClient.Headers.Add(HttpRequestHeader.Referer, url);

    var postValues = new NameValueCollection()
        {
            {"log", username},
            {"pwd", password},
            {"wp-sumbit", "Log In"},
            {"redirect_to", url.Replace("wp-login.php", "wp-admin/") },
            {"testcookie", "1"}
        };

    webClient.AllowRedirects = false; // to handle cookies and redirect manually

    byte[] response = webClient.UploadValues(url, postValues);
    string html = Encoding.UTF8.GetString(response);

    // <FIX>, handle cookies and redirect manually

    string cookies = webClient.ResponseHeaders[HttpResponseHeader.SetCookie];
    var cookiesArr = cookies.Split(',');

    foreach (var cookieStr in cookiesArr)
    {
        var parts = cookieStr.Split(';').Select(s => s.Trim());

        foreach (var part in parts)
        {
            if (part.Contains("path="))
            {
                string path = part.Replace("path=", "");

                var uri = new Uri(url);

                if (path != "/" && !uri.LocalPath.Contains(path))
                {
                    var uriBuilder = new UriBuilder(uri.Scheme, uri.Host, 80,
                        path);

                    webClient.CookieContainer.SetCookies(uriBuilder.Uri, cookieStr);
                }
            }
        }
    }
    // </FIX>

    while (webClient.StatusCode() == HttpStatusCode.Redirect)
    {
        string redirectUrl = webClient.ResponseHeaders[HttpResponseHeader.Location];

        html = webClient.DownloadString(redirectUrl);
    }

    if (html.Contains("?action=logout"))
        Console.WriteLine("Login success");
    else
        Console.WriteLine("Login fail");
}

Web客户端:

// WebClient with CookieContainer
public class CookieAwareWebClient : WebClient
{
    private WebRequest _request = null;
    private WebResponse _response = null;

    public CookieContainer CookieContainer { get; private set; }

    public string UserAgent { get; set; }

    public bool AllowRedirects { get; set; }

    public CookieAwareWebClient()
    {
        UserAgent = "Mozilla/5.0 (Windows NT 6.2; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/29.0.1547.76 Safari/537.36";
        CookieContainer = new CookieContainer();
        AllowRedirects = true;
    }

    protected override WebRequest GetWebRequest(Uri address)
    {
        _request = base.GetWebRequest(address);
        if (_request is HttpWebRequest)
        {
            var httpRequest = (HttpWebRequest)_request;
            httpRequest.CookieContainer = CookieContainer;
            httpRequest.UserAgent = UserAgent;

            httpRequest.AllowAutoRedirect = AllowRedirects;

            httpRequest.ProtocolVersion = HttpVersion.Version11;

            httpRequest.Timeout = 50000;
        }
        return _request;
    }

    protected override WebResponse GetWebResponse(WebRequest request)
    {
        _response = base.GetWebResponse(request);

        return _response;
    }

    // Returns status code of the last response
    public HttpStatusCode StatusCode()
    {
        var httpResponse = _response as HttpWebResponse;

        if (httpResponse == null)
            throw new InvalidOperationException("Unable to retrieve the status code, maybe you have not made a request yet.");

        var result = httpResponse.StatusCode;

        return result;
    }
}

我也尝试过使用.NET 4.5 HttpClient但结果相同(我认为它下面也使用了HttpWebRequest):

public static void Auth2(string url, string username, string password)
{
    using (var httpClient = new HttpClient())
    {
        var loginPageHtml = httpClient.GetStringAsync(url).Result;

        var postContent = new List<KeyValuePair<string, string>>()
            {
                new KeyValuePair<string, string>("log", username),
                new KeyValuePair<string, string>("pwd", password),
                new KeyValuePair<string, string>("wp-submit", "Log In"),
                new KeyValuePair<string, string>("redirect_to", url.Replace("wp-login.php", "wp-admin/")),
                new KeyValuePair<string, string>("testcookie", "1")
            };

        var response = httpClient.PostAsync(url, new FormUrlEncodedContent(postContent)).Result;

        string html = response.Content.ReadAsStringAsync().Result;

        if (html.Contains("?action=logout"))
            Console.WriteLine("Login success");
        else
            Console.WriteLine("Login fail");
    }
}

我做错了什么或是HttpWebRequest / CookieContainer中的错误?

如果有人想测试它,这里是完整的源代码:https://gist.github.com/AlexP11223/5c176972426605ee2112(我的测试网站和登录/密码也应该工作)

而Fiddler记录: