《微信公众平台开发:从零基础到ThinkPHP5高性能框架实践》——3.3 微信开发者中心

简介:

本节书摘来自华章计算机《微信公众平台开发:从零基础到ThinkPHP5高性能框架实践》一书中的第3章,第3.3节,作者 方倍工作室,更多章节内容可以访问云栖社区“华章计算机”公众号查看。

3.3 微信开发者中心

3.3.1 配置和启用服务器

登录微信公众平台后台(微信公众平台地址为https://mp.weixin.qq.com),在左侧列表的最下方找到“基本配置”,如图3-19所示。

单击进入配置页面,可以看到当前有服务器配置信息,状态为未启用,如图3-20所示。
   
screenshot

单击“修改配置”按钮,进入修改页面,如图3-21所示。


screenshot

其中,URL为3.1.1节中介绍的云应用的域名,即http://fbstudio.sinaapp.com ,而Token在index.php中定义为weixin,EncodingAESKey不需要填写,单击“随机生成”按钮,让系统自动生成一个即可,“消息加解密方式”选择“明文模式”,然后单击“提交”按钮,弹出确认框,如图3-22所示。

在弹出的提示框中单击“确定”按钮,相关参数填写成功,如图3-23所示。

再单击右上角的“启用”按钮,启用服务器的配置。系统弹出提示框,询问“是否确定开启服务器配置”,如图3-24所示。

单击“确定”按钮,将启用服务器配置。

如果单击按钮后,上方提示“Token验证失败”,可以重试几次,微信服务器有时不稳定也会造成这样的情况,并不是程序本身有问题。启用成功后的界面如图3-25所示。

screenshot

这样就成功配置并启用了服务器。

screenshot

3.3.2 配置失败常见问题与分析

提交URL和Token的时候,有时会碰到提交不成功的情况,具体有以下几种。

1.请求URL超时

这种情况一般是由于服务器网速或响应速度太慢。此时可以先重试几次或者等一段时间再试,如果还是这样,则需要考虑更换速度更快、性能更好的服务器。

2.系统发生错误,请稍后重试

这种情况一般是由于微信服务器短时间内的异常引起的,一般重试或者过一段时间尝试即可。

3.Token验证失败

这种情况需要具体分析验证过程被卡在哪一个环节了。此时可以通过调用变量$_SERVER来获取服务器和执行环境信息,以便进行分析。

$_SERVER是一个包含诸如头信息(Header)、路径(Path)及脚本位置(Script Location)等信息的数组。这个数组中的项目由Web服务器创建。若要了解更多关于$_SERVER的信息,可访问官方网站http://www.php.net/manual/zh/reserved.variables.server.php

这里需要使用以下两个元素。

  • $_SERVER['REMOTE_ADDR']:来访者的IP地址,此处为微信服务器的IP地址。
  • $_SERVER['QUERY_STRING']:查询请求字符串,此处为微信服务器发过来的GET请求字符串。

将以上两个变量记录到日志中。函数定义如下。

function traceHttp()
{
    $content = date('Y-m-d H:i:s')."\nREMOTE_ADDR:".$_SERVER["REMOTE_ADDR"]."\nQUERY_
    STRING:".$_SERVER["QUERY_STRING"]."\n\n";
    
    if (isset($_SERVER['HTTP_APPNAME'])){   // SAE
        sae_set_display_errors(false);
        sae_debug(trim($content));
        sae_set_display_errors(true);
    }else {
        $max_size = 100000;
        $log_filename = "log.xml";
        if(file_exists($log_filename) and (abs(filesize($log_filename)) > $max_size))
        {unlink($log_filename);}
        file_put_contents($log_filename, $content, FILE_APPEND);
    }
}

上面代码中,当环境为SAE时,使用SAE的调试函数sae_debug()将内容记录到日志中。而在具有读写权限的空间下,使用file_put_contents()函数把字符串写入文件。

然后在程序的数据处理之前调用该函数,记录信息,代码如下。

define("TOKEN", "weixin");
traceHttp();
$wechatObj = new wechatCallbackapiTest();
if (isset($_GET['echostr'])) {
    $wechatObj->valid();
}else{
    $wechatObj->responseMsg();
}

当提交URL和Token验证的时候,程序目录下应当生成一个log.xml文件。内容类似如下。

2016-10-10 11:03:21
REMOTE_ADDR:101.226.61.144
QUERY_STRING:signature=6e35c6f3d3279338781047dbffd09426b9ecdee3&echostr=59794206
53038092664&timestamp=1392001400&nonce=1392192345

下面可以得出初步结论。

没有生成日志文件:微信服务器没有访问到你的服务器,需要先检查你的服务器是否可以通过公网访问,以及URL路径是否正确并且可以访问。如果可以通过公网访问,而微信服务器不能访问,那么可能是防火墙拦截了80端口或微信服务器的IP地址,也可能是服务器所在区域与微信服务器通信不畅,需要更换服务器。

已经生成日志文件:查看REMOTE_ADDR和QUERY_STRING的内容是否与上述类似。确认signature、timestamp、nonce、echostr等4个参数都有值。如果这些都没有问题,则检查程序中定义的Token值是否与提交的一致,再检查程序流程及数据处理是否与官方文档描述的一致。

如果确定以上均没有问题,可以使用下面章节中的微信调试器进行测试,它提供了更为宽松的校验方式,并且可以实时输出当前的XML数据供调试时参考。

3.3.3 自动回复当前时间

在上面的例子中,已经嵌入了一个简单的时间查询功能,发送一个问号“?”就能回复当前的时间,如图3-26所示。


screenshot

这个功能是基于下面的代码实现的。

if($keyword == "?" || $keyword == "?")
{
    $msgType = "text";
    $content = date("Y-m-d H:i:s",time());
    $result = sprintf($textTpl, $fromUsername, $toUsername, $time, $msgType, $content);
    echo $result;
}

上述代码在收到消息后,判断消息内容是否为问号(包括英文输入状态下的问号和中文输入状态下的问号),如果包含,则将当前时间(包括年月日时分秒)作为回复内容,构造成一个消息回复给用户。这样公众号就实现了当前时间的自动回复。

3.3.4 消息交互原理分析

下面结合3.3.3节的代码来分析微信公众平台的消息交互原理。下面的代码基于微信公众平台官方示例代码修改完善而成。

 1 <?php
 2 /*
 3     方倍工作室 http:// www.fangbei.org/
 4     CopyRight 2013 www.doucube.com  All Rights Reserved
 5 */
 6 
 7 define("TOKEN", "weixin");
 8 $wechatObj = new wechatCallbackapiTest();
 9 if (isset($_GET['echostr'])) {
10     $wechatObj->valid();
11 }else{
12     $wechatObj->responseMsg();
13 }
14 
15 class wechatCallbackapiTest
16 {
17     public function valid()
18     {
19         $echoStr = $_GET["echostr"];
20         if($this->checkSignature()){
21             echo $echoStr;
22             exit;
23         }
24     }
25 
26     private function checkSignature()
27     {
28         $signature = $_GET["signature"];
29         $timestamp = $_GET["timestamp"];
30         $nonce = $_GET["nonce"];
31 
32         $token = TOKEN;
33         $tmpArr = array($token, $timestamp, $nonce);
34         sort($tmpArr);
35         $tmpStr = implode( $tmpArr );
36         $tmpStr = sha1( $tmpStr );
37 
38         if( $tmpStr == $signature ){
39             return true;
40         }else{
41             return false;
42         }
43     }
44 
45     public function responseMsg()
46     {
47         $postStr = $GLOBALS["HTTP_RAW_POST_DATA"];
48 
49         if (!empty($postStr)){
50             $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_
               NOCDATA);
51             $fromUsername = $postObj->FromUserName;
52             $toUsername = $postObj->ToUserName;
53             $keyword = trim($postObj->Content);
54             $time = time();
55             $textTpl = "<xml>
56                         <ToUserName><![CDATA[%s]]></ToUserName>
57                         <FromUserName><![CDATA[%s]]></FromUserName>
58                         <CreateTime>%s</CreateTime>
59                         <MsgType><![CDATA[%s]]></MsgType>
60                         <Content><![CDATA[%s]]></Content>
61                         <FuncFlag>0</FuncFlag>
62                         </xml>";
63             if($keyword == "?" || $keyword == "?")
64             {
65                 $msgType = "text";
66                 $content = date("Y-m-d H:i:s",time());
67                 $result = sprintf($textTpl, $fromUsername, $toUsername, $time, 
                   $msgType, $content);
68                 echo $result;
69             }
70         }else{
71             echo "";
72             exit;
73         }
74     }
75 }
76 ?>

