xxxrrrdddd 3 years ago
parent
commit
a717a82feb

+ 1 - 0
addons/qcloudsms/.addonrc

@@ -0,0 +1 @@
+{"license":"regular","licenseto":"39487","licensekey":"5WZLFIqPHk1nj4eV JH\/3+YtThUj0D3m9ql7miJOU5jsrrPnvHW0bIVnEydE="}

+ 197 - 0
addons/qcloudsms/Qcloudsms.php

@@ -0,0 +1,197 @@
+<?php
+
+namespace addons\qcloudsms;
+
+use addons\qcloudsms\library\SmsSingleSender;
+use addons\qcloudsms\library\SmsVoicePromptSender;
+use addons\qcloudsms\library\SmsVoiceverifyCodeSender;
+use addons\qcloudsms\library\TtsVoiceSender;
+use think\Addons;
+use think\Config;
+
+/**
+ * 插件
+ */
+class Qcloudsms extends Addons
+{
+    private $appid = null;
+    private $appkey = null;
+    private $config = null;
+    private $sender = null;
+    private $sendError = '';
+
+    public function ConfigInit()
+    {
+        $this->config = $this->getConfig();
+        //如果使用语音短信  更换成语音短信模板
+        if ($this->config['isVoice'] == 1) {
+            $this->config['template'] = $this->config['voiceTemplate'];
+            //语音短信 需要另行设置Aappid 与Appkey
+            $this->appid = $this->config['voiceAppid'];
+            $this->appkey = $this->config['voiceAppkey'];
+        } else {
+            $this->appid = $this->config['appid'];
+            $this->appkey = $this->config['appkey'];
+        }
+    }
+
+    /**
+     * 短信发送行为
+     * @param Sms $params
+     * @return  boolean
+     */
+    public function smsSend(&$params)
+    {
+        $this->ConfigInit();
+        try {
+            if ($this->config['isTemplateSender'] == 1) {
+                $templateID = $this->config['template'][$params->event];
+                if ($this->config['isVoice'] != 1) {
+                    //普通短信发送
+                    $this->sender = new SmsSingleSender($this->appid, $this->appkey);
+                    $result = $this->sender->sendWithParam("86", $params['mobile'], $templateID, ["{$params->code}"], $this->config['sign'], "", "");
+                } else {
+                    //语音短信发送
+                    $this->sender = new TtsVoiceSender($this->appid, $this->appkey);
+                    //参数: 国家码,手机号、模板ID、模板参数、播放次数(可选字段)、用户的session内容,服务器端原样返回(可选字段)
+                    $result = $this->sender->send("86", $params['mobile'], $templateID, [$params->code]);
+                }
+            } else {
+                //判断是否是语音短信
+                if ($this->config['isVoice'] != 1) {
+                    $this->sender = new SmsSingleSender($this->appid, $this->appkey);
+                    //参数:短信类型{1营销短信,0普通短信 }、国家码、手机号、短信内容、扩展码(可留空)、服务的原样返回的参数
+                    $result = $this->sender->send($params['type'], '86', $params['mobile'], $params['msg'], "", "");
+                } else {
+                    $this->sender = new SmsVoiceVerifyCodeSender($this->appid, $this->appkey);
+                    //参数:国家码、手机号、短信内容、播放次数(默认2次)、服务的原样返回的参数
+                    $result = $this->sender->send('86', $params['mobile'], $params['msg']);
+                }
+            }
+
+            $rsp = json_decode($result, true);
+            if ($rsp['result'] == 0 && $rsp['errmsg'] == 'OK') {
+                return true;
+            } else {
+                //记录错误信息
+                $this->setError($rsp);
+                return false;
+            }
+        } catch (\Exception $e) {
+            $this->setError($e->getMessage());
+        }
+        return false;
+    }
+
+    /**
+     * 短信发送通知
+     * @param array $params
+     * @return  boolean
+     */
+    public function smsNotice(&$params)
+    {
+        $this->ConfigInit();
+        try {
+            if ($this->config['isTemplateSender'] == 1) {
+                $templateID = $this->config['template'][$params['template']];
+
+                if ($this->config['isVoice'] != 1) {
+                    //普通短信发送
+                    $this->sender = new SmsSingleSender($this->appid, $this->appkey);
+                    $result = $this->sender->sendWithParam("86", $params['mobile'], $templateID, ["{$params['msg']}"], $this->config['sign'], "", "");
+                } else {
+                    //语音短信发送
+                    $this->sender = new TtsVoiceSender($this->appid, $this->appkey);
+                    //参数: 国家码,手机号、模板ID、模板参数、播放次数(可选字段)、用户的session内容,服务器端原样返回(可选字段)
+                    $result = $this->sender->send("86", $params['mobile'], $templateID, [$params['msg']]);
+                }
+            } else {
+                //判断是否是语音短信
+                if ($this->config['isVoice'] != 1) {
+                    $this->sender = new SmsSingleSender($this->appid, $this->appkey);
+                    //参数:短信类型{1营销短信,0普通短信 }、国家码、手机号、短信内容、扩展码(可留空)、服务的原样返回的参数
+                    $result = $this->sender->send($params['type'], '86', $params['mobile'], $params['msg'], "", "");
+                } else {
+                    $this->sender = new SmsVoicePromptSender($this->appid, $this->appkey);
+                    //参数:国家码、手机号、语音类型(目前固定为2)、短信内容、播放次数(默认2次)、服务的原样返回的参数
+                    $result = $this->sender->send('86', $params['mobile'], 2, $params['msg']);
+                }
+            }
+            $rsp = (array)json_decode($result, true);
+            if ($rsp['result'] == 0 && $rsp['errmsg'] == 'OK') {
+                return true;
+            } else {
+                //记录错误信息
+                $this->setError($rsp);
+                return false;
+            }
+        } catch (\Exception $e) {
+            var_dump($e);
+            exit();
+        }
+    }
+
+    /**
+     * 记录失败信息
+     * @param [type] $err [description]
+     */
+    private function setError($err)
+    {
+        $this->sendError = $err;
+    }
+
+    /**
+     * 获取失败信息
+     * @return [type] [description]
+     */
+    public function getError()
+    {
+        return $this->sendError;
+    }
+
+    /**
+     * 检测验证是否正确
+     * @param Sms $params
+     * @return  boolean
+     */
+    public function smsCheck(&$params)
+    {
+        return true;
+    }
+
+    /**
+     * 插件安装方法
+     * @return bool
+     */
+    public function install()
+    {
+        return true;
+    }
+
+    /**
+     * 插件卸载方法
+     * @return bool
+     */
+    public function uninstall()
+    {
+        return true;
+    }
+
+    /**
+     * 插件启用方法
+     * @return bool
+     */
+    public function enable()
+    {
+        return true;
+    }
+
+    /**
+     * 插件禁用方法
+     * @return bool
+     */
+    public function disable()
+    {
+        return true;
+    }
+}

