mirror of
https://gitee.com/zongzhige/shopxo.git
synced 2024-11-30 02:49:03 +08:00
382 lines
12 KiB
PHP
Executable File
382 lines
12 KiB
PHP
Executable File
<?php
|
|
// +----------------------------------------------------------------------
|
|
// | ShopXO 国内领先企业级B2C免费开源电商系统
|
|
// +----------------------------------------------------------------------
|
|
// | Copyright (c) 2011~2099 http://shopxo.net All rights reserved.
|
|
// +----------------------------------------------------------------------
|
|
// | Licensed ( https://opensource.org/licenses/mit-license.php )
|
|
// +----------------------------------------------------------------------
|
|
// | Author: Devil
|
|
// +----------------------------------------------------------------------
|
|
namespace base;
|
|
|
|
/**
|
|
* 微信驱动
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2018-06-28
|
|
* @desc 支持所有文件存储到硬盘
|
|
*/
|
|
class Wechat
|
|
{
|
|
// appid
|
|
private $_appid;
|
|
|
|
// appsecret
|
|
private $_appsecret;
|
|
|
|
/**
|
|
* [__construct 构造方法]
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @datetime 2017-12-30T18:04:05+0800
|
|
* @param [string] $app_id [应用appid]
|
|
* @param [string] $app_secret [应用密钥]
|
|
*/
|
|
public function __construct($app_id, $app_secret)
|
|
{
|
|
$this->_appid = $app_id;
|
|
$this->_appsecret = $app_secret;
|
|
}
|
|
|
|
/**
|
|
* [DecryptData 检验数据的真实性,并且获取解密后的明文]
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @datetime 2017-12-30T18:20:53+0800
|
|
* @param [string] $encrypted_data [加密的用户数据]
|
|
* @param [string] $iv [与用户数据一同返回的初始向量]
|
|
* @param [string] $openid [解密后的原文]
|
|
* @return [array|string] [成功返回用户信息数组, 失败返回错误信息]
|
|
*/
|
|
public function DecryptData($encrypted_data, $iv, $openid)
|
|
{
|
|
// 登录授权session
|
|
$login_key = 'wechat_user_login_'.$openid;
|
|
$session_data = MyCache($login_key);
|
|
if(empty($session_data))
|
|
{
|
|
return ['status'=>-1, 'msg'=>'session key不存在'];
|
|
}
|
|
|
|
// iv长度
|
|
if(strlen($iv) != 24)
|
|
{
|
|
return ['status'=>-1, 'msg'=>'iv长度错误'];
|
|
}
|
|
|
|
// 加密函数
|
|
if(!function_exists('openssl_decrypt'))
|
|
{
|
|
return ['status'=>-1, 'msg'=>'openssl不支持'];
|
|
}
|
|
|
|
$result = openssl_decrypt(base64_decode($encrypted_data), "AES-128-CBC", base64_decode($session_data['session_key']), 1, base64_decode($iv));
|
|
$data = json_decode($result, true);
|
|
if($data == NULL)
|
|
{
|
|
return ['status'=>-1, 'msg'=>'请重试!'];
|
|
}
|
|
if($data['watermark']['appid'] != $this->_appid)
|
|
{
|
|
return ['status'=>-1, 'msg'=>'appid不匹配'];
|
|
}
|
|
|
|
// 缓存存储
|
|
$data_key = 'wechat_user_info_'.$openid;
|
|
MyCache($data_key, $data);
|
|
|
|
return ['status'=>0, 'data'=>$data];
|
|
}
|
|
|
|
/**
|
|
* 用户授权
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2018-11-06
|
|
* @desc description
|
|
* @param [array] $params [输入参数]
|
|
*/
|
|
public function GetAuthSessionKey($params = [])
|
|
{
|
|
if(empty($params['authcode']))
|
|
{
|
|
return ['status'=>-1, 'msg'=>'授权码有误'];
|
|
}
|
|
|
|
// 请求获取session_key
|
|
$url = 'https://api.weixin.qq.com/sns/jscode2session?appid='.$this->_appid.'&secret='.$this->_appsecret.'&js_code='.$params['authcode'].'&grant_type=authorization_code';
|
|
$result = $this->HttpRequestGet($url);
|
|
if(!empty($result) && !empty($result['openid']))
|
|
{
|
|
// 缓存SessionKey
|
|
$key = 'wechat_user_login_'.$result['openid'];
|
|
|
|
// 缓存存储
|
|
MyCache($key, $result);
|
|
return ['status'=>0, 'msg'=>'授权成功', 'data'=>$result];
|
|
}
|
|
return ['status'=>-1, 'msg'=>$result['errmsg']];
|
|
}
|
|
|
|
/**
|
|
* [MiniQrCodeCreate 二维码创建]
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @datetime 2018-01-02T19:53:10+0800
|
|
* @param [string] $params['page'] [页面地址]
|
|
* @param [string] $params['scene'] [参数]
|
|
* @return [string] [成功返回文件流, 失败则空]
|
|
*/
|
|
public function MiniQrCodeCreate($params)
|
|
{
|
|
// 请求参数
|
|
$p = [
|
|
[
|
|
'checked_type' => 'empty',
|
|
'key_name' => 'page',
|
|
'error_msg' => 'page地址不能为空',
|
|
],
|
|
[
|
|
'checked_type' => 'length',
|
|
'checked_data' => '1,32',
|
|
'key_name' => 'scene',
|
|
'error_msg' => 'scene参数 1~32 个字符之间',
|
|
],
|
|
];
|
|
$ret = ParamsChecked($params, $p);
|
|
if($ret !== true)
|
|
{
|
|
return DataReturn($ret, -1);
|
|
}
|
|
|
|
// 获取access_token
|
|
$access_token = $this->GetMiniAccessToken();
|
|
if($access_token === false)
|
|
{
|
|
return DataReturn('access_token获取失败', -1);
|
|
}
|
|
|
|
// 获取二维码
|
|
$url = 'https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token='.$access_token;
|
|
$data = [
|
|
'page' => $params['page'],
|
|
'scene' => $params['scene'],
|
|
'width' => empty($params['width']) ? 1000 : intval($params['width']),
|
|
];
|
|
$res = $this->HttpRequestPost($url, json_encode($data), false);
|
|
if(!empty($res))
|
|
{
|
|
if(stripos($res, 'errcode') === false)
|
|
{
|
|
return DataReturn('获取成功', 0, $res);
|
|
}
|
|
$res = json_decode($res, true);
|
|
$msg = isset($res['errmsg']) ? $res['errmsg'] : '获取二维码失败';
|
|
} else {
|
|
$msg = '获取二维码失败';
|
|
}
|
|
return DataReturn($msg, -1);
|
|
}
|
|
|
|
/**
|
|
* 小程序获取access_token
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2020-08-26
|
|
* @desc description
|
|
*/
|
|
public function GetMiniAccessToken()
|
|
{
|
|
return $this->GetAccessToken();
|
|
}
|
|
|
|
/**
|
|
* 获取微信环境签名配置信息
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2020-08-26
|
|
* @desc description
|
|
* @param [array] $params [输入参数]
|
|
*/
|
|
public function GetSignPackage($params = [])
|
|
{
|
|
$access_token = $this->GetAccessToken();
|
|
if(!empty($access_token))
|
|
{
|
|
// 获取 ticket
|
|
$ticket = $this->GetTicket($access_token);
|
|
|
|
// 注意 URL 一定要动态获取,不能 hardcode.
|
|
$url = empty($params['url']) ? __MY_VIEW_URL__ : urldecode(htmlspecialchars_decode($params['url']));
|
|
|
|
$timestamp = time();
|
|
$nonce_str = $this->CreateNonceStr();
|
|
|
|
// 这里参数的顺序要按照 key 值 ASCII 码升序排序
|
|
$string = "jsapi_ticket={$ticket}&noncestr={$nonce_str}×tamp={$timestamp}&url={$url}";
|
|
|
|
return [
|
|
'appId' => $this->_appid,
|
|
'nonceStr' => $nonce_str,
|
|
'timestamp' => $timestamp,
|
|
'url' => $url,
|
|
'signature' => sha1($string),
|
|
'rawString' => $string
|
|
];
|
|
}
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* 签名随机字符串创建
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2020-08-26
|
|
* @desc description
|
|
* @param [int] $length [长度]
|
|
*/
|
|
public function CreateNonceStr($length = 16)
|
|
{
|
|
$chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
|
|
$str = '';
|
|
for($i = 0; $i < $length; $i++)
|
|
{
|
|
$str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
|
|
}
|
|
return $str;
|
|
}
|
|
|
|
/**
|
|
* 公共获取access_token
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2020-08-26
|
|
* @desc description
|
|
*/
|
|
public function GetAccessToken()
|
|
{
|
|
// 缓存key
|
|
$key = $this->_appid.'_access_token';
|
|
$result = MyCache($key);
|
|
if(!empty($result))
|
|
{
|
|
if($result['expires_in'] > time())
|
|
{
|
|
return $result['access_token'];
|
|
}
|
|
}
|
|
|
|
// 网络请求
|
|
$url = 'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid='.$this->_appid.'&secret='.$this->_appsecret;
|
|
$result = $this->HttpRequestGet($url);
|
|
if(!empty($result) && !empty($result['access_token']))
|
|
{
|
|
// 缓存存储
|
|
$result['expires_in'] += time();
|
|
MyCache($key, $result);
|
|
return $result['access_token'];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* 获取授权页ticket
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @date 2020-08-26
|
|
* @desc description
|
|
* @param [string] $access_token [access_token]
|
|
* @param [string] $type [类型(默认jsapi)]
|
|
*/
|
|
public function GetTicket($access_token, $type = 'jsapi')
|
|
{
|
|
// 缓存key
|
|
$key = $this->_appid.'_get_ticket';
|
|
$result = MyCache($key);
|
|
if(!empty($result))
|
|
{
|
|
if($result['expires_in'] > time())
|
|
{
|
|
return $result['ticket'];
|
|
}
|
|
}
|
|
|
|
// 网络请求
|
|
$url = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token='.$access_token.'&type='.$type;
|
|
$result = $this->HttpRequestGet($url);
|
|
if(!empty($result) && !empty($result['ticket']))
|
|
{
|
|
// 缓存存储
|
|
$result['expires_in'] += time();
|
|
MyCache($key, $result);
|
|
return $result['ticket'];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* [HttpRequestGet get请求]
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @datetime 2018-01-03T19:21:38+0800
|
|
* @param [string] $url [url地址]
|
|
* @return [array] [返回数据]
|
|
*/
|
|
public function HttpRequestGet($url)
|
|
{
|
|
$curl = curl_init();
|
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
|
|
curl_setopt($curl, CURLOPT_TIMEOUT, 500);
|
|
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
|
|
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
|
|
curl_setopt($curl, CURLOPT_URL, $url);
|
|
|
|
$res = curl_exec($curl);
|
|
curl_close($curl);
|
|
return json_decode($res, true);
|
|
}
|
|
|
|
/**
|
|
* [HttpRequestPost curl模拟post]
|
|
* @author Devil
|
|
* @blog http://gong.gg/
|
|
* @version 1.0.0
|
|
* @datetime 2017-09-25T09:10:46+0800
|
|
* @param [string] $url [请求地址]
|
|
* @param [array] $data [发送的post数据]
|
|
* @param [array] $is_parsing [是否需要解析数据]
|
|
* @return [array] [返回的数据]
|
|
*/
|
|
public function HttpRequestPost($url, $data, $is_parsing = true)
|
|
{
|
|
$curl = curl_init();
|
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
|
|
curl_setopt($curl, CURLOPT_TIMEOUT, 500);
|
|
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
|
|
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
|
|
curl_setopt($curl, CURLOPT_URL, $url);
|
|
curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
|
|
curl_setopt($curl, CURLOPT_POST, true);
|
|
|
|
$res = curl_exec($curl);
|
|
if($is_parsing === true)
|
|
{
|
|
return json_decode($res, true);
|
|
}
|
|
return $res;
|
|
}
|
|
|
|
}
|
|
?>
|