在圖3-21中,微信公眾平台在配置服務器時,提供了3種加解密模式供開發者選擇,即「明文模式」、「兼容模式」、「安全模式(推薦)」。選擇「兼容模式」和「安全模式(推薦)」前,需在開發者中心填寫AES對稱加密算法的消息加解密密鑰EncodingAESKey。公眾號用此密鑰對收到的密文消息體進行解密,回復消息體也用此密鑰加密。
·明文模式:維持現有模式,沒有適配加解密新特性,消息體明文收發,默認設置為明文模式。
·兼容模式:公眾平台發送消息內容將同時包括明文和密文,消息包長度增加到原來的3倍左右;公眾號回復明文或密文均可,不影響現有消息收發;開發者可在此模式下進行調試。
·安全模式(推薦):公眾平台發送消息體的內容只含有密文,公眾號回復的消息體也為密文,建議開發者在調試成功後使用此模式收發消息。
消息體加解密的實現過程如下。
假設本次的開發配置中URL為
http:// www.fangbei.org/index.php
接口程序中需要配置以下3個參數。
/* 方倍工作室 http:// www.cnblogs.com/txw1958/ CopyRight 2014 All Rights Reserved */ define("TOKEN", "weixin"); define("AppID", "wxbad0b45542aa0b5e"); define("EncodingAESKey", "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG"); require_once('wxBizMsgCrypt.php');
當用戶向公眾號發送消息時,微信公眾號將會在URL中帶上signature、timestamp、nonce、encrypt_type、msg_signature等參數,類似如下。
http:// www.fangbei.org/index.php?signature=35703636de2f9df2a77a662b68e521ce17c34db4×tamp=1414243737&nonce=1792106704&encrypt_type=aes&msg_signature=61479843 31daf7a1a9eed6e0ec3ba69055256154
同時向該接口推送如下XML消息,即一個已加密的消息。
<xml> <ToUserName><![CDATA[gh_680bdefc8c5d]]></ToUserName> <Encrypt><![CDATA[MNn4+jJ/VsFh2gUyKAaOJArwEVYCvVmyN0iXzNarP3O6vXzK62ft1/KG2/ XPZ4y5bPWU/jfIfQxODRQ7sLkUsrDRqsWimuhIT8Eq+w4E/28m+XDAQKEOjWTQIOp1p6kNsIV1Dd C3B+AtcKcKSNAeJDr7x7GHLx5DZYK09qQsYDOjP6R5NqebFjKt/NpEl/GU3gWFwG8LCtRNuIYdK5 axbFSfmXbh5CZ6Bk5wSwj5fu5aS90cMAgUhGsxrxZTY562QR6c+3ydXxb+GHI5w+qA+eqJjrQqR7 u5hS+1x5sEsA7vS+bZ5LYAR3+PZ243avQkGllQ+rg7a6TeSGDxxhvLw+mxxinyk88BNHkJnyK// hM 1k9PuvuLAASdaud4vzRQlAmnYOslZl8CN7gjCjV41skUTZv3wwGPxvEqtm/nf5fQ=]]></Encrypt> </xml>
這時程序需要從URL中獲得以下參數。這些參數將用於加解密過程。
$timestamp = $_GET['timestamp']; $nonce = $_GET["nonce"]; $msg_signature = $_GET['msg_signature']; $encrypt_type = $_GET['encrypt_type'];
接口程序收到消息後,先進行解密,解密部分代碼如下。
$postStr = $GLOBALS["HTTP_RAW_POST_DATA"]; if ($encrypt_type == 'aes'){ $pc = new WXBizMsgCrypt(TOKEN, EncodingAESKey, AppID); $this->logger(" D \r\n".$postStr); $decryptMsg = ""; // 解密後的明文 $errCode = $pc->DecryptMsg($msg_signature, $timestamp, $nonce, $postStr, $decryptMsg); $postStr = $decryptMsg; }
解密完成後,把解密內容又返回給$postStr,這是為了將消息中解密後的內容和明文模式時的消息統一,方便後續處理。解密後的XML如下。
<xml> <ToUserName><![CDATA[gh_680bdefc8c5d]]></ToUserName> <FromUserName><![CDATA[oIDrpjpQ8j8mBuQ8nM26HWzNEZgg]]></FromUserName> <CreateTime>1414243737</CreateTime> <MsgType><![CDATA[text]]></MsgType> <Content><![CDATA[?]]></Content> <MsgId>6074130599188426998</MsgId> </xml>
對消息在自己的原有的代碼流程中處理,完成之後,一個要回復的文本消息如下。
<xml> <ToUserName><![CDATA[oIDrpjpQ8j8mBuQ8nM26HWzNEZgg]]></ToUserName> <FromUserName><![CDATA[gh_680bdefc8c5d]]></FromUserName> <CreateTime>1414243733</CreateTime> <MsgType><![CDATA[text]]></MsgType> <Content><![CDATA[2014-10-25 21:28:53 技術支持 方倍工作室 http:// www.fangbei.org/]]></Content> </xml>
把上述消息加密,返回給微信公眾號,加密過程如下。
// 加密 if ($encrypt_type == 'aes'){ $encryptMsg = ''; // 加密後的密文 $errCode = $pc->encryptMsg($result, $timeStamp, $nonce, $encryptMsg); $result = $encryptMsg; $this->logger(" E \r\n".$result); }
加密後的內容如下。
<xml> <Encrypt><![CDATA[pE6gp6qvVBMHwCXwnM7illFBrh9LmvlKFlPUDuyQo9EKNunqbUFMd2Kj iYoz+3K1B+93JbMWHt+19TI8awdRdyopRS4oUNg5M2jwpwXTmc6TtafkKNjvqlvPXIWmutw0tuMXke 1hDgsqz0SC8h/QjNLxECuwnczrfCMJlt+APHnX2yMMaq/aYUNcndOH387loQvl2suCGucXpglnbx f7frTCz9NQVgKiYrvKOhk6KFiVMnzuxy6WWmoe3GBiUCPTtYf5b1CxzN2IHViEBm28ilV9wWdNOM 9TPG7BSSAcpgY4pcwdIG5+4KhgYmnVU3bc/ZJkk42TIdidigOfFpJwET4UWVrLB/ldUud4aPexp 3aPCR3Fe53S2HHcl3tTxh4iRvDftUKP3svYPctt1MlYuYv/BZ4JyzUQV03H+0XrVyDY2tyVjimgC rA2c1mZMgHttOHTQ6VTnxrMq0GWlRlH0KPQKqtjUpNQzuOH4upQ8boPsEtuY3wDA2RaXQPJrX on]]></Encrypt> <MsgSignature><![CDATA[6c46904dc1f58b2ddf2dd0399f1c6cf41f33ecb9]]></MsgSignature> <TimeStamp>1414243733</TimeStamp> <Nonce><![CDATA[1792106704]]></Nonce> </xml>
這樣一個安全模式下的加解密消息就完成了。
完整的代碼如下。
1 <?php 2 /* 3 方倍工作室 http:// www.cnblogs.com/txw1958/ 4 CopyRight 2014 All Rights Reserved 5 */ 6 define("TOKEN", "weixin"); 7 define("AppID", "wxbad0b45542aa0b5e"); 8 define("EncodingAESKey", "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG"); 9 require_once('wxBizMsgCrypt.php'); 10 11 $wechatObj = new wechatCallbackapiTest; 12 if (!isset($_GET['echostr'])) { 13 $wechatObj->responseMsg; 14 }else{ 15 $wechatObj->valid; 16 } 17 18 class wechatCallbackapiTest 19 { 20 // 驗證簽名 21 public function valid 22 { 23 $echoStr = $_GET["echostr"]; 24 $signature = $_GET["signature"]; 25 $timestamp = $_GET["timestamp"]; 26 $nonce = $_GET["nonce"]; 27 $tmpArr = array(TOKEN, $timestamp, $nonce); 28 sort($tmpArr); 29 $tmpStr = implode($tmpArr); 30 $tmpStr = sha1($tmpStr); 31 if($tmpStr == $signature){ 32 echo $echoStr; 33 exit; 34 } 35 } 36 37 // 響應消息 38 public function responseMsg 39 { 40 $timestamp = $_GET['timestamp']; 41 $nonce = $_GET["nonce"]; 42 $msg_signature = $_GET['msg_signature']; 43 $encrypt_type = (isset($_GET['encrypt_type']) && ($_GET['encrypt_type'] == 'aes')) ? "aes" : "raw"; 44 45 $postStr = $GLOBALS["HTTP_RAW_POST_DATA"]; 46 if (!empty($postStr)){ 47 // 解密 48 if ($encrypt_type == 'aes'){ 49 $pc = new WXBizMsgCrypt(TOKEN, EncodingAESKey, AppID); 50 $this->logger(" D \r\n".$postStr); 51 $decryptMsg = ""; // 解密後的明文 52 $errCode = $pc->DecryptMsg($msg_signature, $timestamp, $nonce, $postStr, $decryptMsg); 53 $postStr = $decryptMsg; 54 } 55 $this->logger(" R \r\n".$postStr); 56 $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_ NOCDATA); 57 $RX_TYPE = trim($postObj->MsgType); 58 59 // 消息類型分離 60 switch ($RX_TYPE) 61 { 62 case "event": 63 $result = $this->receiveEvent($postObj); 64 break; 65 case "text": 66 $result = $this->receiveText($postObj); 67 break; 68 } 69 $this->logger(" R \r\n".$result); 70 // 加密 71 if ($encrypt_type == 'aes'){ 72 $encryptMsg = ''; // 加密後的密文 73 $errCode = $pc->encryptMsg($result, $timeStamp, $nonce, $encry ptMsg); 74 $result = $encryptMsg; 75 $this->logger(" E \r\n".$result); 76 } 77 echo $result; 78 }else { 79 echo ""; 80 exit; 81 } 82 } 83 84 // 接收事件消息 85 private function receiveEvent($object) 86 { 87 $content = ""; 88 switch ($object->Event) 89 { 90 case "subscribe": 91 $content = "歡迎關注方倍工作室 "; 92 break; 93 } 94 95 $result = $this->transmitText($object, $content); 96 return $result; 97 } 98 99 // 接收文本消息 100 private function receiveText($object) 101 { 102 $keyword = trim($object->Content); 103 if (strstr($keyword, "文本")){ 104 $content = "這是個文本消息"; 105 }else if (strstr($keyword, "單圖文")){ 106 $content = array; 107 $content = array("Title"=>"單圖文標題", "Description"=>"單圖文內 容", "PicUrl"=>"http:// discuz.comli.com/weixin/weather/icon/cartoon.jpg", "Url" =>"http:// m.cnblogs.com/?u=txw1958"); 108 }else if (strstr($keyword, "圖文") || strstr($keyword, "多圖文")){ 109 $content = array; 110 $content = array("Title"=>"多圖文1標題", "Description"=>"", "Pic Url"=>"http:// discuz.comli.com/weixin/weather/icon/cartoon.jpg", "Url" => "http:// m.cnblogs.com/?u=txw1958"); 111 $content = array("Title"=>"多圖文2標題", "Description"=>"", "Pic Url"=>"http:// d.hiphotos.bdimg.com/wisegame/pic/item/f3529822720e0cf3ac 9f1ada0846f21fbe09aaa3.jpg", "Url" =>"http:// m.cnblogs.com/?u=txw1958"); 112 $content = array("Title"=>"多圖文3標題", "Description"=>"", "Pic Url"=>"http:// g.hiphotos.bdimg.com/wisegame/pic/item/18cb0a46f21fbe090d33 8acc6a600c338644adfd.jpg", "Url" =>"http:// m.cnblogs.com/?u=txw1958"); 113 }else if (strstr($keyword, "音樂")){ 114 $content = array; 115 $content = array("Title"=>"最炫民族風", "Description"=>"歌手:鳳凰傳 奇", "MusicUrl"=>"http:// 121.199.4.61/music/zxmzf.mp3", "HQMusicUrl"=> "http:// 121.199.4.61/music/zxmzf.mp3"); 116 }else{ 117 $content = date("Y-m-d H:i:s",time)."\n".$object->FromUserName. "\n技術支持 方倍工作室"; 118 } 119 120 if(is_array($content)){ 121 if (isset($content[0])){ 122 $result = $this->transmitNews($object, $content); 123 }else if (isset($content['MusicUrl'])){ 124 $result = $this->transmitMusic($object, $content); 125 } 126 }else{ 127 $result = $this->transmitText($object, $content); 128 } 129 return $result; 130 } 131 132 // 回覆文本消息 133 private function transmitText($object, $content) 134 { 135 $xmlTpl = "<xml> 136 <ToUserName><![CDATA[%s]]></ToUserName> 137 <FromUserName><![CDATA[%s]]></FromUserName> 138 <CreateTime>%s</CreateTime> 139 <MsgType><![CDATA[text]]></MsgType> 140 <Content><![CDATA[%s]]></Content> 141 </xml>"; 142 $result = sprintf($xmlTpl, $object->FromUserName, $object->ToUserName, time, $content); 143 return $result; 144 } 145 146 // 回復圖文消息 147 private function transmitNews($object, $newsArray) 148 { 149 if(!is_array($newsArray)){ 150 return; 151 } 152 $itemTpl = "<item> 153 <Title><![CDATA[%s]]></Title> 154 <Description><![CDATA[%s]]></Description> 155 <PicUrl><![CDATA[%s]]></PicUrl> 156 <Url><![CDATA[%s]]></Url> 157 </item>"; 158 159 $item_str = ""; 160 foreach ($newsArray as $item){ 161 $item_str .= sprintf($itemTpl, $item['Title'], $item['Description'], $item['PicUrl'], $item['Url']); 162 } 163 $xmlTpl = "<xml> 164 <ToUserName><![CDATA[%s]]></ToUserName> 165 <FromUserName><![CDATA[%s]]></FromUserName> 166 <CreateTime>%s</CreateTime> 167 <MsgType><![CDATA[news]]></MsgType> 168 <ArticleCount>%s</ArticleCount> 169 <Articles> 170 $item_str </Articles> 171 </xml>"; 172 173 $result = sprintf($xmlTpl, $object->FromUserName, $object->ToUserName, time, count($newsArray)); 174 return $result; 175 } 176 177 // 回復音樂消息 178 private function transmitMusic($object, $musicArray) 179 { 180 $itemTpl = "<Music> 181 <Title><![CDATA[%s]]></Title> 182 <Description><![CDATA[%s]]></Description> 183 <MusicUrl><![CDATA[%s]]></MusicUrl> 184 <HQMusicUrl><![CDATA[%s]]></HQMusicUrl> 185 </Music>"; 186 187 $item_str = sprintf($itemTpl, $musicArray['Title'], $musicArray['Des cription'], $musicArray['MusicUrl'], $musicArray['HQMusicUrl']); 188 189 $xmlTpl = "<xml> 190 <ToUserName><![CDATA[%s]]></ToUserName> 191 <FromUserName><![CDATA[%s]]></FromUserName> 192 <CreateTime>%s</CreateTime> 193 <MsgType><![CDATA[music]]></MsgType> 194 $item_str 195 </xml>"; 196 197 $result = sprintf($xmlTpl, $object->FromUserName, $object->ToUserName, time); 198 return $result; 199 } 200 201 // 日誌記錄 202 public function logger($log_content) 203 { 204 if(isset($_SERVER['HTTP_APPNAME'])){ // SAE 205 sae_set_display_errors(false); 206 sae_debug($log_content); 207 sae_set_display_errors(true); 208 }else if($_SERVER['REMOTE_ADDR'] != "127.0.0.1"){ // LOCAL 209 $max_size = 500000; 210 $log_filename = "log.xml"; 211 if(file_exists($log_filename) and (abs(filesize($log_filename)) > $max_ size)){unlink($log_filename);} 212 file_put_contents($log_filename, date('Y-m-d H:i:s').$log_content."\r \n", FILE_APPEND); 213 } 214 } 215 } 216 ?>