微信公众平台开发(二)——access_token、日志

本文涉及的产品
云数据库 MongoDB,独享型 2核8GB
推荐场景:
构建全方位客户视图
日志服务 SLS,月写入数据量 50GB 1个月
简介:

一、access_token

1)两种access_token,网页授权access_token和普通access_token

1、微信网页授权是通过OAuth2.0机制实现的,在用户授权给公众号后,公众号可以获取到一个网页授权特有的接口调用凭证(网页授权access_token),通过网页授权access_token可以进行授权后接口调用,如获取用户基本信息。


2、其他微信接口,需要通过基础支持中的“获取access_token”接口来获取到的普通access_token调用。access_token是公众号的全局唯一票据,access_token的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的access_token失效。

 

2)分别获取access_token

1、网页授权的:点击查看网页授权获取用户基本信息文档,通过查看这个文档,

可以看到通过code换取网页授权access_token,而这个code是通过微信的一个授权链接获取到的,然后再根据文档中的请求获取到的,具体的链接地址和参数可以参考文档。

/**
     * 创建一个需要通过微信的OAuth2.0认证的服务url
     * @param $url 服务号需要认证访问的url
     * @param $scope string snsapi_userinfo | snsapi_base
     *      snsapi_userinfo 可以用来获取用户信息
     *      snsapi_base 可以用来获取openid
     * @param string $state 自定义状态值
     *      此处约定为from_weixin代表是从微信认证过来,一般无需轻易变化
     * @return string 返回认证url地址
     */
    public function createAuthUrl($url, $scope = 'snsapi_base', $state = 'from_weixin') { $url = strval($url); $authUrl = 'https://open.weixin.qq.com/connect/oauth2/authorize'; /** * 此处有大坑,请不要打乱param的顺序 * 否则微信认证界面会出现白屏 */ $param = array( 'appid' => $this->appId, 'redirect_uri' => urlencode($url), 'response_type' => 'code', 'scope' => $scope, 'state' => $state ); $seg = array(); foreach ($param as $k => $v) { $seg[] = "{$k}={$v}"; } return $authUrl . '?' . join('&', $seg) . '#wechat_redirect'; }

 

2、普通的:点击查看获取access token文档,通过三个参数获取到。

这里需要注意的是,获取到的token,是有时效性的,2 个小时,所以我会保存在MongoDB中,先从数据库中比对超时了没有,没有的话就直接从数据库中获取,减少不必要的请求。

 

二、推送日志

  在与微信的交互中,会产生很多日志信息,并且开发的时候经常需要分析这些日志,这里我将日志都保存在了MongoDB中。MongoDB方便的地方是任何结构的数据都能放在一个document中,不像MySQL要定义好字段名,我经常调试的时候将各种结构放在一个document中。

 

在微信的入口页面中,也就是前面提到的URL(服务器地址),会在这里面做保存日志的逻辑。逻辑包括关注的时候推送一条消息,二维码扫描关注,点击某个菜单产生事件,点击菜单的超链接等

日志结构如下:

1、代码中包括签名验证逻辑

2、通过file_get_contents('php://input')来获取请求数据,就是下面的getRawMsg方法

3、将推送日志直接塞入到MongoDB

4、将接收到的请求信息SimpleXMLElement对象,就是下面的parseMsg方法

5、handleEventMsg就是在处理各种不同情况了

/**
     * 微信公众号入口
     */
    public function actionPortal()
    {
        $weixin = new Weixin();
        //签名验证逻辑
//         if($weixin->checkSignature()){
//             echo $_GET['echostr'];
//         }
//         exit;
        //读取原始请求数据
        $msg = $weixin->getRawMsg();
        
        //推送日志
        $pushlog = new WeixinPushLog();
        $pushlog->logWeixinPush($msg);
        
        $msgObj = $weixin->parseMsg($msg);
        if ($msgObj === false || !is_object($msgObj)) {
            exit;
        }
        switch ($msgObj->MsgType) {
            case 'event' : //接收事件消息
                $this->handleEventMsg($msgObj);
                break;
            default :
                //todo
                break;
        }
    }
    public function getRawMsg()
    {
        return file_get_contents('php://input');
    }

    /**
     * 解析接收到的消息
     * @param string $msg 消息体
     * @return bool|SimpleXMLElement
     */
    public function parseMsg($msg = '')
    {
        if (!$msg || empty($msg)) {
            return false;
        }
        $msgObj = simplexml_load_string($msg, 'SimpleXMLElement', LIBXML_NOCDATA);
        if ($msgObj === false || !($msgObj instanceof \SimpleXMLElement)) {
            return false;
        }
        return $msgObj;
    } 

 

6、如果要推送消息,die这个方法得要加上

7、下面的代码只列举了两种事件情况,一种是订阅、一种是点击事件

8、createRawTuWenMsg是在拼接XML,点击查看模板消息接口