+ 128 - 0
addons/qcloudsms/config.php

@@ -0,0 +1,128 @@
+<?php
+
+return [
+    [
+        'name' => 'appid',
+        'title' => '应用AppID',
+        'type' => 'string',
+        'content' => [],
+        'value' => 'asdas',
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'appkey',
+        'title' => '应用AppKEY',
+        'type' => 'string',
+        'content' => [],
+        'value' => 'dasdasd',
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'voiceAppid',
+        'title' => '语音短信AppID',
+        'type' => 'string',
+        'content' => [],
+        'value' => 'asdasd',
+        'rule' => 'required',
+        'msg' => '使用语音短信必须设置',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'voiceAppkey',
+        'title' => '语音短信AppKEY',
+        'type' => 'string',
+        'content' => [],
+        'value' => 'asdasd',
+        'rule' => 'required',
+        'msg' => '使用语音短信必须设置',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'sign',
+        'title' => '签名',
+        'type' => 'string',
+        'content' => [],
+        'value' => 'your sign',
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'isVoice',
+        'title' => '是否使用语音短信',
+        'type' => 'radio',
+        'content' => [
+            '否',
+            '是',
+        ],
+        'value' => '0',
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'isTemplateSender',
+        'title' => '是否使用短信模板发送',
+        'type' => 'radio',
+        'content' => [
+            '否',
+            '是',
+        ],
+        'value' => '1',
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'template',
+        'title' => '短信模板',
+        'type' => 'array',
+        'content' => [],
+        'value' => [
+            'register' => '',
+            'resetpwd' => '',
+            'changepwd' => '',
+            'profile' => '',
+        ],
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+    [
+        'name' => 'voiceTemplate',
+        'title' => '语音短信模板',
+        'type' => 'array',
+        'content' => [],
+        'value' => [
+            'register' => '',
+            'resetpwd' => '',
+            'changepwd' => '',
+            'profile' => '',
+        ],
+        'rule' => 'required',
+        'msg' => '',
+        'tip' => '',
+        'ok' => '',
+        'extend' => '',
+    ],
+];

+ 15 - 0
addons/qcloudsms/controller/Index.php

@@ -0,0 +1,15 @@
+<?php
+
+namespace addons\qcloudsms\controller;
+
+use think\addons\Controller;
+
+class Index extends Controller
+{
+
+    public function index()
+    {
+        $this->error("当前插件暂无前台页面");
+    }
+
+}

+ 10 - 0
addons/qcloudsms/info.ini

@@ -0,0 +1,10 @@
+name = qcloudsms
+title = 腾讯云短信发送插件
+intro = 腾讯云短信发送插件
+author = Seacent
+website = https://www.seacent.com
+version = 1.0.3
+state = 1
+url = /addons/qcloudsms
+license = regular
+licenseto = 39487

+ 69 - 0
addons/qcloudsms/library/FileVoiceSender.php

@@ -0,0 +1,69 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+
+/**
+ * 按语音文件fid发送语音通知类
+ *
+ */
+class FileVoiceSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://cloud.tim.qq.com/v5/tlsvoicesvr/sendfvoice";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     *
+     * 按语音文件fid发送语音通知
+     *
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param string $fid         语音文件fid
+     * @param string $playtimes   播放次数,可选,最多3次,默认2次
+     * @param string $ext         用户的session内容,服务端原样返回,可选字段,不需要可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($nationCode, $phoneNumber, $fid, $playtimes = 2, $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+        $data->tel = $tel;
+        $data->fid = $fid;
+        $data->playtimes = $playtimes;
+
+        // app凭证
+        $data->sig = $this->util->calculateSig($this->appkey, $random,
+            $curTime, array($phoneNumber));
+
+        // unix时间戳,请求发起时间,如果和系统时间相差超过10分钟则会返回失败
+        $data->time = $curTime;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 91 - 0
addons/qcloudsms/library/SmsMobileStatusPuller.php

@@ -0,0 +1,91 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 拉取单个手机短信状态类
+ *
+ */
+class SmsMobileStatusPuller
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlssmssvr/pullstatus4mobile";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     * 拉取回执结果
+     *
+     * @param int    $type         拉取类型,0表示回执结果,1表示回复信息
+     * @param string $nationCode   国家码,如 86 为中国
+     * @param string $mobile       不带国家码的手机号
+     * @param int    $beginTime    开始时间(unix timestamp)
+     * @param int    $endTime      结束时间(unix timestamp)
+     * @param int    $max          拉取最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    private function pull($type, $nationCode, $mobile, $beginTime, $endTime, $max)
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        $data = new \stdClass();
+        $data->sig = $this->util->calculateSigForPuller($this->appkey, $random, $curTime);
+        $data->time = $curTime;
+        $data->type = $type;
+        $data->max = $max;
+        $data->begin_time = $beginTime;
+        $data->end_time = $endTime;
+        $data->nationcode = $nationCode;
+        $data->mobile = $mobile;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+
+    /**
+     * 拉取回执结果
+     *
+     * @param string $nationCode   国家码,如 86 为中国
+     * @param string $mobile       不带国家码的手机号
+     * @param int    $beginTime    开始时间(unix timestamp)
+     * @param int    $endTime      结束时间(unix timestamp)
+     * @param int    $max          拉取最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function pullCallback($nationCode, $mobile, $beginTime, $endTime, $max)
+    {
+        return $this->pull(0, $nationCode, $mobile, $beginTime, $endTime, $max);
+    }
+
+    /**
+     * 拉取回复信息
+     *
+     * @param string $nationCode   国家码,如 86 为中国
+     * @param string $mobile       不带国家码的手机号
+     * @param int    $beginTime    开始时间(unix timestamp)
+     * @param int    $endTime      结束时间(unix timestamp)
+     * @param int    $max          拉取最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function pullReply($nationCode, $mobile, $beginTime, $endTime, $max)
+    {
+        return $this->pull(1, $nationCode, $mobile, $beginTime, $endTime, $max);
+    }
+}

+ 99 - 0
addons/qcloudsms/library/SmsMultiSender.php

@@ -0,0 +1,99 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 群发短信类
+ *
+ */
+class SmsMultiSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlssmssvr/sendmultisms2";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     * 普通群发
+     *
+     * 普通群发需明确指定内容,如果有多个签名,请在内容中以【】的方式添加到信息内容中,
+     * 否则系统将使用默认签名。
+     *
+     *
+     * @param int    $type         短信类型,0 为普通短信,1 营销短信
+     * @param string $nationCode   国家码,如 86 为中国
+     * @param array  $phoneNumbers 不带国家码的手机号列表
+     * @param string $msg          信息内容,必须与申请的模板格式一致,否则将返回错误
+     * @param string $extend       扩展码,可填空串
+     * @param string $ext          服务端原样返回的参数,可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($type, $nationCode, $phoneNumbers, $msg, $extend = "", $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        $data = new \stdClass();
+        $data->tel = $this->util->phoneNumbersToArray($nationCode, $phoneNumbers);
+        $data->type = $type;
+        $data->msg = $msg;
+        $data->sig = $this->util->calculateSig($this->appkey, $random,
+            $curTime, $phoneNumbers);
+        $data->time = $curTime;
+        $data->extend = $extend;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+
+    /**
+     * 指定模板群发
+     *
+     *
+     * @param  string $nationCode   国家码,如 86 为中国
+     * @param  array  $phoneNumbers 不带国家码的手机号列表
+     * @param  int    $templId      模板id
+     * @param  array  $params       模板参数列表,如模板 {1}...{2}...{3},那么需要带三个参数
+     * @param  string $sign         签名,如果填空串,系统会使用默认签名
+     * @param  string $extend       扩展码,可填空串
+     * @param  string $ext          服务端原样返回的参数,可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function sendWithParam($nationCode, $phoneNumbers, $templId, $params,
+        $sign = "", $extend = "", $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        $data = new \stdClass();
+        $data->tel = $this->util->phoneNumbersToArray($nationCode, $phoneNumbers);
+        $data->sign = $sign;
+        $data->tpl_id = $templId;
+        $data->params = $params;
+        $data->sig = $this->util->calculateSigForTemplAndPhoneNumbers(
+            $this->appkey, $random, $curTime, $phoneNumbers);
+        $data->time = $curTime;
+        $data->extend = $extend;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 211 - 0
addons/qcloudsms/library/SmsSenderUtil.php

@@ -0,0 +1,211 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+/**
+ * 发送Util类
+ *
+ */
+class SmsSenderUtil
+{
+    /**
+     * 生成随机数
+     *
+     * @return int 随机数结果
+     */
+    public function getRandom()
+    {
+        return rand(100000, 999999);
+    }
+
+    /**
+     * 生成签名
+     *
+     * @param string $appkey        sdkappid对应的appkey
+     * @param string $random        随机正整数
+     * @param string $curTime       当前时间
+     * @param array  $phoneNumbers  手机号码
+     * @return string  签名结果
+     */
+    public function calculateSig($appkey, $random, $curTime, $phoneNumbers)
+    {
+        $phoneNumbersString = $phoneNumbers[0];
+        for ($i = 1; $i < count($phoneNumbers); $i++) {
+            $phoneNumbersString .= ("," . $phoneNumbers[$i]);
+        }
+
+        return hash("sha256", "appkey=".$appkey."&random=".$random
+            ."&time=".$curTime."&mobile=".$phoneNumbersString);
+    }
+
+    /**
+     * 生成签名
+     *
+     * @param string $appkey        sdkappid对应的appkey
+     * @param string $random        随机正整数
+     * @param string $curTime       当前时间
+     * @param array  $phoneNumbers  手机号码
+     * @return string  签名结果
+     */
+    public function calculateSigForTemplAndPhoneNumbers($appkey, $random,
+        $curTime, $phoneNumbers)
+    {
+        $phoneNumbersString = $phoneNumbers[0];
+        for ($i = 1; $i < count($phoneNumbers); $i++) {
+            $phoneNumbersString .= ("," . $phoneNumbers[$i]);
+        }
+
+        return hash("sha256", "appkey=".$appkey."&random=".$random
+            ."&time=".$curTime."&mobile=".$phoneNumbersString);
+    }
+
+    public function phoneNumbersToArray($nationCode, $phoneNumbers)
+    {
+        $i = 0;
+        $tel = array();
+        do {
+            $telElement = new \stdClass();
+            $telElement->nationcode = $nationCode;
+            $telElement->mobile = $phoneNumbers[$i];
+            array_push($tel, $telElement);
+        } while (++$i < count($phoneNumbers));
+
+        return $tel;
+    }
+
+    /**
+     * 生成签名
+     *
+     * @param string $appkey        sdkappid对应的appkey
+     * @param string $random        随机正整数
+     * @param string $curTime       当前时间
+     * @param array  $phoneNumber   手机号码
+     * @return string  签名结果
+     */
+    public function calculateSigForTempl($appkey, $random, $curTime, $phoneNumber)
+    {
+        $phoneNumbers = array($phoneNumber);
+
+        return $this->calculateSigForTemplAndPhoneNumbers($appkey, $random,
+            $curTime, $phoneNumbers);
+    }
+
+    /**
+     * 生成签名
+     *
+     * @param string $appkey        sdkappid对应的appkey
+     * @param string $random        随机正整数
+     * @param string $curTime       当前时间
+     * @return string 签名结果
+     */
+    public function calculateSigForPuller($appkey, $random, $curTime)
+    {
+        return hash("sha256", "appkey=".$appkey."&random=".$random
+            ."&time=".$curTime);
+    }
+
+    /**
+     * 生成上传文件授权
+     *
+     * @param string $appkey        sdkappid对应的appkey
+     * @param string $random        随机正整数
+     * @param string $curTime       当前时间
+     * @param array  $fileSha1Sum   文件sha1sum
+     * @return string  授权结果
+     */
+    public function calculateAuth($appkey, $random, $curTime, $fileSha1Sum)
+    {
+        return hash("sha256", "appkey=".$appkey."&random=".$random
+            ."&time=".$curTime."&content-sha1=".$fileSha1Sum);
+    }
+
+    /**
+     * 生成sha1sum
+     *
+     * @param string $content  内容
+     * @return string  内容sha1散列值
+     */
+    public function sha1sum($content)
+    {
+        return hash("sha1", $content);
+    }
+
+    /**
+     * 发送请求
+     *
+     * @param string $url      请求地址
+     * @param array  $dataObj  请求内容
+     * @return string 应答json字符串
+     */
+    public function sendCurlPost($url, $dataObj)
+    {
+        $curl = curl_init();
+        curl_setopt($curl, CURLOPT_URL, $url);
+        curl_setopt($curl, CURLOPT_HEADER, 0);
+        curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
+        curl_setopt($curl, CURLOPT_POST, 1);
+        curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 60);
+        curl_setopt($curl, CURLOPT_POSTFIELDS, json_encode($dataObj));
+        curl_setopt($curl, CURLOPT_HTTPHEADER, array(
+        'Content-Type: application/json; charset=utf-8',
+        'Content-Length: ' . strlen(json_encode($dataObj)))
+    );
+        curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 0);
+        curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0);
+
+        $ret = curl_exec($curl);
+        if (false == $ret) {
+            // curl_exec failed
+            $result = "{ \"result\":" . -2 . ",\"errmsg\":\"" . curl_error($curl) . "\"}";
+        } else {
+            $rsp = curl_getinfo($curl, CURLINFO_HTTP_CODE);
+            if (200 != $rsp) {
+                $result = "{ \"result\":" . -1 . ",\"errmsg\":\"". $rsp
+                        . " " . curl_error($curl) ."\"}";
+            } else {
+                $result = $ret;
+            }
+        }
+
+        curl_close($curl);
+
+        return $result;
+    }
+
+    /**
+     * 发送请求
+     *
+     * @param string $req  请求对象
+     * @return string 应答json字符串
+     */
+    public function fetch($req)
+    {
+        $curl = curl_init();
+
+        curl_setopt($curl, CURLOPT_URL, $req->url);
+        curl_setopt($curl, CURLOPT_HTTPHEADER, $req->headers);
+        curl_setopt($curl, CURLOPT_POSTFIELDS, $req->body);
+        curl_setopt($curl, CURLOPT_HEADER, 0);
+        curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
+        curl_setopt($curl, CURLOPT_POST, 1);
+        curl_setopt($curl, CURLOPT_CONNECTTIMEOUT, 60);
+        curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 0);
+        curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0);
+
+        $result = curl_exec($curl);
+
+        if (false == $result) {
+            // curl_exec failed
+            $result = "{ \"result\":" . -2 . ",\"errmsg\":\"" . curl_error($curl) . "\"}";
+        } else {
+            $code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
+            if (200 != $code) {
+                $result = "{ \"result\":" . -1 . ",\"errmsg\":\"". $rsp
+                    . " " . curl_error($curl) ."\"}";
+            }
+        }
+        curl_close($curl);
+
+        return $result;
+    }
+}