首先看一下代码的结构。

第2~5行是注释部分。

第7行使用define()函数定义常量,常量名称为TOKEN,常量的值为weixin,这个值就是在启用开发模式时填写的Token。

第15~75行定义了一个类wechatCallbackapiTest,并在类中定义了3个方法valid()、checkSignature()和responseMsg()。

第8~13行为程序执行语句。第8行实例化了一个类对象。在第9行中,判断是否有GET请求有echostr变量,如果有,则执行valid()方法,否则执行responseMsg()方法。

下面分析微信消息交互流程。

提交URL和Token申请验证的时候,微信服务器将发送GET请求到填写的URL上,并且带上4个参数(signature、timestamp、nonce、echostr)。GET请求类似如下。

signature=6e35c6f3d3279338781047dbffd09426b9ecdee3&echostr=5979420653038092664&t
imestamp=1392001400&nonce=1392192345

上述请求的参数说明如表3-1所示。

screenshot

这个GET请求是包含echostr变量的,所以执行valid()方法。在该方法中,又调用了校验签名方法checkSignature()。如果签名校验为真,则原样输出变量$echoStr的值。

加密/校验流程如下。

1)将token、timestamp、nonce等3个参数进行字典序排序,见第33~34行。
2)将3个参数字符串拼接成一个字符串进行sha1加密,见第35~36行。
3)开发者获得加密后的字符串可与signature对比,标识该请求来源于微信,见第38~42行。

