在之前的文章中,我已經介紹過一個社交網站模擬登錄的類庫:imitate-login ,這是一個通過c#的HttpWebRequest來模擬網站登錄的庫,之前實現了微博網頁版和微博Wap版;現在,模擬百度登錄的部分也已經完成。由於個人時間的限制,加上目前有多個項目在同時進行,因此更新頻率會根據項目關注度來決定(Star & fork)。
這個類庫的使用方法非常簡單,僅對外提供一個方法:
LoginResult Login(1: string userName, 2: string password, 3: LoginSite loginSite);
這個方法位於ImitateLogin的LoginHelper類中,使用之前需要先對其進行實例化。通過傳入 用戶名、密碼以及登錄的網站,返回一個包含登錄結果狀態、描述信息和Cookies字典的類。它通過 Thrift 來實現多語言的支持。
下面將通過介紹模擬百度登錄的實現來介紹如何進行擴充與二次開發:
首先,創建百度登錄類 BaiduLogin.cs 繼承 ILogin 接口;實現其生成的 DoLogin 方法。
#region ILogin implementation public LoginResult DoLogin(string UserName, string Password) { throw new NotImplementedException(); } public CookieContainer cookies { set; get;} #endregion
然後我們通過監聽百度登錄過程中的網絡請求,梳理出修改過Cookies和最終提交登錄所需的參數的請求。
Step1: 訪問以下鏈接生成初始Cookies:
HttpHelper.GetHttpContent("https://passport.baidu.com/passApi/html/_blank.html", cookies: cookies, cookiesDomain: "passport.baidu.com");
Step2: 獲取最終登錄提交所需的token:
//1. Get the token. string token_url = string.Format("https://passport.baidu.com/v2/api/?getapi&tpl=mn&apiver=v3&tt={0}&class=login&gid={1}&logintype=dialogLogin&callback=bd__cbs__{2}", TimeHelper.ConvertDateTimeInt(DateTime.Now), Guid.NewGuid().ToString().ToUpper(), build_callback()); string prepareContent = HttpHelper.GetHttpContent(token_url, null, cookies, referer: "https://www.baidu.com/", encode: Encoding.GetEncoding("GB2312"), cookiesDomain: "passport.baidu.com"); //string prepareJson = prepareContent.Split('(')[1].Split(')')[0]; dynamic prepareJson = JsonConvert.DeserializeObject(prepareContent.Split('(')[1].Split(')')[0]); string token = prepareJson.data.token;
其中 build_callback 為隨機生成6位字母或數字的組合的方法。
Step3: 獲取用於加密密碼的publickey:
//2. Get public key string pubkey_url = "https://passport.baidu.com/v2/getpublickey?token={0}&tpl=mn&apiver=v3&tt={1}&gid={2}&callback=bd__cbs__{3}"; string pubkeyContent = HttpHelper.GetHttpContent(string.Format(pubkey_url, token, TimeHelper.ConvertDateTimeInt(DateTime.Now), Guid.NewGuid().ToString().ToUpper(), build_callback()), null, cookies, referer: "https://www.baidu.com/", encode: Encoding.GetEncoding("GB2312"), cookiesDomain: "passport.baidu.com"); dynamic pubkeyJson = JsonConvert.DeserializeObject(pubkeyContent.Split('(')[1].Split(')')[0]); rsa_pub_baidu = pubkeyJson.pubkey; string KEY = pubkeyJson.key;
stopwatch 是一個記錄從最初執行到最終提交之前的耗時的一個計時器,get_pwa_rsa 為加密密碼的方法。
Step4: 模擬執行最終的登錄:
//3. Build post data string login_data = "staticpage=https%3A%2F%2Fwww.baidu.com%2Fcache%2Fuser%2Fhtml%2Fv3Jump.html&charset=UTF-8&token={0}&tpl=mn&subpro=&apiver=v3&tt={1}&codestring=&safeflg=0&u=https%3A%2F%2Fwww.baidu.com%2F&isPhone=&detect=1&gid={2}&quick_user=0&logintype=dialogLogin&logLoginType=pc_loginDialog&idc=&loginmerge=true&splogin=rate&username={3}&password={4}&verifycode=&mem_pass=on&rsakey={5}&crypttype=12&ppui_logintime={6}&countrycode=&callback=parent.bd__pcbs__{7}"; login_data = string.Format(login_data, token, TimeHelper.ConvertDateTimeInt(DateTime.Now), Guid.NewGuid().ToString().ToUpper(), HttpUtility.UrlEncode(UserName), HttpUtility.UrlEncode(get_pwa_rsa(Password)), HttpUtility.UrlEncode(KEY), stopwatch.ElapsedMilliseconds, build_callback()); //4. Post the login data string login_url = "https://passport.baidu.com/v2/api/?login"; HttpHelper.GetHttpContent(login_url, login_data, cookies, referer: "https://www.baidu.com/", cookiesDomain: "passport.baidu.com");
Step5:驗證最終的登錄結果:
string home_url = "https://www.baidu.com"; string result = HttpHelper.GetHttpContent(home_url, cookies: cookies, cookiesDomain: "passport.baidu.com"); //5. Verifty the login result if (string.IsNullOrWhiteSpace(result) || result.Contains("賬號存在異常") || !result.Contains("bds.comm.user=\"")) { return new LoginResult() { Result = ResultType.AccounntLimit, Msg = "Fail, Msg: Login fail! Maybe you account is disable or captcha is needed." }; }
Step6:創建返回結果類:
LoginResult loginResult = new LoginResult() { Result = ResultType.Success, Msg = "Success", Cookies = HttpHelper.GetAllCookies(cookies) };
至此,模擬登錄部分的代碼就完成了,為了能夠被其它程序調用,你還需要在 LoginSite 的枚舉中新增一條來標識這個登錄方法,此處增加了一個 Baidu = 5,並設置 [Description(“Baidu”)]。
然後在 LoginHelper.cs 的 Login 方法中的 switch (loginSite) 裡增加一個 case:
case LoginSite.Baidu: LoginClass = new BaiduLogin (); break;
本文來自 The NewIdea,作者 Carey Tzou 。好了,大功告成!Todo List中還有淘寶、QQ、Facebook、Twitter、Google要做呢,我還想加入GitHub、Wechat…
現在,你可以幫我了嗎?
首發地址:http://www.tnidea.com/login-baidu-throught-imitate-login.html
未經授權,拒絕任何全文及摘要轉載!