Common

<?php
namespace Proxy\Action; use Think\Action;
use Vendor\Func\Red; class CommonAction extends Action
{
public $red;
public function _initialize(){
$this->red = Red::create();
header('Content-type: text/html; charset=utf-8');
} /**
* 错误的json
* @param $code
* @param string $message
* @param array $data
*/
public function jsonError($code, $message = 'error', $data = [])
{
$returnData = [
'code' => $code,
'message' => $message,
'data' => $data
]; header('Content-type:application/json;charset=utf-8');
echo json_encode($returnData);
exit; } /**
* 正确的json
* @param int $code
* @param string $message
* @param array $data
*/
public function jsonSuccess($code = 1, $message = 'success', $data = [])
{
$returnData = [
'code' => $code,
'message' => $message,
'data' => $data
]; header('Content-type:application/json;charset=utf-8');
echo json_encode($returnData);
exit; } /**
* 设置Redis
* @param $k
* @param $v
* @param int $expires
* @return mixed
*/
public function setCache($k, $v, $expires = -1)
{
if ($expires === -1) {
return $this->red->set($k, $v);
} else {
return $this->red->setex($k, $expires, $v);
}
} /**
* 获取Redis
* @param $k
* @return mixed
*/
public function getCache($k)
{
return $this->red->get($k);
} /**
* 删除Redis
* @param $k
* @return mixed
*/
public function delCache($k)
{
return $this->red->delete($k);
} /**
* 获取过期时间
* @param $k
* @return bool
*/
public function getCacheTtl($k)
{
$ttl = $this->red->ttl($k);
if ($ttl != '-2') {
return $ttl;
} else {
return false;
}
} /**
* 检查是否存在
* @param $k
* @return mixed
*/
public function cacheExists($k)
{
return $this->red->exists($k);
}
}

AccessToken

<?php
namespace Proxy\Action; use Vendor\Func\Http; /***
* 基础支持的access_token,每日两千次
* @package Proxy\Action
*/
class AccessTokenAction extends CommonAction
{ const API_TOKEN = 'https://api.weixin.qq.com/cgi-bin/token';
const API_TICKET = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket';
const TOKEN_EXPIRES = 1500;
const TICKET_EXPIRES = 1500; protected $access_token_key = ''; // access_token Redis 键值名
protected $js_ticket_key = ''; // 凭证 Redis 键值名 public function __construct()
{
parent::__construct();
$this->access_token_key = C('WX_APP_ID').':access_token';
$this->js_ticket_key = C('WX_APP_ID').':js_ticket';
} /**
* 获取全局access_token
*/
public function get()
{
// step1 判是存在
$exists = $this->cacheExists($this->access_token_key); // step2 是否强制刷新或者已过期
if ((isset($_GET['flush']) & $_GET['flush']) || !$exists) {
$data = $this->reload();
} else {
$data = [
'access_token' => $this->getCache($this->access_token_key),
'expires_in' => $this->getCacheTtl($this->access_token_key)
];
} if (!$data['access_token']) {
$this->ajaxReturn($this->jsonError(10002, '接口请求失败'));
} $this->ajaxReturn($this->jsonSuccess(1, '成功', $data));
} /**
* 公众号用于调用微信JS接口的临时票据
* @return bool
*/
public function jsTicket()
{
// step1 判是存在
$exists = $this->cacheExists($this->js_ticket_key);
if (!$exists) {
// 获取token
$token = $this->getCache($this->access_token_key);
setlog($token,[],'token');
// 如果token不存在,重新生成
if (!$token) {
// 重新加载一次Token
$this->reload();
$token = $this->getCache($this->access_token_key);
} $params = [
'access_token' => $token,
'type' => 'jsapi',
]; $data = Http::newDoGet(self::API_TICKET, $params); $data = json_decode($data, true); if ((int)$data['errcode'] !== 0) {
$this->delCache($this->access_token_key);
return false;
} // 减少1500秒过期时间,提前过期
$this->setCache($this->js_ticket_key, $data['ticket'], $data['expires_in'] - self::TICKET_EXPIRES);
} else {
$data['ticket'] = $this->getCache($this->js_ticket_key);
} return $data['ticket'];
} /**
* 获取JS-SDK配置信息
*/
public function getConfig()
{
if (!isset($_GET['uri'])) {
$this->ajaxReturn($this->jsonError(0, '参数错误'));
} $url = $_GET['uri'];
$ticket = $this->jsTicket();
setlog($ticket,[],'ticket'); while (!$ticket) {
$this->reload();
$ticket = $this->jsTicket();
} $data = [
'jsapi_ticket' => $ticket,
'nonceStr' => (string)mt_rand(),
'timestamp' => time(),
'url' => $url
]; $sign = $this->getSign($data); $data['signature'] = $sign;
$data['appId'] = C('WX_APP_ID'); $this->ajaxReturn($this->jsonSuccess(1, '成功', $data));
} /**
* 重新载入 access_token
*/
protected function reload()
{
if ($this->cacheExists($this->access_token_key)) {
$data = [
'access_token' => $this->getCache($this->access_token_key),
'expires_in' => $this->getCacheTtl($this->access_token_key)
];
return $data;
}
$params = [
'grant_type' => 'client_credential',
'appid' => C('WX_APP_ID'),
'secret' => C('WX_APP_SECRET')
];
$data = Http::newDoGet(self::API_TOKEN, $params);
if (!$data) {
$this->ajaxReturn($this->jsonError(10002, '接口请求失败'));
}
$data = json_decode($data, true);
if (isset($data['errcode'])) {
$this->ajaxReturn($this->jsonError($data['errcode']), $data['errmsg']);
}
$this->setCache($this->access_token_key, $data['access_token'], $data['expires_in'] - self::TOKEN_EXPIRES);
return $data;
} /**
* 签名算法
* @param $data
* @return string
*/
protected function getSign($data)
{
$str = '';
foreach ($data as $k => $v) {
$str .= strtolower($k) .'='.$v;
$str .= '&';
} $str = trim($str, '&');
$str = sha1($str); return $str;
} }