发送问号的时候,微信服务器也会带上前面3个参数(signature、timestamp、nonce)访问开发者设置的URL,同时还会将消息的XML数据包POST到URL上。XML格式类似如下。

<xml>
    <ToUserName><![CDATA[gh_ba6050bc0be7]]></ToUserName>
    <FromUserName><![CDATA[oDeOAjgSJUX10wvImSRMSwmyQAyA]]></FromUserName>
    <CreateTime>1392043637</CreateTime>
    <MsgType><![CDATA[text]]></MsgType>
    <Content><![CDATA[?]]></Content>
    <MsgId>5978781895719912033</MsgId>
</xml>

而消息请求不包含echostr变量,所以将执行响应消息方法responseMsg()。

响应消息方法首先接收上述原始POST数据,见第47行。

然后它将数据载入对象中,对象名为SimpleXMLElement,LIBXML_NOCDATA?表示将CDATA合并为文本节点,代码中第50行实现此功能。

第51~54行取得XML类对象的值,并赋给新的变量,注意发送方变为接收方,接收方变为发送方。

第55~62行构造要回复的XML数据包。

第63行判断发送过来的关键字是不是问号。

第64~65行设置回复的消息类型为text,内容为当前年月日时分秒。

第66~67行封装回复的XML数据包,并且向微信服务器输出。XML格式如下。

<xml>
    <ToUserName><![CDATA[oDeOAjgSJUX10wvImSRMSwmyQAyA]]></ToUserName>
    <FromUserName><![CDATA[gh_ba6050bc0be7]]></FromUserName>
    <CreateTime>1392043638</CreateTime>
    <MsgType><![CDATA[text]]></MsgType>
    <Content><![CDATA[2014-01-05 11:43:23]]></Content>
</xml>

这样用户就会收到回复的消息,效果如图3-26所示。

3.3.5 消息体加/解密实现

