前言

  其实老喜欢取经,偶尔也得分享下。关于阿里巴巴国际站的登陆,过程有点复杂但是算不上难。一不小心少个东西倒也挺麻烦的。

  主要是看下请求类HttpClient基本请求封装使用,AliClient模拟浏览器的操作与数据封装

  这里只是简单说一下主要的类和注意点,主要步骤与注意点都写在代码注释里了。项目源码下载地址:http://git.oschina.net/GspringG/AliLogin

正文

  1. 主要类/方法

  • HttpClient请求模拟的基础类,也就那么个过程http header设置一下,然后模拟就行了.需要注意的地方在代码注释里

    using System;
    using System.Drawing;
    using System.IO;
    using System.Net;
    using System.Net.Security;
    using System.Security.Cryptography.X509Certificates;
    using System.Text; namespace Main.Http
    {
    public class HttpClient
    {
    const string Accept = "text/html, application/xhtml+xml, */*";
    public const string UserAgent = "Mozilla/5.0 (Windows NT 6.1; WOW64; Trident/7.0; rv:11.0) like Gecko";
    const int Timeout = ;//超时时间,但前提是请求允许超时才行。
    const string AcceptLanguage = "Accept-Language:zh-CN";
    const string AcceptEncoding = "Accept-Encoding:gzip, deflate";
    const string Contenttype = "application/x-www-form-urlencoded";
    /// <summary>
    /// 静态构造方法
    /// </summary>
    static HttpClient()
    {
    //伪造证书,验证服务器证书回调自动验证
    ServicePointManager.ServerCertificateValidationCallback = CheckValidationResult;
    //客户端系统 win7或者winxp上可能会出现 could not create ssl/tls secure channel的问题导致加载ali登陆验证码报错
    //这里必须设置
    ServicePointManager.Expect100Continue = false;//默认是true,要手动设为false
    ServicePointManager.SecurityProtocol=SecurityProtocolType.Ssl3; ServicePointManager.DefaultConnectionLimit = ;
    } public static string Get(string url, CookieContainer cookie, int retryCount = )
    {
    return Request(url, cookie, "GET", null, retryCount);
    }
    public static string Post(string url, CookieContainer cookie, string postData, int retryCount = )
    {
    return Request(url, cookie, "POST", postData, retryCount);
    } public static string Post(string url, CookieContainer cookie, string postData, Action<HttpWebRequest> beginRequest, int reTry = )
    {
    return Request(url, cookie, "POST", postData, reTry, beginRequest);
    } private static string Request(string url, CookieContainer cookie, string method, string postData, int retryCount, Action<HttpWebRequest> beginRequest = null)
    {
    string html = string.Empty;
    for (var i = ; i <= retryCount; i++)
    {
    try
    {
    html = Request(url, cookie, method, postData, beginRequest);
    return html;
    }
    catch (Exception e)
    {
    if (i == retryCount)
    {
    throw new Exception(e.ToString());
    }
    }
    }
    return html;
    } public static bool CheckValidationResult(object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors)
    { // 伪造证书,总是接受
    return true;
    } private static string Request(string url, CookieContainer cookie, string method, string postData, Action<HttpWebRequest> beginRequestHandle = null)
    {
    var request = (HttpWebRequest)WebRequest.Create(url);
    request.Method = method;
    request.CookieContainer = cookie;
    request.AllowAutoRedirect = true;
    request.ContentType = Contenttype;
    request.Accept = Accept;
    request.Timeout = Timeout;
    request.UserAgent = UserAgent;
    request.Headers.Add(AcceptLanguage);
    request.Headers.Add(AcceptEncoding); request.AutomaticDecompression = DecompressionMethods.GZip; if (beginRequestHandle != null)
    beginRequestHandle(request); if (!string.IsNullOrEmpty(postData))
    {
    byte[] byteRequest = Encoding.UTF8.GetBytes(postData);
    request.ContentLength = byteRequest.Length;
    var stream = request.GetRequestStream();
    stream.Write(byteRequest, , byteRequest.Length);
    stream.Close();
    }
    var httpWebResponse = (HttpWebResponse)request.GetResponse(); var responseStream = httpWebResponse.GetResponseStream();
    if (responseStream == null) return string.Empty;
    if (responseStream.CanTimeout)
    {
    responseStream.ReadTimeout = ;
    }
    var encoding = Encoding.GetEncoding(httpWebResponse.CharacterSet ?? "UTF8"); var streamReader = new StreamReader(responseStream, encoding); string html = streamReader.ReadToEnd();
    streamReader.Close();
    responseStream.Close();
    request.Abort();
    httpWebResponse.Close();
    return html;
    } public static byte[] DownloadFile(string url, CookieContainer cookie)
    {
    var webClient = new CookieWebClient { Cookie = cookie };
    byte[] data = webClient.DownloadData(url);
    return data;
    }
    public static Image DownloadImage(string url, CookieContainer cookie)
    {
    byte[] data = DownloadFile(url, cookie);
    var ms = new MemoryStream(data);
    var image = Image.FromStream(ms);
    ms.Close();
    return image;
    }
    }
    }
  • AliClient:相当于一个浏览器,存储着cookie、session、_csrf_token_、userName等信息。客户端发出的模拟请求都通过AliClient发出保证全部在一个会话中。(想想一个浏览器需要些什么,浏览器是怎么做的。就感觉很合理了)

    using System;
    using System.Collections.Specialized;
    using System.Net;
    using System.Web;
    using Main.Http; namespace Main.Ali
    {
    public class AliClient
    {
    const string _csrf_token_ = "_csrf_token_";
    public CookieContainer Cookie { get; set; }
    public string SessionId { get; set; } public string CsrfToken { get; set; }
    public AliLoginUser AliLoginUser { get; set; }
    public string DmtrackPageid { get; set; }
    public string UserName
    {
    get
    {
    if (AliLoginUser == null) return null;
    if (AliLoginUser.person_data == null) return null;
    return AliLoginUser.person_data.login_id;
    }
    }
    public AliClient()
    {
    Cookie = new CookieContainer();
    }
    public string Get(string url)
    {
    return HttpClient.Get(AddCsrfTokenToUrl(url), this.Cookie, );
    }
    public string Post(string url, string postData)
    {
    return HttpClient.Post(AddCsrfTokenToUrl(url), this.Cookie, AddCsrfTokenToPostData(postData), );
    }
    //post请求时不用手动加了,默认都加上
    private string AddCsrfTokenToPostData(string postData)
    {
    NameValueCollection queryString = HttpUtility.ParseQueryString(postData);
    if (queryString[_csrf_token_] == null)
    {
    queryString.Add("_csrf_token_", this.CsrfToken);
    }
    return queryString.ToString();
    }
    //Get请求时不用手动加了,默认都加上
    private string AddCsrfTokenToUrl(string url)
    { if (string.IsNullOrWhiteSpace(this.CsrfToken)) return url;
    UriBuilder ub = new UriBuilder(url);
    NameValueCollection queryString = HttpUtility.ParseQueryString(ub.Query);
    if (queryString[_csrf_token_] == null)
    {
    queryString.Add("_csrf_token_", this.CsrfToken);
    }
    ub.Query = queryString.ToString();
    string newurl = ub.Uri.ToString();
    return newurl;
    } }
    }
  • AliPassporter:阿里巴巴登陆核心类(具体个注意细节点都在代码内部已注释)

    登陆过程为:从上到下依次

      1.PrepareLogin:

         /// <summary>