微信开发核心AccessToken实现的更多相关文章

  1. 微信开发 获取AccessToken

    获取AccessToken的方法 public static Access_token GetAccessToken() { string formatString = String.Format(& ...

  2. 微信开发笔记(accesstoken)

    access_token分两种 一种是公众号权限获取用,调用cgi-bin接口 ,此种token一个公众号同时只有一个,用这一个就够了. 服务器最好缓存. 用这个token前提是用户关注了此公众号. ...

  3. 使用delphi+intraweb进行微信开发5—准备实现微信API,先从获取AccessToken开始

    在前4讲中我们已经使iw开发的应用成功和微信进行了对接,再接下来的章节中我们开始逐一尝试和实现微信的各个API,开始前先来点准备工作 首先需要明确的是,微信的API都是通过https调用实现的,分为p ...

  4. 微信开发学习日记(八):7步看懂weiphp插件机制,核心目标是响应微信请求

    又经过了几个小时的梳理.回顾,截至目前,终于对weiphp这个框架的机制搞明白了些.想要完全明白,自然还需要大把的时间.第1步:   配置微信公众号,http://weiphp.jiutianniao ...

  5. 微信开发-ACCESS TOKEN 过期失效解决方案

    微信开发-ACCESS TOKEN 过期失效解决方案 起因:因为access_token的重要性,开发过微信的都知道,但是他有自己的生命周期,官方解释为:"有效期为7200秒",一 ...

  6. 微信开发系列----02:实现POST请求响应

    继续昨天的,现在我们的微信测试成功了,可以开发实现微信的各种功能,今天主要实现微信的简单交互,比如发送语音,图片,文本等请求,网站服务器发送对应的响应. 项目GitHub地址:  https://gi ...

  7. 2016北京PHP39期 ThinkPHP Discuz Dedecms 微信开发视频教程

    2016北京PHP39期 ThinkPHP Discuz Dedecms 微信开发视频教程 所有项目均带有软件,笔记,视频,源码 日期   课程(空内容代表放假) 2015/7/10 星期五 开学典礼 ...

  8. 《微信开发日志》之OAuth2验证接口

    OAuth2接口说明: 企业应用中的URL链接(包括自定义菜单或者消息中的链接),可以通过OAuth2.0验证接口来获取员工的身份信息. 通过此接口获取用户身份会有一定的时间开销.对于频繁获取用户身份 ...

  9. html 微信开发——微信授权

    微信JS-SDK说明文档 链接地址:http://mp.weixin.qq.com/wiki/7/aaa137b55fb2e0456bf8dd9148dd613f.html 微信web开发:http: ...

随机推荐

  1. jquery.nicescroll.js Unable to preventDefault inside passive event listener due to target being treated as passive.

    解决办法就是:https://github.com/bestjhh/Plugin 下载替换. 参考: https://github.com/bestjhh/Plugin https://blog.cs ...

  2. Win7如何设置怎样在局域网内共享打印机

    首先进入桌面,点击开始按钮,然后打开控制面板 2 在控制面板设置界面,找到“管理工具”选项 3 接着打开“计算机管理”   选择“本地用户和组”的Guest账户   确保Guest账户被禁用   下面 ...

  3. 依赖倒置原则(DIP)

    1. 定义 (1)高层模块不应依赖于低层模块,两者都应该依赖于抽象.(2)抽象不应该依赖于细节,细节应该依赖于抽象. 为什么是“倒置”这个词? 这是由于许多传统的软件开发方法,比如结构化分析和设计,总 ...

  4. Simplifying Failures

    # # Finish the delta debug function ddmin # import re def test(s): print s, len(s),repr(s) if re.sea ...

  5. Spring boot POST 提交错误 Request header is too large

    解决方法 application.yml server: # 单位 KB max-http-header-size: 100000 java.lang.IllegalArgumentException ...

  6. python中requests里.text和.content方法的区别

    requests对象的get和post方法都会返回一个Response对象,这个对象里面存的是服务器返回的所有信息,包括响应头,响应状态码等.其中返回的网页部分会存在.content和.text两个对 ...

  7. 我们一起分析一下这个刚刚修复的RDP漏洞CVE-2019-0708

    写在前面的话 在微软今年五月份的漏洞更新安全公告中,提到了一个跟远程桌面协议(RDP)有关的漏洞.我们之所以要在这里专门针对这个漏洞进行分析,是因为这个漏洞更新涉及到Windows XP以及其他多个W ...

  8. CEF3开发者系列之Cookies管理和共享<转>

    原帖地址:https://www.cnblogs.com/guolixiucai/p/6994559.html 涉及网页登录相关的技术,Cookies肯定是忽略不了的.由于项目的需要,要做一个双核的产 ...

  9. Visual Studio 2019更新到16.2.2

    Visual Studio 2019更新到16.2.2 此次更新的变化有以下几点: (1)修复了Visual Studio在关闭时的停止响应问题.(2)修复CVE-2019-1211权限提升漏洞.(3 ...

  10. android studio 把 ionic 打包时修改应用名称、修改应用图标、修改启动画面,升级打包

    在项目中resources中替换成自己的图标和启动画面即可 在config.xml 修改包名 打正式包和升级打包同原生的类似,在Androidmanifest.xml修改版本号和版本名