在图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=35703636de2f9df2a77a662b68e521ce17c34d
b4&timestamp=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 ?>
相关文章
|
7天前
|
小程序 前端开发 API
微信小程序全栈开发中的异常处理与日志记录
【4月更文挑战第12天】本文探讨了微信小程序全栈开发中的异常处理和日志记录,强调其对确保应用稳定性和用户体验的重要性。异常处理涵盖前端(网络、页面跳转、用户输入、逻辑异常)和后端(数据库、API、业务逻辑)方面;日志记录则关注关键操作和异常情况的追踪。实践中,前端可利用try-catch处理异常,后端借助日志框架记录异常,同时采用集中式日志管理工具提升分析效率。开发者应注意安全性、性能和团队协作,以优化异常处理与日志记录流程。
|
7天前
|
小程序 安全 数据安全/隐私保护
微信小程序全栈开发中的身份认证与授权机制
【4月更文挑战第12天】本文探讨了微信小程序全栈开发中的身份认证与授权机制。身份认证包括手机号验证、微信登录和第三方登录,而授权机制涉及角色权限控制、ACL和OAuth 2.0。实践中,开发者可利用微信登录获取用户信息,集成第三方登录,以及实施角色和ACL进行权限控制。注意点包括安全性、用户体验和合规性,以保障小程序的安全运行和良好体验。通过这些方法,开发者能有效掌握小程序全栈开发技术。
|
7天前
|
JavaScript 前端开发 小程序
微信小程序全栈开发之性能优化策略
【4月更文挑战第12天】本文探讨了微信小程序全栈开发的性能优化策略,包括前端的资源和渲染优化,如图片压缩、虚拟DOM、代码分割;后端的数据库和API优化,如索引创建、缓存使用、RESTful API设计;以及服务器的负载均衡和CDN加速。通过这些方法,开发者可提升小程序性能,优化用户体验,增强商业价值。
|
1月前
|
SQL 存储 数据库
SQL实践篇(二):为什么微信用SQLite存储聊天记录
SQL实践篇(二):为什么微信用SQLite存储聊天记录
59 1
|
7天前
|
小程序 前端开发 JavaScript
微信小程序全栈开发中的PWA技术应用
【4月更文挑战第12天】本文探讨了微信小程序全栈开发中PWA技术的应用,PWA结合Web的开放性和原生应用的性能,提供离线访问、后台运行、桌面图标和原生体验。开发者可利用Service Worker实现离线访问,Worker处理后台运行,Web App Manifest添加桌面图标,CSS和JavaScript提升原生体验。实践中需注意兼容性、性能优化和用户体验。PWA技术能提升小程序的性能和用户体验,助力开发者打造优质小程序。
|
1月前
|
小程序 前端开发 程序员
微信小程序开发入门教程-小程序账号注册及开通
微信小程序开发入门教程-小程序账号注册及开通
|
7天前
|
SQL 安全 小程序
探索微信小程序全栈开发的安全性问题
【4月更文挑战第12天】本文探讨了微信小程序全栈开发中的安全性问题,包括数据安全、接口安全、隐私保护和代码安全。为解决这些问题,建议采取数据加密、使用HTTPS协议、身份认证与授权、输入验证、安全审计及漏洞扫描以及安全培训等措施。通过这些方法,开发者可提升小程序安全性,保护用户隐私和数据。
|
29天前
|
存储 缓存 监控
微信团队分享:微信后端海量数据查询从1000ms降到100ms的技术实践
针对大数据量带来的查询性能问题,微信团队对数据层查询接口进行了针对性的优化,将平均查询速度从1000ms+优化到了100ms级别。本文为各位分享优化过程,希望对你有用!
32 2
|
30天前
|
小程序 JavaScript
在使用微信小程序开发中用vant2框架中的Uploader 文件上传wx.uploadFile无反应和使用多图上传
网上有的说是bind:after-read="afterRead"的命名问题不支持-,但是我这儿执行了console.log("file",file);证明函数运行了。后来发现是multiple="true"原因开启了多图上传,如果是多图上传的话file就是数组了
25 2
|
1月前
|
小程序 前端开发 JavaScript
微信小程序MINA框架
微信小程序MINA框架
40 0

热门文章

最新文章