/// 登陆前先请求一下页面获取sessionId,DmtrackPageid,初始化cookie的值,像正常访问浏览器一样
/// </summary>
/// <param name="aliClient"> AliClient当需要多账号登陆时,每个账号的cookie 各种验证id等独有的东西放到各自的Client中,相当于每个独立的浏览器</param>
/// <param name="isSpec">针对某种特殊情况取值方法不一样,默认false</param>
public static void PrepareLogin(AliClient aliClient, bool isSpec = false);

      2.DoLoginStep1:

        /// <summary>
/// 登录阿里巴巴
/// </summary>
/// <param name="aliClient"> AliClient当需要多账号登陆时,每个账号的cookie 各种验证id等独有的东西放到各自的Client中</param>
/// <param name="account">帐户名</param>
/// <param name="password">密码</param>
/// <param name="checkCode">验证码</param>
/// <returns>登陆结果(true/false)</returns>
public static bool DoLoginStep1(AliClient aliClient, string account, string password, string checkCode);

      3.DoLoginStep2:

         /// <summary>
/// 登录处理第二步
/// </summary>
/// <param name="aliClient"></param>
/// <param name="userId">账户</param>
/// <param name="password">密码</param>
/// <param name="dmtrackPageid">令牌1</param>
/// <param name="st">令牌三</param>
/// <returns>登陆结果</returns>
private static bool DoLoginStep2(AliClient aliClient, string userId, string password, string dmtrackPageid, string st);

      4.DoCheckCode:

         /// <summary>
