微信公众平台消息体加解密实现

一、消息体加解密

微信公众平台在配置服务器时,提供了3种加解密的模式供开发者选择,即明文模式、兼容模式、安全模式,选择兼容模式和安全模式前,需在开发者中心填写消息加解密密钥EncodingAESKey。

  • 明文模式:维持现有模式,没有适配加解密新特性,消息体明文收发,默认设置为明文模式
  • 兼容模式:公众平台发送消息内容将同时包括明文和密文,消息包长度增加到原来的3倍左右;公众号回复明文或密文均可,不影响现有消息收发;开发者可在此模式下进行调试
  • 安全模式(推荐):公众平台发送消息体的内容只含有密文,公众账号回复的消息体也为密文,建议开发者在调试成功后使用此模式收发消息

什么是EncodingAESKey?

  • 微信公众平台采用AES对称加密算法对推送给公众帐号的消息体对行加密,EncodingAESKey则是加密所用的秘钥。公众帐号用此秘钥对收到的密文消息体进行解密,回复消息体也用此秘钥加密。

  加解密的详细技术方案可以参考官方文档 http://mp.weixin.qq.com/wiki/index.php?title=%E6%8A%80%E6%9C%AF%E6%96%B9%E6%A1%88

适用公众账号类型

  • 已认证订阅号
  • 服务号
  • 企业号

不能用于未认证订阅号,因为其没有appid参数

二、开发实现及数据分析

1. 配置

假设本次的开发配置中URL为

http://www.fangbei.org/index.php

接口程序中需要配置以下三项参数

/*

    方倍工作室 http://www.cnblogs.com/txw1958/
    CopyRight 2014 All Rights Reserved
*/
define("TOKEN", "weixin");
define("AppID", "wxbad0b45542aa0b5e");
define("EncodingAESKey", "abcdefghijklmnopqrstuvwxyz0123456789ABCDEFG");
require_once(‘wxBizMsgCrypt.php‘);

2. 加解密实现

当用户向公众账号发送消息时,微信公众账号将会在URL中带上signature、timestamp、nonce、encrypt_type、msg_signature等参数,如下所示

http://www.fangbei.org/aes/index.php?signature=35703636de2f9df2a77a662b68e521ce17c34db4&timestamp=1414243737&nonce=1792106704&encrypt_type=aes&msg_signature=6147984331daf7a1a9eed6e0ec3ba69055256154

同时向该接口推送如下XML消息 ,即一个已加密的消息