+ 107 - 0
addons/qcloudsms/library/SmsSingleSender.php

@@ -0,0 +1,107 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 单发短信类
+ *
+ */
+class SmsSingleSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlssmssvr/sendsms";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     * 普通单发
+     *
+     * 普通单发需明确指定内容,如果有多个签名,请在内容中以【】的方式添加到信息内容中,否则系统将使用默认签名。
+     *
+     * @param int    $type        短信类型,0 为普通短信,1 营销短信
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param string $msg         信息内容,必须与申请的模板格式一致,否则将返回错误
+     * @param string $extend      扩展码,可填空串
+     * @param string $ext         服务端原样返回的参数,可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($type, $nationCode, $phoneNumber, $msg, $extend = "", $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+
+        $data->tel = $tel;
+        $data->type = (int)$type;
+        $data->msg = $msg;
+        $data->sig = hash("sha256",
+            "appkey=".$this->appkey."&random=".$random."&time="
+            .$curTime."&mobile=".$phoneNumber, FALSE);
+        $data->time = $curTime;
+        $data->extend = $extend;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+
+    /**
+     * 指定模板单发
+     *
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param int    $templId     模板 id
+     * @param array  $params      模板参数列表,如模板 {1}...{2}...{3},那么需要带三个参数
+     * @param string $sign        签名,如果填空串,系统会使用默认签名
+     * @param string $extend      扩展码,可填空串
+     * @param string $ext         服务端原样返回的参数,可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function sendWithParam($nationCode, $phoneNumber, $templId = 0, $params,
+        $sign = "", $extend = "", $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+
+        $data->tel = $tel;
+        $data->sig = $this->util->calculateSigForTempl($this->appkey, $random,
+            $curTime, $phoneNumber);
+        $data->tpl_id = $templId;
+        $data->params = $params;
+        $data->sign = $sign;
+        $data->time = $curTime;
+        $data->extend = $extend;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 75 - 0
addons/qcloudsms/library/SmsStatusPuller.php

@@ -0,0 +1,75 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 拉取短信状态类
+ *
+ */
+class SmsStatusPuller
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlssmssvr/pullstatus";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     * 拉取回执结果
+     *
+     * @param int $type 拉取类型,0表示回执结果,1表示回复信息
+     * @param int $max  最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    private function pull($type, $max)
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        $data = new \stdClass();
+        $data->sig = $this->util->calculateSigForPuller($this->appkey, $random, $curTime);
+        $data->time = $curTime;
+        $data->type = $type;
+        $data->max = $max;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+
+    /**
+     * 拉取回执结果
+     *
+     * @param int $max 拉取最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function pullCallback($max)
+    {
+        return $this->pull(0, $max);
+    }
+
+    /**
+     * 拉取回复信息
+     *
+     * @param int $max 拉取最大条数,最多100
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function pullReply($max)
+    {
+        return $this->pull(1, $max);
+    }
+}

+ 71 - 0
addons/qcloudsms/library/SmsVoicePromptSender.php

@@ -0,0 +1,71 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 发送语音通知类
+ *
+ */
+class SmsVoicePromptSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlsvoicesvr/sendvoiceprompt";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     *
+     * 发送语音通知
+     *
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param string $prompttype  语音类型,目前固定为2
+     * @param string $msg         信息内容,必须与申请的模板格式一致,否则将返回错误
+     * @param string $playtimes   播放次数,可选,最多3次,默认2次
+     * @param string $ext         用户的session内容,服务端原样返回,可选字段,不需要可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($nationCode, $phoneNumber, $prompttype, $msg, $playtimes = 2, $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+
+        $data->tel = $tel;
+        // 通知内容,utf8编码,支持中文英文、数字及组合,需要和语音内容模版相匹配
+        $data->promptfile = $msg;
+        // 固定值 2
+        $data->prompttype = $prompttype;
+        $data->playtimes = $playtimes;
+        // app凭证
+        $data->sig = hash("sha256",
+            "appkey=".$this->appkey."&random=".$random."&time="
+            .$curTime."&mobile=".$phoneNumber, FALSE);
+        // unix时间戳,请求发起时间,如果和系统时间相差超过10分钟则会返回失败
+        $data->time = $curTime;
+        $data->ext = $ext;
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 67 - 0
addons/qcloudsms/library/SmsVoiceVerifyCodeSender.php

@@ -0,0 +1,67 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+/**
+ * 发送语音验证码类
+ *
+ */
+class SmsVoiceVerifyCodeSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://yun.tim.qq.com/v5/tlsvoicesvr/sendvoice";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     * 发送语音验证码
+     *
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param string $msg         信息内容,必须与申请的模板格式一致,否则将返回错误
+     * @param int    $playtimes   播放次数,可选,最多3次,默认2次
+     * @param string $ext         用户的session内容,服务端原样返回,可选字段,不需要可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($nationCode, $phoneNumber, $msg, $playtimes = 2, $ext = "")
+    {
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+
+        $data->tel = $tel;
+        $data->msg = $msg;
+        $data->playtimes = $playtimes;
+        // app凭证
+        $data->sig = hash("sha256",
+            "appkey=".$this->appkey."&random=".$random."&time="
+            .$curTime."&mobile=".$phoneNumber, FALSE);
+        // unix时间戳,请求发起时间,如果和系统时间相差超过10分钟则会返回失败
+        $data->time = $curTime;
+        $data->ext = $ext;
+
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 77 - 0
addons/qcloudsms/library/TtsVoiceSender.php

@@ -0,0 +1,77 @@
+<?php
+
+namespace addons\qcloudsms\library;
+
+use addons\qcloudsms\library\SmsSenderUtil;
+
+
+/**
+ * 指定模板发送语音通知类
+ *
+ */
+class TtsVoiceSender
+{
+    private $url;
+    private $appid;
+    private $appkey;
+    private $util;
+
+    /**
+     * 构造函数
+     *
+     * @param string $appid  sdkappid
+     * @param string $appkey sdkappid对应的appkey
+     */
+    public function __construct($appid, $appkey)
+    {
+        $this->url = "https://cloud.tim.qq.com/v5/tlsvoicesvr/sendtvoice";
+        $this->appid =  $appid;
+        $this->appkey = $appkey;
+        $this->util = new SmsSenderUtil();
+    }
+
+    /**
+     *
+     * 指定模板发送语音短信
+     *
+     * @param string $nationCode  国家码,如 86 为中国
+     * @param string $phoneNumber 不带国家码的手机号
+     * @param int    $templId     模板 id
+     * @param array  $params      模板参数列表,如模板 {1}...{2}...{3},需要带三个参数
+     * @param string $playtimes   播放次数,可选,最多3次,默认2次
+     * @param string $ext         用户的session内容,服务端原样返回,可选字段,不需要可填空串
+     * @return string 应答json字符串,详细内容参见腾讯云协议文档
+     */
+    public function send($nationCode, $phoneNumber, $templId, $params, $playtimes = 2, $ext = "")
+    {
+        /*var_dump($nationCode);
+        var_dump($phoneNumber);
+        var_dump($templId);
+        var_dump($params);
+        var_dump($playtimes);
+        exit();*/
+        $random = $this->util->getRandom();
+        $curTime = time();
+        $wholeUrl = $this->url . "?sdkappid=" . $this->appid . "&random=" . $random;
+
+        // 按照协议组织 post 包体
+        $data = new \stdClass();
+        $tel = new \stdClass();
+        $tel->nationcode = "".$nationCode;
+        $tel->mobile = "".$phoneNumber;
+        $data->tel = $tel;
+        $data->tpl_id = $templId;
+        $data->params = $params;
+        $data->playtimes = $playtimes;
+
+        // app凭证
+        $data->sig = $this->util->calculateSig($this->appkey, $random,
+            $curTime, array($phoneNumber));
+
+        // unix时间戳,请求发起时间,如果和系统时间相差超过10分钟则会返回失败
+        $data->time = $curTime;
+        $data->ext = $ext;
+        //var_dump($data);exit();
+        return $this->util->sendCurlPost($wholeUrl, $data);
+    }
+}

+ 74 - 0
application/admin/controller/Agreement.php

@@ -0,0 +1,74 @@
+<?php
+
+namespace app\admin\controller;
+
+use app\common\controller\Backend;
+
+/**
+ * 
+ *
+ * @icon fa fa-circle-o
+ */
+class Agreement extends Backend
+{
+    
+    /**
+     * Agreement模型对象
+     * @var \app\admin\model\Agreement
+     */
+    protected $model = null;
+
+    public function _initialize()
+    {
+        parent::_initialize();
+        $this->model = new \app\admin\model\Agreement;
+
+    }
+
+    public function import()
+    {
+        parent::import();
+    }
+
+    /**
+     * 默认生成的控制器所继承的父类中有index/add/edit/del/multi五个基础方法、destroy/restore/recyclebin三个回收站方法
+     * 因此在当前控制器中可不用编写增删改查的代码,除非需要自己控制这部分逻辑
+     * 需要将application/admin/library/traits/Backend.php中对应的方法复制到当前控制器,然后进行修改
+     */
+    
+
+    /**
+     * 查看
+     */
+    public function index()
+    {
+        //当前是否为关联查询
+        $this->relationSearch = false;
+        //设置过滤方法
+        $this->request->filter(['strip_tags', 'trim']);
+        if ($this->request->isAjax()) {
+            //如果发送的来源是Selectpage,则转发到Selectpage
+            if ($this->request->request('keyField')) {
+                return $this->selectpage();
+            }
+            list($where, $sort, $order, $offset, $limit) = $this->buildparams();
+
+            $list = $this->model
+                    
+                    ->where($where)
+                    ->order($sort, $order)
+                    ->paginate($limit);
+
+            foreach ($list as $row) {
+                $row->visible(['id','title','content']);
+                
+            }
+
+            $result = array("total" => $list->total(), "rows" => $list->items());
+
+            return json($result);
+        }
+        return $this->view->fetch();
+    }
+
+}

+ 6 - 0
application/admin/lang/zh-cn/agreement.php

@@ -0,0 +1,6 @@
+<?php
+
+return [
+    'Title'   => '标题',
+    'Content' => '内容'
+];

+ 40 - 0
application/admin/model/Agreement.php

@@ -0,0 +1,40 @@
+<?php
+
+namespace app\admin\model;
+
+use think\Model;
+
+
+class Agreement extends Model
+{
+
+    
+
+    
+
+    // 表名
+    protected $table = 'agreement';
+    
+    // 自动写入时间戳字段
+    protected $autoWriteTimestamp = false;
+
+    // 定义时间戳字段名
+    protected $createTime = false;
+    protected $updateTime = false;
+    protected $deleteTime = false;
+
+    // 追加属性
+    protected $append = [
+
+    ];
+    
+
+    
+
+
+
+
+
+
+
+}

+ 27 - 0
application/admin/validate/Agreement.php

@@ -0,0 +1,27 @@
+<?php
+
+namespace app\admin\validate;
+
+use think\Validate;
+
+class Agreement extends Validate
+{
+    /**
+     * 验证规则
+     */
+    protected $rule = [
+    ];
+    /**
+     * 提示消息
+     */
+    protected $message = [
+    ];
+    /**
+     * 验证场景
+     */
+    protected $scene = [
+        'add'  => [],
+        'edit' => [],
+    ];
+    
+}

+ 22 - 0
application/admin/view/agreement/add.html

@@ -0,0 +1,22 @@
+<form id="add-form" class="form-horizontal" role="form" data-toggle="validator" method="POST" action="">
+
+    <div class="form-group">
+        <label class="control-label col-xs-12 col-sm-2">{:__('Title')}:</label>
+        <div class="col-xs-12 col-sm-8">
+            <input id="c-title" class="form-control" name="row[title]" type="text">
+        </div>
+    </div>
+    <div class="form-group">
+        <label class="control-label col-xs-12 col-sm-2">{:__('Content')}:</label>
+        <div class="col-xs-12 col-sm-8">
+            <textarea id="c-content" class="form-control editor" rows="5" name="row[content]" cols="50"></textarea>
+        </div>
+    </div>
+    <div class="form-group layer-footer">
+        <label class="control-label col-xs-12 col-sm-2"></label>
+        <div class="col-xs-12 col-sm-8">
+            <button type="submit" class="btn btn-success btn-embossed disabled">{:__('OK')}</button>
+            <button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
+        </div>
+    </div>
+</form>

+ 22 - 0
application/admin/view/agreement/edit.html

@@ -0,0 +1,22 @@
+<form id="edit-form" class="form-horizontal" role="form" data-toggle="validator" method="POST" action="">
+
+    <div class="form-group">
+        <label class="control-label col-xs-12 col-sm-2">{:__('Title')}:</label>
+        <div class="col-xs-12 col-sm-8">
+            <input id="c-title" class="form-control" name="row[title]" type="text" value="{$row.title|htmlentities}">
+        </div>
+    </div>
+    <div class="form-group">
+        <label class="control-label col-xs-12 col-sm-2">{:__('Content')}:</label>
+        <div class="col-xs-12 col-sm-8">
+            <textarea id="c-content" class="form-control editor" rows="5" name="row[content]" cols="50">{$row.content|htmlentities}</textarea>
+        </div>
+    </div>
+    <div class="form-group layer-footer">
+        <label class="control-label col-xs-12 col-sm-2"></label>
+        <div class="col-xs-12 col-sm-8">
+            <button type="submit" class="btn btn-success btn-embossed disabled">{:__('OK')}</button>
+            <button type="reset" class="btn btn-default btn-embossed">{:__('Reset')}</button>
+        </div>
+    </div>
+</form>

+ 37 - 0
application/admin/view/agreement/index.html

@@ -0,0 +1,37 @@
+<div class="panel panel-default panel-intro">
+    {:build_heading()}
+
+    <div class="panel-body">
+        <div id="myTabContent" class="tab-content">
+            <div class="tab-pane fade active in" id="one">
+                <div class="widget-body no-padding">
+                    <div id="toolbar" class="toolbar">
+                        <a href="javascript:;" class="btn btn-primary btn-refresh" title="{:__('Refresh')}" ><i class="fa fa-refresh"></i> </a>
+                        <a href="javascript:;" class="btn btn-success btn-add {:$auth->check('agreement/add')?'':'hide'}" title="{:__('Add')}" ><i class="fa fa-plus"></i> {:__('Add')}</a>
+                        <a href="javascript:;" class="btn btn-success btn-edit btn-disabled disabled {:$auth->check('agreement/edit')?'':'hide'}" title="{:__('Edit')}" ><i class="fa fa-pencil"></i> {:__('Edit')}</a>
+                        <a href="javascript:;" class="btn btn-danger btn-del btn-disabled disabled {:$auth->check('agreement/del')?'':'hide'}" title="{:__('Delete')}" ><i class="fa fa-trash"></i> {:__('Delete')}</a>
+<!--
+                        <a href="javascript:;" class="btn btn-danger btn-import {:$auth->check('agreement/import')?'':'hide'}" title="{:__('Import')}" id="btn-import-file" data-url="ajax/upload" data-mimetype="csv,xls,xlsx" data-multiple="false"><i class="fa fa-upload"></i> {:__('Import')}</a>
+-->
+
+                        <div class="dropdown btn-group {:$auth->check('agreement/multi')?'':'hide'}">
+                            <a class="btn btn-primary btn-more dropdown-toggle btn-disabled disabled" data-toggle="dropdown"><i class="fa fa-cog"></i> {:__('More')}</a>
+                            <ul class="dropdown-menu text-left" role="menu">
+                                <li><a class="btn btn-link btn-multi btn-disabled disabled" href="javascript:;" data-params="status=normal"><i class="fa fa-eye"></i> {:__('Set to normal')}</a></li>
+                                <li><a class="btn btn-link btn-multi btn-disabled disabled" href="javascript:;" data-params="status=hidden"><i class="fa fa-eye-slash"></i> {:__('Set to hidden')}</a></li>
+                            </ul>
+                        </div>
+
+                        
+                    </div>
+                    <table id="table" class="table table-striped table-bordered table-hover table-nowrap"
+                           data-operate-edit="{:$auth->check('agreement/edit')}" 
+                           data-operate-del="{:$auth->check('agreement/del')}" 
+                           width="100%">
+                    </table>
+                </div>
+            </div>
+
+        </div>
+    </div>
+</div>

+ 5 - 3
application/api/controller/User.php

@@ -361,12 +361,14 @@ class User extends Api
             'type'=>'require|in:1,2',
         ]);
         $session=Mini::mini($data['type'])->auth->session($data['code']);
-        $user=\app\admin\model\User::where('openid',$session['openid'])->where('type',1)->find();
+        $decryptedData = Mini::mini($data['type'])->encryptor->decryptData($session, $data['iv'], $data['encryptedData']);
+        $user=\app\admin\model\User::where('openid',$session['openid'])->where('type',$data['type'])->find();
+        $user1=\app\admin\model\User::where('mobile',$decryptedData['phoneNumber'])->where('type',$data['type'])->find();
+        $user=$user?:$user1;
         if(!$user){
-            $decryptedData = Mini::mini($data['type'])->encryptor->decryptData($session, $data['iv'], $data['encryptedData']);
             $ret = $this->auth->register(session_create_id(), '', '', $decryptedData['phoneNumber'], [
                 'openid'=>$session['openid'],
-                'type'=>1,
+                'type'=>$data['type'],
                 'unionid'=>$session['unionid'],
             ]);
             if ($ret) {

+ 17 - 0
application/common/model/User.php

@@ -23,6 +23,19 @@ class User extends Model
     // 追加属性
     protected $append = [
         'url',
+        'level_text'
+    ];
+
+    const LEVEL_0=0;
+    const LEVEL_1=1;
+    const LEVEL_10=10;
+    const LEVEL_20=20;
+
+    public static $levels=[
+        self::LEVEL_0=>'普通会员',
+        self::LEVEL_1=>'黄金会员',
+        self::LEVEL_10=>'白金会员',
+        self::LEVEL_20=>'钻石会员',
     ];
 
     /**
@@ -35,6 +48,10 @@ class User extends Model
     {
         return fullUrl("/u/" . $data['id']);
     }
+    public function getLevelTextAttr($value, $data)
+    {
+        return self::$levels[$data['level']]??'普通会员';
+    }
 
     /**
      * 获取头像

+ 10 - 0
application/extra/addons.php

@@ -4,8 +4,18 @@ return [
     'autoload' => false,
     'hooks' => [
         'config_init' => [
+            'qcloudsms',
             'third',
         ],
+        'sms_send' => [
+            'qcloudsms',
+        ],
+        'sms_notice' => [
+            'qcloudsms',
+        ],
+        'sms_check' => [
+            'qcloudsms',
+        ],
     ],
     'route' => [
         '/third$' => 'third/index/index',

+ 51 - 0
public/assets/js/backend/agreement.js

@@ -0,0 +1,51 @@
+define(['jquery', 'bootstrap', 'backend', 'table', 'form'], function ($, undefined, Backend, Table, Form) {
+
+    var Controller = {
+        index: function () {
+            // 初始化表格参数配置
+            Table.api.init({
+                extend: {
+                    index_url: 'agreement/index' + location.search,
+                    add_url: 'agreement/add',
+                    edit_url: 'agreement/edit',
+                    del_url: 'agreement/del',
+                    multi_url: 'agreement/multi',
+                    import_url: 'agreement/import',
+                    table: 'agreement',
+                }
+            });
+
+            var table = $("#table");
+
+            // 初始化表格
+            table.bootstrapTable({
+                url: $.fn.bootstrapTable.defaults.extend.index_url,
+                pk: 'id',
+                sortName: 'id',
+                columns: [
+                    [
+                        {checkbox: true},
+                        {field: 'id', title: __('Id')},
+                        {field: 'title', title: __('Title')},
+                        {field: 'operate', title: __('Operate'), table: table, events: Table.api.events.operate, formatter: Table.api.formatter.operate}
+                    ]
+                ]
+            });
+
+            // 为表格绑定事件
+            Table.api.bindevent(table);
+        },
+        add: function () {
+            Controller.api.bindevent();
+        },
+        edit: function () {
+            Controller.api.bindevent();
+        },
+        api: {
+            bindevent: function () {
+                Form.api.bindevent($("form[role=form]"));
+            }
+        }
+    };
+    return Controller;
+});