private function handleEventMsg($msgObj)
    {
        $weixin = new Weixin();
        $openId = $msgObj->FromUserName;
        $fromUserName = $msgObj->ToUserName;
        //未关注,关注后推送
        if ($msgObj->Event == 'subscribe') {
            $pushData['PicUrl'] = 'http://mmbiz.qpic.cn/';
            $pushData['Title'] = '基因检测,带你一起探索生命的奥妙 ';
            $pushData['Description'] = '为什么不同人在身高、体重、肤色和形状上长得不一样?但是往往又和自己的父母相似?';
            $pushData['Url'] = 'http://mp.weixin.qq.com';
            $msg = $weixin->createRawTuWenMsg($fromUserName, $openId, array($pushData));

            die($msg);
        }elseif($msgObj->Event == 'CLICK'){
            //die($msg);
        }
    }
public function createRawTuWenMsg($fromUserName, $toUserName, $items = array())
    {
        if (!is_array($items)) {
            return '';
        }
        $count = count($items);
        $its = '';
        foreach ($items as $item) {
            $its .= <<<ITEMTPL
<item>
<Title><![CDATA[{$item['Title']}]]></Title>
<Description><![CDATA[{$item['Description']}]]></Description>
<PicUrl><![CDATA[{$item['PicUrl']}]]></PicUrl>
<Url><![CDATA[{$item['Url']}]]></Url>
</item>
ITEMTPL;
        }
    
        $msg = <<<MSG
<xml>
<ToUserName><![CDATA[{$toUserName}]]></ToUserName>
<FromUserName><![CDATA[{$fromUserName}]]></FromUserName>
<CreateTime>12345678</CreateTime>
<MsgType><![CDATA[news]]></MsgType>
<ArticleCount>{$count}</ArticleCount>
<Articles>
{$its}
</Articles>
</xml>
MSG;
    return $msg;
    }

 

demo下载:

github地址:https://github.com/pwstrick/weixin_demo

CSDN地址:http://download.csdn.net/detail/loneleaf1/9045731



    本文转自 咖啡机(K.F.J)   博客园博客,原文链接:http://www.cnblogs.com/strick/p/4747374.html,如需转载请自行联系原作者


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
3月前
|
存储 监控 数据库
Django 后端架构开发:高效日志规范与实践
Django 后端架构开发:高效日志规范与实践
59 1
|
5月前
|
存储 数据采集 JavaScript
深入理解数仓开发(一)数据技术篇之日志采集
深入理解数仓开发(一)数据技术篇之日志采集
|
19天前
|
Rust 前端开发 JavaScript
Tauri 开发实践 — Tauri 日志记录功能开发
本文介绍了如何为 Tauri 应用配置日志记录。Tauri 是一个利用 Web 技术构建桌面应用的框架。文章详细说明了如何在 Rust 和 JavaScript 代码中设置和集成日志记录,并控制日志输出。通过添加 `log` crate 和 Tauri 日志插件,可以轻松实现多平台日志记录,包括控制台输出、Webview 控制台和日志文件。文章还展示了如何调整日志级别以优化输出内容。配置完成后,日志记录功能将显著提升开发体验和程序稳定性。
50 1
Tauri 开发实践 — Tauri 日志记录功能开发
|
3月前
|
SQL 关系型数据库 MySQL
【MySQL】根据binlog日志获取回滚sql的一个开发思路
【MySQL】根据binlog日志获取回滚sql的一个开发思路
|
24天前
|
开发框架 缓存 安全
开发日志:IIS安全配置
开发日志:IIS安全配置
开发日志:IIS安全配置
|
25天前
|
开发工具 git
git显示开发日志+WinSW——将.exe文件注册为服务的一个工具+图床PicGo+kubeconfig 多个集群配置 如何切换
git显示开发日志+WinSW——将.exe文件注册为服务的一个工具+图床PicGo+kubeconfig 多个集群配置 如何切换
32 1
|
3月前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
120 1
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
3月前
|
小程序 前端开发 API
微信小程序全栈开发中的异常处理与日志记录是一个重要而复杂的问题。
微信小程序作为业务拓展的新渠道,其全栈开发涉及前端与后端的紧密配合。本文聚焦小程序开发中的异常处理与日志记录,从前端的网络、页面跳转等异常,到后端的数据库、API调用等问题,详述了如何利用try-catch及日志框架进行有效管理。同时强调了集中式日志管理的重要性,并提醒开发者注意安全性、性能及团队协作等方面,以构建稳定可靠的小程序应用。
59 1
|
4月前
|
开发框架 缓存 前端开发
基于SqlSugar的开发框架循序渐进介绍(8)-- 在基类函数封装实现用户操作日志记录
基于SqlSugar的开发框架循序渐进介绍(8)-- 在基类函数封装实现用户操作日志记录
|
3月前
|
人工智能 Java Spring
Spring框架下,如何让你的日志管理像‘AI’一样智能,提升开发效率的秘密武器!
【8月更文挑战第31天】日志管理在软件开发中至关重要,不仅能帮助开发者追踪问题和调试程序,还是系统监控和运维的重要工具。在Spring框架下,通过合理配置Logback等日志框架,可大幅提升日志管理效率。本文将介绍如何引入日志框架、配置日志级别、在代码中使用Logger,以及利用ELK等工具进行日志聚合和分析,帮助你构建高效、可靠的日志管理系统,为开发和运维提供支持。
55 0