+ *
+ * 当用户使用自己的缓存驱动时,直接实例化对象后可直接设置 AccessToekn
+ * - 多用于分布式项目时保持 AccessToken 统一
+ * - 使用此方法后就由用户来保证传入的 AccessToekn 为有效 AccessToekn
+ */
+ public function setAccessToken($access_token)
+ {
+ if (!is_string($access_token)) {
+ throw new InvalidArgumentException("Invalid AccessToken type, need string.");
+ }
+ $cache = $this->config->get('appid') . '_access_token';
+ Tools::setCache($cache, $this->access_token = $access_token);
+ }
+
+ /**
+ * 清理删除 AccessToken
+ * @return bool
+ */
+ public function delAccessToken()
+ {
+ $this->access_token = '';
+ return Tools::delCache($this->config->get('appid') . '_access_token');
+ }
+
+ /**
+ * 以GET获取接口数据并转为数组
+ * @param string $url 接口地址
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function httpGetForJson($url)
+ {
+ try {
+ return Tools::json2arr(Tools::get($url));
+ } catch (InvalidResponseException $e) {
+ if (isset($this->currentMethod['method']) && empty($this->isTry)) {
+ if (in_array($e->getCode(), ['40014', '40001', '41001', '42001'])) {
+ $this->delAccessToken();
+ $this->isTry = true;
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ }
+ throw new InvalidResponseException($e->getMessage(), $e->getCode());
+ }
+ }
+
+ /**
+ * 以POST获取接口数据并转为数组
+ * @param string $url 接口地址
+ * @param array $data 请求数据
+ * @param bool $buildToJson
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function httpPostForJson($url, array $data, $buildToJson = true)
+ {
+ try {
+ return Tools::json2arr(Tools::post($url, $buildToJson ? Tools::arr2json($data) : $data));
+ } catch (InvalidResponseException $e) {
+ if (!$this->isTry && in_array($e->getCode(), ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ throw new InvalidResponseException($e->getMessage(), $e->getCode());
+ }
+ }
+
+ /**
+ * 注册当前请求接口
+ * @param string $url 接口地址
+ * @param string $method 当前接口方法
+ * @param array $arguments 请求参数
+ * @return mixed
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function registerApi(&$url, $method, $arguments = [])
+ {
+ $this->currentMethod = ['method' => $method, 'arguments' => $arguments];
+ if (empty($this->access_token)) {
+ $this->access_token = $this->getAccessToken();
+ }
+ return $url = str_replace('ACCESS_TOKEN', $this->access_token, $url);
+ }
+
+ /**
+ * 接口通用POST请求方法
+ * @param string $url 接口URL
+ * @param array $data POST提交接口参数
+ * @param bool $isBuildJson
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function callPostApi($url, array $data, $isBuildJson = true)
+ {
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data, $isBuildJson);
+ }
+
+ /**
+ * 接口通用GET请求方法
+ * @param string $url 接口URL
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function callGetApi($url)
+ {
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/BasicWePay.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/BasicWePay.php
new file mode 100644
index 000000000..55a708004
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/BasicWePay.php
@@ -0,0 +1,203 @@
+config = new DataArray($options);
+ // 商户基础参数
+ $this->params = new DataArray([
+ 'appid' => $this->config->get('appid'),
+ 'mch_id' => $this->config->get('mch_id'),
+ 'nonce_str' => Tools::createNoncestr(),
+ ]);
+ // 商户参数支持
+ if ($this->config->get('sub_appid')) {
+ $this->params->set('sub_appid', $this->config->get('sub_appid'));
+ }
+ if ($this->config->get('sub_mch_id')) {
+ $this->params->set('sub_mch_id', $this->config->get('sub_mch_id'));
+ }
+ }
+
+ /**
+ * 静态创建对象
+ * @param array $config
+ * @return static
+ */
+ public static function instance(array $config)
+ {
+ $key = md5(get_called_class() . serialize($config));
+ if (isset(self::$cache[$key])) return self::$cache[$key];
+ return self::$cache[$key] = new static($config);
+ }
+
+ /**
+ * 获取微信支付通知
+ * @return array
+ * @throws InvalidResponseException
+ */
+ public function getNotify()
+ {
+ $data = Tools::xml2arr(file_get_contents('php://input'));
+ if (isset($data['sign']) && $this->getPaySign($data) === $data['sign']) {
+ return $data;
+ }
+ throw new InvalidResponseException('Invalid Notify.', '0');
+ }
+
+ /**
+ * 获取微信支付通知回复内容
+ * @return string
+ */
+ public function getNotifySuccessReply()
+ {
+ return Tools::arr2xml(['return_code' => 'SUCCESS', 'return_msg' => 'OK']);
+ }
+
+ /**
+ * 生成支付签名
+ * @param array $data 参与签名的数据
+ * @param string $signType 参与签名的类型
+ * @param string $buff 参与签名字符串前缀
+ * @return string
+ */
+ public function getPaySign(array $data, $signType = 'MD5', $buff = '')
+ {
+ ksort($data);
+ if (isset($data['sign'])) unset($data['sign']);
+ foreach ($data as $k => $v) $buff .= "{$k}={$v}&";
+ $buff .= ("key=" . $this->config->get('mch_key'));
+ if (strtoupper($signType) === 'MD5') {
+ return strtoupper(md5($buff));
+ }
+ return strtoupper(hash_hmac('SHA256', $buff, $this->config->get('mch_key')));
+ }
+
+ /**
+ * 转换短链接
+ * @param string $longUrl 需要转换的URL,签名用原串,传输需URLencode
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function shortUrl($longUrl)
+ {
+ $url = 'https://api.mch.weixin.qq.com/tools/shorturl';
+ return $this->callPostApi($url, ['long_url' => $longUrl]);
+ }
+
+
+ /**
+ * 数组直接转xml数据输出
+ * @param array $data
+ * @param bool $isReturn
+ * @return string
+ */
+ public function toXml(array $data, $isReturn = false)
+ {
+ $xml = Tools::arr2xml($data);
+ if ($isReturn) {
+ return $xml;
+ }
+ echo $xml;
+ }
+
+ /**
+ * 以Post请求接口
+ * @param string $url 请求
+ * @param array $data 接口参数
+ * @param bool $isCert 是否需要使用双向证书
+ * @param string $signType 数据签名类型 MD5|SHA256
+ * @param bool $needSignType 是否需要传签名类型参数
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function callPostApi($url, array $data, $isCert = false, $signType = 'HMAC-SHA256', $needSignType = true)
+ {
+ $option = [];
+ if ($isCert) {
+ $option['ssl_p12'] = $this->config->get('ssl_p12');
+ $option['ssl_cer'] = $this->config->get('ssl_cer');
+ $option['ssl_key'] = $this->config->get('ssl_key');
+ if (is_string($option['ssl_p12']) && file_exists($option['ssl_p12'])) {
+ $content = file_get_contents($option['ssl_p12']);
+ if (openssl_pkcs12_read($content, $certs, $this->config->get('mch_id'))) {
+ $option['ssl_key'] = Tools::pushFile(md5($certs['pkey']) . '.pem', $certs['pkey']);
+ $option['ssl_cer'] = Tools::pushFile(md5($certs['cert']) . '.pem', $certs['cert']);
+ } else throw new InvalidArgumentException("P12 certificate does not match MCH_ID --- ssl_p12");
+ }
+ if (empty($option['ssl_cer']) || !file_exists($option['ssl_cer'])) {
+ throw new InvalidArgumentException("Missing Config -- ssl_cer", '0');
+ }
+ if (empty($option['ssl_key']) || !file_exists($option['ssl_key'])) {
+ throw new InvalidArgumentException("Missing Config -- ssl_key", '0');
+ }
+ }
+ $params = $this->params->merge($data);
+ $needSignType && ($params['sign_type'] = strtoupper($signType));
+ $params['sign'] = $this->getPaySign($params, $signType);
+ $result = Tools::xml2arr(Tools::post($url, Tools::arr2xml($params), $option));
+ if ($result['return_code'] !== 'SUCCESS') {
+ throw new InvalidResponseException($result['return_msg'], '0');
+ }
+ return $result;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataArray.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataArray.php
new file mode 100644
index 000000000..c932c7daf
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataArray.php
@@ -0,0 +1,124 @@
+config = $options;
+ }
+
+ /**
+ * 设置配置项值
+ * @param string $offset
+ * @param string|array|null|integer $value
+ */
+ public function set($offset, $value)
+ {
+ $this->offsetSet($offset, $value);
+ }
+
+ /**
+ * 获取配置项参数
+ * @param string|null $offset
+ * @return array|string|null
+ */
+ public function get($offset = null)
+ {
+ return $this->offsetGet($offset);
+ }
+
+ /**
+ * 合并数据到对象
+ * @param array $data 需要合并的数据
+ * @param bool $append 是否追加数据
+ * @return array
+ */
+ public function merge(array $data, $append = false)
+ {
+ if ($append) {
+ return $this->config = array_merge($this->config, $data);
+ }
+ return array_merge($this->config, $data);
+ }
+
+ /**
+ * 设置配置项值
+ * @param string $offset
+ * @param string|array|null|integer $value
+ */
+ public function offsetSet($offset, $value)
+ {
+ if (is_null($offset)) {
+ $this->config[] = $value;
+ } else {
+ $this->config[$offset] = $value;
+ }
+ }
+
+ /**
+ * 判断配置Key是否存在
+ * @param string $offset
+ * @return bool
+ */
+ public function offsetExists($offset)
+ {
+ return isset($this->config[$offset]);
+ }
+
+ /**
+ * 清理配置项
+ * @param string|null $offset
+ */
+ public function offsetUnset($offset = null)
+ {
+ if (is_null($offset)) {
+ $this->config = [];
+ } else {
+ unset($this->config[$offset]);
+ }
+ }
+
+ /**
+ * 获取配置项参数
+ * @param string|null $offset
+ * @return array|string|null
+ */
+ public function offsetGet($offset = null)
+ {
+ if (is_null($offset)) {
+ return $this->config;
+ }
+ return isset($this->config[$offset]) ? $this->config[$offset] : null;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataError.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataError.php
new file mode 100644
index 000000000..51cd9a2ee
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/DataError.php
@@ -0,0 +1,194 @@
+ '系统繁忙,此时请开发者稍候再试',
+ 0 => '请求成功',
+ 40001 => '获取 access_token 时 AppSecret 错误,或者 access_token 无效。请开发者认真比对 AppSecret 的正确性,或查看是否正在为恰当的公众号调用接口',
+ 40002 => '不合法的凭证类型',
+ 40003 => '不合法的 OpenID ,请开发者确认 OpenID (该用户)是否已关注公众号,或是否是其他公众号的 OpenID',
+ 40004 => '不合法的媒体文件类型',
+ 40005 => '不合法的文件类型',
+ 40006 => '不合法的文件大小',
+ 40007 => '不合法的媒体文件 id',
+ 40008 => '不合法的消息类型',
+ 40009 => '不合法的图片文件大小',
+ 40010 => '不合法的语音文件大小',
+ 40011 => '不合法的视频文件大小',
+ 40012 => '不合法的缩略图文件大小',
+ 40013 => '不合法的 AppID ,请开发者检查 AppID 的正确性,避免异常字符,注意大小写',
+ 40014 => '不合法的 access_token ,请开发者认真比对 access_token 的有效性(如是否过期),或查看是否正在为恰当的公众号调用接口',
+ 40015 => '不合法的菜单类型',
+ 40016 => '不合法的按钮个数',
+ 40017 => '不合法的按钮个数',
+ 40018 => '不合法的按钮名字长度',
+ 40019 => '不合法的按钮 KEY 长度',
+ 40020 => '不合法的按钮 URL 长度',
+ 40021 => '不合法的菜单版本号',
+ 40022 => '不合法的子菜单级数',
+ 40023 => '不合法的子菜单按钮个数',
+ 40024 => '不合法的子菜单按钮类型',
+ 40025 => '不合法的子菜单按钮名字长度',
+ 40026 => '不合法的子菜单按钮 KEY 长度',
+ 40027 => '不合法的子菜单按钮 URL 长度',
+ 40028 => '不合法的自定义菜单使用用户',
+ 40029 => '不合法的 oauth_code',
+ 40030 => '不合法的 refresh_token',
+ 40031 => '不合法的 openid 列表',
+ 40032 => '不合法的 openid 列表长度',
+ 40033 => '不合法的请求字符,不能包含 \\uxxxx 格式的字符',
+ 40035 => '不合法的参数',
+ 40038 => '不合法的请求格式',
+ 40039 => '不合法的 URL 长度',
+ 40050 => '不合法的分组 id',
+ 40051 => '分组名字不合法',
+ 40060 => '删除单篇图文时,指定的 article_idx 不合法',
+ 40117 => '分组名字不合法',
+ 40118 => 'media_id 大小不合法',
+ 40119 => 'button 类型错误',
+ 40120 => 'button 类型错误',
+ 40121 => '不合法的 media_id 类型',
+ 40132 => '微信号不合法',
+ 40137 => '不支持的图片格式',
+ 40155 => '请勿添加其他公众号的主页链接',
+ 41001 => '缺少 access_token 参数',
+ 41002 => '缺少 appid 参数',
+ 41003 => '缺少 refresh_token 参数',
+ 41004 => '缺少 secret 参数',
+ 41005 => '缺少多媒体文件数据',
+ 41006 => '缺少 media_id 参数',
+ 41007 => '缺少子菜单数据',
+ 41008 => '缺少 oauth code',
+ 41009 => '缺少 openid',
+ 42001 => 'access_token 超时,请检查 access_token 的有效期,请参考基础支持 - 获取 access_token 中,对 access_token 的详细机制说明',
+ 42002 => 'refresh_token 超时',
+ 42003 => 'oauth_code 超时',
+ 42007 => '用户修改微信密码, accesstoken 和 refreshtoken 失效,需要重新授权',
+ 43001 => '需要 GET 请求',
+ 43002 => '需要 POST 请求',
+ 43003 => '需要 HTTPS 请求',
+ 43004 => '需要接收者关注',
+ 43005 => '需要好友关系',
+ 43019 => '需要将接收者从黑名单中移除',
+ 44001 => '多媒体文件为空',
+ 44002 => 'POST 的数据包为空',
+ 44003 => '图文消息内容为空',
+ 44004 => '文本消息内容为空',
+ 45001 => '多媒体文件大小超过限制',
+ 45002 => '消息内容超过限制',
+ 45003 => '标题字段超过限制',
+ 45004 => '描述字段超过限制',
+ 45005 => '链接字段超过限制',
+ 45006 => '图片链接字段超过限制',
+ 45007 => '语音播放时间超过限制',
+ 45008 => '图文消息超过限制',
+ 45009 => '接口调用超过限制',
+ 45010 => '创建菜单个数超过限制',
+ 45011 => 'API 调用太频繁,请稍候再试',
+ 45015 => '回复时间超过限制',
+ 45016 => '系统分组,不允许修改',
+ 45017 => '分组名字过长',
+ 45018 => '分组数量超过上限',
+ 45047 => '客服接口下行条数超过上限',
+ 46001 => '不存在媒体数据',
+ 46002 => '不存在的菜单版本',
+ 46003 => '不存在的菜单数据',
+ 46004 => '不存在的用户',
+ 47001 => '解析 JSON/XML 内容错误',
+ 48001 => 'api 功能未授权,请确认公众号已获得该接口,可以在公众平台官网 - 开发者中心页中查看接口权限',
+ 48002 => '粉丝拒收消息(粉丝在公众号选项中,关闭了 “ 接收消息 ” )',
+ 48004 => 'api 接口被封禁,请登录 mp.weixin.qq.com 查看详情',
+ 48005 => 'api 禁止删除被自动回复和自定义菜单引用的素材',
+ 48006 => 'api 禁止清零调用次数,因为清零次数达到上限',
+ 48008 => '没有该类型消息的发送权限',
+ 50001 => '用户未授权该 api',
+ 50002 => '用户受限,可能是违规后接口被封禁',
+ 61451 => '参数错误 (invalid parameter)',
+ 61452 => '无效客服账号 (invalid kf_account)',
+ 61453 => '客服帐号已存在 (kf_account exsited)',
+ 61454 => '客服帐号名长度超过限制 ( 仅允许 10 个英文字符,不包括 @ 及 @ 后的公众号的微信号 )(invalid kf_acount length)',
+ 61455 => '客服帐号名包含非法字符 ( 仅允许英文 + 数字 )(illegal character in kf_account)',
+ 61456 => '客服帐号个数超过限制 (10 个客服账号 )(kf_account count exceeded)',
+ 61457 => '无效头像文件类型 (invalid file type)',
+ 61450 => '系统错误 (system error)',
+ 61500 => '日期格式错误',
+ 65301 => '不存在此 menuid 对应的个性化菜单',
+ 65302 => '没有相应的用户',
+ 65303 => '没有默认菜单,不能创建个性化菜单',
+ 65304 => 'MatchRule 信息为空',
+ 65305 => '个性化菜单数量受限',
+ 65306 => '不支持个性化菜单的帐号',
+ 65307 => '个性化菜单信息为空',
+ 65308 => '包含没有响应类型的 button',
+ 65309 => '个性化菜单开关处于关闭状态',
+ 65310 => '填写了省份或城市信息,国家信息不能为空',
+ 65311 => '填写了城市信息,省份信息不能为空',
+ 65312 => '不合法的国家信息',
+ 65313 => '不合法的省份信息',
+ 65314 => '不合法的城市信息',
+ 65316 => '该公众号的菜单设置了过多的域名外跳(最多跳转到 3 个域名的链接)',
+ 65317 => '不合法的 URL',
+ 9001001 => 'POST 数据参数不合法',
+ 9001002 => '远端服务不可用',
+ 9001003 => 'Ticket 不合法',
+ 9001004 => '获取摇周边用户信息失败',
+ 9001005 => '获取商户信息失败',
+ 9001006 => '获取 OpenID 失败',
+ 9001007 => '上传文件缺失',
+ 9001008 => '上传素材的文件类型不合法',
+ 9001009 => '上传素材的文件尺寸不合法',
+ 9001010 => '上传失败',
+ 9001020 => '帐号不合法',
+ 9001021 => '已有设备激活率低于 50% ,不能新增设备',
+ 9001022 => '设备申请数不合法,必须为大于 0 的数字',
+ 9001023 => '已存在审核中的设备 ID 申请',
+ 9001024 => '一次查询设备 ID 数量不能超过 50',
+ 9001025 => '设备 ID 不合法',
+ 9001026 => '页面 ID 不合法',
+ 9001027 => '页面参数不合法',
+ 9001028 => '一次删除页面 ID 数量不能超过 10',
+ 9001029 => '页面已应用在设备中,请先解除应用关系再删除',
+ 9001030 => '一次查询页面 ID 数量不能超过 50',
+ 9001031 => '时间区间不合法',
+ 9001032 => '保存设备与页面的绑定关系参数错误',
+ 9001033 => '门店 ID 不合法',
+ 9001034 => '设备备注信息过长',
+ 9001035 => '设备申请参数不合法',
+ 9001036 => '查询起始值 begin 不合法',
+ ];
+
+ /**
+ * 异常代码解析描述
+ * @param string $code
+ * @return string
+ */
+ public static function toMessage($code)
+ {
+ return isset(self::$message[$code]) ? self::$message[$code] : $code;
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/MyCurlFile.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/MyCurlFile.php
new file mode 100644
index 000000000..e81e3da46
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/MyCurlFile.php
@@ -0,0 +1,63 @@
+ $v) $this->{$k} = $v;
+ } else {
+ $this->mimetype = $mimetype;
+ $this->postname = $postname;
+ $this->extension = pathinfo($filename, PATHINFO_EXTENSION);
+ if (empty($this->extension)) $this->extension = 'tmp';
+ if (empty($this->mimetype)) $this->mimetype = Tools::getExtMine($this->extension);
+ if (empty($this->postname)) $this->postname = pathinfo($filename, PATHINFO_BASENAME);
+ $this->content = base64_encode(file_get_contents($filename));
+ $this->tempname = md5($this->content) . ".{$this->extension}";
+ }
+ }
+
+ /**
+ * 获取文件信息
+ * @return \CURLFile|string
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function get()
+ {
+ $this->filename = Tools::pushFile($this->tempname, base64_decode($this->content));
+ if (class_exists('CURLFile')) {
+ return new \CURLFile($this->filename, $this->mimetype, $this->postname);
+ }
+ return "@{$this->tempname};filename={$this->postname};type={$this->mimetype}";
+ }
+
+ /**
+ * 类销毁处理
+ */
+ public function __destruct()
+ {
+ // Tools::delCache($this->tempname);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/Prpcrypt.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/Prpcrypt.php
new file mode 100644
index 000000000..2b93c5038
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/Prpcrypt.php
@@ -0,0 +1,189 @@
+ PKCS7Encoder::$blockSize) {
+ $pad = 0;
+ }
+ return substr($text, 0, strlen($text) - $pad);
+ }
+
+}
+
+/**
+ * 公众号消息 - 加解密
+ * Class Prpcrypt
+ */
+class Prpcrypt
+{
+
+ public $key;
+
+ /**
+ * Prpcrypt constructor.
+ * @param $key
+ */
+ function __construct($key)
+ {
+ $this->key = base64_decode("{$key}=");
+ }
+
+ /**
+ * 对明文进行加密
+ * @param string $text 需要加密的明文
+ * @param string $appid 公众号APPID
+ * @return array
+ */
+ public function encrypt($text, $appid)
+ {
+ try {
+ $random = $this->getRandomStr();
+ $iv = substr($this->key, 0, 16);
+ $pkcEncoder = new PKCS7Encoder();
+ $text = $pkcEncoder->encode($random . pack("N", strlen($text)) . $text . $appid);
+ $encrypted = openssl_encrypt($text, 'AES-256-CBC', substr($this->key, 0, 32), OPENSSL_ZERO_PADDING, $iv);
+ return [ErrorCode::$OK, $encrypted];
+ } catch (Exception $e) {
+ return [ErrorCode::$EncryptAESError, null];
+ }
+ }
+
+ /**
+ * 对密文进行解密
+ * @param string $encrypted 需要解密的密文
+ * @return array
+ */
+ public function decrypt($encrypted)
+ {
+ try {
+ $iv = substr($this->key, 0, 16);
+ $decrypted = openssl_decrypt($encrypted, 'AES-256-CBC', substr($this->key, 0, 32), OPENSSL_ZERO_PADDING, $iv);
+ } catch (Exception $e) {
+ return [ErrorCode::$DecryptAESError, null];
+ }
+ try {
+ $pkcEncoder = new PKCS7Encoder();
+ $result = $pkcEncoder->decode($decrypted);
+ if (strlen($result) < 16) {
+ return [ErrorCode::$DecryptAESError, null];
+ }
+ $content = substr($result, 16, strlen($result));
+ $len_list = unpack("N", substr($content, 0, 4));
+ $xml_len = $len_list[1];
+ return [0, substr($content, 4, $xml_len), substr($content, $xml_len + 4)];
+ } catch (Exception $e) {
+ return [ErrorCode::$IllegalBuffer, null];
+ }
+ }
+
+ /**
+ * 随机生成16位字符串
+ * @param string $str
+ * @return string 生成的字符串
+ */
+ function getRandomStr($str = "")
+ {
+ $str_pol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
+ $max = strlen($str_pol) - 1;
+ for ($i = 0; $i < 16; $i++) {
+ $str .= $str_pol[mt_rand(0, $max)];
+ }
+ return $str;
+ }
+
+}
+
+/**
+ * 仅用作类内部使用
+ * 不用于官方API接口的errCode码
+ * Class ErrorCode
+ */
+class ErrorCode
+{
+
+ public static $OK = 0;
+ public static $ParseXmlError = 40002;
+ public static $IllegalAesKey = 40004;
+ public static $IllegalBuffer = 40008;
+ public static $EncryptAESError = 40006;
+ public static $DecryptAESError = 40007;
+ public static $EncodeBase64Error = 40009;
+ public static $DecodeBase64Error = 40010;
+ public static $GenReturnXmlError = 40011;
+ public static $ValidateAppidError = 40005;
+ public static $ComputeSignatureError = 40003;
+ public static $ValidateSignatureError = 40001;
+ public static $errCode = [
+ '0' => '处理成功',
+ '40001' => '校验签名失败',
+ '40002' => '解析xml失败',
+ '40003' => '计算签名失败',
+ '40004' => '不合法的AESKey',
+ '40005' => '校验AppID失败',
+ '40006' => 'AES加密失败',
+ '40007' => 'AES解密失败',
+ '40008' => '公众平台发送的xml不合法',
+ '40009' => 'Base64编码失败',
+ '40010' => 'Base64解码失败',
+ '40011' => '公众帐号生成回包xml失败',
+ ];
+
+ /**
+ * 获取错误消息内容
+ * @param string $code 错误代码
+ * @return bool
+ */
+ public static function getErrText($code)
+ {
+ if (isset(self::$errCode[$code])) {
+ return self::$errCode[$code];
+ }
+ return false;
+ }
+
+}
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Contracts/Tools.php b/vendor/zoujingli/wechat-developer/WeChat/Contracts/Tools.php
new file mode 100644
index 000000000..60a435d5c
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Contracts/Tools.php
@@ -0,0 +1,450 @@
+ null, // 写入缓存
+ 'get' => null, // 获取缓存
+ 'del' => null, // 删除缓存
+ 'put' => null, // 写入文件
+ ];
+
+ /**
+ * 网络缓存
+ * @var array
+ */
+ private static $cache_curl = [];
+
+ /**
+ * 产生随机字符串
+ * @param int $length 指定字符长度
+ * @param string $str 字符串前缀
+ * @return string
+ */
+ public static function createNoncestr($length = 32, $str = "")
+ {
+ $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
+ for ($i = 0; $i < $length; $i++) {
+ $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
+ }
+ return $str;
+ }
+
+
+ /**
+ * 根据文件后缀获取文件类型
+ * @param string|array $ext 文件后缀
+ * @param array $mine 文件后缀MINE信息
+ * @return string
+ * @throws LocalCacheException
+ */
+ public static function getExtMine($ext, $mine = [])
+ {
+ $mines = self::getMines();
+ foreach (is_string($ext) ? explode(',', $ext) : $ext as $e) {
+ $mine[] = isset($mines[strtolower($e)]) ? $mines[strtolower($e)] : 'application/octet-stream';
+ }
+ return join(',', array_unique($mine));
+ }
+
+ /**
+ * 获取所有文件扩展的类型
+ * @return array
+ * @throws LocalCacheException
+ */
+ private static function getMines()
+ {
+ $mines = self::getCache('all_ext_mine');
+ if (empty($mines)) {
+ $content = file_get_contents('http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types');
+ preg_match_all('#^([^\s]{2,}?)\s+(.+?)$#ism', $content, $matches, PREG_SET_ORDER);
+ foreach ($matches as $match) foreach (explode(" ", $match[2]) as $ext) $mines[$ext] = $match[1];
+ self::setCache('all_ext_mine', $mines);
+ }
+ return $mines;
+ }
+
+ /**
+ * 创建CURL文件对象
+ * @param $filename
+ * @param string $mimetype
+ * @param string $postname
+ * @return \CURLFile|string
+ * @throws LocalCacheException
+ */
+ public static function createCurlFile($filename, $mimetype = null, $postname = null)
+ {
+ if (is_string($filename) && file_exists($filename)) {
+ if (is_null($postname)) $postname = basename($filename);
+ if (is_null($mimetype)) $mimetype = self::getExtMine(pathinfo($filename, 4));
+ if (function_exists('curl_file_create')) {
+ return curl_file_create($filename, $mimetype, $postname);
+ }
+ return "@{$filename};filename={$postname};type={$mimetype}";
+ }
+ return $filename;
+ }
+
+ /**
+ * 数组转XML内容
+ * @param array $data
+ * @return string
+ */
+ public static function arr2xml($data)
+ {
+ return "" . self::_arr2xml($data) . "";
+ }
+
+ /**
+ * XML内容生成
+ * @param array $data 数据
+ * @param string $content
+ * @return string
+ */
+ private static function _arr2xml($data, $content = '')
+ {
+ foreach ($data as $key => $val) {
+ is_numeric($key) && $key = 'item';
+ $content .= "<{$key}>";
+ if (is_array($val) || is_object($val)) {
+ $content .= self::_arr2xml($val);
+ } elseif (is_string($val)) {
+ $content .= '';
+ } else {
+ $content .= $val;
+ }
+ $content .= "{$key}>";
+ }
+ return $content;
+ }
+
+ /**
+ * 解析XML内容到数组
+ * @param string $xml
+ * @return array
+ */
+ public static function xml2arr($xml)
+ {
+ $entity = libxml_disable_entity_loader(true);
+ $data = (array)simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
+ libxml_disable_entity_loader($entity);
+ return json_decode(json_encode($data), true);
+ }
+
+ /**
+ * 数组转xml内容
+ * @param array $data
+ * @return null|string|string
+ */
+ public static function arr2json($data)
+ {
+ $json = json_encode(self::buildEnEmojiData($data), JSON_UNESCAPED_UNICODE);
+ return $json === '[]' ? '{}' : $json;
+ }
+
+ /**
+ * 数组对象Emoji编译处理
+ * @param array $data
+ * @return array
+ */
+ public static function buildEnEmojiData(array $data)
+ {
+ foreach ($data as $key => $value) {
+ if (is_array($value)) {
+ $data[$key] = self::buildEnEmojiData($value);
+ } elseif (is_string($value)) {
+ $data[$key] = self::emojiEncode($value);
+ } else {
+ $data[$key] = $value;
+ }
+ }
+ return $data;
+ }
+
+ /**
+ * 数组对象Emoji反解析处理
+ * @param array $data
+ * @return array
+ */
+ public static function buildDeEmojiData(array $data)
+ {
+ foreach ($data as $key => $value) {
+ if (is_array($value)) {
+ $data[$key] = self::buildDeEmojiData($value);
+ } elseif (is_string($value)) {
+ $data[$key] = self::emojiDecode($value);
+ } else {
+ $data[$key] = $value;
+ }
+ }
+ return $data;
+ }
+
+ /**
+ * Emoji原形转换为String
+ * @param string $content
+ * @return string
+ */
+ public static function emojiEncode($content)
+ {
+ return json_decode(preg_replace_callback("/(\\\u[ed][0-9a-f]{3})/i", function ($string) {
+ return addslashes($string[0]);
+ }, json_encode($content)));
+ }
+
+ /**
+ * Emoji字符串转换为原形
+ * @param string $content
+ * @return string
+ */
+ public static function emojiDecode($content)
+ {
+ return json_decode(preg_replace_callback('/\\\\\\\\/i', function () {
+ return '\\';
+ }, json_encode($content)));
+ }
+
+ /**
+ * 解析JSON内容到数组
+ * @param string $json
+ * @return array
+ * @throws InvalidResponseException
+ */
+ public static function json2arr($json)
+ {
+ $result = json_decode($json, true);
+ if (empty($result)) {
+ throw new InvalidResponseException('invalid response.', '0');
+ }
+ if (!empty($result['errcode'])) {
+ throw new InvalidResponseException($result['errmsg'], $result['errcode'], $result);
+ }
+ return $result;
+ }
+
+ /**
+ * 以get访问模拟访问
+ * @param string $url 访问URL
+ * @param array $query GET数
+ * @param array $options
+ * @return boolean|string
+ * @throws LocalCacheException
+ */
+ public static function get($url, $query = [], $options = [])
+ {
+ $options['query'] = $query;
+ return self::doRequest('get', $url, $options);
+ }
+
+ /**
+ * 以post访问模拟访问
+ * @param string $url 访问URL
+ * @param array $data POST数据
+ * @param array $options
+ * @return boolean|string
+ * @throws LocalCacheException
+ */
+ public static function post($url, $data = [], $options = [])
+ {
+ $options['data'] = $data;
+ return self::doRequest('post', $url, $options);
+ }
+
+ /**
+ * CURL模拟网络请求
+ * @param string $method 请求方法
+ * @param string $url 请求方法
+ * @param array $options 请求参数[headers,data,ssl_cer,ssl_key]
+ * @return boolean|string
+ * @throws LocalCacheException
+ */
+ public static function doRequest($method, $url, $options = [])
+ {
+ $curl = curl_init();
+ // GET参数设置
+ if (!empty($options['query'])) {
+ $url .= (stripos($url, '?') !== false ? '&' : '?') . http_build_query($options['query']);
+ }
+ // CURL头信息设置
+ if (!empty($options['headers'])) {
+ curl_setopt($curl, CURLOPT_HTTPHEADER, $options['headers']);
+ }
+ // POST数据设置
+ if (strtolower($method) === 'post') {
+ curl_setopt($curl, CURLOPT_POST, true);
+ curl_setopt($curl, CURLOPT_POSTFIELDS, self::_buildHttpData($options['data']));
+ }
+ // 证书文件设置
+ if (!empty($options['ssl_cer'])) if (file_exists($options['ssl_cer'])) {
+ curl_setopt($curl, CURLOPT_SSLCERTTYPE, 'PEM');
+ curl_setopt($curl, CURLOPT_SSLCERT, $options['ssl_cer']);
+ } else throw new InvalidArgumentException("Certificate files that do not exist. --- [ssl_cer]");
+ // 证书文件设置
+ if (!empty($options['ssl_key'])) if (file_exists($options['ssl_key'])) {
+ curl_setopt($curl, CURLOPT_SSLKEYTYPE, 'PEM');
+ curl_setopt($curl, CURLOPT_SSLKEY, $options['ssl_key']);
+ } else throw new InvalidArgumentException("Certificate files that do not exist. --- [ssl_key]");
+ curl_setopt($curl, CURLOPT_URL, $url);
+ curl_setopt($curl, CURLOPT_TIMEOUT, 60);
+ curl_setopt($curl, CURLOPT_HEADER, false);
+ curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
+ curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
+ curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, false);
+ list($content) = [curl_exec($curl), curl_close($curl)];
+ // 清理 CURL 缓存文件
+ if (!empty(self::$cache_curl)) foreach (self::$cache_curl as $key => $file) {
+ Tools::delCache($file);
+ unset(self::$cache_curl[$key]);
+ }
+ return $content;
+ }
+
+ /**
+ * POST数据过滤处理
+ * @param array $data 需要处理的数据
+ * @param boolean $build 是否编译数据
+ * @return array|string
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ private static function _buildHttpData($data, $build = true)
+ {
+ if (!is_array($data)) return $data;
+ foreach ($data as $key => $value) if (is_object($value) && $value instanceof \CURLFile) {
+ $build = false;
+ } elseif (is_object($value) && isset($value->datatype) && $value->datatype === 'MY_CURL_FILE') {
+ $build = false;
+ $mycurl = new MyCurlFile((array)$value);
+ $data[$key] = $mycurl->get();
+ array_push(self::$cache_curl, $mycurl->tempname);
+ } elseif (is_string($value) && class_exists('CURLFile', false) && stripos($value, '@') === 0) {
+ if (($filename = realpath(trim($value, '@'))) && file_exists($filename)) {
+ $build = false;
+ $data[$key] = self::createCurlFile($filename);
+ }
+ }
+ return $build ? http_build_query($data) : $data;
+ }
+
+ /**
+ * 写入文件
+ * @param string $name 文件名称
+ * @param string $content 文件内容
+ * @return string
+ * @throws LocalCacheException
+ */
+ public static function pushFile($name, $content)
+ {
+ if (is_callable(self::$cache_callable['put'])) {
+ return call_user_func_array(self::$cache_callable['put'], func_get_args());
+ }
+ $file = self::_getCacheName($name);
+ if (!file_put_contents($file, $content)) {
+ throw new LocalCacheException('local file write error.', '0');
+ }
+ return $file;
+ }
+
+ /**
+ * 缓存配置与存储
+ * @param string $name 缓存名称
+ * @param string $value 缓存内容
+ * @param int $expired 缓存时间(0表示永久缓存)
+ * @return string
+ * @throws LocalCacheException
+ */
+ public static function setCache($name, $value = '', $expired = 3600)
+ {
+ if (is_callable(self::$cache_callable['set'])) {
+ return call_user_func_array(self::$cache_callable['set'], func_get_args());
+ }
+ $file = self::_getCacheName($name);
+ $data = ['name' => $name, 'value' => $value, 'expired' => time() + intval($expired)];
+ if (!file_put_contents($file, serialize($data))) {
+ throw new LocalCacheException('local cache error.', '0');
+ }
+ return $file;
+ }
+
+ /**
+ * 获取缓存内容
+ * @param string $name 缓存名称
+ * @return null|mixed
+ */
+ public static function getCache($name)
+ {
+ if (is_callable(self::$cache_callable['get'])) {
+ return call_user_func_array(self::$cache_callable['get'], func_get_args());
+ }
+ $file = self::_getCacheName($name);
+ if (file_exists($file) && ($content = file_get_contents($file))) {
+ $data = unserialize($content);
+ if (isset($data['expired']) && (intval($data['expired']) === 0 || intval($data['expired']) >= time())) {
+ return $data['value'];
+ }
+ self::delCache($name);
+ }
+ return null;
+ }
+
+ /**
+ * 移除缓存文件
+ * @param string $name 缓存名称
+ * @return boolean
+ */
+ public static function delCache($name)
+ {
+ if (is_callable(self::$cache_callable['del'])) {
+ return call_user_func_array(self::$cache_callable['del'], func_get_args());
+ }
+ $file = self::_getCacheName($name);
+ return file_exists($file) ? unlink($file) : true;
+ }
+
+ /**
+ * 应用缓存目录
+ * @param string $name
+ * @return string
+ */
+ private static function _getCacheName($name)
+ {
+ if (empty(self::$cache_path)) {
+ self::$cache_path = dirname(dirname(__DIR__)) . DIRECTORY_SEPARATOR . 'Cache' . DIRECTORY_SEPARATOR;
+ }
+ self::$cache_path = rtrim(self::$cache_path, '/\\') . DIRECTORY_SEPARATOR;
+ file_exists(self::$cache_path) || mkdir(self::$cache_path, 0755, true);
+ return self::$cache_path . $name;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Custom.php b/vendor/zoujingli/wechat-developer/WeChat/Custom.php
new file mode 100644
index 000000000..6d0309140
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Custom.php
@@ -0,0 +1,247 @@
+ $kf_account, 'nickname' => $nickname];
+ $url = "https://api.weixin.qq.com/customservice/kfaccount/add?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 修改客服帐号
+ * @param string $kf_account 客服账号
+ * @param string $nickname 客服昵称
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateAccount($kf_account, $nickname)
+ {
+ $data = ['kf_account' => $kf_account, 'nickname' => $nickname];
+ $url = "https://api.weixin.qq.com/customservice/kfaccount/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 删除客服帐号
+ * @param string $kf_account 客服账号
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function deleteAccount($kf_account)
+ {
+ $data = ['kf_account' => $kf_account];
+ $url = "https://api.weixin.qq.com/customservice/kfaccount/del?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 邀请绑定客服帐号
+ * @param string $kf_account 完整客服帐号,格式为:帐号前缀@公众号微信号
+ * @param string $invite_wx 接收绑定邀请的客服微信号
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function inviteWorker($kf_account, $invite_wx)
+ {
+ $url = 'https://api.weixin.qq.com/customservice/kfaccount/inviteworker?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['kf_account' => $kf_account, 'invite_wx' => $invite_wx]);
+ }
+
+ /**
+ * 获取所有客服账号
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getAccountList()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/customservice/getkflist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 设置客服帐号的头像
+ * @param string $kf_account 客户账号
+ * @param string $image 头像文件位置
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function uploadHeadimg($kf_account, $image)
+ {
+ $url = "http://api.weixin.qq.com/customservice/kfaccount/uploadheadimg?access_token=ACCESS_TOKEN&kf_account={$kf_account}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media' => Tools::createCurlFile($image)]);
+ }
+
+ /**
+ * 客服接口-发消息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function send(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/custom/send?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 客服输入状态
+ * @param string $openid 普通用户(openid)
+ * @param string $command Typing:正在输入,CancelTyping:取消正在输入
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function typing($openid, $command = 'Typing')
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/custom/typing?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['touser' => $openid, 'command' => $command]);
+ }
+
+ /**
+ * 根据标签进行群发【订阅号与服务号认证后均可用】
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massSendAll(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/sendall?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 根据OpenID列表群发【订阅号不可用,服务号认证后可用】
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massSend(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/send?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 删除群发【订阅号与服务号认证后均可用】
+ * @param integer $msg_id 发送出去的消息ID
+ * @param null|integer $article_idx 要删除的文章在图文消息中的位置,第一篇编号为1,该字段不填或填0会删除全部文章
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massDelete($msg_id, $article_idx = null)
+ {
+ $data = ['msg_id' => $msg_id];
+ is_null($article_idx) || $data['article_idx'] = $article_idx;
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/delete?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 预览接口【订阅号与服务号认证后均可用】
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massPreview(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/preview?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询群发消息发送状态【订阅号与服务号认证后均可用】
+ * @param integer $msg_id 群发消息后返回的消息id
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massGet($msg_id)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['msg_id' => $msg_id]);
+ }
+
+ /**
+ * 获取群发速度
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massGetSeed()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/speed/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, []);
+ }
+
+ /**
+ * 设置群发速度
+ * @param integer $speed 群发速度的级别
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function massSetSeed($speed)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/mass/speed/set?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['speed' => $speed]);
+ }
+
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidArgumentException.php b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidArgumentException.php
new file mode 100644
index 000000000..993c4e008
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidArgumentException.php
@@ -0,0 +1,40 @@
+raw = $raw;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidDecryptException.php b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidDecryptException.php
new file mode 100644
index 000000000..f45c0581e
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidDecryptException.php
@@ -0,0 +1,40 @@
+raw = $raw;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidInstanceException.php b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidInstanceException.php
new file mode 100644
index 000000000..e4c6816f8
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidInstanceException.php
@@ -0,0 +1,40 @@
+raw = $raw;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidResponseException.php b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidResponseException.php
new file mode 100644
index 000000000..c3cb89267
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/InvalidResponseException.php
@@ -0,0 +1,41 @@
+raw = $raw;
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Exceptions/LocalCacheException.php b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/LocalCacheException.php
new file mode 100644
index 000000000..35be83a8d
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Exceptions/LocalCacheException.php
@@ -0,0 +1,42 @@
+raw = $raw;
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Limit.php b/vendor/zoujingli/wechat-developer/WeChat/Limit.php
new file mode 100644
index 000000000..6ccc6f488
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Limit.php
@@ -0,0 +1,68 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['appid' => $this->config->get('appid')]);
+ }
+
+ /**
+ * 网络检测
+ * @param string $action 执行的检测动作
+ * @param string $operator 指定平台从某个运营商进行检测
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function ping($action = 'all', $operator = 'DEFAULT')
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/callback/check?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['action' => $action, 'check_operator' => $operator]);
+ }
+
+ /**
+ * 获取微信服务器IP地址
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getCallbackIp()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/getcallbackip?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Media.php b/vendor/zoujingli/wechat-developer/WeChat/Media.php
new file mode 100644
index 000000000..8c7ca02d2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Media.php
@@ -0,0 +1,201 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media' => Tools::createCurlFile($filename)], false);
+ }
+
+ /**
+ * 获取临时素材
+ * @param string $media_id
+ * @param string $outType 返回处理函数
+ * @return array|string
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function get($media_id, $outType = null)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/media/get?access_token=ACCESS_TOKEN&media_id={$media_id}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = Tools::get($url);
+ if (is_array($json = json_decode($result, true))) {
+ if (!$this->isTry && isset($json['errcode']) && in_array($json['errcode'], ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+ /**
+ * 新增图文素材
+ * @param array $data 文件名称
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function addNews($data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/material/add_news?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 更新图文素材
+ * @param string $media_id 要修改的图文消息的id
+ * @param int $index 要更新的文章在图文消息中的位置(多图文消息时,此字段才有意义),第一篇为0
+ * @param array $news 文章内容
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function updateNews($media_id, $index, $news)
+ {
+ $data = ['media_id' => $media_id, 'index' => $index, 'articles' => $news];
+ $url = "https://api.weixin.qq.com/cgi-bin/material/update_news?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 上传图文消息内的图片获取URL
+ * @param string $filename
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function uploadImg($filename)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/media/uploadimg?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media' => Tools::createCurlFile($filename)], false);
+ }
+
+ /**
+ * 新增其他类型永久素材
+ * @param string $filename 文件名称
+ * @param string $type 媒体文件类型(image|voice|video|thumb)
+ * @param array $description 包含素材的描述信息
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function addMaterial($filename, $type = 'image', $description = [])
+ {
+ if (!in_array($type, ['image', 'voice', 'video', 'thumb'])) {
+ throw new InvalidResponseException('Invalid Media Type.', '0');
+ }
+ $url = "https://api.weixin.qq.com/cgi-bin/material/add_material?access_token=ACCESS_TOKEN&type={$type}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media' => Tools::createCurlFile($filename), 'description' => Tools::arr2json($description)], false);
+ }
+
+ /**
+ * 获取永久素材
+ * @param string $media_id
+ * @param null|string $outType 输出类型
+ * @return array|string
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function getMaterial($media_id, $outType = null)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/material/get_material?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = Tools::post($url, ['media_id' => $media_id]);
+ if (is_array($json = json_decode($result, true))) {
+ if (!$this->isTry && isset($json['errcode']) && in_array($json['errcode'], ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+ /**
+ * 删除永久素材
+ * @param string $media_id
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function delMaterial($media_id)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/material/del_material?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media_id' => $media_id]);
+ }
+
+ /**
+ * 获取素材总数
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function getMaterialCount()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/material/get_materialcount?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 获取素材列表
+ * @param string $type
+ * @param int $offset
+ * @param int $count
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function batchGetMaterial($type = 'image', $offset = 0, $count = 20)
+ {
+ if (!in_array($type, ['image', 'voice', 'video', 'news'])) {
+ throw new InvalidResponseException('Invalid Media Type.', '0');
+ }
+ $url = "https://api.weixin.qq.com/cgi-bin/material/batchget_material?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['type' => $type, 'offset' => $offset, 'count' => $count]);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Menu.php b/vendor/zoujingli/wechat-developer/WeChat/Menu.php
new file mode 100644
index 000000000..c006a8eae
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Menu.php
@@ -0,0 +1,109 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 自定义菜单删除接口
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function delete()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/menu/delete?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 自定义菜单创建
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function create(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/menu/create?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 创建个性化菜单
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addConditional(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/menu/addconditional?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 删除个性化菜单
+ * @param string $menuid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function delConditional($menuid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/menu/delconditional?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['menuid' => $menuid]);
+ }
+
+ /**
+ * 测试个性化菜单匹配结果
+ * @param string $openid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function tryConditional($openid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/menu/trymatch?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['user_id' => $openid]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Oauth.php b/vendor/zoujingli/wechat-developer/WeChat/Oauth.php
new file mode 100644
index 000000000..a8be2729e
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Oauth.php
@@ -0,0 +1,99 @@
+config->get('appid');
+ $redirect_uri = urlencode($redirect_url);
+ return "https://open.weixin.qq.com/connect/oauth2/authorize?appid={$appid}&redirect_uri={$redirect_uri}&response_type=code&scope={$scope}&state={$state}#wechat_redirect";
+ }
+
+ /**
+ * 通过 code 获取 AccessToken 和 openid
+ * @return bool|array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getOauthAccessToken()
+ {
+ $appid = $this->config->get('appid');
+ $appsecret = $this->config->get('appsecret');
+ $code = isset($_GET['code']) ? $_GET['code'] : '';
+ $url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid={$appid}&secret={$appsecret}&code={$code}&grant_type=authorization_code";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 刷新AccessToken并续期
+ * @param string $refresh_token
+ * @return bool|array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getOauthRefreshToken($refresh_token)
+ {
+ $appid = $this->config->get('appid');
+ $url = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid={$appid}&grant_type=refresh_token&refresh_token={$refresh_token}";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 检验授权凭证(access_token)是否有效
+ * @param string $access_token 网页授权接口调用凭证,注意:此access_token与基础支持的access_token不同
+ * @param string $openid 用户的唯一标识
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function checkOauthAccessToken($access_token, $openid)
+ {
+ $url = "https://api.weixin.qq.com/sns/auth?access_token={$access_token}&openid={$openid}";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 拉取用户信息(需scope为 snsapi_userinfo)
+ * @param string $access_token 网页授权接口调用凭证,注意:此access_token与基础支持的access_token不同
+ * @param string $openid 用户的唯一标识
+ * @param string $lang 返回国家地区语言版本,zh_CN 简体,zh_TW 繁体,en 英语
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getUserInfo($access_token, $openid, $lang = 'zh_CN')
+ {
+ $url = "https://api.weixin.qq.com/sns/userinfo?access_token={$access_token}&openid={$openid}&lang={$lang}";
+ return $this->httpGetForJson($url);
+ }
+
+}
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Pay.php b/vendor/zoujingli/wechat-developer/WeChat/Pay.php
new file mode 100644
index 000000000..dcc87cede
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Pay.php
@@ -0,0 +1,232 @@
+config->get())->create($options);
+ }
+
+ /**
+ * 刷卡支付
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function createMicropay($options)
+ {
+ return Order::instance($this->config->get())->micropay($options);
+ }
+
+ /**
+ * 创建JsApi及H5支付参数
+ * @param string $prepay_id 统一下单预支付码
+ * @return array
+ */
+ public function createParamsForJsApi($prepay_id)
+ {
+ return Order::instance($this->config->get())->jsapiParams($prepay_id);
+ }
+
+ /**
+ * 获取APP支付参数
+ * @param string $prepay_id 统一下单预支付码
+ * @return array
+ */
+ public function createParamsForApp($prepay_id)
+ {
+ return Order::instance($this->config->get())->appParams($prepay_id);
+ }
+
+ /**
+ * 获取支付规则二维码
+ * @param string $product_id 商户定义的商品id 或者订单号
+ * @return string
+ */
+ public function createParamsForRuleQrc($product_id)
+ {
+ return Order::instance($this->config->get())->qrcParams($product_id);
+ }
+
+ /**
+ * 查询订单
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function queryOrder(array $options)
+ {
+ return Order::instance($this->config->get())->query($options);
+ }
+
+ /**
+ * 关闭订单
+ * @param string $out_trade_no 商户订单号
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function closeOrder($out_trade_no)
+ {
+ return Order::instance($this->config->get())->close($out_trade_no);
+ }
+
+ /**
+ * 申请退款
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function createRefund(array $options)
+ {
+ return Refund::instance($this->config->get())->create($options);
+ }
+
+ /**
+ * 查询退款
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function queryRefund(array $options)
+ {
+ return Refund::instance($this->config->get())->query($options);
+ }
+
+ /**
+ * 交易保障
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function report(array $options)
+ {
+ return Order::instance($this->config->get())->report($options);
+ }
+
+ /**
+ * 授权码查询openid
+ * @param string $authCode 扫码支付授权码,设备读取用户微信中的条码或者二维码信息
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function queryAuthCode($authCode)
+ {
+ return Order::instance($this->config->get())->queryAuthCode($authCode);
+ }
+
+ /**
+ * 下载对账单
+ * @param array $options 静音参数
+ * @param null|string $outType 输出类型
+ * @return bool|string
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function billDownload(array $options, $outType = null)
+ {
+ return Bill::instance($this->config->get())->download($options, $outType);
+ }
+
+ /**
+ * 拉取订单评价数据
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function billCommtent(array $options)
+ {
+ return Bill::instance($this->config->get())->comment($options);
+ }
+
+ /**
+ * 企业付款到零钱
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function createTransfers(array $options)
+ {
+ return Transfers::instance($this->config->get())->create($options);
+ }
+
+ /**
+ * 查询企业付款到零钱
+ * @param string $partner_trade_no 商户调用企业付款API时使用的商户订单号
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function queryTransfers($partner_trade_no)
+ {
+ return Transfers::instance($this->config->get())->query($partner_trade_no);
+ }
+
+ /**
+ * 企业付款到银行卡
+ * @param array $options
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws Exceptions\InvalidDecryptException
+ * @throws Exceptions\InvalidResponseException
+ */
+ public function createTransfersBank(array $options)
+ {
+ return TransfersBank::instance($this->config->get())->create($options);
+ }
+
+ /**
+ * 商户企业付款到银行卡操作进行结果查询
+ * @param string $partner_trade_no 商户订单号,需保持唯一
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function queryTransFresBank($partner_trade_no)
+ {
+ return TransfersBank::instance($this->config->get())->query($partner_trade_no);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Product.php b/vendor/zoujingli/wechat-developer/WeChat/Product.php
new file mode 100644
index 000000000..74eb3eed4
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Product.php
@@ -0,0 +1,177 @@
+ $keystandard, 'keystr' => $keystr, 'status' => $status];
+ $url = "https://api.weixin.qq.com/scan/product/modstatus?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 设置测试人员白名单
+ * @param array $openids 测试人员的openid列表
+ * @param array $usernames 测试人员的微信号列表
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setTestWhiteList(array $openids = [], array $usernames = [])
+ {
+ $data = ['openid' => $openids, 'username' => $usernames];
+ $url = "https://api.weixin.qq.com/scan/testwhitelist/set?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 获取商品二维码
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @param integer $qrcode_size 二维码的尺寸(整型),数值代表边长像素数,不填写默认值为100
+ * @param array $extinfo 由商户自定义传入,建议仅使用大小写字母、数字及-_().*这6个常用字符
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getQrcode($keystandard, $keystr, $qrcode_size, $extinfo = [])
+ {
+ $data = ['keystandard' => $keystandard, 'keystr' => $keystr, 'qrcode_size' => $qrcode_size];
+ empty($extinfo) || $data['extinfo'] = $extinfo;
+ $url = "https://api.weixin.qq.com/scan/product/getqrcode?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询商品信息
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getProduct($keystandard, $keystr)
+ {
+ $data = ['keystandard' => $keystandard, 'keystr' => $keystr];
+ empty($extinfo) || $data['extinfo'] = $extinfo;
+ $url = "https://api.weixin.qq.com/scan/product/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 批量查询商品信息
+ * @param integer $offset 批量查询的起始位置,从0开始,包含该起始位置
+ * @param integer $limit 批量查询的数量
+ * @param null|string $status 支持按状态拉取。on为发布状态,off为未发布状态,check为审核中状态,reject为审核未通过状态,all为所有状态
+ * @param string $keystr 支持按部分编码内容拉取。填写该参数后,可将编码内容中包含所传参数的商品信息拉出。类似关键词搜索
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getProductList($offset, $limit = 10, $status = null, $keystr = '')
+ {
+ $data = ['offset' => $offset, 'limit' => $limit];
+ is_null($status) || $data['status'] = $status;
+ empty($keystr) || $data['keystr'] = $keystr;
+ $url = "https://api.weixin.qq.com/scan/product/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+
+ /**
+ * 更新商品信息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateProduct(array $data)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 清除商品信息
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function clearProduct($keystandard, $keystr)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/clear?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['keystandard' => $keystandard, 'keystr' => $keystr]);
+ }
+
+
+ /**
+ * 检查wxticket参数
+ * @param string $ticket
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function scanTicketCheck($ticket)
+ {
+ $url = "https://api.weixin.qq.com/scan/scanticket/check?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['ticket' => $ticket]);
+ }
+
+ /**
+ * 清除扫码记录
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @param string $extinfo 调用“获取商品二维码接口”时传入的extinfo,为标识参数
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function clearScanticket($keystandard, $keystr, $extinfo)
+ {
+ $data = ['keystandard' => $keystandard, 'keystr' => $keystr, 'extinfo' => $extinfo];
+ $url = "https://api.weixin.qq.com/scan/scanticket/check?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Qrcode.php b/vendor/zoujingli/wechat-developer/WeChat/Qrcode.php
new file mode 100644
index 000000000..3b6251101
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Qrcode.php
@@ -0,0 +1,77 @@
+ ['scene' => ['scene_id' => $scene]]];
+ } else {
+ $data = ['action_info' => ['scene' => ['scene_str' => $scene]]];
+ }
+ if ($expire_seconds > 0) { // 临时二维码
+ $data['expire_seconds'] = $expire_seconds;
+ $data['action_name'] = is_integer($scene) ? 'QR_SCENE' : 'QR_STR_SCENE';
+ } else { // 永久二维码
+ $data['action_name'] = is_integer($scene) ? 'QR_LIMIT_SCENE' : 'QR_LIMIT_STR_SCENE';
+ }
+ $url = "https://api.weixin.qq.com/cgi-bin/qrcode/create?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 通过ticket换取二维码
+ * @param string $ticket 获取的二维码ticket,凭借此ticket可以在有效时间内换取二维码。
+ * @return string
+ */
+ public function url($ticket)
+ {
+ return "https://mp.weixin.qq.com/cgi-bin/showqrcode?ticket={$ticket}";
+ }
+
+ /**
+ * 长链接转短链接接口
+ * @param string $longUrl 需要转换的长链接
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function shortUrl($longUrl)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/shorturl?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['action' => 'long2short', 'long_url' => $longUrl]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Receive.php b/vendor/zoujingli/wechat-developer/WeChat/Receive.php
new file mode 100644
index 000000000..7f4d07006
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Receive.php
@@ -0,0 +1,164 @@
+message = [
+ 'CreateTime' => time(),
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'MsgType' => 'transfer_customer_service',
+ ];
+ empty($account) || $this->message['TransInfo'] = ['KfAccount' => $account];
+ return $this;
+ }
+
+ /**
+ * 设置文本消息
+ * @param string $content 文本内容
+ * @return $this
+ */
+ public function text($content = '')
+ {
+ $this->message = [
+ 'MsgType' => 'text',
+ 'CreateTime' => time(),
+ 'Content' => $content,
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ ];
+ return $this;
+ }
+
+ /**
+ * 设置回复图文
+ * @param array $newsData
+ * @return $this
+ */
+ public function news($newsData = [])
+ {
+ $this->message = [
+ 'CreateTime' => time(),
+ 'MsgType' => 'news',
+ 'Articles' => $newsData,
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'ArticleCount' => count($newsData),
+ ];
+ return $this;
+ }
+
+ /**
+ * 设置图片消息
+ * @param string $mediaId 图片媒体ID
+ * @return $this
+ */
+ public function image($mediaId = '')
+ {
+ $this->message = [
+ 'MsgType' => 'image',
+ 'CreateTime' => time(),
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'Image' => ['MediaId' => $mediaId],
+ ];
+ return $this;
+ }
+
+ /**
+ * 设置语音回复消息
+ * @param string $mediaid 语音媒体ID
+ * @return $this
+ */
+ public function voice($mediaid = '')
+ {
+ $this->message = [
+ 'CreateTime' => time(),
+ 'MsgType' => 'voice',
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'Voice' => ['MediaId' => $mediaid],
+ ];
+ return $this;
+ }
+
+ /**
+ * 设置视频回复消息
+ * @param string $mediaid 视频媒体ID
+ * @param string $title 视频标题
+ * @param string $description 视频描述
+ * @return $this
+ */
+ public function video($mediaid = '', $title = '', $description = '')
+ {
+ $this->message = [
+ 'CreateTime' => time(),
+ 'MsgType' => 'video',
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'Video' => [
+ 'Title' => $title,
+ 'MediaId' => $mediaid,
+ 'Description' => $description,
+ ],
+ ];
+ return $this;
+ }
+
+ /**
+ * 设置音乐回复消息
+ * @param string $title 音乐标题
+ * @param string $desc 音乐描述
+ * @param string $musicurl 音乐地址
+ * @param string $hgmusicurl 高清音乐地址
+ * @param string $thumbmediaid 音乐图片缩略图的媒体id(可选)
+ * @return $this
+ */
+ public function music($title, $desc, $musicurl, $hgmusicurl = '', $thumbmediaid = '')
+ {
+ $this->message = [
+ 'CreateTime' => time(),
+ 'MsgType' => 'music',
+ 'ToUserName' => $this->getOpenid(),
+ 'FromUserName' => $this->getToOpenid(),
+ 'Music' => [
+ 'Title' => $title,
+ 'Description' => $desc,
+ 'MusicUrl' => $musicurl,
+ 'HQMusicUrl' => $hgmusicurl,
+ ],
+ ];
+ if ($thumbmediaid) {
+ $this->message['Music']['ThumbMediaId'] = $thumbmediaid;
+ }
+ return $this;
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Scan.php b/vendor/zoujingli/wechat-developer/WeChat/Scan.php
new file mode 100644
index 000000000..ad272eb43
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Scan.php
@@ -0,0 +1,199 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 创建商品
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addProduct(array $data)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/create?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 商品发布
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @param string $status 设置发布状态。on为提交审核,off为取消发布
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function modProduct($keystandard, $keystr, $status = 'on')
+ {
+ $data = ['keystandard' => $keystandard, 'keystr' => $keystr, 'status' => $status];
+ $url = "https://api.weixin.qq.com/scan/product/modstatus?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 设置测试人员白名单
+ * @param array $openids 测试人员的openid列表
+ * @param array $usernames 测试人员的微信号列表
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setTestWhiteList($openids = [], $usernames = [])
+ {
+ $data = ['openid' => $openids, 'username' => $usernames];
+ $url = "https://api.weixin.qq.com/scan/product/modstatus?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 获取商品二维码
+ * @param string $keystandard
+ * @param string $keystr
+ * @param null|string $extinfo
+ * @param integer $qrcode_size
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getQrc($keystandard, $keystr, $extinfo = null, $qrcode_size = 64)
+ {
+ $data = ['keystandard' => $keystandard, 'keystr' => $keystr, 'qrcode_size' => $qrcode_size];
+ is_null($extinfo) || $data['extinfo'] = $extinfo;
+ $url = "https://api.weixin.qq.com/scan/product/getqrcode?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询商品信息
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getProductInfo($keystandard, $keystr)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['keystandard' => $keystandard, 'keystr' => $keystr]);
+ }
+
+ /**
+ * 批量查询商品信息
+ * @param integer $offset 批量查询的起始位置,从0开始,包含该起始位置。
+ * @param integer $limit 批量查询的数量。
+ * @param string $status 支持按状态拉取。on为发布状态,off为未发布状态,check为审核中状态,reject为审核未通过状态,all为所有状态。
+ * @param string $keystr 支持按部分编码内容拉取。填写该参数后,可将编码内容中包含所传参数的商品信息拉出。类似关键词搜索。
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getProductList($offset = 1, $limit = 10, $status = null, $keystr = null)
+ {
+ $data = ['offset' => $offset, 'limit' => $limit];
+ is_null($status) || $data['status'] = $status;
+ is_null($keystr) || $data['keystr'] = $keystr;
+ $url = "https://api.weixin.qq.com/scan/product/getlist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 更新商品信息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateProduct(array $data)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 清除商品信息
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function clearProduct($keystandard, $keystr)
+ {
+ $url = "https://api.weixin.qq.com/scan/product/clear?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['keystandard' => $keystandard, 'keystr' => $keystr]);
+ }
+
+ /**
+ * 检查wxticket参数
+ * @param string $ticket
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function checkTicket($ticket)
+ {
+ $url = "https://api.weixin.qq.com/scan/scanticket/check?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['ticket' => $ticket]);
+ }
+
+ /**
+ * 清除扫码记录
+ * @param string $keystandard 商品编码标准
+ * @param string $keystr 商品编码内容
+ * @param string $extinfo 调用“获取商品二维码接口”时传入的extinfo,为标识参数
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function clearScanTicket($keystandard, $keystr, $extinfo)
+ {
+ $url = "https://api.weixin.qq.com/scan/scanticket/check?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['keystandard' => $keystandard, 'keystr' => $keystr, 'extinfo' => $extinfo]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Script.php b/vendor/zoujingli/wechat-developer/WeChat/Script.php
new file mode 100644
index 000000000..145e51945
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Script.php
@@ -0,0 +1,113 @@
+config->get('appid');
+ $cache_name = "{$appid}_ticket_{$type}";
+ Tools::delCache($cache_name);
+ }
+
+ /**
+ * 获取JSAPI_TICKET接口
+ * @param string $type TICKET类型(wx_card|jsapi)
+ * @param string $appid 强制指定有效APPID
+ * @return string
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getTicket($type = 'jsapi', $appid = null)
+ {
+ is_null($appid) && $appid = $this->config->get('appid');
+ $cache_name = "{$appid}_ticket_{$type}";
+ $ticket = Tools::getCache($cache_name);
+ if (empty($ticket)) {
+ $url = "https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=ACCESS_TOKEN&type={$type}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = $this->httpGetForJson($url);
+ if (empty($result['ticket'])) {
+ throw new InvalidResponseException('Invalid Resoponse Ticket.', '0');
+ }
+ $ticket = $result['ticket'];
+ Tools::setCache($cache_name, $ticket, 5000);
+ }
+ return $ticket;
+ }
+
+ /**
+ * 获取JsApi使用签名
+ * @param string $url 网页的URL
+ * @param string $appid 用于多个appid时使用(可空)
+ * @param string $ticket 强制指定ticket
+ * @return array
+ * @throws Exceptions\LocalCacheException
+ * @throws InvalidResponseException
+ */
+ public function getJsSign($url, $appid = null, $ticket = null)
+ {
+ list($url,) = explode('#', $url);
+ is_null($ticket) && $ticket = $this->getTicket('jsapi');
+ is_null($appid) && $appid = $this->config->get('appid');
+ $data = ["url" => $url, "timestamp" => '' . time(), "jsapi_ticket" => $ticket, "noncestr" => Tools::createNoncestr(16)];
+ return [
+ 'debug' => false,
+ "appId" => $appid,
+ "nonceStr" => $data['noncestr'],
+ "timestamp" => $data['timestamp'],
+ "signature" => $this->getSignature($data, 'sha1'),
+ 'jsApiList' => [
+ 'updateAppMessageShareData', 'updateTimelineShareData', 'onMenuShareTimeline', 'onMenuShareAppMessage', 'onMenuShareQQ', 'onMenuShareWeibo', 'onMenuShareQZone',
+ 'startRecord', 'stopRecord', 'onVoiceRecordEnd', 'playVoice', 'pauseVoice', 'stopVoice', 'onVoicePlayEnd', 'uploadVoice', 'downloadVoice',
+ 'chooseImage', 'previewImage', 'uploadImage', 'downloadImage', 'translateVoice', 'getNetworkType', 'openLocation', 'getLocation',
+ 'hideOptionMenu', 'showOptionMenu', 'hideMenuItems', 'showMenuItems', 'hideAllNonBaseMenuItem', 'showAllNonBaseMenuItem',
+ 'closeWindow', 'scanQRCode', 'chooseWXPay', 'openProductSpecificView', 'addCard', 'chooseCard', 'openCard',
+ ],
+ ];
+ }
+
+ /**
+ * 数据生成签名
+ * @param array $data 签名数组
+ * @param string $method 签名方法
+ * @param array $params 签名参数
+ * @return bool|string 签名值
+ */
+ protected function getSignature($data, $method = "sha1", $params = [])
+ {
+ ksort($data);
+ if (!function_exists($method)) return false;
+ foreach ($data as $k => $v) array_push($params, "{$k}={$v}");
+ return $method(join('&', $params));
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Shake.php b/vendor/zoujingli/wechat-developer/WeChat/Shake.php
new file mode 100644
index 000000000..2d259581c
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Shake.php
@@ -0,0 +1,364 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询审核状态
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function auditStatus()
+ {
+ $url = "https://api.weixin.qq.com/shakearound/account/auditstatus?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 申请设备ID
+ * @param string $quantity 申请的设备ID的数量,单次新增设备超过500个,需走人工审核流程
+ * @param string $apply_reason 申请理由,不超过100个汉字或200个英文字母
+ * @param null|string $comment 备注,不超过15个汉字或30个英文字母
+ * @param null|string $poi_id 设备关联的门店ID,关联门店后,在门店1KM的范围内有优先摇出信息的机会。
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function createApply($quantity, $apply_reason, $comment = null, $poi_id = null)
+ {
+ $data = ['quantity' => $quantity, 'apply_reason' => $apply_reason];
+ is_null($poi_id) || $data['poi_id'] = $poi_id;
+ is_null($comment) || $data['comment'] = $comment;
+ $url = "https://api.weixin.qq.com/shakearound/device/applyid?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询设备ID申请审核状态
+ * @param integer $applyId 批次ID,申请设备ID时所返回的批次ID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getApplyStatus($applyId)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/applyid?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['apply_id' => $applyId]);
+ }
+
+ /**
+ * 编辑设备信息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateApply(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 配置设备与门店的关联关系
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function bindLocation(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/bindlocation?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询设备列表
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function search(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/search?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 页面管理
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function createPage(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/page/add?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 编辑页面信息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updatePage(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/page/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询页面列表
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function searchPage(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/page/search?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 删除页面
+ * @param integer page_id 指定页面的id
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function deletePage($page_id)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/page/delete?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['page_id' => $page_id]);
+ }
+
+ /**
+ * 上传图片素材
+ * @param string $filename 图片名字
+ * @param string $type Icon:摇一摇页面展示的icon图;License:申请开通摇一摇周边功能时需上传的资质文件;若不传type,则默认type=icon
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function upload($filename, $type = 'icon')
+ {
+ $url = "https://api.weixin.qq.com/shakearound/material/add?access_token=ACCESS_TOKEN&type={$type}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['media' => Tools::createCurlFile($filename)]);
+ }
+
+ /**
+ * 配置设备与页面的关联关系
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function bindPage(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/bindpage?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询设备与页面的关联关系
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function queryPage(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/relation/search?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 以设备为维度的数据统计接口
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function totalDevice(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/statistics/device?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 批量查询设备统计数据接口
+ * @param integer $date 指定查询日期时间戳,单位为秒
+ * @param integer $page_index 指定查询的结果页序号;返回结果按摇周边人数降序排序,每50条记录为一页
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function totalDeviceList($date, $page_index = 1)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/statistics/devicelist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['date' => $date, 'page_index' => $page_index]);
+ }
+
+ /**
+ * 以页面为维度的数据统计接口
+ * @param integer $page_id 指定页面的设备ID
+ * @param integer $begin_date 起始日期时间戳,最长时间跨度为30天,单位为秒
+ * @param integer $end_date 结束日期时间戳,最长时间跨度为30天,单位为秒
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function totalPage($page_id, $begin_date, $end_date)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/statistics/page?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['page_id' => $page_id, 'begin_date' => $begin_date, 'end_date' => $end_date]);
+ }
+
+ /**
+ * 编辑分组信息
+ * @param integer $group_id 分组唯一标识,全局唯一
+ * @param string $group_name 分组名称,不超过100汉字或200个英文字母
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateGroup($group_id, $group_name)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['group_id' => $group_id, 'group_name' => $group_name]);
+ }
+
+ /**
+ * 删除分组
+ * @param integer $group_id 分组唯一标识,全局唯一
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function deleteGroup($group_id)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/delete?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['group_id' => $group_id]);
+ }
+
+ /**
+ * 查询分组列表
+ * @param integer $begin 分组列表的起始索引值
+ * @param integer $count 待查询的分组数量,不能超过1000个
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getGroupList($begin = 0, $count = 10)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/getlist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['begin' => $begin, 'count' => $count]);
+ }
+
+
+ /**
+ * 查询分组详情
+ * @param integer $group_id 分组唯一标识,全局唯一
+ * @param integer $begin 分组里设备的起始索引值
+ * @param integer $count 待查询的分组里设备的数量,不能超过1000个
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getGroupDetail($group_id, $begin = 0, $count = 100)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/getdetail?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['group_id' => $group_id, 'begin' => $begin, 'count' => $count]);
+ }
+
+ /**
+ * 添加设备到分组
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addDeviceGroup(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/adddevice?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 从分组中移除设备
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function deleteDeviceGroup(array $data)
+ {
+ $url = "https://api.weixin.qq.com/shakearound/device/group/deletedevice?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Tags.php b/vendor/zoujingli/wechat-developer/WeChat/Tags.php
new file mode 100644
index 000000000..a13c79f79
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Tags.php
@@ -0,0 +1,124 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 创建粉丝标签
+ * @param string $name
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function createTags($name)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/tags/create?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['tag' => ['name' => $name]]);
+ }
+
+ /**
+ * 更新粉丝标签
+ * @param integer $id 标签ID
+ * @param string $name 标签名称
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function updateTags($id, $name)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/tags/update?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['tag' => ['name' => $name, 'id' => $id]]);
+ }
+
+ /**
+ * 删除粉丝标签
+ * @param int $tagId
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function deleteTags($tagId)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/tags/delete?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['tag' => ['id' => $tagId]]);
+ }
+
+ /**
+ * 批量为用户打标签
+ * @param array $openids
+ * @param integer $tagId
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function batchTagging(array $openids, $tagId)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/tags/members/batchtagging?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid_list' => $openids, 'tagid' => $tagId]);
+ }
+
+ /**
+ * 批量为用户取消标签
+ * @param array $openids
+ * @param integer $tagId
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function batchUntagging(array $openids, $tagId)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/tags/members/batchuntagging?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid_list' => $openids, 'tagid' => $tagId]);
+ }
+
+ /**
+ * 获取用户身上的标签列表
+ * @param string $openid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getUserTagId($openid)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/tags/getidlist?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid' => $openid]);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Template.php b/vendor/zoujingli/wechat-developer/WeChat/Template.php
new file mode 100644
index 000000000..eb50ab4a2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Template.php
@@ -0,0 +1,110 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['industry_id1' => $industry_id1, 'industry_id2' => $industry_id2]);
+ }
+
+ /**
+ * 获取设置的行业信息
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getIndustry()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/template/get_industry?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 获得模板ID
+ * @param string $tpl_id 板库中模板的编号,有“TM**”和“OPENTMTM**”等形式
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addTemplate($tpl_id)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/template/api_add_template?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['template_id_short' => $tpl_id]);
+ }
+
+ /**
+ * 获取模板列表
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getAllPrivateTemplate()
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/template/get_all_private_template?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 删除模板ID
+ * @param string $tpl_id 公众帐号下模板消息ID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function delPrivateTemplate($tpl_id)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/template/del_private_template?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['template_id' => $tpl_id]);
+ }
+
+ /**
+ * 发送模板消息
+ * @param array $data
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function send(array $data)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/User.php b/vendor/zoujingli/wechat-developer/WeChat/User.php
new file mode 100644
index 000000000..2156cf7b2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/User.php
@@ -0,0 +1,147 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid' => $openid, 'remark' => $remark]);
+ }
+
+ /**
+ * 获取用户基本信息(包括UnionID机制)
+ * @param string $openid
+ * @param string $lang
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getUserInfo($openid, $lang = 'zh_CN')
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token=ACCESS_TOKEN&openid={$openid}&lang={$lang}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 批量获取用户基本信息
+ * @param array $openids
+ * @param string $lang
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getBatchUserInfo(array $openids, $lang = 'zh_CN')
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/user/info/batchget?access_token=ACCESS_TOKEN';
+ $data = ['user_list' => []];
+ foreach ($openids as $openid) {
+ $data['user_list'][] = ['openid' => $openid, 'lang' => $lang];
+ }
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 获取用户列表
+ * @param string $next_openid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getUserList($next_openid = '')
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/user/get?access_token=ACCESS_TOKEN&next_openid={$next_openid}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 获取标签下粉丝列表
+ * @param integer $tagid 标签ID
+ * @param string $next_openid 第一个拉取的OPENID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getUserListByTag($tagid, $next_openid = '')
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/user/tag/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['tagid' => $tagid, 'next_openid' => $next_openid]);
+ }
+
+ /**
+ * 获取公众号的黑名单列表
+ * @param string $begin_openid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getBlackList($begin_openid = '')
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/tags/members/getblacklist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['begin_openid' => $begin_openid]);
+ }
+
+ /**
+ * 批量拉黑用户
+ * @param array $openids
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function batchBlackList(array $openids)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/tags/members/batchblacklist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid_list' => $openids]);
+ }
+
+ /**
+ * 批量取消拉黑用户
+ * @param array $openids
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function batchUnblackList(array $openids)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/tags/members/batchunblacklist?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['openid_list' => $openids]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeChat/Wifi.php b/vendor/zoujingli/wechat-developer/WeChat/Wifi.php
new file mode 100644
index 000000000..ff22afffc
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeChat/Wifi.php
@@ -0,0 +1,284 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['pageindex' => $pageindex, 'pagesize' => $pagesize]);
+ }
+
+ /**
+ * 查询门店Wi-Fi信息
+ * @param integer $shop_id 门店ID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getShopWifi($shop_id)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/shop/list?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id]);
+ }
+
+ /**
+ * 修改门店网络信息
+ * @param integer $shop_id 门店ID
+ * @param string $old_ssid 旧的无线网络设备的ssid
+ * @param string $ssid 新的无线网络设备的ssid
+ * @param string $password 无线网络设备的密码(可选)
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function upShopWifi($shop_id, $old_ssid, $ssid, $password = null)
+ {
+ $data = ['shop_id' => $shop_id, 'old_ssid' => $old_ssid, 'ssid' => $ssid];
+ is_null($password) || $data['password'] = $password;
+ $url = 'https://api.weixin.qq.com/bizwifi/shop/update?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 清空门店网络及设备
+ * @param integer $shop_id
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function clearShopWifi($shop_id)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/shop/clean?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id]);
+ }
+
+ /**
+ * 添加密码型设备
+ * @param integer $shop_id 门店ID
+ * @param string $ssid 无线网络设备的ssid
+ * @param null|string $password 无线网络设备的密码
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addShopWifi($shop_id, $ssid, $password = null)
+ {
+ $data = ['shop_id' => $shop_id, 'ssid' => $ssid, 'password' => $password];
+ $url = 'https://api.weixin.qq.com/bizwifi/device/add?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 添加portal型设备
+ * @param integer $shop_id 门店ID
+ * @param string $ssid 无线网络设备的ssid
+ * @param bool $reset 重置secretkey,false-不重置,true-重置,默认为false
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function addShopPortal($shop_id, $ssid, $reset = false)
+ {
+ $data = ['shop_id' => $shop_id, 'ssid' => $ssid, 'reset' => $reset];
+ $url = 'https://api.weixin.qq.com/bizwifi/apportal/register?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询设备
+ * @param null|integer $shop_id 根据门店id查询
+ * @param null|integer $pageindex 分页下标,默认从1开始
+ * @param null|integer $pagesize 每页的个数,默认10个,最大20个
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function queryShopWifi($shop_id = null, $pageindex = null, $pagesize = null)
+ {
+ $data = [];
+ is_null($pagesize) || $data['pagesize'] = $pagesize;
+ is_null($pageindex) || $data['pageindex'] = $pageindex;
+ is_null($shop_id) || $data['shop_id'] = $shop_id;
+ $url = 'https://api.weixin.qq.com/bizwifi/device/list?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 删除设备
+ * @param string $bssid 需要删除的无线网络设备无线mac地址,格式冒号分隔,字符长度17个,并且字母小写,例如:00:1f:7a:ad:5c:a8
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function delShopWifi($bssid)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/device/delete?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['bssid' => $bssid]);
+ }
+
+ /**
+ * 获取物料二维码
+ * @param integer $shop_id 门店ID
+ * @param string $ssid 已添加到门店下的无线网络名称
+ * @param integer $img_id 物料样式编号:0-纯二维码,可用于自由设计宣传材料;1-二维码物料,155mm×215mm(宽×高),可直接张贴
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getQrc($shop_id, $ssid, $img_id = 1)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/qrcode/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id, 'ssid' => $ssid, 'img_id' => $img_id]);
+ }
+
+ /**
+ * 设置商家主页
+ * @param integer $shop_id 门店ID
+ * @param integer $template_id 模板ID,0-默认模板,1-自定义url
+ * @param null|string $url 自定义链接,当template_id为1时必填
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setHomePage($shop_id, $template_id, $url = null)
+ {
+ $data = ['shop_id' => $shop_id, 'template_id' => $template_id];
+ is_null($url) && $data['struct'] = ['url' => $url];
+ $url = 'https://api.weixin.qq.com/bizwifi/homepage/set?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询商家主页
+ * @param integer $shop_id 查询的门店id
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getHomePage($shop_id)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/homepage/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id]);
+ }
+
+ /**
+ * 设置微信首页欢迎语
+ * @param integer $shop_id 门店ID
+ * @param integer $bar_type 微信首页欢迎语的文本内容:0--欢迎光临+公众号名称;1--欢迎光临+门店名称;2--已连接+公众号名称+WiFi;3--已连接+门店名称+Wi-Fi。
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setBar($shop_id, $bar_type = 1)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/bar/set?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id, 'bar_type' => $bar_type]);
+ }
+
+ /**
+ * 设置连网完成页
+ * @param integer $shop_id 门店ID
+ * @param string $finishpage_url 连网完成页URL
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setFinishPage($shop_id, $finishpage_url)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/finishpage/set?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id, 'finishpage_url' => $finishpage_url]);
+ }
+
+ /**
+ * Wi-Fi 数据统计
+ * @param string $begin_date 起始日期时间,格式yyyy-mm-dd,最长时间跨度为30天
+ * @param string $end_date 结束日期时间戳,格式yyyy-mm-dd,最长时间跨度为30天
+ * @param integer $shop_id 按门店ID搜索,-1为总统计
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function staticList($begin_date, $end_date, $shop_id = -1)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/statistics/list?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id, 'begin_date' => $begin_date, 'end_date' => $end_date]);
+ }
+
+ /**
+ * 设置门店卡券投放信息
+ * @param integer $shop_id 门店ID,可设置为0,表示所有门店
+ * @param integer $card_id 卡券ID
+ * @param string $card_describe 卡券描述,不能超过18个字符
+ * @param string $start_time 卡券投放开始时间(单位是秒)
+ * @param string $end_time 卡券投放结束时间(单位是秒) 注:不能超过卡券的有效期时间
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function setCouponput($shop_id, $card_id, $card_describe, $start_time, $end_time)
+ {
+ $data = ['shop_id' => $shop_id, 'card_id' => $card_id, 'card_describe' => $card_describe, 'start_time' => $start_time, 'end_time' => $end_time];
+ $url = 'https://api.weixin.qq.com/bizwifi/couponput/set?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 查询门店卡券投放信息
+ * @param integer $shop_id 门店ID,可设置为0,表示所有门店
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function getCouponput($shop_id)
+ {
+ $url = 'https://api.weixin.qq.com/bizwifi/couponput/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['shop_id' => $shop_id]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Crypt.php b/vendor/zoujingli/wechat-developer/WeMini/Crypt.php
new file mode 100644
index 000000000..38fe360d2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Crypt.php
@@ -0,0 +1,105 @@
+config->get('appid'), $sessionKey);
+ $errCode = $pc->decryptData($encryptedData, $iv, $data);
+ if ($errCode == 0) {
+ return json_decode($data, true);
+ }
+ return false;
+ }
+
+ /**
+ * 登录凭证校验
+ * @param string $code 登录时获取的 code
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function session($code)
+ {
+ $appid = $this->config->get('appid');
+ $secret = $this->config->get('appsecret');
+ $url = "https://api.weixin.qq.com/sns/jscode2session?appid={$appid}&secret={$secret}&js_code={$code}&grant_type=authorization_code";
+ return json_decode(Tools::get($url), true);
+ }
+
+ /**
+ * 换取用户信息
+ * @param string $code 用户登录凭证(有效期五分钟)
+ * @param string $iv 加密算法的初始向量
+ * @param string $encryptedData 加密数据( encryptedData )
+ * @return array
+ * @throws InvalidDecryptException
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function userInfo($code, $iv, $encryptedData)
+ {
+ $result = $this->session($code);
+ if (empty($result['session_key'])) {
+ throw new InvalidResponseException('Code 换取 SessionKey 失败', 403);
+ }
+ $userinfo = $this->decode($iv, $result['session_key'], $encryptedData);
+ if (empty($userinfo)) {
+ throw new InvalidDecryptException('用户信息解析失败', 403);
+ }
+ return array_merge($result, $userinfo);
+ }
+
+ /**
+ * 用户支付完成后,获取该用户的 UnionId
+ * @param string $openid 支付用户唯一标识
+ * @param null|string $transaction_id 微信支付订单号
+ * @param null|string $mch_id 微信支付分配的商户号,和商户订单号配合使用
+ * @param null|string $out_trade_no 微信支付商户订单号,和商户号配合使用
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getPaidUnionId($openid, $transaction_id = null, $mch_id = null, $out_trade_no = null)
+ {
+ $url = "https://api.weixin.qq.com/wxa/getpaidunionid?access_token=ACCESS_TOKEN&openid={$openid}";
+ if (is_null($mch_id)) $url .= "&mch_id={$mch_id}";
+ if (is_null($out_trade_no)) $url .= "&out_trade_no={$out_trade_no}";
+ if (is_null($transaction_id)) $url .= "&transaction_id={$transaction_id}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callGetApi($url);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Delivery.php b/vendor/zoujingli/wechat-developer/WeMini/Delivery.php
new file mode 100644
index 000000000..6d6d6aff5
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Delivery.php
@@ -0,0 +1,181 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 下配送单接口
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/add?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 可以对待接单状态的订单增加小费
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addTip($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/addtips?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 取消配送单接口
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function cancelOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/cancel?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 获取已支持的配送公司列表接口
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAllImmeDelivery($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/delivery/getall?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 拉取已绑定账号
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getBindAccount($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/shop/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 拉取配送单信息
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 模拟配送公司更新配送单状态
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function mockUpdateOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/test_update_order?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 预下配送单接口
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function preAddOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/pre_add?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 预取消配送单接口
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function preCancelOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/precancel?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 重新下单
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function reOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/local/business/order/readd?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Image.php b/vendor/zoujingli/wechat-developer/WeMini/Image.php
new file mode 100644
index 000000000..16db57544
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Image.php
@@ -0,0 +1,72 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['img_url' => $img_url, 'img' => $img], true);
+ }
+
+ /**
+ * 本接口提供基于小程序的条码/二维码识别的API
+ * @param string $img_url 要检测的图片 url,传这个则不用传 img 参数。
+ * @param string $img form-data 中媒体文件标识,有filename、filelength、content-type等信息,传这个则不用穿 img_url
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function scanQRCode($img_url, $img)
+ {
+ $url = "https://api.weixin.qq.com/cv/img/qrcode?img_url=ENCODE_URL&access_token=ACCESS_TOCKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['img_url' => $img_url, 'img' => $img], true);
+ }
+
+ /**
+ * 本接口提供基于小程序的图片高清化能力
+ * @param string $img_url 要检测的图片 url,传这个则不用传 img 参数
+ * @param string $img form-data 中媒体文件标识,有filename、filelength、content-type等信息,传这个则不用穿 img_url
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function superresolution($img_url, $img)
+ {
+ $url = "https://api.weixin.qq.com/cv/img/qrcode?img_url=ENCODE_URL&access_token=ACCESS_TOCKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['img_url' => $img_url, 'img' => $img], true);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Logistics.php b/vendor/zoujingli/wechat-developer/WeMini/Logistics.php
new file mode 100644
index 000000000..dc048bb5e
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Logistics.php
@@ -0,0 +1,206 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 取消运单
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function cancelOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/order/cancel?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 获取支持的快递公司列表
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAllDelivery()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/delivery/getall?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callGetApi($url);
+ }
+
+ /**
+ * 获取运单数据
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/order/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 查询运单轨迹
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getPath($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/path/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 获取打印员。若需要使用微信打单 PC 软件,才需要调用
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getPrinter()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/printer/getall?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callGetApi($url);
+ }
+
+ /**
+ * 获取电子面单余额。仅在使用加盟类快递公司时,才可以调用
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getQuota($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/path/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 模拟快递公司更新订单状态, 该接口只能用户测试
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function testUpdateOrder($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/test_update_order?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 配置面单打印员,若需要使用微信打单 PC 软件,才需要调用
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function updatePrinter($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/business/printer/update?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 获取面单联系人信息
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getContact($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/delivery/contact/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 预览面单模板。用于调试面单模板使用
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function previewTemplate($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/delivery/template/preview?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 更新商户审核结果
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function updateBusiness($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/delivery/service/business/update?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 更新运单轨迹
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function updatePath($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/express/delivery/path/update?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Message.php b/vendor/zoujingli/wechat-developer/WeMini/Message.php
new file mode 100644
index 000000000..fdb06a4c6
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Message.php
@@ -0,0 +1,67 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 动态消息,修改被分享的动态消息
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setUpdatableMsg($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/message/wxopen/updatablemsg/send?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 下发小程序和公众号统一的服务消息
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function uniformSend($data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/message/wxopen/template/uniform_send?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Ocr.php b/vendor/zoujingli/wechat-developer/WeMini/Ocr.php
new file mode 100644
index 000000000..22b5048d7
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Ocr.php
@@ -0,0 +1,110 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 本接口提供基于小程序的营业执照 OCR 识别
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function businessLicense($data)
+ {
+ $url = 'https://api.weixin.qq.com/cv/ocr/bizlicense?access_token=ACCESS_TOCKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 本接口提供基于小程序的驾驶证 OCR 识别
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function driverLicense($data)
+ {
+ $url = 'https://api.weixin.qq.com/cv/ocr/drivinglicense?access_token=ACCESS_TOCKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 本接口提供基于小程序的身份证 OCR 识别
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function idcard($data)
+ {
+ $url = 'https://api.weixin.qq.com/cv/ocr/idcard?access_token=ACCESS_TOCKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 本接口提供基于小程序的通用印刷体 OCR 识别
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function printedText($data)
+ {
+ $url = 'https://api.weixin.qq.com/cv/ocr/comm?access_token=ACCESS_TOCKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 本接口提供基于小程序的行驶证 OCR 识别
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function vehicleLicense($data)
+ {
+ $url = 'https://api.weixin.qq.com/cv/ocr/driving?access_token=ACCESS_TOCKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Plugs.php b/vendor/zoujingli/wechat-developer/WeMini/Plugs.php
new file mode 100644
index 000000000..6110e89cd
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Plugs.php
@@ -0,0 +1,112 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['action' => 'apply', 'plugin_appid' => $plugin_appid], true);
+ }
+
+ /**
+ * 2.查询已添加的插件
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getList()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/plugin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['action' => 'list'], true);
+ }
+
+ /**
+ * 3.删除已添加的插件
+ * @param string $plugin_appid 插件appid
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function unbind($plugin_appid)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/plugin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['action' => 'unbind', 'plugin_appid' => $plugin_appid], true);
+ }
+
+ /**
+ * 获取当前所有插件使用方
+ * 修改插件使用申请的状态
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function devplugin($data)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/devplugin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 4.获取当前所有插件使用方(供插件开发者调用)
+ * @param integer $page 拉取第page页的数据
+ * @param integer $num 表示每页num条记录
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function devApplyList($page = 1, $num = 10)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/plugin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $data = ['action' => 'dev_apply_list', 'page' => $page, 'num' => $num];
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 5.修改插件使用申请的状态(供插件开发者调用)
+ * @param string $action dev_agree:同意申请;dev_refuse:拒绝申请;dev_delete:删除已拒绝的申请者
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function devAgree($action = 'dev_agree')
+ {
+ $url = 'https://api.weixin.qq.com/wxa/plugin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['action' => $action], true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Poi.php b/vendor/zoujingli/wechat-developer/WeMini/Poi.php
new file mode 100644
index 000000000..c81ef5d2d
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Poi.php
@@ -0,0 +1,91 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ $data = [
+ 'related_name' => $related_name, 'related_credential' => $related_credential,
+ 'related_address' => $related_address, 'related_proof_material' => $related_proof_material,
+ ];
+ return $this->callPostApi($url, $data, true);
+ }
+
+ /**
+ * 查看地点列表
+ * @param integer $page 起始页id(从1开始计数)
+ * @param integer $page_rows 每页展示个数(最多1000个)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getNearByPoiList($page = 1, $page_rows = 1000)
+ {
+ $url = "https://api.weixin.qq.com/wxa/getnearbypoilist?page={$page}&page_rows={$page_rows}&access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callGetApi($url);
+ }
+
+ /**
+ * 删除地点
+ * @param string $poi_id 附近地点ID
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function delNearByPoiList($poi_id)
+ {
+ $url = "https://api.weixin.qq.com/wxa/delnearbypoi?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['poi_id' => $poi_id], true);
+ }
+
+ /**
+ * 展示/取消展示附近小程序
+ * @param string $poi_id 附近地点ID
+ * @param string $status 0:取消展示;1:展示
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setNearByPoiShowStatus($poi_id, $status)
+ {
+ $url = "https://api.weixin.qq.com/wxa/setnearbypoishowstatus?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['poi_id' => $poi_id, 'status' => $status], true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Qrcode.php b/vendor/zoujingli/wechat-developer/WeMini/Qrcode.php
new file mode 100644
index 000000000..4c238b34d
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Qrcode.php
@@ -0,0 +1,112 @@
+ "0", "g" => "0", "b" => "0"], $is_hyaline = true, $outType = null)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/getwxacode?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $data = ['path' => $path, 'width' => $width, 'auto_color' => $auto_color, 'line_color' => $line_color, 'is_hyaline' => $is_hyaline];
+ $result = Tools::post($url, Tools::arr2json($data));
+ if (is_array($json = json_decode($result, true))) {
+ if (!$this->isTry && isset($json['errcode']) && in_array($json['errcode'], ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+ /**
+ * 获取小程序码(永久有效)
+ * 接口B:适用于需要的码数量极多的业务场景
+ * @param string $scene 最大32个可见字符,只支持数字
+ * @param string $page 必须是已经发布的小程序存在的页面
+ * @param integer $width 二维码的宽度
+ * @param bool $auto_color 自动配置线条颜色,如果颜色依然是黑色,则说明不建议配置主色调
+ * @param array $line_color auto_color 为 false 时生效
+ * @param boolean $is_hyaline 是否需要透明底色
+ * @param null|string $outType 输出类型
+ * @return array|string
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function createMiniScene($scene, $page, $width = 430, $auto_color = false, $line_color = ["r" => "0", "g" => "0", "b" => "0"], $is_hyaline = true, $outType = null)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=ACCESS_TOKEN';
+ $data = ['scene' => $scene, 'width' => $width, 'auto_color' => $auto_color, 'page' => $page, 'line_color' => $line_color, 'is_hyaline' => $is_hyaline];
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = Tools::post($url, Tools::arr2json($data));
+ if (is_array($json = json_decode($result, true))) {
+ if (!$this->isTry && isset($json['errcode']) && in_array($json['errcode'], ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+ /**
+ * 获取小程序二维码(永久有效)
+ * 接口C:适用于需要的码数量较少的业务场景
+ * @param string $path 不能为空,最大长度 128 字节
+ * @param integer $width 二维码的宽度
+ * @param null|string $outType 输出类型
+ * @return array|string
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function createDefault($path, $width = 430, $outType = null)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxaapp/createwxaqrcode?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = Tools::post($url, Tools::arr2json(['path' => $path, 'width' => $width]));
+ if (is_array($json = json_decode($result, true))) {
+ if (!$this->isTry && isset($json['errcode']) && in_array($json['errcode'], ['40014', '40001', '41001', '42001'])) {
+ [$this->delAccessToken(), $this->isTry = true];
+ return call_user_func_array([$this, $this->currentMethod['method']], $this->currentMethod['arguments']);
+ }
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Security.php b/vendor/zoujingli/wechat-developer/WeMini/Security.php
new file mode 100644
index 000000000..5b1901fa4
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Security.php
@@ -0,0 +1,70 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['media' => $media], true);
+ }
+
+ /**
+ * 异步校验图片/音频是否含有违法违规内容
+ * @param string $media_url
+ * @param string $media_type
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function mediaCheckAsync($media_url, $media_type)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/media_check_async?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['media_url' => $media_url, 'media_type' => $media_type], true);
+ }
+
+ /**
+ * 检查一段文本是否含有违法违规内容
+ * @param string $content
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function msgSecCheck($content)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/msg_sec_check?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['content' => $content], true);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Soter.php b/vendor/zoujingli/wechat-developer/WeMini/Soter.php
new file mode 100644
index 000000000..789056896
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Soter.php
@@ -0,0 +1,40 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Template.php b/vendor/zoujingli/wechat-developer/WeMini/Template.php
new file mode 100644
index 000000000..516d40741
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Template.php
@@ -0,0 +1,110 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['offset' => '0', 'count' => '20'], true);
+ }
+
+ /**
+ * 获取模板库某个模板标题下关键词库
+ * @param string $template_id 模板标题id,可通过接口获取,也可登录小程序后台查看获取
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getTemplateLibrary($template_id)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/template/library/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['id' => $template_id], true);
+ }
+
+ /**
+ * 组合模板并添加至帐号下的个人模板库
+ * @param string $template_id 模板标题id,可通过接口获取,也可登录小程序后台查看获取
+ * @param array $keyword_id_list 开发者自行组合好的模板关键词列表,关键词顺序可以自由搭配(例如[3,5,4]或[4,5,3]),最多支持10个关键词组合
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addTemplate($template_id, array $keyword_id_list)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/template/add?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['id' => $template_id, 'keyword_id_list' => $keyword_id_list], true);
+ }
+
+ /**
+ * 获取帐号下已存在的模板列表
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getTemplateList()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/template/list?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['offset' => '0', 'count' => '20'], true);
+ }
+
+ /**
+ * 删除模板消息
+ * @param string $template_id 要删除的模板id
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function delTemplate($template_id)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/template/del?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['template_id' => $template_id], true);
+ }
+
+ /**
+ * 发送模板消息
+ * @param array $data 发送的消息对象数组
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function send(array $data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/message/wxopen/template/send?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, $data, true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/Total.php b/vendor/zoujingli/wechat-developer/WeMini/Total.php
new file mode 100644
index 000000000..9d55a6b5b
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/Total.php
@@ -0,0 +1,176 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 访问分析
+ * @param string $begin_date 开始日期
+ * @param string $end_date 结束日期,限定查询1天数据,end_date允许设置的最大值为昨日
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidDailyVisittrend($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappiddailyvisittrend?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 周趋势
+ * @param string $begin_date 开始日期,为周一日期
+ * @param string $end_date 结束日期,为周日日期,限定查询一周数据
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidWeeklyVisittrend($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidweeklyvisittrend?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 月趋势
+ * @param string $begin_date 开始日期,为自然月第一天
+ * @param string $end_date 结束日期,为自然月最后一天,限定查询一个月数据
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidMonthlyVisittrend($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidmonthlyvisittrend?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 访问分布
+ * @param string $begin_date 开始日期
+ * @param string $end_date 结束日期,限定查询1天数据,end_date允许设置的最大值为昨日
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidVisitdistribution($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidvisitdistribution?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 日留存
+ * @param string $begin_date 开始日期
+ * @param string $end_date 结束日期,限定查询1天数据,end_date允许设置的最大值为昨日
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidDailyRetaininfo($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappiddailyretaininfo?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 周留存
+ * @param string $begin_date 开始日期,为周一日期
+ * @param string $end_date 结束日期,为周日日期,限定查询一周数据
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidWeeklyRetaininfo($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidweeklyretaininfo?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 月留存
+ * @param string $begin_date 开始日期,为自然月第一天
+ * @param string $end_date 结束日期,为自然月最后一天,限定查询一个月数据
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidMonthlyRetaininfo($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidmonthlyretaininfo?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 访问页面
+ * @param string $begin_date 开始日期
+ * @param string $end_date 结束日期,限定查询1天数据,end_date允许设置的最大值为昨日
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidVisitPage($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappidvisitpage?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+ /**
+ * 用户画像
+ * @param string $begin_date 开始日期
+ * @param string $end_date 结束日期,开始日期与结束日期相差的天数限定为0/6/29,分别表示查询最近1/7/30天数据,end_date允许设置的最大值为昨日
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeanalysisAppidUserportrait($begin_date, $end_date)
+ {
+ $url = 'https://api.weixin.qq.com/datacube/getweanalysisappiduserportrait?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->callPostApi($url, ['begin_date' => $begin_date, 'end_date' => $end_date], true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/crypt/errorCode.php b/vendor/zoujingli/wechat-developer/WeMini/crypt/errorCode.php
new file mode 100644
index 000000000..a4f8e7267
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/crypt/errorCode.php
@@ -0,0 +1,20 @@
+
+ * -41001: encodingAesKey 非法
+ * -41003: aes 解密失败
+ * -41004: 解密后得到的buffer非法
+ * -41005: base64加密失败
+ * -41016: base64解密失败
+ *
+ */
+class ErrorCode
+{
+ public static $OK = 0;
+ public static $IllegalAesKey = -41001;
+ public static $IllegalIv = -41002;
+ public static $IllegalBuffer = -41003;
+ public static $DecodeBase64Error = -41004;
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WeMini/crypt/wxBizDataCrypt.php b/vendor/zoujingli/wechat-developer/WeMini/crypt/wxBizDataCrypt.php
new file mode 100644
index 000000000..d969638ef
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WeMini/crypt/wxBizDataCrypt.php
@@ -0,0 +1,57 @@
+appid = $appid;
+ $this->sessionKey = $sessionKey;
+ include_once __DIR__ . DIRECTORY_SEPARATOR . "errorCode.php";
+ }
+
+ /**
+ * 检验数据的真实性,并且获取解密后的明文.
+ * @param $encryptedData string 加密的用户数据
+ * @param $iv string 与用户数据一同返回的初始向量
+ * @param $data string 解密后的原文
+ *
+ * @return int 成功0,失败返回对应的错误码
+ */
+ public function decryptData($encryptedData, $iv, &$data)
+ {
+ if (strlen($this->sessionKey) != 24) {
+ return \ErrorCode::$IllegalAesKey;
+ }
+ $aesKey = base64_decode($this->sessionKey);
+ if (strlen($iv) != 24) {
+ return \ErrorCode::$IllegalIv;
+ }
+ $aesIV = base64_decode($iv);
+ $aesCipher = base64_decode($encryptedData);
+ $result = openssl_decrypt($aesCipher, "AES-128-CBC", $aesKey, 1, $aesIV);
+ $dataObj = json_decode($result);
+ if ($dataObj == null) {
+ return \ErrorCode::$IllegalBuffer;
+ }
+ if ($dataObj->watermark->appid != $this->appid) {
+ return \ErrorCode::$IllegalBuffer;
+ }
+ $data = $result;
+ return \ErrorCode::$OK;
+ }
+
+}
+
diff --git a/vendor/zoujingli/wechat-developer/WePay/Bill.php b/vendor/zoujingli/wechat-developer/WePay/Bill.php
new file mode 100644
index 000000000..4c853c700
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Bill.php
@@ -0,0 +1,63 @@
+params->set('sign_type', 'MD5');
+ $params = $this->params->merge($options);
+ $params['sign'] = $this->getPaySign($params, 'MD5');
+ $result = Tools::post('https://api.mch.weixin.qq.com/pay/downloadbill', Tools::arr2xml($params));
+ if (($jsonData = Tools::xml2arr($result))) {
+ if ($jsonData['return_code'] !== 'SUCCESS') {
+ throw new InvalidResponseException($jsonData['return_msg'], '0');
+ }
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+
+ /**
+ * 拉取订单评价数据
+ * @param array $options
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function comment(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/billcommentsp/batchquerycomment';
+ return $this->callPostApi($url, $options, true);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/Coupon.php b/vendor/zoujingli/wechat-developer/WePay/Coupon.php
new file mode 100644
index 000000000..92ca5c06f
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Coupon.php
@@ -0,0 +1,65 @@
+callPostApi($url, $options, true);
+ }
+
+ /**
+ * 查询代金券批次
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function queryStock(array $options)
+ {
+ $url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/query_coupon_stock";
+ return $this->callPostApi($url, $options, false);
+ }
+
+ /**
+ * 查询代金券信息
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function queryInfo(array $options)
+ {
+ $url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/query_coupon_stock";
+ return $this->callPostApi($url, $options, false);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/Order.php b/vendor/zoujingli/wechat-developer/WePay/Order.php
new file mode 100644
index 000000000..1d93288fe
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Order.php
@@ -0,0 +1,173 @@
+callPostApi($url, $options, false, 'MD5');
+ }
+
+ /**
+ * 刷卡支付
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function micropay(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/pay/micropay';
+ return $this->callPostApi($url, $options, false, 'MD5');
+ }
+
+ /**
+ * 查询订单
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function query(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/pay/orderquery';
+ return $this->callPostApi($url, $options);
+ }
+
+ /**
+ * 关闭订单
+ * @param string $outTradeNo 商户订单号
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function close($outTradeNo)
+ {
+ $url = 'https://api.mch.weixin.qq.com/pay/closeorder';
+ return $this->callPostApi($url, ['out_trade_no' => $outTradeNo]);
+ }
+
+ /**
+ * 创建JsApi及H5支付参数
+ * @param string $prepayId 统一下单预支付码
+ * @return array
+ */
+ public function jsapiParams($prepayId)
+ {
+ $option = [];
+ $option["appId"] = $this->config->get('appid');
+ $option["timeStamp"] = (string)time();
+ $option["nonceStr"] = Tools::createNoncestr();
+ $option["package"] = "prepay_id={$prepayId}";
+ $option["signType"] = "MD5";
+ $option["paySign"] = $this->getPaySign($option, 'MD5');
+ $option['timestamp'] = $option['timeStamp'];
+ return $option;
+ }
+
+ /**
+ * 获取支付规则二维码
+ * @param string $productId 商户定义的商品id或者订单号
+ * @return string
+ */
+ public function qrcParams($productId)
+ {
+ $data = [
+ 'appid' => $this->config->get('appid'),
+ 'mch_id' => $this->config->get('mch_id'),
+ 'time_stamp' => (string)time(),
+ 'nonce_str' => Tools::createNoncestr(),
+ 'product_id' => (string)$productId,
+ ];
+ $data['sign'] = $this->getPaySign($data, 'MD5');
+ return "weixin://wxpay/bizpayurl?" . http_build_query($data);
+ }
+
+ /**
+ * 获取微信App支付秘需参数
+ * @param string $prepayId 统一下单预支付码
+ * @return array
+ */
+ public function appParams($prepayId)
+ {
+ $data = [
+ 'appid' => $this->config->get('appid'),
+ 'partnerid' => $this->config->get('mch_id'),
+ 'prepayid' => (string)$prepayId,
+ 'package' => 'Sign=WXPay',
+ 'timestamp' => (string)time(),
+ 'noncestr' => Tools::createNoncestr(),
+ ];
+ $data['sign'] = $this->getPaySign($data, 'MD5');
+ return $data;
+ }
+
+ /**
+ * 刷卡支付 撤销订单
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function reverse(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/secapi/pay/reverse';
+ return $this->callPostApi($url, $options, true);
+ }
+
+ /**
+ * 刷卡支付 授权码查询openid
+ * @param string $authCode 扫码支付授权码,设备读取用户微信中的条码或者二维码信息
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function queryAuthCode($authCode)
+ {
+ $url = 'https://api.mch.weixin.qq.com/tools/authcodetoopenid';
+ return $this->callPostApi($url, ['auth_code' => $authCode]);
+ }
+
+ /**
+ * 刷卡支付 交易保障
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function report(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/payitil/report';
+ return $this->callPostApi($url, $options);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/Redpack.php b/vendor/zoujingli/wechat-developer/WePay/Redpack.php
new file mode 100644
index 000000000..bf97ad89f
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Redpack.php
@@ -0,0 +1,72 @@
+params->offsetUnset('appid');
+ $this->params->set('wxappid', $this->config->get('appid'));
+ $url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/sendredpack";
+ return $this->callPostApi($url, $options, true, 'MD5', false);
+ }
+
+ /**
+ * 发放裂变红包
+ * @param array $options
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function groups(array $options)
+ {
+ $this->params->offsetUnset('appid');
+ $this->params->set('wxappid', $this->config->get('appid'));
+ $url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/sendgroupredpack";
+ return $this->callPostApi($url, $options, true, 'MD5', false);
+ }
+
+ /**
+ * 查询红包记录
+ * @param string $mchBillno 商户发放红包的商户订单号
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function query($mchBillno)
+ {
+ $this->params->offsetUnset('wxappid');
+ $this->params->set('appid', $this->config->get('appid'));
+ $url = "https://api.mch.weixin.qq.com/mmpaymkttransfers/gethbinfo";
+ return $this->callPostApi($url, ['mch_billno' => $mchBillno, 'bill_type' => 'MCHT'], true, 'MD5', false);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/Refund.php b/vendor/zoujingli/wechat-developer/WePay/Refund.php
new file mode 100644
index 000000000..8e3c199fc
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Refund.php
@@ -0,0 +1,78 @@
+callPostApi($url, $options, true);
+ }
+
+ /**
+ * 查询退款
+ * @param array $options
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function query(array $options)
+ {
+ $url = 'https://api.mch.weixin.qq.com/pay/refundquery';
+ return $this->callPostApi($url, $options);
+ }
+
+ /**
+ * 获取退款通知
+ * @return array
+ * @throws InvalidResponseException
+ */
+ public function getNotify()
+ {
+ $data = Tools::xml2arr(file_get_contents("php://input"));
+ if (!isset($data['return_code']) || $data['return_code'] !== 'SUCCESS') {
+ throw new InvalidResponseException('获取退款通知XML失败!');
+ }
+ if (!class_exists('Prpcrypt', false)) {
+ include dirname(__DIR__) . '/WeChat/Contracts/Prpcrypt.php';
+ }
+ $pc = new \Prpcrypt(md5($this->config->get('mch_key')));
+ $array = $pc->decrypt(base64_decode($data['req_info']));
+ if (intval($array[0]) > 0) {
+ throw new InvalidResponseException($array[1], $array[0]);
+ }
+ $data['decode'] = $array[1];
+ return $data;
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/Transfers.php b/vendor/zoujingli/wechat-developer/WePay/Transfers.php
new file mode 100644
index 000000000..89ff1d8c1
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/Transfers.php
@@ -0,0 +1,61 @@
+params->offsetUnset('appid');
+ $this->params->offsetUnset('mch_id');
+ $this->params->set('mchid', $this->config->get('mch_id'));
+ $this->params->set('mch_appid', $this->config->get('appid'));
+ $url = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers';
+ return $this->callPostApi($url, $options, true, 'MD5', false);
+ }
+
+ /**
+ * 查询企业付款到零钱
+ * @param string $partnerTradeNo 商户调用企业付款API时使用的商户订单号
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function query($partnerTradeNo)
+ {
+ $this->params->offsetUnset('mchid');
+ $this->params->offsetUnset('mch_appid');
+ $this->params->set('appid', $this->config->get('appid'));
+ $this->params->set('mch_id', $this->config->get('mch_id'));
+ $url = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/gettransferinfo';
+ return $this->callPostApi($url, ['partner_trade_no' => $partnerTradeNo], true, 'MD5', false);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/WePay/TransfersBank.php b/vendor/zoujingli/wechat-developer/WePay/TransfersBank.php
new file mode 100644
index 000000000..0509d010a
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/WePay/TransfersBank.php
@@ -0,0 +1,125 @@
+params->offsetUnset('appid');
+ return $this->callPostApi('https://api.mch.weixin.qq.com/mmpaysptrans/pay_bank', [
+ 'amount' => $options['amount'],
+ 'bank_code' => $options['bank_code'],
+ 'partner_trade_no' => $options['partner_trade_no'],
+ 'enc_bank_no' => $this->rsaEncode($options['enc_bank_no']),
+ 'enc_true_name' => $this->rsaEncode($options['enc_true_name']),
+ 'desc' => isset($options['desc']) ? $options['desc'] : '',
+ ], true, 'MD5', false);
+ }
+
+ /**
+ * 商户企业付款到银行卡操作进行结果查询
+ * @param string $partnerTradeNo 商户订单号,需保持唯一
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function query($partnerTradeNo)
+ {
+ $this->params->offsetUnset('appid');
+ $url = 'https://api.mch.weixin.qq.com/mmpaysptrans/query_bank';
+ return $this->callPostApi($url, ['partner_trade_no' => $partnerTradeNo], true, 'MD5', false);
+ }
+
+ /**
+ * RSA加密处理
+ * @param string $string
+ * @param string $encrypted
+ * @return string
+ * @throws \WeChat\Exceptions\InvalidDecryptException
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ private function rsaEncode($string, $encrypted = '')
+ {
+ $search = ['-----BEGIN RSA PUBLIC KEY-----', '-----END RSA PUBLIC KEY-----', "\n", "\r"];
+ $pkc1 = str_replace($search, '', $this->getRsaContent());
+ $publicKey = '-----BEGIN PUBLIC KEY-----' . PHP_EOL .
+ wordwrap('MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8A' . $pkc1, 64, PHP_EOL, true) . PHP_EOL .
+ '-----END PUBLIC KEY-----';
+ if (!openssl_public_encrypt("{$string}", $encrypted, $publicKey, OPENSSL_PKCS1_OAEP_PADDING)) {
+ throw new InvalidDecryptException('Rsa Encrypt Error.');
+ }
+ return base64_encode($encrypted);
+ }
+
+ /**
+ * 获取签名文件内容
+ * @return string
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ private function getRsaContent()
+ {
+ $cacheKey = "pub_ras_key_" . $this->config->get('mch_id');
+ if (($pub_key = Tools::getCache($cacheKey))) {
+ return $pub_key;
+ }
+ $data = $this->callPostApi('https://fraud.mch.weixin.qq.com/risk/getpublickey', [], true, 'MD5');
+ if (!isset($data['return_code']) || $data['return_code'] !== 'SUCCESS' || $data['result_code'] !== 'SUCCESS') {
+ $error = 'ResultError:' . $data['return_msg'];
+ $error .= isset($data['err_code_des']) ? ' - ' . $data['err_code_des'] : '';
+ throw new InvalidResponseException($error, 20000, $data);
+ }
+ Tools::setCache($cacheKey, $data['pub_key'], 600);
+ return $data['pub_key'];
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-app.php b/vendor/zoujingli/wechat-developer/_test/alipay-app.php
new file mode 100644
index 000000000..0070eb31c
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-app.php
@@ -0,0 +1,39 @@
+apply([
+ 'out_trade_no' => time(), // 商户订单号
+ 'total_amount' => '1', // 支付金额
+ 'subject' => 'test subject', // 支付订单描述
+ ]);
+ echo '';
+ var_export($result);
+} catch (\Exception $e) {
+ echo $e->getMessage();
+}
+
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-bill.php b/vendor/zoujingli/wechat-developer/_test/alipay-bill.php
new file mode 100644
index 000000000..af5c493cf
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-bill.php
@@ -0,0 +1,36 @@
+apply([
+ 'bill_date' => '2018-10-03', // 账单时间(日账单yyyy-MM-dd,月账单 yyyy-MM)
+ 'bill_type' => 'signcustomer', // 账单类型(trade指商户基于支付宝交易收单的业务账单,signcustomer是指基于商户支付宝余额收入及支出等资金变动的帐务账单)
+ ]);
+ echo '';
+ var_export($result);
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-notify.php b/vendor/zoujingli/wechat-developer/_test/alipay-notify.php
new file mode 100644
index 000000000..0d5911682
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-notify.php
@@ -0,0 +1,39 @@
+notify();
+ if (in_array($data['trade_status'], ['TRADE_SUCCESS', 'TRADE_FINISHED'])) {
+ // @todo 更新订单状态,支付完成
+ file_put_contents('notify.txt', "收到来自支付宝的异步通知\r\n", FILE_APPEND);
+ file_put_contents('notify.txt', '订单号:' . $data['out_trade_no'] . "\r\n", FILE_APPEND);
+ file_put_contents('notify.txt', '订单金额:' . $data['total_amount'] . "\r\n\r\n", FILE_APPEND);
+ } else {
+ file_put_contents('notify.txt', "收到异步通知\r\n", FILE_APPEND);
+ }
+} catch (\Exception $e) {
+ // 异常处理
+ echo $e->getMessage();
+}
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-pos.php b/vendor/zoujingli/wechat-developer/_test/alipay-pos.php
new file mode 100644
index 000000000..e55c4e2e0
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-pos.php
@@ -0,0 +1,41 @@
+apply([
+ 'out_trade_no' => '4312412343', // 订单号
+ 'total_amount' => '13', // 订单金额,单位:元
+ 'subject' => '订单商品标题', // 订单商品标题
+ 'auth_code' => '123456', // 授权码
+ ]);
+
+ echo '';
+ var_export($result);
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
+
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-refund.php b/vendor/zoujingli/wechat-developer/_test/alipay-refund.php
new file mode 100644
index 000000000..487ca7b27
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-refund.php
@@ -0,0 +1,39 @@
+refund($out_trade_no, $refund_fee);
+
+ echo '';
+ var_export($result);
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-scan.php b/vendor/zoujingli/wechat-developer/_test/alipay-scan.php
new file mode 100644
index 000000000..2bc880632
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-scan.php
@@ -0,0 +1,40 @@
+apply([
+ 'out_trade_no' => '14321412', // 订单号
+ 'total_amount' => '13', // 订单金额,单位:元
+ 'subject' => '订单商品标题', // 订单商品标题
+ ]);
+
+ echo '';
+ var_export($result);
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
+
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-transfer.php b/vendor/zoujingli/wechat-developer/_test/alipay-transfer.php
new file mode 100644
index 000000000..4eaf6fb1f
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-transfer.php
@@ -0,0 +1,43 @@
+apply([
+ 'out_biz_no' => time(), // 订单号
+ 'payee_type' => 'ALIPAY_LOGONID', // 收款方账户类型(ALIPAY_LOGONID | ALIPAY_USERID)
+ 'payee_account' => 'demo@sandbox.com', // 收款方账户
+ 'amount' => '10', // 转账金额
+ 'payer_show_name' => '未寒', // 付款方姓名
+ 'payee_real_name' => '张三', // 收款方真实姓名
+ 'remark' => '张三', // 转账备注
+ ]);
+
+ echo '';
+ var_export($result);
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-wap.php b/vendor/zoujingli/wechat-developer/_test/alipay-wap.php
new file mode 100644
index 000000000..ac04021d2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-wap.php
@@ -0,0 +1,42 @@
+apply([
+ 'out_trade_no' => time(), // 商户订单号
+ 'total_amount' => '1', // 支付金额
+ 'subject' => '支付订单描述', // 支付订单描述
+ ]);
+
+ echo $result;
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
+
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay-web.php b/vendor/zoujingli/wechat-developer/_test/alipay-web.php
new file mode 100644
index 000000000..46f0efcf0
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay-web.php
@@ -0,0 +1,43 @@
+apply([
+ 'out_trade_no' => time(), // 商户订单号
+ 'total_amount' => '1', // 支付金额
+ 'subject' => '支付订单描述', // 支付订单描述
+ ]);
+
+ echo $result;
+} catch (Exception $e) {
+ echo $e->getMessage();
+}
+
+
diff --git a/vendor/zoujingli/wechat-developer/_test/alipay.php b/vendor/zoujingli/wechat-developer/_test/alipay.php
new file mode 100644
index 000000000..25a7379d3
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/alipay.php
@@ -0,0 +1,30 @@
+ true,
+ // 签名类型(RSA|RSA2)
+ 'sign_type' => "RSA2",
+ // 应用ID
+ 'appid' => '2016090900468879',
+ // 支付宝公钥(1行填写,特别注意,这里是支付宝公钥,不是应用公钥,最好从开发者中心的网页上去复制)
+ 'public_key' => 'MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtU71NY53UDGY7JNvLYAhsNa+taTF6KthIHJmGgdio9bkqeJGhHk6ttkTKkLqFgwIfgAkHpdKiOv1uZw6gVGZ7TCu5LfHTqKrCd6Uz+N7hxhY+4IwicLgprcV1flXQLmbkJYzFMZqkXGkSgOsR2yXh4LyQZczgk9N456uuzGtRy7MoB4zQy34PLUkkxR6W1B2ftNbLRGXv6tc7p/cmDcrY6K1bSxnGmfRxFSb8lRfhe0V0UM6pKq2SGGSeovrKHN0OLp+Nn5wcULVnFgATXGCENshRlp96piPEBFwneXs19n+sX1jx60FTR7/rME3sW3AHug0fhZ9mSqW4x401WjdnwIDAQAB',
+ // 支付宝私钥(1行填写)
+ 'private_key' => 'MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQC3pbN7esinxgjE8uxXAsccgGNKIq+PR1LteNTFOy0fsete43ObQCrzd9DO0zaUeBUzpIOnxrKxez7QoZROZMYrinttFZ/V5rbObEM9E5AR5Tv/Fr4IBywoS8ZtN16Xb+fZmibfU91yq9O2RYSvscncU2qEYmmaTenM0QlUO80ZKqPsM5JkgCNdcYZTUeHclWeyER3dSImNtlSKiSBSSTHthb11fkudjzdiUXua0NKVWyYuAOoDMcpXbD6NJmYqEA/iZ/AxtQt08pv0Mow581GPB0Uop5+qA2hCV85DpagE94a067sKcRui0rtkJzHem9k7xVL+2RoFm1fv3RnUkMwhAgMBAAECggEAAetkddzxrfc+7jgPylUIGb8pyoOUTC4Vqs/BgZI9xYAJksNT2QKRsFvHPfItNt4Ocqy8h4tnIL3GCU43C564B4p6AcjhE85GiN/O0BudPOKlfuQQ9mqExqMMHuYeQfz0cmzPDTSGMwWiv9v4KBH2pyvkCCAzNF6uG+rvawb4/NNVuiI7C8Ku/wYsamtbgjMZVOFFdScYgIw1BgA99RUU/fWBLMnTQkoyowSRb9eSmEUHjt/WQt+/QgKAT2WmuX4RhaGy0qcQLbNaJNKXdJ+PVhQrSiasINNtqYMa8GsQuuKsk3X8TCg9K6/lowivt5ruhyWcP2sx93zY/LGzIHgHcQKBgQDoZlcs9RWxTdGDdtH8kk0J/r+QtMijNzWI0a+t+ZsWOyd3rw+uM/8O4JTNP4Y98TvvxhJXewITbfiuOIbW1mxh8bnO/fcz7+RXZKgPDeoTeNo717tZFZGBEyUdH9M9Inqvht7+hjVDIMCYBDomYebdk3Xqo4mDBjLRdVNGrhGmVQKBgQDKS/MgTMK8Ktfnu1KzwCbn/FfHTOrp1a1t1wWPv9AW0rJPYeaP6lOkgIoO/1odG9qDDhdB6njqM+mKY5Yr3N94PHamHbwJUCmbkqEunCWpGzgcQZ1Q254xk9D7UKq/XUqW2WDqDq80GQeNial+fBc46yelQzokwdA+JdIFKoyinQKBgQCBems9V/rTAtkk1nFdt6EGXZEbLS3PiXXhGXo4gqV+OEzf6H/i/YMwJb2hsK+5GQrcps0XQihA7PctEb9GOMa/tu5fva0ZmaDtc94SLR1p5d4okyQFGPgtIp594HpPSEN0Qb9BrUJFeRz0VP6U3dzDPGHo7V4yyqRLgIN6EIcy1QKBgAqdh6mHPaTAHspDMyjJiYEc5cJIj/8rPkmIQft0FkhMUB0IRyAALNlyAUyeK61hW8sKvz+vPR8VEEk5xpSQp41YpuU6pDZc5YILZLfca8F+8yfQbZ/jll6Foi694efezl4yE/rUQG9cbOAJfEJt4o4TEOaEK5XoMbRBKc8pl22lAoGARTq0qOr9SStihRAy9a+8wi2WEwL4QHcmOjH7iAuJxy5b5TRDSjlk6h+0dnTItiFlTXdfpO8KhWA8EoSJVBZ1kcACQDFgMIA+VM+yXydtzMotOn21W4stfZ4I6dHFiujMsnKpNYVpQh3oCrJf4SeXiQDdiSCodqb1HlKkEc6naHQ=',
+ // 支付成功通知地址
+ 'notify_url' => '',
+ // 网页支付回跳地址
+ 'return_url' => '',
+];
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/config.php b/vendor/zoujingli/wechat-developer/_test/config.php
new file mode 100644
index 000000000..e6f807584
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/config.php
@@ -0,0 +1,45 @@
+ function ($name, $value, $expired = 360) {
+// var_dump(func_get_args());
+// },
+// 'get' => function ($name) {
+// var_dump(func_get_args());
+// },
+// 'del' => function ($name) {
+// var_dump(func_get_args());
+// },
+// 'put' => function ($name) {
+// var_dump(func_get_args());
+// },
+//];
+
+return [
+ 'token' => 'test',
+ 'appid' => 'wx60a43dd8161666d4',
+ 'appsecret' => 'b4e28746f1bd73b5c6684f5e01883c36',
+ 'encodingaeskey' => 'BJIUzE0gqlWy0GxfPp4J1oPTBmOrNDIGPNav1YFH5Z5',
+ // 配置商户支付参数
+ 'mch_id' => "1332187001",
+ 'mch_key' => 'A82DC5BD1F3359081049C568D8502BC5',
+ // 配置商户支付双向证书目录 (p12 | key,cert 二选一,两者都配置时p12优先)
+ 'ssl_p12' => __DIR__ . DIRECTORY_SEPARATOR . 'cert' . DIRECTORY_SEPARATOR . '1332187001_20181030_cert.p12',
+ // 'ssl_key' => __DIR__ . DIRECTORY_SEPARATOR . 'cert' . DIRECTORY_SEPARATOR . '1332187001_20181030_key.pem',
+ // 'ssl_cer' => __DIR__ . DIRECTORY_SEPARATOR . 'cert' . DIRECTORY_SEPARATOR . '1332187001_20181030_cert.pem',
+ // 配置缓存目录,需要拥有写权限
+ 'cache_path' => '',
+];
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/mini-login.php b/vendor/zoujingli/wechat-developer/_test/mini-login.php
new file mode 100644
index 000000000..c6f56bac0
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/mini-login.php
@@ -0,0 +1,24 @@
+ 'wx6bb7b70258da09c6',
+ 'appsecret' => '78b7b8d65bd67b078babf951d4342b42',
+];
+
+// 解码数据
+$iv = 'ltM/wT7hsAl0TijEBI4v/g==';
+$code = '013LyiTR0TwjC92QjJRR0mEsTR0LyiT3';
+$decode = 'eIoVtIC2YzLCnrwiIs1IBbXMvC0vyL8bo1IhD38fUQIRbk3lgTWa0Hdw/Ty7NTs3iu7YlqqZBti+cxd6dCfeXBUQwTO2QpbHg0WTeDAdrihsHRHm4dCWdfTx8rzDloGbNOIsKdRElIhUH5YFdiTr5AYiufUDb34cwJ4GNWLAUq4bR0dmFeVEi+3nfwe2MAjGYDl4aq719VLsHodOggK6lXZvM5wjoDyuZsK2dPqJr3/Ji30Z0mdyFq32R4uR3rtJH/h+Rj0+/QmE9QYG7Y6Z48hgPE8cpnhRQNwH49jnC/zKZ9wtDkQ/J8J3Ed2i58zcuY01v8IV+pZ8oBUKXfO5ha+APOxtBSTzyHraU/2RGo8UWtOF6h64OQZhd/UQQy362eyc/qoq8sF9JnEFRP0mRmTDJ+u9oyDhxswCu6x8V73ERWaJeEGSCyjiGpep7/DxZ6eSSBq36OB0BWBkJqsq9Q==';
+$sessionKey = 'OetNxl86B/yMpbwG6wtMEw==';
+
+// $mini = \We::WeMiniCrypt($config);
+// $mini = new WeMini\Crypt($config);
+$mini = \WeMini\Crypt::instance($config);
+
+echo '';
+//print_r($mini->session($code));
+print_r($mini->decode($iv, $sessionKey, $decode));
+//print_r($mini->userInfo($code, $iv, $decode));
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/mini-qrc.php b/vendor/zoujingli/wechat-developer/_test/mini-qrc.php
new file mode 100644
index 000000000..84ca9d689
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/mini-qrc.php
@@ -0,0 +1,25 @@
+ 'wx6bb7b70258da09c6',
+ 'appsecret' => '78b7b8d65bd67b078babf951d4342b42',
+];
+
+//We::config($config);
+
+// $mini = We::WeMiniQrcode($config);
+// $mini = new WeMini\Qrcode($config);
+$mini = \WeMini\Qrcode::instance($config);
+
+//echo '';
+try {
+ header('Content-type:image/jpeg'); //输出的类型
+// echo $mini->createDefault('pages/index?query=1');
+// echo $mini->createMiniScene('432432', 'pages/index/index');
+ echo $mini->createMiniPath('pages/index?query=1');
+} catch (Exception $e) {
+ var_dump($e->getMessage());
+}
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-download-bill.php b/vendor/zoujingli/wechat-developer/_test/pay-download-bill.php
new file mode 100644
index 000000000..ffac45b0f
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-download-bill.php
@@ -0,0 +1,42 @@
+ '20171001',
+ 'bill_type' => 'ALL',
+ ];
+ $result = $wechat->billDownload($options);
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-order-close.php b/vendor/zoujingli/wechat-developer/_test/pay-order-close.php
new file mode 100644
index 000000000..eeead4af5
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-order-close.php
@@ -0,0 +1,39 @@
+closeOrder($options);
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-order-create.php b/vendor/zoujingli/wechat-developer/_test/pay-order-create.php
new file mode 100644
index 000000000..e9c9f20a7
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-order-create.php
@@ -0,0 +1,55 @@
+ '测试商品',
+ 'out_trade_no' => time(),
+ 'total_fee' => '1',
+ 'openid' => 'o38gpszoJoC9oJYz3UHHf6bEp0Lo',
+ 'trade_type' => 'JSAPI',
+ 'notify_url' => 'http://a.com/text.html',
+ 'spbill_create_ip' => '127.0.0.1',
+ ];
+ // 生成预支付码
+ $result = $wechat->createOrder($options);
+ // 创建JSAPI参数签名
+ $options = $wechat->createParamsForJsApi($result['prepay_id']);
+
+ echo '';
+ echo "\n--- 创建预支付码 ---\n";
+ var_export($result);
+
+ echo "\n\n--- JSAPI 及 H5 参数 ---\n";
+ var_export($options);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-order-notify.php b/vendor/zoujingli/wechat-developer/_test/pay-order-notify.php
new file mode 100644
index 000000000..7484aace9
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-order-notify.php
@@ -0,0 +1,44 @@
+getNotify();
+ if ($data['return_code'] === 'SUCCESS' && $data['result_code'] === 'SUCCESS') {
+ // @todo 去更新下原订单的支付状态
+ $order_no = $data['out_trade_no'];
+
+ // 返回接收成功的回复
+ ob_clean();
+ echo $wechat->getNotifySuccessReply();
+ }
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-order-query.php b/vendor/zoujingli/wechat-developer/_test/pay-order-query.php
new file mode 100644
index 000000000..b7edf34a1
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-order-query.php
@@ -0,0 +1,42 @@
+ '1008450740201411110005820873',
+// 'out_trade_no' => '商户订单号',
+ ];
+ $result = $wechat->queryOrder($options);
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-redpack-create.php b/vendor/zoujingli/wechat-developer/_test/pay-redpack-create.php
new file mode 100644
index 000000000..c45dadeb4
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-redpack-create.php
@@ -0,0 +1,53 @@
+ time(),
+ 're_openid' => 'o38gps3vNdCqaggFfrBRCRikwlWY',
+ 'send_name' => '商户名称😍',
+ 'act_name' => '活动名称',
+ 'total_amount' => '100',
+ 'total_num' => '1',
+ 'wishing' => '感谢您参加猜灯谜活动,祝您元宵节快乐!',
+ 'remark' => '猜越多得越多,快来抢!',
+ 'client_ip' => '127.0.0.1',
+ ];
+ // 发送红包记录
+ $result = $wechat->create($options);
+ echo '';
+ var_export($result);
+ // 查询红包记录
+ $result = $wechat->query($options['mch_billno']);
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-refund-create.php b/vendor/zoujingli/wechat-developer/_test/pay-refund-create.php
new file mode 100644
index 000000000..8e3df01ff
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-refund-create.php
@@ -0,0 +1,44 @@
+ '1008450740201411110005820873',
+ 'out_refund_no' => '商户退款单号',
+ 'total_fee' => '1',
+ 'refund_fee' => '1',
+ ];
+ $result = $wechat->createRefund($options);
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-refund-query.php b/vendor/zoujingli/wechat-developer/_test/pay-refund-query.php
new file mode 100644
index 000000000..fa63acc9e
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-refund-query.php
@@ -0,0 +1,44 @@
+ '1008450740201411110005820873',
+ // 'out_trade_no' => '商户订单号',
+ // 'out_refund_no' => '商户退款单号'
+ // 'refund_id' => '微信退款单号',
+ ];
+ $result = $wechat->queryRefund($options);
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-transfers-create.php b/vendor/zoujingli/wechat-developer/_test/pay-transfers-create.php
new file mode 100644
index 000000000..659d888e2
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-transfers-create.php
@@ -0,0 +1,48 @@
+ time(),
+ 'openid' => 'o38gps3vNdCqaggFfrBRCRikwlWY',
+ 'check_name' => 'NO_CHECK',
+ 'amount' => '100',
+ 'desc' => '企业付款操作说明信息',
+ 'spbill_create_ip' => '127.0.0.1',
+ ];
+ $result = $wechat->createTransfers($options);
+ echo '';
+ var_export($result);
+ $result = $wechat->queryTransfers($options['partner_trade_no']);
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/pay-transfersbank-create.php b/vendor/zoujingli/wechat-developer/_test/pay-transfersbank-create.php
new file mode 100644
index 000000000..ca3979edf
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/pay-transfersbank-create.php
@@ -0,0 +1,46 @@
+ time(),
+ 'enc_bank_no' => '6212263602037318102',
+ 'enc_true_name' => '邹景立',
+ 'bank_code' => '1002',
+ 'amount' => '100',
+ 'desc' => '打款测试',
+ ];
+ echo '';
+ $result = $wechat->createTransfersBank($options);
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/wechat-jssdk-sign.php b/vendor/zoujingli/wechat-developer/_test/wechat-jssdk-sign.php
new file mode 100644
index 000000000..6a3b6985c
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/wechat-jssdk-sign.php
@@ -0,0 +1,38 @@
+getJsSign('http://a.com/test.php');
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/wechat-menu-get.php b/vendor/zoujingli/wechat-developer/_test/wechat-menu-get.php
new file mode 100644
index 000000000..5447aad17
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/wechat-menu-get.php
@@ -0,0 +1,38 @@
+get();
+
+ var_export($result);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/wechat-qrcode-create.php b/vendor/zoujingli/wechat-developer/_test/wechat-qrcode-create.php
new file mode 100644
index 000000000..256e30ad9
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/wechat-qrcode-create.php
@@ -0,0 +1,42 @@
+create('场景内容');
+ echo var_export($result, true) . PHP_EOL;
+
+ // 5. 创建二维码链接
+ $url = $wechat->url($result['ticket']);
+ echo var_export($url, true);
+
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/_test/wechat-user-get.php b/vendor/zoujingli/wechat-developer/_test/wechat-user-get.php
new file mode 100644
index 000000000..d42cb5632
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/_test/wechat-user-get.php
@@ -0,0 +1,45 @@
+getUserList();
+
+ echo '';
+ var_export($result);
+
+ // 5. 批量获取用户资料
+ foreach (array_chunk($result['data']['openid'], 100) as $item) {
+ $userList = $wechat->getBatchUserInfo($item);
+ var_export($userList);
+ }
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/composer.json b/vendor/zoujingli/wechat-developer/composer.json
new file mode 100644
index 000000000..d8ee7c14c
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/composer.json
@@ -0,0 +1,42 @@
+{
+ "type": "library",
+ "name": "zoujingli/wechat-developer",
+ "homepage": "https://github.com/zoujingli/WeChatDeveloper",
+ "description": "WeChat platform and WeChat payment development tools",
+ "license": "MIT",
+ "authors": [
+ {
+ "name": "Anyon",
+ "email": "zoujingli@qq.com",
+ "homepage": "http://ctolog.com"
+ }
+ ],
+ "keywords": [
+ "WePay",
+ "AliPay",
+ "WeMini",
+ "WeChat",
+ "WeChatPay",
+ "WeChatDeveloper"
+ ],
+ "require": {
+ "php": ">=5.4",
+ "ext-json": "*",
+ "ext-curl": "*",
+ "ext-libxml": "*",
+ "ext-openssl": "*",
+ "ext-mbstring": "*",
+ "ext-simplexml": "*"
+ },
+ "autoload": {
+ "classmap": [
+ "We.php"
+ ],
+ "psr-4": {
+ "WePay\\": "WePay",
+ "WeMini\\": "WeMini",
+ "WeChat\\": "WeChat",
+ "AliPay\\": "AliPay"
+ }
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/wechat-developer/include.php b/vendor/zoujingli/wechat-developer/include.php
new file mode 100644
index 000000000..e88c70ece
--- /dev/null
+++ b/vendor/zoujingli/wechat-developer/include.php
@@ -0,0 +1,27 @@
+ 'test',
+ 'appid' => 'wx60a43dd8161666d4',
+ 'appsecret' => '71308e96a204296c57d7cd4b21b883e8',
+ 'encodingaeskey' => 'BJIUzE0gqlWy0GxfPp4J1oPTBmOrNDIGPNav1YFH5Z5',
+ // 配置商户支付参数(可选,在使用支付功能时需要)
+ 'mch_id' => "1235704602",
+ 'mch_key' => 'IKI4kpHjU94ji3oqre5zYaQMwLHuZPmj',
+ // 配置商户支付双向证书目录(可选,在使用退款|打款|红包时需要)
+ 'ssl_key' => '',
+ 'ssl_cer' => '',
+ // 缓存目录配置(可选,需拥有读写权限)
+ 'cache_path' => '',
+];
+```
+
+3.1 实例指定接口
+```php
+try {
+
+ // 实例对应的接口对象
+ $user = new \WeChat\User($config);
+
+ // 调用接口对象方法
+ $list = $user->getUserList();
+
+ // 处理返回的结果
+ echo '';
+ var_export($list);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
+```
+
+微信支付
+---
+```php
+ // 创建接口实例
+ $wechat = new \WeChat\Pay($config);
+
+ // 组装参数,可以参考官方商户文档
+ $options = [
+ 'body' => '测试商品',
+ 'out_trade_no' => time(),
+ 'total_fee' => '1',
+ 'openid' => 'o38gpszoJoC9oJYz3UHHf6bEp0Lo',
+ 'trade_type' => 'JSAPI',
+ 'notify_url' => 'http://a.com/text.html',
+ 'spbill_create_ip' => '127.0.0.1',
+ ];
+
+try {
+
+ // 生成预支付码
+ $result = $wechat->createOrder($options);
+
+ // 创建JSAPI参数签名
+ $options = $wechat->createParamsForJsApi($result['prepay_id']);
+
+ // @todo 把 $options 传到前端用js发起支付就可以了
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
+```
+* 更多功能请阅读测试代码或SDK封装源码
+
+支付宝支付
+----
+* 支付参数配置(可用沙箱模式)
+```php
+$config = [
+ // 沙箱模式
+ 'debug' => true,
+ // 应用ID
+ 'appid' => '2016090900468879',
+ // 支付宝公钥(1行填写)
+ 'public_key' => 'MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtU71NY53UDGY7JNvLYAhsNa+taTF6KthIHJmGgdio9bkqeJGhHk6ttkTKkLqFgwIfgAkHpdKiOv1uZw6gVGZ7TCu5LfHTqKrCd6Uz+N7hxhY+4IwicLgprcV1flXQLmbkJYzFMZqkXGkSgOsR2yXh4LyQZczgk9N456uuzGtRy7MoB4zQy34PLUkkxR6W1B2ftNbLRGXv6tc7p/cmDcrY6K1bSxnGmfRxFSb8lRfhe0V0UM6pKq2SGGSeovrKHN0OLp+Nn5wcULVnFgATXGCENshRlp96piPEBFwneXs19n+sX1jx60FTR7/rME3sW3AHug0fhZ9mSqW4x401WjdnwIDAQAB',
+ // 支付宝私钥(1行填写)
+ 'private_key' => 'MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQC3pbN7esinxgjE8uxXAsccgGNKIq+PR1LteNTFOy0fsete43ObQCrzd9DO0zaUeBUzpIOnxrKxez7QoZROZMYrinttFZ/V5rbObEM9E5AR5Tv/Fr4IBywoS8ZtN16Xb+fZmibfU91yq9O2RYSvscncU2qEYmmaTenM0QlUO80ZKqPsM5JkgCNdcYZTUeHclWeyER3dSImNtlSKiSBSSTHthb11fkudjzdiUXua0NKVWyYuAOoDMcpXbD6NJmYqEA/iZ/AxtQt08pv0Mow581GPB0Uop5+qA2hCV85DpagE94a067sKcRui0rtkJzHem9k7xVL+2RoFm1fv3RnUkMwhAgMBAAECggEAAetkddzxrfc+7jgPylUIGb8pyoOUTC4Vqs/BgZI9xYAJksNT2QKRsFvHPfItNt4Ocqy8h4tnIL3GCU43C564B4p6AcjhE85GiN/O0BudPOKlfuQQ9mqExqMMHuYeQfz0cmzPDTSGMwWiv9v4KBH2pyvkCCAzNF6uG+rvawb4/NNVuiI7C8Ku/wYsamtbgjMZVOFFdScYgIw1BgA99RUU/fWBLMnTQkoyowSRb9eSmEUHjt/WQt+/QgKAT2WmuX4RhaGy0qcQLbNaJNKXdJ+PVhQrSiasINNtqYMa8GsQuuKsk3X8TCg9K6/lowivt5ruhyWcP2sx93zY/LGzIHgHcQKBgQDoZlcs9RWxTdGDdtH8kk0J/r+QtMijNzWI0a+t+ZsWOyd3rw+uM/8O4JTNP4Y98TvvxhJXewITbfiuOIbW1mxh8bnO/fcz7+RXZKgPDeoTeNo717tZFZGBEyUdH9M9Inqvht7+hjVDIMCYBDomYebdk3Xqo4mDBjLRdVNGrhGmVQKBgQDKS/MgTMK8Ktfnu1KzwCbn/FfHTOrp1a1t1wWPv9AW0rJPYeaP6lOkgIoO/1odG9qDDhdB6njqM+mKY5Yr3N94PHamHbwJUCmbkqEunCWpGzgcQZ1Q254xk9D7UKq/XUqW2WDqDq80GQeNial+fBc46yelQzokwdA+JdIFKoyinQKBgQCBems9V/rTAtkk1nFdt6EGXZEbLS3PiXXhGXo4gqV+OEzf6H/i/YMwJb2hsK+5GQrcps0XQihA7PctEb9GOMa/tu5fva0ZmaDtc94SLR1p5d4okyQFGPgtIp594HpPSEN0Qb9BrUJFeRz0VP6U3dzDPGHo7V4yyqRLgIN6EIcy1QKBgAqdh6mHPaTAHspDMyjJiYEc5cJIj/8rPkmIQft0FkhMUB0IRyAALNlyAUyeK61hW8sKvz+vPR8VEEk5xpSQp41YpuU6pDZc5YILZLfca8F+8yfQbZ/jll6Foi694efezl4yE/rUQG9cbOAJfEJt4o4TEOaEK5XoMbRBKc8pl22lAoGARTq0qOr9SStihRAy9a+8wi2WEwL4QHcmOjH7iAuJxy5b5TRDSjlk6h+0dnTItiFlTXdfpO8KhWA8EoSJVBZ1kcACQDFgMIA+VM+yXydtzMotOn21W4stfZ4I6dHFiujMsnKpNYVpQh3oCrJf4SeXiQDdiSCodqb1HlKkEc6naHQ=',
+ // 支付成功通知地址
+ 'notify_url' => '', // 可以应用的时候配置哦
+ // 网页支付回跳地址
+ 'return_url' => '', // 可以应用的时候配置哦
+];
+```
+* 支付宝发起PC网站支付
+```php
+// 参考公共参数 https://docs.open.alipay.com/203/107090/
+$config['notify_url'] = 'http://pay.thinkadmin.top/test/alipay-notify.php';
+$config['return_url'] = 'http://pay.thinkadmin.top/test/alipay-success.php';
+
+try {
+
+ // 实例支付对象
+ $pay = We::AliPayWeb($config);
+ // $pay = new \AliPay\Web($config);
+
+ // 参考链接:https://docs.open.alipay.com/api_1/alipay.trade.page.pay
+ $result = $pay->apply([
+ 'out_trade_no' => time(), // 商户订单号
+ 'total_amount' => '1', // 支付金额
+ 'subject' => '支付订单描述', // 支付订单描述
+ ]);
+
+ echo $result; // 直接输出HTML(提交表单跳转)
+
+} catch (Exception $e) {
+
+ // 异常处理
+ echo $e->getMessage();
+
+}
+```
+* 支付宝发起手机网站支付
+```php
+// 参考公共参数 https://docs.open.alipay.com/203/107090/
+$config['notify_url'] = 'http://pay.thinkadmin.top/test/alipay-notify.php';
+$config['return_url'] = 'http://pay.thinkadmin.top/test/alipay-success.php';
+
+try {
+
+ // 实例支付对象
+ $pay = We::AliPayWap($config);
+ // $pay = new \AliPay\Wap($config);
+
+ // 参考链接:https://docs.open.alipay.com/api_1/alipay.trade.wap.pay
+ $result = $pay->apply([
+ 'out_trade_no' => time(), // 商户订单号
+ 'total_amount' => '1', // 支付金额
+ 'subject' => '支付订单描述', // 支付订单描述
+ ]);
+
+ echo $result; // 直接输出HTML(提交表单跳转)
+
+} catch (Exception $e) {
+
+ // 异常处理
+ echo $e->getMessage();
+
+}
+```
+* 更多功能请阅读测试代码或SDK封装源码
+
+开源协议
+----
+* WeChatDeveloper 基于`MIT`协议发布,任何人可以用在任何地方,不受约束
+* WeChatDeveloper 部分代码来自互联网,若有异议,可以联系作者进行删除
+
+赞助支持
+----
+
+
+
diff --git a/vendor/zoujingli/weopen-developer/.gitignore b/vendor/zoujingli/weopen-developer/.gitignore
new file mode 100644
index 000000000..93d1e095c
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/.gitignore
@@ -0,0 +1,4 @@
+.git
+.idea
+vendor
+composer.lock
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/LICENSE b/vendor/zoujingli/weopen-developer/LICENSE
new file mode 100644
index 000000000..e19cbbdf9
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (c) 2019 邹景立
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+SOFTWARE.
diff --git a/vendor/zoujingli/weopen-developer/README.md b/vendor/zoujingli/weopen-developer/README.md
new file mode 100644
index 000000000..894004423
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/README.md
@@ -0,0 +1,127 @@
+[](https://packagist.org/packages/zoujingli/weopen-developer)
+[](https://packagist.org/packages/zoujingli/weopen-developer)
+[](https://packagist.org/packages/zoujingli/weopen-developer)
+[](https://packagist.org/packages/zoujingli/weopen-developer)
+
+# WeOpenDeveloper
+WeOpenDeveloper 为微信开放平台服务开发工具,基于 WeChatDeveloper 可对公众号进行管理。
+更多功能可以参考下面的文档。
+
+Documentation
+--
+PHP开发技术交流(QQ群 513350915)
+
+[](http://shang.qq.com/wpa/qunwpa?idkey=ae25cf789dafbef62e50a980ffc31242f150bc61a61164458216dd98c411832a)
+
+> WeChatDeveloper 是基于官方接口封装,在做微信开发前,必需先阅读微信官方文档。
+>* 微信官方文档:http://mp.weixin.qq.com/wiki
+>* 开放平台文档:https://open.weixin.qq.com
+>* 商户支付文档:https://pay.weixin.qq.com/wiki/doc/api/index.html
+
+> 针对 WeChatDeveloper 也有一准备了帮助资料可供参考。
+>* WeChatDeveloper 文档:http://www.kancloud.cn/zoujingli/wechat-developer
+
+Repository
+--
+ WeOpenDeveloper 为开源项目,允许把它用于任何地方,不受任何约束,欢迎 fork 项目。
+>* GitHub 托管地址:https://github.com/zoujingli/WeOpenDeveloper
+>* OSChina 托管地址:http://git.oschina.net/zoujingli/WeOpenDeveloper
+
+更多开发可以参考项目 [ThinkService](https://github.com/zoujingli/ThinkService) 。
+此项目已经实现对接,[ThinkAdmin](https://github.com/zoujingli/ThinkAdmin) + [ThinkService](https://github.com/zoujingli/ThinkService) 组合。
+
+Install
+--
+* 通过 Composer 来管理安装
+```shell
+# 首次安装 线上版本(稳定)
+composer require zoujingli/weopen-developer
+
+# 首次安装 开发版本
+composer require zoujingli/weopen-developer dev-master
+
+# 更新 WeChatDeveloper
+composer update zoujingli/weopen-developer
+```
+
+* 接口实例所需参数
+```php
+
+// 配置参数(可以公众号服务平台获取)
+$config = [
+ 'component_appid' => 'wx4e63e993e222df8d',
+ 'component_token' => 'P8QHTIxpBEq88IrxatqhgpBm2OAQROkI',
+ 'component_appsecret' => '7cfa1afa87a41e2ea3445cea015c0974',
+ 'component_encodingaeskey' => 'L5uFIa0U6KLalPyXckyqoVIJYLhsfrg8k9YzybZIHsx',
+];
+
+// 注册授权公众号 AccessToken 处理
+$config['GetAccessTokenCallback'] = function ($authorizer_appid) use ($config) {
+ $open = new \WeOpen\Service($config);
+ $authorizer_refresh_token = ''; // 通过$authorizer_appid从数据库去找吧,在授权绑定的时候获取
+ $result = $open->refreshAccessToken($authorizer_appid, $authorizer_refresh_token);
+ if (empty($result['authorizer_access_token'])) {
+ throw new \WeChat\Exceptions\InvalidResponseException($result['errmsg'], '0');
+ }
+ $data = [
+ 'authorizer_access_token' => $result['authorizer_access_token'],
+ 'authorizer_refresh_token' => $result['authorizer_refresh_token'],
+ ];
+ // 需要把$data记录到数据库
+ return $result['authorizer_access_token'];
+};
+```
+
+* Ticket 接收处理
+```php
+
+try{
+
+ // 实例公众号服务接口
+ $server = new \WeOpen\Service($config);
+
+ // 获取并更新Ticket推送
+ if (!($data = $server->getComonentTicket())) {
+ return "Ticket event handling failed.";
+ }
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
+```
+
+* 实例指定接口
+```php
+try{
+
+ // 实例公众号服务接口
+ $open = new \WeOpen\Service($config);
+
+ // 获取公众号接口操作实例
+ $wechat = $open->instance('User', 'wx60a43dd8161666d4');
+
+ // 获取公众号粉丝列表
+ $list = $wechat->getUserList();
+ var_export($list);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
+
+```
+
+Copyright
+--
+* WeOpenDeveloper 基于`MIT`协议发布,任何人可以用在任何地方,不受约束
+* WeOpenDeveloper 部分代码来自互联网,若有异议,可以联系作者进行删除
+
+
+Sponsor
+--
+
diff --git a/vendor/zoujingli/weopen-developer/WeChat/Bind.php b/vendor/zoujingli/weopen-developer/WeChat/Bind.php
new file mode 100644
index 000000000..31994873a
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeChat/Bind.php
@@ -0,0 +1,80 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $this->config->get('appid')]);
+ }
+
+ /**
+ * 将公众号绑定到开放平台帐号下
+ * @param string $openidAppid 开放平台帐号APPID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function link($openidAppid)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/open/bind?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $this->config->get('appid'), 'open_appid' => $openidAppid]);
+ }
+
+ /**
+ * 将公众号从开放平台帐号下解绑
+ * @param string $openidAppid 开放平台帐号APPID
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function unlink($openidAppid)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/open/unbind?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $this->config->get('appid'), 'open_appid' => $openidAppid]);
+ }
+
+ /**
+ * 获取公众号所绑定的开放平台帐号
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function get()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/open/get?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $this->config->get('appid')]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeChat/Mini.php b/vendor/zoujingli/weopen-developer/WeChat/Mini.php
new file mode 100644
index 000000000..65ddb49a7
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeChat/Mini.php
@@ -0,0 +1,87 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, [], true);
+ }
+
+ /**
+ * 2. 关联小程序
+ * @param string $miniAppid 小程序appid
+ * @param integer $notifyUsers 是否发送模板消息通知公众号粉丝
+ * @param integer $showProfile 是否展示公众号主页中
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function linkWxamp($miniAppid, $notifyUsers = 1, $showProfile = 1)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/wxopen/wxamplink?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, [
+ 'appid' => $miniAppid,
+ 'notify_users' => $notifyUsers,
+ 'show_profile' => $showProfile,
+ ]);
+ }
+
+ /**
+ * 3.解除已关联的小程序
+ * @param string $miniAppid 小程序appid
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function unlinkWxamp($miniAppid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/wxopen/wxampunlink?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $miniAppid]);
+ }
+
+ /**
+ * 第三方平台调用快速注册API完成注册
+ * @param string $ticket 公众号扫码授权的凭证(公众平台扫码页面回跳到第三方平台时携带)
+ * @return array
+ * @throws Exceptions\InvalidResponseException
+ * @throws Exceptions\LocalCacheException
+ */
+ public function fastRegister($ticket)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/account/fastregister?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['ticket' => $ticket]);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/Account.php b/vendor/zoujingli/weopen-developer/WeMini/Account.php
new file mode 100644
index 000000000..569b33019
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/Account.php
@@ -0,0 +1,198 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 2.2 小程序名称设置及改名
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setNickname(array $data)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/setnickname?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 2.3 小程序改名审核状态查询
+ * @param integer $audit_id 审核单id
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function queryChangeNicknameAuditStatus($audit_id)
+ {
+ $url = "https://api.weixin.qq.com/wxa/api_wxa_querynickname?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['audit_id' => $audit_id]);
+ }
+
+ /**
+ *
+ * 2.4 微信认证名称检测
+ * @param string $nickname 微信认证名称
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function checkWxVerifyNickname($nickname)
+ {
+ $url = "https://api.weixin.qq.com/wxa/api_wxa_querynickname?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['nick_name' => $nickname]);
+ }
+
+ /**
+ * 2.5 修改头像
+ * @param string $headImgMediaId 头像素材media_id
+ * @param integer $x1 裁剪框左上角x坐标(取值范围:[0, 1])
+ * @param integer $y1 裁剪框左上角y坐标(取值范围:[0, 1])
+ * @param integer $x2 裁剪框右下角x坐标(取值范围:[0, 1])
+ * @param integer $y2 裁剪框右下角y坐标(取值范围:[0, 1])
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function modifyHeadImage($headImgMediaId, $x1 = 0, $y1 = 0, $x2 = 1, $y2 = 1)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/account/modifyheadimage?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['head_img_media_id' => $headImgMediaId]);
+ }
+
+ /**
+ * 2.6 修改功能介绍
+ * @param string $signature 功能介绍(简介)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException]
+ */
+ public function modifySignature($signature)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/account/modifysignature?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['signature' => $signature]);
+ }
+
+ /**
+ * 2.7.3跳转至第三方平台,第三方平台调用快速注册API完成管理员换绑。
+ * @param string $taskid 换绑管理员任务序列号(公众平台最终点击提交回跳到第三方平台时携带)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function componentreBindAdmin($taskid)
+ {
+ $url = 'https://api.weixin.qq.com/cgi- bin/account/componentrebindadmin?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['taskid' => $taskid]);
+ }
+
+ /**
+ * 3.1 获取账号可以设置的所有类目
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAllCategories()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/getallcategories?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 3.2 添加类目
+ * @param array $categories
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addCategory($categories)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/addcategory?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['categories' => $categories]);
+ }
+
+ /**
+ * 3.3 删除类目
+ * @param string $first 一级类目ID
+ * @param string $second 二级类目ID
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function delCategroy($first, $second)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/deletecategory?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['first' => $first, 'second' => $second]);
+ }
+
+ /**
+ * 3.4 获取账号已经设置的所有类目
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getCategory()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/getcategory?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 3.5 修改类目
+ * @param string $first 一级类目ID
+ * @param string $second 二级类目ID
+ * @param array $certicates
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function modifyCategory($first, $second, $certicates)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/modifycategory?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['first' => $first, 'second' => $second, 'categories' => $categories]);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/Basic.php b/vendor/zoujingli/weopen-developer/WeMini/Basic.php
new file mode 100644
index 000000000..dca27944a
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/Basic.php
@@ -0,0 +1,54 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['status' => $status], true);
+ }
+
+ /**
+ * 2. 查询小程序当前隐私设置(是否可被搜索)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWxaSearchStatus()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/getwxasearchstatus?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/Code.php b/vendor/zoujingli/weopen-developer/WeMini/Code.php
new file mode 100644
index 000000000..1b9f264af
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/Code.php
@@ -0,0 +1,346 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ $data = [
+ 'template_id' => $templateId,
+ 'ext_json' => $extJson,
+ 'user_version' => $userVersion,
+ 'user_desc' => $userDesc,
+ ];
+ return $this->httpPostForJson($url, $data, true);
+ }
+
+ /**
+ * 2. 获取体验小程序的体验二维码
+ * @param null|string $path 指定体验版二维码跳转到某个具体页面
+ * @param null|string $outType 指定输出类型
+ * @return array|bool|string
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getQrcode($path = null, $outType = null)
+ {
+ $pathStr = is_null($path) ? '' : ("&path=" . urlencode($path));
+ $url = "https://api.weixin.qq.com/wxa/get_qrcode?access_token=ACCESS_TOKEN{$pathStr}";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ $result = Tools::get($url);
+ if (json_decode($result)) {
+ return Tools::json2arr($result);
+ }
+ return is_null($outType) ? $result : $outType($result);
+ }
+
+ /**
+ * 3. 获取授权小程序帐号的可选类目
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getCategory()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/get_category?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 4. 获取小程序的第三方提交代码的页面配置(仅供第三方开发者代小程序调用)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getPage()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/get_page?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 5. 将第三方提交的代码包提交审核(仅供第三方开发者代小程序调用)
+ * @param array $itemList 提交审核项的一个列表
+ * @param string $feedbackInfo 反馈内容不超过200字
+ * @param string $feedbackStuff 图片 media_id 列表
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function submitAudit(array $itemList, $feedbackInfo = '', $feedbackStuff = '')
+ {
+ $url = 'https://api.weixin.qq.com/wxa/submit_audit?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['item_list' => $itemList, 'feedback_info' => '', 'feedback_stuff' => $feedbackStuff], true);
+ }
+
+ /**
+ * 6. 获取审核结果
+ * @return array
+ */
+ public function getNotify()
+ {
+ return Tools::xml2arr(file_get_contents('php://input'));
+ }
+
+ /**
+ * 7. 查询某个指定版本的审核状态(仅供第三方代小程序调用)
+ * @param string $auditid 提交审核时获得的审核id
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAuditstatus($auditid)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/get_auditstatus?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['auditid' => $auditid], true);
+ }
+
+ /**
+ * 8、查询最新一次提交的审核状态(仅供第三方代小程序调用)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getLatestAuditatus()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/get_latest_auditstatus?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 9、发布已通过审核的小程序(仅供第三方代小程序调用)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function publishRelease()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/release?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, [], true);
+ }
+
+ /**
+ * 10、修改小程序线上代码的可见状态(仅供第三方代小程序调用)
+ * @param string $action 设置可访问状态,发布后默认可访问,close为不可见,open为可见
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function changeVisitStatus($action)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/change_visitstatus?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['action' => $action], true);
+ }
+
+ /**
+ * 11. 小程序版本回退(仅供第三方代小程序调用)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function revertCodeRelease()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/revertcoderelease?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 12. 查询当前设置的最低基础库版本及各版本用户占比 (仅供第三方代小程序调用)
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getWeappSupportVersion()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/getweappsupportversion?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, []);
+ }
+
+ /**
+ * 13. 设置最低基础库版本(仅供第三方代小程序调用)
+ * @param string $version 版本
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setWeappSupportVersion($version)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/setweappsupportversion?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['version' => $version]);
+ }
+
+ /**
+ * 14. 设置小程序“扫普通链接二维码打开小程序”能力
+ * (1) 增加或修改二维码规则
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addQrcodeJump(array $data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/qrcodejumpadd?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 14. 设置小程序“扫普通链接二维码打开小程序”能力
+ * (2) 获取已设置的二维码规则
+ * @param array $data
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getQrcodeJump(array $data)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/qrcodejumpget?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data);
+ }
+
+ /**
+ * 14. 设置小程序“扫普通链接二维码打开小程序”能力
+ * (3)获取校验文件名称及内容
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function downloadQrcodeJump()
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/qrcodejumpdownload?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, []);
+ }
+
+ /**
+ * 14. 设置小程序“扫普通链接二维码打开小程序”能力
+ * (4)删除已设置的二维码规则
+ * @param string $prefix 二维码规则
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function deleteQrcodeJump($prefix)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/qrcodejumpdelete?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['prefix' => $prefix]);
+ }
+
+ /**
+ * 14. 设置小程序“扫普通链接二维码打开小程序”能力
+ * (5)发布已设置的二维码规则
+ * @param string $prefix 二维码规则
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function publishQrcodeJump($prefix)
+ {
+ $url = 'https://api.weixin.qq.com/cgi-bin/wxopen/qrcodejumppublish?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['prefix' => $prefix]);
+ }
+
+ /**
+ * 16. 小程序审核撤回
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function undoCodeAudit()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/undocodeaudit?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 17.小程序分阶段发布
+ * (1)分阶段发布接口
+ * @param integer $gray_percentage 灰度的百分比,1到100的整数
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function grayRelease($gray_percentage)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/grayrelease?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['gray_percentage' => $gray_percentage]);
+ }
+
+ /**
+ * 17.小程序分阶段发布
+ * (2)取消分阶段发布
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function revertGrayRelease()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/revertgrayrelease?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 17.小程序分阶段发布
+ * (3)查询当前分阶段发布详情
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getGrayreLeasePlan()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/getgrayreleaseplan?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpGetForJson($url);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/Domain.php b/vendor/zoujingli/weopen-developer/WeMini/Domain.php
new file mode 100644
index 000000000..7751e2a77
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/Domain.php
@@ -0,0 +1,60 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, $data, true);
+ }
+
+ /**
+ * 2、设置小程序业务域名(仅供第三方代小程序调用)
+ * @param string $action add添加, delete删除, set覆盖, get获取。
+ * 当参数是get时不需要填webviewdomain字段。
+ * 如果没有action字段参数,则默认见开放平台第三方登记的小程序业务域名全部添加到授权的小程序中
+ * @param string $webviewdomain 小程序业务域名,当action参数是get时不需要此字段
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setWebViewDomain($action, $webviewdomain)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/setwebviewdomain?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['action' => $action, 'webviewdomain' => $webviewdomain], true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/Tester.php b/vendor/zoujingli/weopen-developer/WeMini/Tester.php
new file mode 100644
index 000000000..9b0828e91
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/Tester.php
@@ -0,0 +1,68 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['wechatid' => $testid], true);
+ }
+
+ /**
+ * 2、解除绑定小程序的体验者
+ * @param string $testid 微信号
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function unbindTester($testid)
+ {
+ $url = 'https://api.weixin.qq.com/wxa/unbind_tester?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['wechatid' => $testid], true);
+ }
+
+ /**
+ * 3. 获取体验者列表
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getTesterList()
+ {
+ $url = 'https://api.weixin.qq.com/wxa/memberauth?access_token=ACCESS_TOKEN';
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['action' => 'get_experiencer'], true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeMini/User.php b/vendor/zoujingli/weopen-developer/WeMini/User.php
new file mode 100644
index 000000000..d3a1f8ffe
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeMini/User.php
@@ -0,0 +1,83 @@
+registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $appid], true);
+ }
+
+ /**
+ * 2. 将公众号/小程序绑定到开放平台帐号下
+ * @param string $appid 授权公众号或小程序的appid
+ * @param string $openAppid 开放平台帐号appid
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function bind($appid, $openAppid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/open/bind?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $appid, 'open_appid' => $openAppid]);
+ }
+
+ /**
+ * 3. 将公众号/小程序从开放平台帐号下解绑
+ * @param string $appid 授权公众号或小程序的appid
+ * @param string $openAppid 开放平台帐号appid
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function unbind($appid, $openAppid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/open/unbind?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $appid, 'open_appid' => $openAppid]);
+ }
+
+ /**
+ * 3. 获取公众号/小程序所绑定的开放平台帐号
+ * @param string $appid 授权公众号或小程序的appid
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function get($appid)
+ {
+ $url = "https://api.weixin.qq.com/cgi-bin/open/get?access_token=ACCESS_TOKEN";
+ $this->registerApi($url, __FUNCTION__, func_get_args());
+ return $this->httpPostForJson($url, ['appid' => $appid]);
+ }
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeOpen/Login.php b/vendor/zoujingli/weopen-developer/WeOpen/Login.php
new file mode 100644
index 000000000..2c45101c5
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeOpen/Login.php
@@ -0,0 +1,114 @@
+config = new DataArray($options);
+ if (empty($options['appid'])) {
+ throw new InvalidArgumentException("Missing Config -- [appid]");
+ }
+ if (empty($options['appsecret'])) {
+ throw new InvalidArgumentException("Missing Config -- [appsecret]");
+ }
+ }
+
+ /**
+ * 第一步:请求CODE
+ * @param string $redirectUri 请使用urlEncode对链接进行处理
+ * @return string
+ */
+ public function auth($redirectUri)
+ {
+ $appid = $this->config->get('appid');
+ $redirectUri = urlencode($redirectUri);
+ return "https://open.weixin.qq.com/connect/qrconnect?appid={$appid}&redirect_uri={$redirectUri}&response_type=code&scope=snsapi_login&state={$appid}#wechat_redirect";
+ }
+
+ /**
+ * 第二步:通过code获取access_token
+ * @return mixed
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAccessToken()
+ {
+ $appid = $this->config->get('appid');
+ $secret = $this->config->get('appsecret');
+ $code = isset($_GET['code']) ? $_GET['code'] : '';
+ $url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid={$appid}&secret={$secret}&code={$code}&grant_type=authorization_code";
+ return json_decode(Tools::get($url));
+ }
+
+ /**
+ * 刷新AccessToken有效期
+ * @param string $refreshToken
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function refreshToken($refreshToken)
+ {
+ $appid = $this->config->get('appid');
+ $url = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid={$appid}&grant_type=refresh_token&refresh_token={$refreshToken}";
+ return json_decode(Tools::get($url));
+ }
+
+ /**
+ * 检验授权凭证(access_token)是否有效
+ * @param string $accessToken 调用凭证
+ * @param string $openid 普通用户的标识,对当前开发者帐号唯一
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function checkAccessToken($accessToken, $openid)
+ {
+ $url = "https://api.weixin.qq.com/sns/auth?access_token={$accessToken}&openid={$openid}";
+ return json_decode(Tools::get($url));
+ }
+
+ /**
+ * 获取用户个人信息(UnionID机制)
+ * @param string $accessToken 调用凭证
+ * @param string $openid 普通用户的标识,对当前开发者帐号唯一
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getUserinfo($accessToken, $openid)
+ {
+ $url = "https://api.weixin.qq.com/sns/userinfo?access_token={$accessToken}&openid={$openid}";
+ return json_decode(Tools::get($url));
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/WeOpen/MiniApp.php b/vendor/zoujingli/weopen-developer/WeOpen/MiniApp.php
new file mode 100644
index 000000000..2125615cc
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeOpen/MiniApp.php
@@ -0,0 +1,125 @@
+config->get('component_appid');
+ $component_access_token = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/sns/component/jscode2session?appid={$appid}&js_code={$code}&grant_type=authorization_code&component_appid={$component_appid}&component_access_token={$component_access_token}";
+ return json_decode(Tools::get($url), true);
+ }
+
+ /**
+ * 1.注册流程及接口说明
+ * @param string $authorizerAppid 公众号的appid
+ * @param integer $copyWxVerify 是否复用公众号的资质进行微信认证(1:申请复用资质进行微信 认证 0:不申请)
+ * @param string $redirectUri 用户扫码授权后,MP 扫码页面将跳转到该地址(注:1.链接需 urlencode 2.Host 需和第三方平台在微信开放平台上面填写的登 录授权的发起页域名一致)
+ * @return string
+ */
+ public function getCopyRegisterMiniUrl($authorizerAppid, $copyWxVerify, $redirectUri)
+ {
+ $redirectUri = urlencode($redirectUri);
+ $componentAppid = $this->config->get('component_appid');
+ return "https://mp.weixin.qq.com/cgi-bin/fastregisterauth?appid={$authorizerAppid}&component_appid={$componentAppid}©_wx_verify={$copyWxVerify}&redirect_uri={$redirectUri}";
+ }
+
+
+ /**
+ * 2.7.1 从第三方平台跳转至微信公众平台授权注册页面
+ * @param string $authorizerAppid 公众号的 appid
+ * @param string $redirectUri 新管理员信息填写完成点击提交后,将跳转到该地址
+ * @return string
+ */
+ public function getComponentreBindAdmin($authorizerAppid, $redirectUri)
+ {
+ $redirectUri = urlencode($redirectUri);
+ $componentAppid = $this->config->get('component_appid');
+ return "https://mp.weixin.qq.com/wxopen/componentrebindadmin?appid={$authorizerAppid}&component_appid={$componentAppid}&redirect_uri={$redirectUri}";
+ }
+
+ /**
+ * 1、获取草稿箱内的所有临时代码草稿
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getTemplateDraftList()
+ {
+ $component_access_token = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/wxa/gettemplatedraftlist?access_token={$component_access_token}";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 2、获取代码模版库中的所有小程序代码模版
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getTemplateList()
+ {
+ $component_access_token = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/wxa/gettemplatelist?access_token={$component_access_token}";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 3、将草稿箱的草稿选为小程序代码模版
+ * @param integer $draft_id 草稿ID,本字段可通过“ 获取草稿箱内的所有临时代码草稿 ”接口获得
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function addToTemplate($draft_id)
+ {
+ $component_access_token = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/wxa/addtotemplate?access_token={$component_access_token}";
+ return $this->httpPostForJson($url, ['draft_id' => $draft_id]);
+ }
+
+ /**
+ * 4、删除指定小程序代码模版
+ * @param integer $template_id 要删除的模版ID
+ * @return array
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function deleteTemplate($template_id)
+ {
+ $component_access_token = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/wxa/deletetemplate?access_token={$component_access_token}";
+ return $this->httpPostForJson($url, ['template_id' => $template_id]);
+ }
+
+}
diff --git a/vendor/zoujingli/weopen-developer/WeOpen/Service.php b/vendor/zoujingli/weopen-developer/WeOpen/Service.php
new file mode 100644
index 000000000..5d8558f5c
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/WeOpen/Service.php
@@ -0,0 +1,396 @@
+config = new DataArray($options);
+ }
+
+ /**
+ * 接收公众平台推送的 Ticket
+ * @return bool|array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getComonentTicket()
+ {
+ $receive = new Receive([
+ 'token' => $this->config->get('component_token'),
+ 'appid' => $this->config->get('component_appid'),
+ 'appsecret' => $this->config->get('component_appsecret'),
+ 'encodingaeskey' => $this->config->get('component_encodingaeskey'),
+ 'cache_path' => $this->config->get('cache_path'),
+ ]);
+ $data = $receive->getReceive();
+ if (!empty($data['ComponentVerifyTicket'])) {
+ Tools::setCache('component_verify_ticket', $data['ComponentVerifyTicket']);
+ }
+ return $data;
+ }
+
+ /**
+ * 获取或刷新服务 AccessToken
+ * @return bool|string
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getComponentAccessToken()
+ {
+ $cache = 'wechat_component_access_token';
+ if (($componentAccessToken = Tools::getCache($cache))) {
+ return $componentAccessToken;
+ }
+ $data = [
+ 'component_appid' => $this->config->get('component_appid'),
+ 'component_appsecret' => $this->config->get('component_appsecret'),
+ 'component_verify_ticket' => Tools::getCache('component_verify_ticket'),
+ ];
+ $url = 'https://api.weixin.qq.com/cgi-bin/component/api_component_token';
+ $result = $this->httpPostForJson($url, $data);
+ if (empty($result['component_access_token'])) {
+ throw new InvalidResponseException($result['errmsg'], $result['errcode'], $data);
+ }
+ Tools::setCache($cache, $result['component_access_token'], 7000);
+ return $result['component_access_token'];
+ }
+
+ /**
+ * 获取授权方的帐号基本信息
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAuthorizerInfo($authorizerAppid)
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_get_authorizer_info?component_access_token={$componentAccessToken}";
+ $data = [
+ 'authorizer_appid' => $authorizerAppid,
+ 'component_appid' => $this->config->get('component_appid'),
+ ];
+ $result = $this->httpPostForJson($url, $data);
+ if (empty($result['authorizer_info'])) {
+ throw new InvalidResponseException($result['errmsg'], $result['errcode'], $data);
+ }
+ return $result['authorizer_info'];
+ }
+
+ /**
+ * 确认接受公众号将某权限集高级权限的授权
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @param string $funcscopeCategoryId 权限集ID
+ * @param string $confirmValue 是否确认(1.确认授权, 2.取消确认)
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setAuthorization($authorizerAppid, $funcscopeCategoryId, $confirmValue)
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_confirm_authorization?component_access_token={$componentAccessToken}";
+ return $this->httpPostForJson($url, [
+ 'confirm_value' => $confirmValue,
+ 'authorizer_appid' => $authorizerAppid,
+ 'funcscope_category_id' => $funcscopeCategoryId,
+ 'component_appid' => $this->config->get('component_appid'),
+ ]);
+ }
+
+ /**
+ * 设置授权方的选项信息
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @param string $optionName 选项名称
+ * @param string $optionValue 设置的选项值
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function setAuthorizerOption($authorizerAppid, $optionName, $optionValue)
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_set_authorizer_option?component_access_token={$componentAccessToken}";
+ return $this->httpPostForJson($url, [
+ 'option_name' => $optionName,
+ 'option_value' => $optionValue,
+ 'authorizer_appid' => $authorizerAppid,
+ 'component_appid' => $this->config->get('component_appid'),
+ ]);
+ }
+
+ /**
+ * 获取授权方的选项设置信息
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @param string $optionName 选项名称
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAuthorizerOption($authorizerAppid, $optionName)
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_get_authorizer_option?component_access_token={$componentAccessToken}";
+ return $this->httpPostForJson($url, [
+ 'option_name' => $optionName,
+ 'authorizer_appid' => $authorizerAppid,
+ 'component_appid' => $this->config->get('component_appid'),
+ ]);
+ }
+
+ /**
+ * 获取预授权码 pre_auth_code
+ * @return string
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getPreauthCode()
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_create_preauthcode?component_access_token={$componentAccessToken}";
+ $result = $this->httpPostForJson($url, ['component_appid' => $this->config->get('component_appid')]);
+ if (empty($result['pre_auth_code'])) {
+ throw new InvalidResponseException('GetPreauthCode Faild.', '0', $result);
+ }
+ return $result['pre_auth_code'];
+ }
+
+ /**
+ * 获取授权回跳地址
+ * @param string $redirectUri 回调URI
+ * @param integer $authType 要授权的帐号类型
+ * @return bool
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAuthRedirect($redirectUri, $authType = 3)
+ {
+ $redirectUri = urlencode($redirectUri);
+ $preAuthCode = $this->getPreauthCode();
+ $componentAppid = $this->config->get('component_appid');
+ return "https://mp.weixin.qq.com/cgi-bin/componentloginpage?component_appid={$componentAppid}&pre_auth_code={$preAuthCode}&redirect_uri={$redirectUri}&auth_type={$authType}";
+ }
+
+ /**
+ * 使用授权码换取公众号或小程序的接口调用凭据和授权信息
+ * @param null $authCode 授权码
+ * @return bool|array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getQueryAuthorizerInfo($authCode = null)
+ {
+ if (is_null($authCode) && isset($_GET['auth_code'])) {
+ $authCode = $_GET['auth_code'];
+ }
+ if (empty($authCode)) {
+ return false;
+ }
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_query_auth?component_access_token={$componentAccessToken}";
+ $data = [
+ 'authorization_code' => $authCode,
+ 'component_appid' => $this->config->get('component_appid'),
+ ];
+ $result = $this->httpPostForJson($url, $data);
+ if (empty($result['authorization_info'])) {
+ throw new InvalidResponseException($result['errmsg'], $result['errcode'], $data);
+ }
+ $authorizerAppid = $result['authorization_info']['authorizer_appid'];
+ $authorizerAccessToken = $result['authorization_info']['authorizer_access_token'];
+ // 缓存授权公众号访问 ACCESS_TOKEN
+ Tools::setCache("{$authorizerAppid}_access_token", $authorizerAccessToken, 7000);
+ return $result['authorization_info'];
+ }
+
+ /**
+ * 获取(刷新)授权公众号的令牌
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @param string $authorizerRefreshToken 授权方的刷新令牌
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function refreshAccessToken($authorizerAppid, $authorizerRefreshToken)
+ {
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_authorizer_token?component_access_token={$componentAccessToken}";
+ $data = [
+ 'authorizer_appid' => $authorizerAppid,
+ 'authorizer_refresh_token' => $authorizerRefreshToken,
+ 'component_appid' => $this->config->get('component_appid'),
+ ];
+ $result = $this->httpPostForJson($url, $data);
+ if (empty($result['authorizer_access_token'])) {
+ throw new InvalidResponseException($result['errmsg'], $result['errcode'], $data);
+ }
+ // 缓存授权公众号访问 ACCESS_TOKEN
+ Tools::setCache("{$authorizerAppid}_access_token", $result['authorizer_access_token'], 7000);
+ return $result;
+ }
+
+ /**
+ * oauth 授权跳转接口
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @param string $redirectUri 回调地址
+ * @param string $scope snsapi_userinfo|snsapi_base
+ * @return string
+ */
+ public function getOauthRedirect($authorizerAppid, $redirectUri, $scope = 'snsapi_userinfo')
+ {
+ $redirectUri = urlencode($redirectUri);
+ $componentAppid = $this->config->get('component_appid');
+ return "https://open.weixin.qq.com/connect/oauth2/authorize?appid={$authorizerAppid}&redirect_uri={$redirectUri}&response_type=code&scope={$scope}&state={$authorizerAppid}&component_appid={$componentAppid}#wechat_redirect";
+ }
+
+ /**
+ * 通过code获取AccessToken
+ * @param string $authorizerAppid 授权公众号或小程序的appid
+ * @return bool|array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getOauthAccessToken($authorizerAppid)
+ {
+ if (empty($_GET['code'])) {
+ return false;
+ }
+ $componentAppid = $this->config->get('component_appid');
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/sns/oauth2/component/access_token?appid={$authorizerAppid}&code={$_GET['code']}&grant_type=authorization_code&component_appid={$componentAppid}&component_access_token={$componentAccessToken}";
+ return $this->httpGetForJson($url);
+ }
+
+ /**
+ * 取当前所有已授权的帐号基本信息
+ * @param integer $count 拉取数量,最大为500
+ * @param integer $offset 偏移位置/起始位置
+ * @return array|bool
+ * @throws \WeChat\Exceptions\InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function getAuthorizerList($count = 500, $offset = 0)
+ {
+ $componentAppid = $this->config->get('component_appid');
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/api_get_authorizer_list?component_access_token={$componentAccessToken}";
+ return $this->httpPostForJson($url, ['count' => $count, 'offset' => $offset, 'component_appid' => $componentAppid]);
+ }
+
+ /**
+ * 对第三方平台所有API调用次数清零
+ * @return array
+ * @throws InvalidResponseException
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ public function clearQuota()
+ {
+ $componentAppid = $this->config->get('component_appid');
+ $componentAccessToken = $this->getComponentAccessToken();
+ $url = "https://api.weixin.qq.com/cgi-bin/component/clear_quota?component_access_token={$componentAccessToken}";
+ return $this->httpPostForJson($url, ['component_appid' => $componentAppid]);
+ }
+
+ /**
+ * 创建指定授权公众号接口实例
+ * @param string $name 需要加载的接口实例名称
+ * @param string $authorizerAppid 授权公众号的appid
+ * @param string $type 加载SDK类型 WeChat|WeMini
+ * @return \WeChat\Card|\WeChat\Custom|\WeChat\Media|\WeChat\Menu|\WeChat\Oauth|\WeChat\Pay|\WeChat\Product|\WeChat\Qrcode|\WeChat\Receive|\WeChat\Scan|\WeChat\Script|\WeChat\Shake|\WeChat\Tags|\WeChat\Template|\WeChat\User|\WeChat\Wifi
+ */
+ public function instance($name, $authorizerAppid, $type = 'WeChat')
+ {
+ $className = "{$type}\\" . ucfirst(strtolower($name));
+ return new $className($this->getConfig($authorizerAppid));
+ }
+
+ /**
+ * 获取授权公众号配置参数
+ * @param string $authorizerAppid 授权公众号的appid
+ * @return array
+ */
+ public function getConfig($authorizerAppid)
+ {
+ $config = $this->config->get();
+ $config['appid'] = $authorizerAppid;
+ $config['token'] = $this->config->get('component_token');
+ $config['appsecret'] = $this->config->get('component_appsecret');
+ $config['encodingaeskey'] = $this->config->get('component_encodingaeskey');
+ return $config;
+ }
+
+ /**
+ * 以POST获取接口数据并转为数组
+ * @param string $url 接口地址
+ * @param array $data 请求数据
+ * @param bool $buildToJson
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function httpPostForJson($url, array $data, $buildToJson = true)
+ {
+ return json_decode(Tools::post($url, $buildToJson ? Tools::arr2json($data) : $data), true);
+ }
+
+ /**
+ * 以GET获取接口数据并转为数组
+ * @param string $url 接口地址
+ * @return array
+ * @throws \WeChat\Exceptions\LocalCacheException
+ */
+ protected function httpGetForJson($url)
+ {
+ return json_decode(Tools::get($url), true);
+ }
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/_test/open.php b/vendor/zoujingli/weopen-developer/_test/open.php
new file mode 100644
index 000000000..669992722
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/_test/open.php
@@ -0,0 +1,57 @@
+ 'wx4e63e993e222df8d',
+ 'component_token' => 'P8QHTIxpBEq88IrxatqhgpBm2OAQROkI',
+ 'component_appsecret' => '7cfa1afa87a41e2ea3445cea015c0974',
+ 'component_encodingaeskey' => 'L5uFIa0U6KLalPyXckyqoVIJYLhsfrg8k9YzybZIHsx',
+ ];
+
+ // 注册授权公众号 AccessToken 处理
+ $config['GetAccessTokenCallback'] = function ($authorizer_appid) use ($config) {
+ $open = new \WeOpen\Service($config);
+ $authorizer_refresh_token = ''; // 从数据库去找吧,在授权绑定的时候获取到了
+ $authorizer_refresh_token = 'L5uFIa0U6KLalPyXckyqoVIJYLhsfrg8k9YzybZIHsx'; // 从数据库去找吧,在授权绑定的时候获取到了
+ $result = $open->refreshAccessToken($authorizer_appid, $authorizer_refresh_token);
+ if (empty($result['authorizer_access_token'])) {
+ throw new \WeChat\Exceptions\InvalidResponseException($result['errmsg'], '0');
+ }
+ $data = [
+ 'authorizer_access_token' => $result['authorizer_access_token'],
+ 'authorizer_refresh_token' => $result['authorizer_refresh_token'],
+ ];
+ // 需要把$data记录到数据库
+ return $result['authorizer_access_token'];
+ };
+
+ // 3 使用第三方服务创建接口实例
+ $open = new \WeOpen\Service($config);
+ $wechat = $open->instance('User', 'wx60a43dd8161666d4');
+ $list = $wechat->getUserList();
+ var_export($list);
+
+} catch (Exception $e) {
+
+ // 出错啦,处理下吧
+ echo $e->getMessage() . PHP_EOL;
+
+}
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/_test/wechat.sql b/vendor/zoujingli/weopen-developer/_test/wechat.sql
new file mode 100644
index 000000000..99a42e050
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/_test/wechat.sql
@@ -0,0 +1,27 @@
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `authorizer_appid` varchar(100) DEFAULT NULL COMMENT '公众号APPID',
+ `authorizer_access_token` varchar(200) DEFAULT NULL COMMENT '公众号Token',
+ `authorizer_refresh_token` varchar(200) DEFAULT NULL COMMENT '公众号刷新Token',
+ `func_info` varchar(100) DEFAULT NULL COMMENT '公众号集权',
+ `nick_name` varchar(50) DEFAULT NULL COMMENT '公众号昵称',
+ `head_img` varchar(200) DEFAULT NULL COMMENT '公众号头像',
+ `expires_in` bigint(20) DEFAULT NULL COMMENT 'Token有效时间',
+ `service_type` tinyint(2) DEFAULT NULL COMMENT '公众号实际类型',
+ `service_type_info` tinyint(2) DEFAULT NULL COMMENT '服务类型信息',
+ `verify_type` tinyint(2) DEFAULT NULL COMMENT '公众号实际认证类型',
+ `verify_type_info` tinyint(2) DEFAULT NULL COMMENT '公众号认证类型',
+ `user_name` varchar(100) DEFAULT NULL COMMENT '众众号原始账号',
+ `alias` varchar(100) DEFAULT NULL COMMENT '公众号别名',
+ `qrcode_url` varchar(200) DEFAULT NULL COMMENT '公众号二维码地址',
+ `business_info` varchar(255) DEFAULT NULL,
+ `principal_name` varchar(255) DEFAULT NULL COMMENT '公司名称',
+ `idc` tinyint(1) unsigned DEFAULT NULL,
+ `status` tinyint(1) unsigned DEFAULT '1' COMMENT '状态(1正常授权,0取消授权)',
+ `total` bigint(20) unsigned DEFAULT '0' COMMENT '统计调用次数',
+ `appkey` char(32) DEFAULT NULL COMMENT '接口KEY',
+ `appuri` varchar(255) DEFAULT NULL COMMENT '响应接口APP',
+ `create_by` bigint(20) DEFAULT NULL COMMENT '创建人ID',
+ `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `index_wechat_config_authorizer_appid` (`authorizer_appid`) USING BTREE
+) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='公众号授权参数';
\ No newline at end of file
diff --git a/vendor/zoujingli/weopen-developer/composer.json b/vendor/zoujingli/weopen-developer/composer.json
new file mode 100644
index 000000000..9b1d4b985
--- /dev/null
+++ b/vendor/zoujingli/weopen-developer/composer.json
@@ -0,0 +1,33 @@
+{
+ "type": "library",
+ "name": "zoujingli/weopen-developer",
+ "homepage": "https://github.com/zoujingli/WeOpenDeveloper",
+ "description": "WeChat Open development of SDK",
+ "license": "MIT",
+ "authors": [
+ {
+ "name": "Anyon",
+ "email": "zoujingli@qq.com",
+ "homepage": "http://ctolog.com"
+ }
+ ],
+ "keywords": [
+ "WeChat",
+ "WeChatOpen",
+ "WeChatOpenDeveloper"
+ ],
+ "require": {
+ "php": ">=5.4",
+ "ext-curl": "*",
+ "ext-json": "*",
+ "ext-openssl": "*",
+ "zoujingli/wechat-developer": "^1.0"
+ },
+ "autoload": {
+ "psr-4": {
+ "WeOpen\\": "WeOpen",
+ "WeChat\\": "WeChat",
+ "WeMini\\": "WeMini"
+ }
+ }
+}
\ No newline at end of file