<xml>
    <ToUserName><![CDATA[gh_680bdefc8c5d]]></ToUserName>
    <Encrypt><![CDATA[MNn4+jJ/VsFh2gUyKAaOJArwEVYCvVmyN0iXzNarP3O6vXzK62ft1/KG2/XPZ4y5bPWU/jfIfQxODRQ7sLkUsrDRqsWimuhIT8Eq+w4E/28m+XDAQKEOjWTQIOp1p6kNsIV1DdC3B+AtcKcKSNAeJDr7x7GHLx5DZYK09qQsYDOjP6R5NqebFjKt/NpEl/GU3gWFwG8LCtRNuIYdK5axbFSfmXbh5CZ6Bk5wSwj5fu5aS90cMAgUhGsxrxZTY562QR6c+3ydXxb+GHI5w+qA+eqJjrQqR7u5hS+1x5sEsA7vS+bZ5LYAR3+PZ243avQkGllQ+rg7a6TeSGDxxhvLw+mxxinyk88BNHkJnyK//hM1k9PuvuLAASdaud4vzRQlAmnYOslZl8CN7gjCjV41skUTZv3wwGPxvEqtm/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[pE6gp6qvVBMHwCXwnM7illFBrh9LmvlKFlPUDuyQo9EKNunqbUFMd2KjiYoz+3K1B+93JbMWHt+19TI8awdRdyopRS4oUNg5M2jwpwXTmc6TtafkKNjvqlvPXIWmutw0tuMXke1hDgsqz0SC8h/QjNLxECuwnczrfCMJlt+APHnX2yMMaq/aYUNcndOH387loQvl2suCGucXpglnbxf7frTCz9NQVgKiYrvKOhk6KFiVMnzuxy6WWmoe3GBiUCPTtYf5b1CxzN2IHViEBm28ilV9wWdNOM9TPG7BSSAcpgY4pcwdIG5+4KhgYmnVU3bc/ZJkk42TIdidigOfFpJwET4UWVrLB/ldUud4aPexp3aPCR3Fe53S2HHcl3tTxh4iRvDftUKP3svYPctt1MlYuYv/BZ4JyzUQV03H+0XrVyDY2tyVjimgCrA2c1mZMgHttOHTQ6VTnxrMq0GWlRlH0KPQKqtjUpNQzuOH4upQ8boPsEtuY3wDA2RaXQPJrXon]]></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, $encryptMsg);
 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"=>"", "PicUrl"=>"http://discuz.comli.com/weixin/weather/icon/cartoon.jpg", "Url" =>"http://m.cnblogs.com/?u=txw1958");
111             $content[] = array("Title"=>"多图文2标题", "Description"=>"", "PicUrl"=>"http://d.hiphotos.bdimg.com/wisegame/pic/item/f3529822720e0cf3ac9f1ada0846f21fbe09aaa3.jpg", "Url" =>"http://m.cnblogs.com/?u=txw1958");
112             $content[] = array("Title"=>"多图文3标题", "Description"=>"", "PicUrl"=>"http://g.hiphotos.bdimg.com/wisegame/pic/item/18cb0a46f21fbe090d338acc6a600c338644adfd.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[‘Description‘], $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 ?>
时间: 2024-10-03 13:10:04

微信公众平台消息体加解密实现的相关文章

微信公众平台消息体签名及加解密实例(Java)

前言: 最近在研究微信公众平台的开发,玩得不亦乐乎.基本的回复功能已经实现了,而且回复用到了图灵机器人的接口.其实图灵机器人已经有微信接口可以直接调用.如果项目的需要,想要做个性化需求的话,用这种方式是行不通的.我现在的解决方案是,我开发的应用A作为中间层,连接微信接口与图灵机器人接口.有点扯远了,如题,微信官方为了更高的安全性,10月份推出了消息体签名验证.网上关于此项的实例不多,其实根据官方的例子,重新封装一下,就可以了. 一.准备工作: 1.已申请了相关的订阅号或者服务号(可以用虚拟器,具

微信公众平台消息接口开发(32)空气质量指数查询

原文:微信公众平台消息接口开发(32)空气质量指数查询 微信公众平台开发 微信公众平台开发者 微信公众平台开发模式 空气质量指数 PM2.5 作者:方倍工作室 原文:http://www.cnblogs.com/txw1958/archive/2013/05/30/weixin-if32-air-quality.html 一.获取原版数据 在中国环境监测总站http://www.cnemc.cn/ 可以找到全国城市空气质量实时发布平台,其地址为 http://113.108.142.147:20

爬虫-微信公众平台消息获取

帮朋友抓取微信公众平台的用户评论信息. 下面只说核心的部分,怎么获取评论信息. 查看HTML代码,没有发现关于评论部分的标签.看来是用JS动态生成的,但是查找ajax请求也没有找到哪里有返回数据. 最后搜索一下,原来是在这里,很直白的写在了JS里: <script type="text/javascript"> wx.cgiData = { total_count : 91, latest_msg_id : '200325222', count : "20&quo

微信公众平台消息接口开发-封装weixin.class.php(转)

一.封装weixin.class.php 由于微信公众平台的通信使用的是特定格式的XML数据,每次接受和回复都要去做一大堆的数据处理. 我们就考虑在这个基础上做一次封装,weixin.class.php,代码如下: <?php class Weixin {     public $token = '';//token     public $debug =  false;//是否debug的状态标示,方便我们在调试的时候记录一些中间数据     public $setFlag = false;

微信公众平台消息接口开发-封装weixin.class.php

原文:微信公众平台消息接口开发-封装weixin.class.php 一.封装weixin.class.php 由于微信公众平台的通信使用的是特定格式的XML数据,每次接受和回复都要去做一大堆的数据处理. 我们就考虑在这个基础上做一次封装,weixin.class.php,代码如下: <?php class Weixin {     public $token = '';//token     public $debug =  false;//是否debug的状态标示,方便我们在调试的时候记录一

微信公众平台消息接口PHP版开发教程

原文:微信公众平台消息接口PHP版开发教程  一.写好接口程序 在你的服务器上上传好一个接口程序文件,如http://www.yourdomain.com/weixin.php  内容如下: <?php define("TOKEN", "weixin");//自己定义的token 就是个通信的私钥 $wechatObj = new wechatCallbackapiTest(); $wechatObj->valid(); //$wechatObj-&

微信公众平台消息接口开发 小黄鸡(小贱鸡)机器人 微信公众平台 公众号聊天机器人 ,消息,接口,小黄鸡,小贱鸡,机器人

第一部分 基于模拟请求的方式 一.模拟请求数据 先看一下小黄鸡的网页版界面 我们通过模拟http请求来实现,上面对话抓包如下: 发送消息的包 接收消息的包: 根据上面的包,模拟发起请求如下: 二.与微信对接 小黄鸡还可以使用API方式调用,但免费时间有限,代码和上面基本一样,就不多写了. 三.演示效果 第二部分 基于接口的方式 一.申请小黄鸡接口SimSimi,发音为〝shim-shimee〞,中文翻译:小黄鸡,由韩文simsim(??)演变而成. “??”原意为“无聊”.SimSimi 由IS

微信公众平台消息接口开发之微信浏览器HTTP_USER_AGENT判断

在微信公众平台的开发过程中,我们有时需要开发网页并判断是否是是来自微信浏览器访问,本文介绍如何做出这一判断. 一.$_SERVER数组 $_SERVER 是一个包含了诸如头信息(header).路径(path).以及脚本位置(script locations)等等信息的数组.这个数组中的项目由 Web 服务器创建.不能保证每个服务器都提供全部项目:服务器可能会忽略一些. 二.获取HTTP_USER_AGENT 以下方法可获得 <?php echo $_SERVER["HTTP_USER_A

微信公众平台消息接口开发 彩票查询

一.获取数据 目前很多网站都提供彩票信息查询,所以取得彩票数据是件很容易的事.方倍工作室开发出彩票查询接口 API,目前已开通'双色球','3D','七乐彩','大乐透','七星彩','排列3','排列5','胜负彩','六场半全场','四场进球' 10种数据,每日同步更新 使用方式为直接在URL中提交彩票名称即可,名称需要先做urlencode调用url方法:以下是调用双色球方法 http://api2.sinaapp.com/search/lottery/?appkey=0020130430