/// 获取checkcode,图片验证码,其实有时候是不需要的。可以先判断一下,不行再获取验证码
/// </summary>
/// <param name="aliClient">aliClient</param>
/// <returns>验证码图片,如果是web应用,直接把checkCodeUrl(img src=checkCodeUrl)写进去就行了</returns>
public static Image DoCheckCode(AliClient aliClient);

      5.登陆过程中还会调用GetToken、GetSt、GetCsrfToken等方法提供登录所需要的get/post参数

      6.前端调用流程:登录过程关键代码(详情参见详细代码)

           //1.初始化_aliClient,相当于打开一个浏览器,并设置一个空的cookie
AliClient _aliClient=new AliClient {Cookie = new CookieContainer()};
//2.无需验证码登陆时这里替换为PrepareLogin(_aliClient);
RefreshCheckCode();
//3.登陆系统,异步方法防止页面假死
var data=await AliPassporter.DoLoginAsync(_aliClient, UserName, Password, CheckCode);
if (data)
{
//登陆成功,登陆后其他数据就随便抓了,这里要注意,需要手机验证码的,要在网页中先把手机验证码输入了
//我的测试账号默认写在里面,别乱玩就行了
//进入管理个人信息页面
var html = _aliClient.Get("http://accounts.alibaba.com/user/organization/manage_person_profile.htm");
MessageBox.Show("登陆成功!");
//_aliClient.Post("");
}
else
{
MessageBox.Show("登陆失败!");
}

总结

  以上就是阿里巴巴国际站登陆的全部过程,总体大同小异,很久之前写的了(建议先把注释看一遍再用)。现在拿出来分享一下,顺便复习一下HttpWebRequest的相关使用。

