微信網(wǎng)頁(yè)版的通信協(xié)議,很多人都想自己寫(xiě)了個(gè)程序,實(shí)現(xiàn)微信的登錄、初始化、讀取聯(lián)系人列表、發(fā)送微信、接收微信等功能,其實(shí)大家在網(wǎng)上看一下也有不少人做過(guò)這方面的內(nèi)容。我主要用的工具是HTTPAnalyzer,我認(rèn)為這個(gè)是目前分析http/https協(xié)議最好用的工具了,比wireshark和fiddler都清晰明確,推薦大家分析http/https協(xié)議內(nèi)容使用這個(gè)工具。其實(shí)分析了協(xié)議自己再寫(xiě)代碼就很容易了。
自己開(kāi)發(fā)程序來(lái)收發(fā)微信,整個(gè)流程基本如下:
1、獲取會(huì)話(huà)UUID
微信Web版本不使用用戶(hù)名和密碼登錄,而是采用掃描二維碼登錄,所以服務(wù)器需要首先分配一個(gè)唯一的會(huì)話(huà)ID,用來(lái)標(biāo)識(shí)當(dāng)前的一次登錄。
使用get方法,通過(guò)請(qǐng)求地址:https://login.weixin.qq.com/jslogin?appid=wx782c26e4c19acffb&fun=new&lang=zh_CN&_=時(shí)間戳
其中,時(shí)間戳這個(gè)值是當(dāng)前距離林威治標(biāo)準(zhǔn)時(shí)間的毫秒。
get成功,則返回:window.QRLogin.code = 200; window.QRLogin.uuid = 'XXXXXXX'
其中的XXXXXXX就是我們需要的uuid
2、獲取登錄二維碼
訪問(wèn)網(wǎng)址:https://login.weixin.qq.com/qrcode/XXXXXX
這里的XXXXXXX就是我們剛才獲取的uuid,這個(gè)網(wǎng)址直接顯示的就是二維碼
3、查詢(xún)是否掃描二維碼登錄
顯示了二維碼以后,用戶(hù)必須用手機(jī)微信掃描這個(gè)二維碼才能登錄。(有人會(huì)說(shuō)微信為啥要這么設(shè)計(jì)?很奇怪的思維。我用電腦很多情況不就是因?yàn)槭謾C(jī)沒(méi)在旁邊嗎。其實(shí)是因?yàn)榘踩目紤])
使用get方法,查詢(xún)地址:https://login.weixin.qq.com/cgi-bin/mmwebwx-bin/login?uuid=XXXXXX&tip=1&_=時(shí)間戳
這里的XXXXXX是我們剛才獲取的uuid,時(shí)間戳同上。tip在第一次獲取時(shí)應(yīng)為1,這個(gè)數(shù)是每次查詢(xún)要變的。
如果服務(wù)器返回:window.code=201,則說(shuō)明此時(shí)用戶(hù)在手機(jī)端已經(jīng)完成掃描,但還沒(méi)有點(diǎn)擊確認(rèn),繼續(xù)使用上面的地址查詢(xún),但tip要變成0;
如果服務(wù)器返回:
window.code=200
window.redirect_uri='XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX'
則說(shuō)明此時(shí)用戶(hù)在手機(jī)端已經(jīng)確認(rèn)登錄,window.redirect_uri=后面的這個(gè)網(wǎng)址要記下來(lái),接著要訪問(wèn)這個(gè)地址。
如果服務(wù)器返回:window.code=408,則說(shuō)明等待超時(shí),繼續(xù)使用上面的地址查詢(xún),但tip=1
4、訪問(wèn)登錄地址,獲得uin、sid、pass_ticket、skey
用get方法,訪問(wèn)在上一步驟獲得訪問(wèn)地址,并在參數(shù)后面加上:&fun=new,會(huì)返回一個(gè)xml格式的文本,類(lèi)似這樣:
<error>
<ret>0</ret>
<message>OK</message>
<skey>xxx</skey>
<wxsid>xxx</wxsid>
<wxuin>xxx</wxuin>
<pass_ticket>xxx</pass_ticket>
<isgrayscale>1</isgrayscale>
</error>
把這里的wxuin,wxsid,skey,pass_ticket都記下來(lái),這是重要數(shù)據(jù)。
5、微信初始化
這個(gè)是很重要的一步,我在這個(gè)步驟折騰了很久。。。
要使用POST方法,訪問(wèn)地址:https://wx2.qq.com/cgi-bin/mmwebwx-bin/webwxinit?r=時(shí)間戳&lang=ch_ZN&pass_ticket=XXXXXX
其中,時(shí)間戳不用解釋?zhuān)琾ass_ticket是我們?cè)谏厦娅@取的一長(zhǎng)串字符。
POST的內(nèi)容是個(gè)json串,{'BaseRequest':{'Uin':'XXXXXXXX','Sid':'XXXXXXXX','Skey':XXXXXXXXXXXXX','DeviceID':'e123456789012345'}}
uin、sid、skey分別對(duì)應(yīng)上面步驟4獲取的字符串,DeviceID是e后面跟著一個(gè)15字節(jié)的隨機(jī)數(shù)。
程序里面要注意使用UTF8編碼方式。
POST成功,則服務(wù)器返回一個(gè)很長(zhǎng)的JSON串,格式是這樣:
{
'BaseResponse': {
'Ret': 0,
'ErrMsg': ''
},
'Count': 8,
'ContactList': [...],
'SyncKey': {
'Count': 4,
'List': [
{
'Key': 1,
'Val': 635705559
},
...
]
},
'User': {
'Uin': xxx,
'UserName': xxx,
'NickName': xxx,
'HeadImgUrl': xxx,
'RemarkName': '',
'PYInitial': '',
'PYQuanPin': '',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'HideInputBarFlag': 0,
'StarFriend': 0,
'Sex': 1,
'Signature': 'Apt-get install B',
'AppAccountFlag': 0,
'VerifyFlag': 0,
'ContactFlag': 0,
'WebWxPluginSwitch': 0,
'HeadImgFlag': 1,
'SnsFlag': 17
},
'ChatSet': xxx,
'SKey': xxx,
'ClientVersion': 369297683,
'SystemTime': 1453124908,
'GrayScale': 1,
'InviteStartCount': 40,
'MPSubscribeMsgCount': 2,
'MPSubscribeMsgList': [...],
'ClickReportInterval': 600000
}
這其中,User里面是自己的信息,UserName是用戶(hù)名,NickName是昵稱(chēng)。要注意UserName是每次都會(huì)變的。。。也就是說(shuō),獲取不到唯一的不變標(biāo)識(shí)表示身份,無(wú)語(yǔ)。。。SyncKey一般是4個(gè),要記下這里的Key和val,后面每次收信息都要用。至于count和contactList先不用管。
6、獲取好友列表
使用POST方法,訪問(wèn):https://wx2.qq.com/cgi-bin/mmwebwx-bin/webwxgetcontact?r=時(shí)間戳
POST的內(nèi)容為空。成功則以JSON格式返回所有聯(lián)系人的信息。格式類(lèi)似:
{
'BaseResponse': {
'Ret': 0,
'ErrMsg': ''
},
'MemberCount': 21,
'MemberList': [
{
'Uin': 0,
'UserName': xxx,
'NickName': 'Urinx',
'HeadImgUrl': xxx,
'ContactFlag': 3,
'MemberCount': 0,
'MemberList': [],
'RemarkName': '',
'HideInputBarFlag': 0,
'Sex': 0,
'Signature': 'xxxx',
'VerifyFlag': 8,
'OwnerUin': 0,
'PYInitial': 'URINX',
'PYQuanPin': 'Urinx',
'RemarkPYInitial': '',
'RemarkPYQuanPin': '',
'StarFriend': 0,
'AppAccountFlag': 0,
'Statues': 0,
'AttrStatus': 0,
'Province': '',
'City': '',
'Alias': 'Urinxs',
'SnsFlag': 0,
'UniFriend': 0,
'DisplayName': '',
'ChatRoomId': 0,
'KeyWord': 'gh_',
'EncryChatRoomId': ''
},
...
],
'Seq': 0
}
其中,MemberCount表示總共有多少聯(lián)系人,里面的內(nèi)容都比較清晰。
7、開(kāi)啟微信狀態(tài)通知
用POST方法,訪問(wèn):https://wx.qq.com/cgi-bin/mmwebwx-bin/webwxstatusnotify
POST的內(nèi)容是JSON串,格式:
{
BaseRequest: { Uin: xxx, Sid: xxx, Skey: xxx, DeviceID: xxx },
Code: 3,
FromUserName: 自己ID,
ToUserName: 自己ID,
ClientMsgId: 時(shí)間戳
}
8、心跳包,與服務(wù)器同步并獲取狀態(tài)
以上步驟完成以后,就可以進(jìn)入收發(fā)微信的循環(huán)了,可以用線程方式發(fā)送心跳包。
使用get方法,設(shè)置超時(shí)為60秒,訪問(wèn):https://webpush.wx2.qq.com/cgi-bin/mmwebwx-bin/synccheck?sid=XXXXXX&uin=XXXXXX&synckey=XXXXXX&r=時(shí)間戳&skey=XXXXXX&deviceid=XXXXXX&_=時(shí)間戳
其他幾個(gè)參數(shù)不用解釋?zhuān)@里的synckey需要說(shuō)一下,前面的步驟獲取的json串中有多個(gè)key信息,需要把這些信息拼起來(lái),key_val,中間用|分割,類(lèi)似這樣:
1_652651920|2_652651939|3_652651904|1000_0
服務(wù)器返回:window.synccheck={retcode:”0”,selector:”0”}
retcode為0表示成功,selector為2和6表示有新信息。4表示公眾號(hào)新信息。
9、讀取新信息
檢測(cè)到有信息以后,用POST方法,訪問(wèn):https://wx2.qq.com/cgi-bin/mmwebwx-bin/webwxsync?sid=XXXXXX&skey=XXXXXX
POST的內(nèi)容:
{'BaseRequest' : {'DeviceID':'XXXXXX,'Sid':'XXXXXX', 'Skey':'XXXXXX', 'Uin':'XXXXXX'},'SyncKey' : {'Count':4,'List':[{'Key':1,'Val':652653204},{'Key':2,'Val':652653674},{'Key':3,'Val':652653544},{'Key':1000,'Val':0}]},'rr' :時(shí)間戳}
注意這里的SyncKey格式,參考前面的說(shuō)明。
請(qǐng)求成功之后服務(wù)器會(huì)返回一個(gè)JSON串,其中AddMsgCount表示有多少信息,AddMsgList中是一個(gè)數(shù)組,包含了所有新消息,里面的MsgType表示信息類(lèi)型,Content就是信息內(nèi)容。
注意again,返回的信息中,會(huì)有新的synckey,要更新這個(gè)內(nèi)容,下次獲取信息訪問(wèn)要用這個(gè)新的key。
10、發(fā)送信息
這個(gè)比較簡(jiǎn)單,用POST方法,訪問(wèn):https://wx2.qq.com/cgi-bin/mmwebwx-bin/webwxsendmsg
POST的還是json格式,類(lèi)似這樣:
{'Msg':{'Type':1,'Content':'測(cè)試信息','FromUserName':'XXXXXX','ToUserName':'XXXXXX','LocalID':'時(shí)間戳','ClientMsgId':'時(shí)間戳'},'BaseRequest':{'Uin':'XXXXXX','Sid':'XXXXXX','Skey':'XXXXXX','DeviceID':'XXXXXX'}}
這里的Content是信息內(nèi)容,LocalID和ClientMsgId都用當(dāng)前時(shí)間戳。
以上就是基本的微信收發(fā)流程了。參考這個(gè),可以自己去開(kāi)發(fā)其他相關(guān)內(nèi)容
聯(lián)系客服