相信很多人會(huì)跟我一樣,token驗(yàn)證之后,發(fā)送消息給訂閱號(hào),沒(méi)有消息返回。
以下,說(shuō)一下我辛苦調(diào)試得到的解決辦法:
首先,token驗(yàn)證:
自己寫的token一直驗(yàn)證失敗,找了好久,沒(méi)有發(fā)現(xiàn)bug。實(shí)在沒(méi)辦法,就用了官方的示例代碼。并且通過(guò)示例代碼調(diào)試,發(fā)現(xiàn)了一個(gè)讓我吐血的bug(也不算bug):
token驗(yàn)證貌似要求字符編碼格式!?。?!
官方的示例代碼,直接上傳到服務(wù)器,token直接過(guò)!
把官方示例代碼改為UTF-8格式,再上傳覆蓋,token失??!失敗!失??!
后來(lái),把自己寫的修改為ANSI格式還是token失??!醉了醉了!那只好用官方示例代碼。在此,說(shuō)下,token是一次握手驗(yàn)證,驗(yàn)證過(guò)一次就不用了。
下面,言歸正傳,貌似偏題了...orz
token驗(yàn)證之后,直接用官方示例代碼,趕緊測(cè)試自己的訂閱號(hào),結(jié)果....發(fā)出去的消息就跟潑出去的水一樣,什么鬼都沒(méi)有返回...orz
又各種找bug,各種群?jiǎn)枺鞣N搜索....歷經(jīng)本博主九九八十一的努力,終于找出了問(wèn)題所在(這里是指我自己開(kāi)發(fā)的,并不包括全部,如果你有不同的bug,歡迎交流):
1、最容易被忽視的一個(gè)bug,官方給的示例代碼,壓根就沒(méi)調(diào)用寫好的那個(gè)responseMsg()函數(shù)!
2、把之前的token代碼注釋,也就是$wechatObj->valid();這行代碼。因?yàn)閠oke驗(yàn)證那段代碼會(huì)有一個(gè)echo $echostr,會(huì)把responseMsg()函數(shù)里的echo $resultStr;(56行)xml格式混亂,輸回給微信服務(wù)器就無(wú)法識(shí)別了(貌似只能識(shí)別xml格式,還有json格式)。(token驗(yàn)證是一次握手驗(yàn)證,驗(yàn)證開(kāi)發(fā)者之后,就可以不用了,趕緊讓它消失在我們整潔的代碼orz...)
3、最惡心的一個(gè)bug,還是字符編碼問(wèn)題!orz...xml要求UTF-8編碼,所以,把示例代碼改回UTF-8編碼!這個(gè)bug找的讓我崩潰!??!
下面是我修改后的代碼,能正常運(yùn)行,無(wú)bug,需要的可以參考一下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
|
<?php /** * wechat php test */ //define your token define( "TOKEN" , "codcodog" ); $wechatObj = new wechatCallbackapiTest(); //$wechatObj->valid(); $wechatObj ->responseMsg(); class wechatCallbackapiTest { public function valid() { $echoStr = $_GET [ "echostr" ]; //valid signature , option if ( $this ->checkSignature()){ header( 'content-type:text' ); echo $echoStr ; exit ; } } public function responseMsg() { //get post data, May be due to the different environments $postStr = $GLOBALS [ "HTTP_RAW_POST_DATA" ]; //$postStr = file_get_contents("php://input"); file_put_contents ( "log.txt" , $postStr ,FILE_APPEND ); //extract post data if (! empty ( $postStr )){ /* libxml_disable_entity_loader is to prevent XML eXternal Entity Injection, the best way is to check the validity of xml by yourself */ libxml_disable_entity_loader(true); $postObj = simplexml_load_string( $postStr , 'SimpleXMLElement' , LIBXML_NOCDATA); $fromUsername = $postObj ->FromUserName; //用戶 $toUsername = $postObj ->ToUserName; //公眾平臺(tái) $keyword = trim( $postObj ->Content); $time = time(); $textTpl = "<xml> <ToUserName><![CDATA[%s]]></ToUserName> <FromUserName><![CDATA[%s]]></FromUserName> <CreateTime>%s</CreateTime> <MsgType><![CDATA[%s]]></MsgType> <Content><![CDATA[%s]]></Content> <FuncFlag></FuncFlag> </xml>"; if (! empty ( $keyword )) { $msgType = "text" ; $contentStr = "Welcome to wechat world!" ; $resultStr = sprintf( $textTpl , $fromUsername , $toUsername , $time , $msgType , $contentStr ); echo $resultStr ; } else { echo "Input something..." ; } } else { echo "" ; exit ; } } private function checkSignature() { // you must define TOKEN by yourself if (!defined( "TOKEN" )) { throw new Exception( 'TOKEN is not defined!' ); } $signature = $_GET [ "signature" ]; $timestamp = $_GET [ "timestamp" ]; $nonce = $_GET [ "nonce" ]; $token = TOKEN; $tmpArr = array ( $token , $timestamp , $nonce ); // use SORT_STRING rule sort( $tmpArr , SORT_STRING); $tmpStr = implode( $tmpArr ); $tmpStr = sha( $tmpStr ); if ( $tmpStr == $signature ){ return true; } else { return false; } } } ?> |
以上所述是小編給大家分享的php微信訂閱號(hào)開(kāi)發(fā)之token驗(yàn)證后自動(dòng)發(fā)送消息給訂閱號(hào)但是沒(méi)有消息返回的解決方案,希望大家喜歡。