使用HttpWebRequest模拟登陆阿里巴巴(alibaba、httpwebrequest、login)的更多相关文章

  1. c# 使用 HttpWebRequest模拟登陆

    c# 使用 HttpWebRequest模拟登陆(附带验证码) 分类: C# .net2010-06-04 00:50 35647人阅读 评论(43) 收藏 举报 c#exceptionstreams ...

  2. 使用C#的HttpWebRequest模拟登陆访问人人网(转)

    无论使用任何语言做模拟登陆或者抓取访问页面,无外乎以下思路:第一 启用一个web访问会话方法或者实例化一个web访问类,如.net中的HttpWebRequest:第二 模拟POST或者GET方式提交 ...

  3. c# 使用 HttpWebRequest模拟登陆(附带验证码)

    在C#中,可以使用HttpWebRequest进行相关的模拟登陆,登陆后进行相关的操作,比如抓取数据,页面分析,制作相关登陆助手等等. 先说下流程 1.使用httpwebrequest先进入你要登录的 ...

  4. 使用C#的HttpWebRequest模拟登陆访问人人网

    使用任何语言做模拟登陆或者抓取访问页面,无外乎以下思路: 第一 启用一个web访问会话方法或者实例化一个web访问类,如.net中的HttpWebRequest:第二 模拟POST或者GET方式提交的 ...

  5. 转:使用C#的HttpWebRequest模拟登陆网站

    这篇文章是有关模拟登录网站方面的. 实现步骤: 启用一个web会话 发送模拟数据请求(POST或者GET) 获取会话的CooKie 并根据该CooKie继续访问登录后的页面,获取后续访问的页面数据. ...

  6. 使用C#的HttpWebRequest模拟登陆网站

    很久没有写新的东西了,今天在工作中遇到的一个问题,感觉很有用,有种想记下来的冲动. 这篇文章是有关模拟登录网站方面的. 实现步骤: 启用一个web会话 发送模拟数据请求(POST或者GET) 获取会话 ...

  7. HttpWebRequest模拟登陆,存储Cookie以便登录请求后使用

    [一篮饭特稀原创,转载请注明出自http://www.cnblogs.com/wanghafan/p/3284481.html] PostLogin :登录,并保存Cookie 1 public st ...

  8. C#如何HttpWebRequest模拟登陆,获取服务端返回Cookie以便登录请求后使用

    public static string GetCookie(string requestUrlString, Encoding encoding, ref CookieContainer cooki ...

  9. 通过HttpWebRequest实现模拟登陆

    1>通过HttpWebRequest模拟登陆 using System; using System.Collections.Generic; using System.Linq; using S ...

随机推荐

  1. 验证(C#和正则表达式)

    原文:验证(C#和正则表达式) 我们经常会需要验证字符串的格式,比如密码长度范围.电子邮件格式.固定电话号码和手机号码格式等,这个时候我们经常会需要用到正则表达式.但是正则表达式用起来性能会低一点,所 ...

  2. PHP 17: MySQL的简单介绍

    原文:PHP 17: MySQL的简单介绍 这一章将简单介绍MySQL的基本知识. 本文来自http://lib.hackbase.com/html/8/35125.htm. MySQL是最受欢迎的开 ...

  3. testNg的安装与卸载

    1.testNG的安装 打开eclips,点击Help菜单.选择Install New Software. 在弹出的窗口的work with的输入框,输入http://beust.com/eclips ...

  4. CreateFont详细解释

    CFont * f;    f = new CFont;    f->CreateFont(10, // nHeight         0, // nWidth         0, // n ...

  5. rem测试用实现移动端自适应页面

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  6. GIT+云盘作 做 文档管理工具

    GIT+云盘作 做 文档管理工具 在工作中, 会遇到公司的文档 和 自己家里的 文档进行同步的问题, 通常我们使用U盘作为传输节制, 但是不是非常好,文档的改动都不能发现, 导致回家同步的时候, 出各 ...

  7. Javascript多线程引擎(七)

    Javascript多线程引擎(七)--synchronized关键字 经过两天的努力, 今天synchronzied关键字终于支持了, 如下是测试代码 thread() 是一个开启新线程的API, ...

  8. Oracle单行函数笔记

    Oracle中单行函数的分类:1.字符函数substr():字符串截取select substr('我爱你,你知道么?',0,4) from dual执行结果:我爱你,length函数:字符串长度se ...

  9. Java类之间的关联关系(转载)

    Java类之间的关联关系 UML类图中的关系分为四种:泛化.依赖.关联.实现:关联关系又可以细化为聚合和组合. 一.泛化(Generalization) 泛化是父类和子类之间的关系,子类继承父类的所有 ...

  10. 一个Shift的后门程序,可以让你可以进入你不知道密码的电脑

    1.前提 你可以在平时亲身接触状态电脑,哪怕是在电脑主人不在的时候(虽然主人不在,或者关机了,进入电脑是要密码的). 2.原理 利用电脑连续按5次Shift会触发粘滞键,它会运行c:\winows\s ...