zhangguidong 2 anni fa
parent
commit
88fe5dc673

+ 173 - 0
application/api/controller/Area.php

@@ -0,0 +1,173 @@
+<?php
+
+// +----------------------------------------------------------------------
+// | ThinkAdmin
+// +----------------------------------------------------------------------
+// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
+// +----------------------------------------------------------------------
+// | 官方网站: http://demo.thinkadmin.top
+// +----------------------------------------------------------------------
+// | 开源协议 ( https://mit-license.org )
+// +----------------------------------------------------------------------
+// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
+// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
+// +----------------------------------------------------------------------
+
+namespace app\api\controller;
+
+use app\common\controller\Api;
+use library\File;
+use think\Db;
+/**
+ * @title 地址库
+ * @controller Area
+ * @group common
+ */
+class Area extends Base
+{
+    /**
+     * @title 获取省市区
+     * @desc 获取省市区
+     * @author QC
+     * @url /api/Area/getAllArea
+     * @method GET
+     * @tag 省市区
+     */
+    public function getAllArea(){
+        $area_set = Db::table('platform_set')->where('name','all_area')->find();
+        if(!empty($area_set) && $area_set['content']) {
+           $list = json_decode($area_set['content'],true);
+        }else{
+            $field=['id','pid','shortname','name'];
+            $list=Db::table('store_area')->where('pid',0)->field($field)->select();
+            foreach ($list as $k=>&$v){
+                $v['children']= Db::table('store_area')->where('pid',$v['id'])->field($field)->select();
+                if(!empty($v['children'])){
+                    foreach ($v['children'] as $kk=>&$vv){
+                        $vv['children']=Db::table('store_area')->where('pid',$vv['id'])->field($field)->select();
+                    }
+                }
+            }
+            if(!empty($area_set)){
+                Db::table('platform_set')->where('name','all_area')->update(['content'=>json_encode($list)]);
+            }else{
+                Db::table('platform_set')->insert(['name'=>'all_area','content'=>json_encode($list)]);
+            }
+        }
+        $this->success('获取成功',$list);
+    }
+
+    /**
+     * @title 获取下级地区
+     * @desc 获取下级地区
+     * @author QC
+     * @url /api/Area/getChildrenArea
+     * @method GET
+     * @param  name:id type:int  default:1 desc:省id或市id
+     */
+    public function getChildrenArea()
+    {
+        $field=['id','pid','shortname','name'];
+        $list=Db::table('store_area')->where('pid',$this->request->get('id'))->field($field)->select();
+        $this->success('获取成功',$list);
+    }
+
+    /**
+     * 获取定位城市
+     */
+    public function getCity(){
+        $list=Db::table('store_area')->where('level',2)->field('first')->group('first')->order('first asc')->select();
+        $field=['id','pid','shortname','name','first'];
+        foreach ($list as $k=>&$v){
+            $v['list']=Db::table('store_area')->where('level',2)->where('first',$v['first'])->field($field)->select();
+        }
+        $hot=Db::table('store_area')->whereIn('id',[2,802,1965])->field($field)->select();
+        $history=Db::table('store_area')->whereIn('id',[2,802,1965])->field($field)->select();
+        array_unshift($list,['first'=>'历史','list'=>$history],['first'=>'热门城市','list'=>$hot]);
+        $this->success('获取成功',$list);
+    }
+
+    /**
+     * @title 通过省市区id返回具体地址信息
+     * @desc 通过省市区id返回具体地址信息
+     * @author QC
+     * @url /api/Area/getCity
+     * @method GET
+     * @tag 省市区
+     */
+    public function get_area($province_id,$city_id,$county_id){
+        $province = '';
+        if(!empty($province_id)){
+            $quarters = Db::name('store_area')->field('name')->where('id',$province_id)->find();
+            $province = $quarters['name'];
+        }
+        $city = '';
+        if(!empty($city_id)){
+            $quarters = Db::name('store_area')->field('name')->where('id',$city_id)->find();
+            $city = $quarters['name'];
+        }
+        $county = '';
+        if(!empty($county_id)){
+            $quarters = Db::name('store_area')->field('name')->where('id',$county_id)->find();
+            $county = $quarters['name'];
+        }
+        return $province.$city.$county;
+    }
+    /**
+     * 通过省市区名称返回省市区id
+     */
+    public function get_area_id($province = '',$city = '',$county = ''){
+        $province_id = '';
+        if(!empty($province)){
+            $province_id = Db::name('store_area')->where('name',$province)->value('id');
+        }
+        $city_id = '';
+        if(!empty($city)){
+            $city_id = Db::name('store_area')->where('name',$city)->value('id');
+        }
+        $county_id = '';
+        if(!empty($county)){
+            $county_id = Db::name('store_area')->where('name',$county)->value('id');
+        }
+        return $province_id.$city_id.$county_id;
+    }
+    /**
+     * 通过id获取名称
+     */
+    public function get_name($id){
+        $quarters = Db::name('store_area')->field('name')->where('id',$id)->find();
+        return $quarters['name'];
+    }
+    /**
+     * 通过省/市的ID获取所有县区ID
+     * $type 1代表省 2代表市
+     */
+    public function get_county_id($id,$type=1){
+        $county_id_str = '';
+        if($type == 1){
+            $city_id_arr = Db::name('store_area')->field('id')->where('pid',$id)->column('id');
+            foreach ($city_id_arr as $value){
+                $county_id_arr = Db::name('store_area')->field('id')->where('pid',$value)->column('id');
+                $county_id_str .= implode(',',$county_id_arr).',';
+            }
+        }else{
+            $county_id_arr = Db::name('store_area')->field('id')->where('pid',$id)->column('id');
+            $county_id_str .= implode(',',$county_id_arr).',';
+        }
+        return substr($county_id_str,0,strlen($county_id_str)-1);
+    }
+    /**
+     * 通过省的ID获取所有市的ID
+     */
+    public function get_city_id($id){
+        $city_id_arr = Db::name('store_area')->field('id')->where('pid',$id)->column('id');
+        return implode(',',$city_id_arr);
+    }
+    /**
+     * 通过市的ID获取省的ID
+     */
+    public function get_province_id($id){
+        $province_id = Db::name('store_area')->field('id')->where('id',$id)->value('pid');
+        return $province_id;
+    }
+}

+ 172 - 0
application/api/controller/Base.php

@@ -0,0 +1,172 @@
+<?php
+
+// +----------------------------------------------------------------------
+// | ThinkAdmin
+// +----------------------------------------------------------------------
+// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
+// +----------------------------------------------------------------------
+// | 官方网站: http://demo.thinkadmin.top
+// +----------------------------------------------------------------------
+// | 开源协议 ( https://mit-license.org )
+// +----------------------------------------------------------------------
+// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
+// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
+// +----------------------------------------------------------------------
+
+namespace app\api\controller;
+
+use app\common\controller\Api;
+use Firebase\JWT\JWT;
+use think\Controller;
+use think\Db;
+use think\Exception;
+use think\exception\HttpResponseException;
+use think\Request;
+use think\Response;
+
+/**
+ * 会员管理基类
+ * Class Member
+ * @package app\store\controller\api
+ */
+class Base extends Controller
+{
+    protected $uid = 10;
+
+    protected $page; // 页数
+    protected $page_num;
+    protected $off_set;
+    public function initialize(){
+        $this->page = input('page',1);
+        $this->page_num = input('page_num',20);
+        $this->off_set   =   $this->page * $this->page_num - $this->page_num;
+    }
+    //校验jwt权限API
+    protected function check_login()
+    {
+        $authorization = app()->request->header('Authorization');
+        //var_dump($authorization.'111');exit();
+        if(empty($authorization) || $authorization == null){
+            $this->error('未登录,请先登录','',0);
+        }
+        $key = md5(config('app.jwt'));
+        try {
+            $jwtAuth = json_encode(JWT::decode($authorization, $key, array('HS256')));
+            $authInfo = json_decode($jwtAuth, true);
+            if (!empty($authInfo['uid'])) {
+                $member = Db::name('store_member')->field('status')->where('id',$authInfo['uid'])->find();
+                if($member['status']){
+                    $this->uid = $authInfo['uid'];
+                    return $this->uid;
+                }else{
+                    $this->error('该会员已被禁用','',0);
+                }
+            } else {
+                $this->error('Token验证不通过,用户不存在','',0);
+            }
+        } catch (\Firebase\JWT\SignatureInvalidException $e) {
+            $this->error('Token无效','',0);
+        } catch (\Firebase\JWT\ExpiredException $e) {
+            $this->error('Token过期','',0);
+        } catch (Exception $e) {
+            return $e;
+        }
+    }
+
+    protected function set_uid(){
+        $authorization = app()->request->header('Authorization');
+        $key = md5(config('app.jwt'));
+        if(empty($authorization) || $authorization == null){
+            return false;
+        }
+        $jwtAuth = json_encode(JWT::decode($authorization, $key, array('HS256')));
+        $authInfo = json_decode($jwtAuth, true);
+        if (!empty($authInfo['uid'])) {
+            $member = Db::name('store_member')->field('status')->where('id',$authInfo['uid'])->find();
+            if($member['status']){
+                $this->uid = $authInfo['uid'];
+                return $this->uid;
+            }
+        }
+    }
+
+    /**
+     * 操作成功返回的数据
+     * @param string $msg   提示信息
+     * @param mixed $data   要返回的数据
+     * @param int   $code   错误码,默认为1
+     * @param string $type  输出类型
+     * @param array $header 发送的 Header 信息
+     */
+    protected function success($msg = '', $data = null , $is_login = 1, $code = 1, $type = null, array $header = [])
+    {
+        $this->results($msg, $data, $is_login, $code, $type, $header);
+    }
+
+    /**
+     * 操作失败返回的数据
+     * @param string $msg   提示信息
+     * @param mixed $data   要返回的数据
+     * @param int   $code   错误码,默认为0
+     * @param string $type  输出类型
+     * @param array $header 发送的 Header 信息
+     */
+    protected function error($msg = '', $data = null, $is_login = 1, $code = 0, $type = null, array $header = [])
+    {
+        if(empty($this->uid)){
+            $is_login = 0;
+        }
+        $this->results($msg, $data, $is_login, $code, $type, $header);
+    }
+    /**
+     * 返回封装后的 API 数据到客户端
+     * @access protected
+     * @param mixed  $msg    提示信息
+     * @param mixed  $data   要返回的数据
+     * @param int    $code   错误码,默认为0
+     * @param string $type   输出类型,支持json/xml/jsonp
+     * @param array  $header 发送的 Header 信息
+     * @return void
+     * @throws HttpResponseException
+     */
+    protected function results($msg, $data = null, $is_login, $code = 0, $type = null, array $header = [])
+    {
+        $result = [
+            'code' => $code,
+            'is_login' => $is_login,
+            'msg'  => $msg,
+            'time' => \think\facade\Request::instance()->server('REQUEST_TIME'),
+            'data' => $data,
+        ];
+        // 如果未设置类型则自动判断
+        $type = $type ? $type : 'json';
+
+        if (isset($header['statuscode']))
+        {
+            $code = $header['statuscode'];
+            unset($header['statuscode']);
+        }
+        else
+        {
+            //未设置状态码,根据code值判断
+            $code = $code >= 1000 || $code < 200 ? 200 : $code;
+        }
+        $response = Response::create($result, $type, $code)->header($header);
+        throw new HttpResponseException($response);
+    }
+    protected function get_uid(){
+        $uid = 0;
+        $authorization = app()->request->header('Authorization');
+        if(!empty($authorization)){
+            $key = md5(config('app.jwt'));
+            $jwtAuth = json_encode(JWT::decode($authorization, $key, array('HS256')));
+            $authInfo = json_decode($jwtAuth, true);
+            if (!empty($authInfo['uid'])) {
+                $uid = $authInfo['uid'];
+            }
+        }
+        return $uid;
+    }
+
+
+}

+ 253 - 0
application/api/controller/Box.php

@@ -0,0 +1,253 @@
+<?php
+
+
+namespace app\api\controller;
+use app\common\library\AliPay;
+use EasyWeChat\Factory;
+use think\Db;
+/**
+ * @title 盲盒
+ * @controller Box
+ * @package app\api\controller
+ */
+class Box extends Base
+{
+
+    public function initialize(){
+        parent::check_login();
+    }
+
+    /**
+     * @title 抽奖
+     * @desc  抽奖
+     * @author  Gavin
+     * @url /api/Box/lucky_draw
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     */
+    public function lucky_draw(){
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        //获取每日免费次数
+        $free_number = getConfigValue('free_lucky_number');
+        //获取当日免费抽奖记录
+        $date = date('Y-m-d');
+        $now_date_count = Db::name('store_blind_box_log')
+            ->where('m_id',$this->uid)
+            ->where('date',$date)
+            ->where('is_free',0)
+            ->count();
+        $free = $free_number-$now_date_count; //剩余免费次数
+        $number = $user['lottery_number'] + $free;
+        if ($number<=0) $this->error('抽奖次数不足');
+        checkBox();
+        $prize_arr = Db::name('store_blind_box')
+            ->where('status',1)
+            ->where('is_del',1)
+            ->field('id,rate')
+            ->select();
+        foreach ($prize_arr as $key => $val) {
+            $arr[$val['id']] = $val['rate'];//概率数组
+        }
+        $rid = get_rand($arr); //根据概率获取奖项id
+        $info = Db::name('store_blind_box')->where('id',$rid)->field('id,title,prize,is_prize')->find();
+        $return = [
+            'is_prize'=>$info['is_prize']
+        ];
+        $coll_info =  getCollectionInfoHash($info['prize']);
+        if ($info['is_prize']==1){
+            $json_info = json_encode($coll_info,true);
+            $return['name'] = $info['title'];
+            $return['cover'] = $coll_info['cover'];
+
+        }else{
+            $json_info = '';
+            $return['name'] = '未中奖';
+            $return['cover'] = '';
+        }
+        $com = true;
+        Db::startTrans();
+        try {
+            $data = [
+                'm_id'=>$this->uid,
+                'b_id'=>$rid,
+                'is_prize'=>$info['is_prize'],
+                'name'=>$return['name'],
+                'cover'=>$return['cover'],
+                'info'=>$json_info,
+                'date'=>$date,
+                'is_free'=>$free>0 ? 0 : 1
+            ];
+            Db::name('store_blind_box_log')->insert($data);
+            if (!$free){
+                Db::name('store_member')->where('id',$this->uid)->setDec('lottery_number');
+            }
+            //减少数据库库存
+            if ($info['is_prize']==1){
+                Db::name('store_collection')->where('id',$info['prize'])->setDec('now_inventory');
+                //获取排名
+                $rank = getRanking($info['prize'])+1;
+                $tag = getTag($info['prize'],$rank,$coll_info['inventory']);
+                saveRanking($info['prize']);
+                $nfttype = Db::name('hash2')->where('goods_id',$info['prize'])->where('success',1)->value('NFTType');
+                $collectors_hash = '';
+                $date = [
+                    'order_id'=>0,
+                    'order_no'=>get_order_sn(),
+                    'tag'=>$tag,
+                    'mid'=>$this->uid,
+                    'c_id'=>$info['prize'],
+                    'name'=>$coll_info['name'],
+                    'cover'=>$coll_info['cover'],
+                    'pro_info'=>$json_info,
+                    'nfttype'=>$nfttype,
+                    'collectors_hash'=>$collectors_hash,
+                    'collectors_hash_time'=>'',
+                    'status'=>4
+                ];
+                Db::name('store_order_info')->insert($date);
+            }
+            Db::commit();
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($this->uid);
+            if ($info['is_prize']==1){
+                //减掉库存
+                loseCollectionInventory($info['prize'],1);
+            }
+
+
+            $this->success('成功',$return);
+        }
+        $this->error('系统错误,请稍后重试');
+    }
+
+    /**
+     * @title 充值次数
+     * @desc  充值次数
+     * @author  Gavin
+     * @url /api/Box/recharge
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:num type:int require:1 default:1 desc:充值次数
+     * @param name:pay_type type:string require:1 default:wx desc:wx:微信zfb:支付宝
+     * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
+     *
+     * @return  name:order_no type:int require:0 default:0 desc:订单号
+     * @return  name:pay type:string require:0 default:0 desc:支付信息
+     */
+    public function recharge(){
+        $num = input('num',1);
+        $pay_type = input('pay_type','wx');
+        $from = input('from','wx');
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        if ($num<1) $this->error('数量错误');
+        $com = true;
+        Db::startTrans();
+        try {
+            $order_no = get_order_sn();
+            //获取价格
+            $price = getConfigValue('lucky_recharge_price');
+            $total_fee = bcmul($price,$num,2);
+            $data = [
+                'order_no'=>$order_no,
+                'm_id'=>$this->uid,
+                'num'=>$num,
+                'price'=>$price,
+                'pay_price'=>$total_fee,
+                'pay_type'=>$pay_type
+            ];
+            Db::name('store_blind_recharge')->insert($data);
+            $body = 'top艺术充值盲盒次数';
+            switch ($pay_type){
+                case 'wx':
+                    $config = retrunWxConfig();
+                    $total_fee = $total_fee * 100;
+                    $config['notify_url'] = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/BlindRechargeNotify';
+                    $app = Factory::payment($config);
+
+                    $post_data = [
+                        'body' => $body,
+                        'out_trade_no' => $order_no,
+                        'total_fee' => $total_fee,
+                        'attach'=>$this->uid,     //自定义传值
+                    ];
+                    //trade_type   SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
+                    if ($from=='wx'){
+                        $post_data['openid'] = $user['openid'];
+                        $post_data['trade_type'] = 'JSAPI';
+                    }elseif ($from=='h5'){
+                        $post_data['trade_type'] = 'MWEB';
+                    }
+                    $result = $app->order->unify($post_data);
+
+                    if ($result['return_msg']=='OK'){
+                        if ($result['result_code']=='FAIL'){
+                            $com = false;
+                            Db::rollback();
+                        }else{
+                            $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
+                            $retrun_data['order_no'] = $order_no;
+                            $retrun_data['pay'] = json_decode($order1,true);
+                            Db::commit();
+                        }
+                    }else{
+                        $com = false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'zfb':
+
+                    $zfb = new AliPay();
+                    $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/alipayBlindRechargeNotify';//回调地址
+                    $order = $zfb->ali_pay_pc($body, $total_fee, $order_no, $notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/manghe/manghe');//调用支付宝支付的方法
+                    $retrun_data['order_no'] = $order_no;
+                    $retrun_data['pay'] = $order;
+                    Db::commit();
+                    break;
+            }
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            $this->success('成功',$retrun_data);
+        }
+        $this->error('失败,请稍后重试');
+    }
+
+
+
+    /**
+     * @title 用户中奖纪录
+     * @desc 用户中奖纪录
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/Box/boxLog
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @return  name:id type:int require:0 default:0 desc:ID
+     * @return  name:name type:string require:0 default:0 desc:奖品名称
+     * @return  name:cover type:string require:0 default:0 desc:藏品图片
+     * @return  name:member_name type:string require:0 default:0 desc:用户昵称
+     * @return  name:create_at type:string require:0 default:0 desc:时间
+     */
+    public function boxLog(){
+        $list = Db::name('store_blind_box_log')
+            ->whereIn('is_prize',1)
+            ->field('id,m_id,name,cover,create_at')
+            ->order('id desc')
+            ->limit(10)
+            ->select();
+        foreach ($list as &$v){
+            $v['member_name'] = Db::name('store_member')->where('id',$v['m_id'])->value('name');
+        }
+        $this->success('成功',$list);
+    }
+
+
+}

+ 244 - 0
application/api/controller/Index.php

@@ -0,0 +1,244 @@
+<?php
+
+
+namespace app\api\controller;
+use app\common\library\Jiyan;
+use think\Db;
+/**
+ * @title 首页、详情
+ * @controller Index
+ * @package app\api\controller
+ */
+class Index extends Base
+{
+
+    public function initialize(){
+        //parent::check_login();
+    }
+
+    /**
+     * @title 获取轮播图
+     * @desc  获取轮播图
+     * @author  Gavin
+     * @url /api/Index/getBanner
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     */
+    public function getBanner()
+    {
+        $list =  Db::table('store_banner')
+            ->field('id,name,cover')
+            ->where(['is_deleted'=>0])
+            ->order('sort desc ,id desc')
+            ->select();
+        $this->success('ok',$list);
+    }
+
+    /**
+     * @title 首页藏品列表
+     * @desc 首页藏品列表
+     * @author  Gavin
+     * @url /api/Index/indexCollectionList
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @return name:name type:string default:-- desc:藏品名称
+     * @return name:cover type:string default:-- desc:藏品图
+     * @return name:label type:string default:-- desc:标签
+     * @return name:price type:DECIMAL default:-- desc:价格
+     * @return name:inventory type:int default:-- desc:库存
+     * @return name:now_inventory type:int default:-- desc:剩余库存
+     * @return name:state type:float default:-- desc:藏品状态(1:进行中2:即将开售3:已售罄)
+     * @return name:sell_time type:string default:-- desc:发行时间
+     * @return name:is_remind type:string default:-- desc:是否设置提醒true:已设置false:未设置
+     */
+    public function indexCollectionList(){
+        checkCollectionState();
+        parent::set_uid();
+        $type = input('type',1);
+        $list = Db::name('store_collection')
+            ->where('is_deleted',0)
+            ->where('status',1)
+            ->whereIn('type',$type)
+            ->field('id,cover,name,label,price,inventory,now_inventory,state,sell_time,detail_img,sold_out,buy_count')
+            ->order('state asc,sell_time desc')
+            ->select();
+        foreach ($list as &$v){
+            $now_inventory = getCollectionInventory($v['id']);
+            $v['now_inventory'] = $now_inventory<=0 ? 0 : $now_inventory;
+            if (!$v['sold_out']){
+                $v['state'] =3;
+                $v['now_inventory'] = 0;
+            }
+            $v['id'] = (string)$v['id'];
+            $v['is_remind'] = getRemind($this->uid,$v['id']) ? true : false;
+        }
+       $this->success('成功',$list);
+    }
+
+    /**
+     * @title 发售日历
+     * @desc 发售日历
+     * @author  Gavin
+     * @url /api/Index/CollectionCalendar
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @return name:name type:string default:-- desc:藏品名称
+     * @return name:cover type:string default:-- desc:藏品图
+     * @return name:label type:string default:-- desc:标签
+     * @return name:price type:DECIMAL default:-- desc:价格
+     * @return name:inventory type:int default:-- desc:库存
+     * @return name:now_inventory type:int default:-- desc:剩余库存
+     * @return name:state type:float default:-- desc:藏品状态(1:进行中2:即将开售3:已售罄)
+     * @return name:sell_time type:string default:-- desc:发行时间
+     * @return name:date type:string default:-- desc:日期
+     * @return name:time type:string default:-- desc:时间
+     * @return name:is_remind type:string default:-- desc:是否设置提醒true:已设置false:未设置
+     */
+    public function CollectionCalendar(){
+        checkCollectionState();
+        parent::set_uid();
+        $datelist = Db::name('store_collection')
+            ->where('is_deleted',0)
+            ->where('status',1)
+            ->where('state',2)
+            ->whereIn('type','1,3')
+            ->where('date','>=',date('Y-m-d',time()))
+            ->group('date')
+            ->order('date asc')
+            ->column('date');
+        if ($datelist){
+            foreach ($datelist as $k=>&$v){
+                $array[$k]['date'] = $v;
+                $list = Db::name('store_collection')
+                    ->where('is_deleted',0)
+                    ->where('status',1)
+                    ->where('state',2)
+                    ->where('date',$v)
+                    ->field('id,cover,name,label,price,inventory,now_inventory,state,sell_time,date,detail_img,sold_out,buy_count')
+                    ->order('state asc,sell_time asc')
+                    ->select();
+                foreach ($list as &$a){
+                    $a['time'] = date('H:i',strtotime($a['sell_time']));
+                    $a['is_remind'] = getRemind($this->uid,$a['id']) ? true : false;
+                    $now_inventory = getCollectionInventory($a['id']);
+                    $a['now_inventory'] = $now_inventory<=0 ? 0 : $now_inventory;
+                    if (!$a['sold_out']){
+                        $a['state'] =3;
+                        $a['now_inventory'] = 0;
+                    }
+                    $a['id'] = (string)$a['id'];
+                }
+                $array[$k]['list'] = $list;
+            }
+        }else $array=[];
+        $this->success('成功',$array);
+    }
+
+
+
+    /**
+     * @title 藏品详情
+     * @desc 藏品详情
+     * @author  Gavin
+     * @url /api/Index/CollectionDetail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:string require:1 default:-- desc:藏品ID
+     *
+     * @return name:name type:string default:-- desc:藏品名称
+     * @return name:cover type:string default:-- desc:藏品图
+     * @return name:label type:string default:-- desc:标签
+     * @return name:price type:DECIMAL default:-- desc:价格
+     * @return name:inventory type:int default:-- desc:库存
+     * @return name:now_inventory type:int default:-- desc:剩余库存
+     * @return name:state type:float default:-- desc:藏品状态(1:进行中2:即将开售3:已售罄)
+     * @return name:sell_time type:string default:-- desc:发行时间
+     * @return name:describe type:string default:-- desc:商品描述
+     * @return name:buy_count type:int default:1 desc:限购数量
+     * @return name:instructions type:int default:1 desc:权益说明
+     * @return name:buy_notice type:int default:1 desc:购买须知
+     * @return name:warm_prompt type:int default:1 desc:温馨提示
+     * @return name:auth_img type:int default:1 desc:作者头像
+     * @return name:auth_name type:int default:1 desc:作者姓名
+     * @return name:is_remind type:string default:-- desc:是否设置提醒true:已设置false:未设置
+     * @return name:share_img type:string default:-- desc:分享二维码
+     */
+    public function CollectionDetail(){
+        parent::set_uid();
+        $collect_id = input('id');
+        if (!$collect_id) $this->error('参数错误');
+        checkCollectionState($collect_id);
+        $info = Db::name('store_collection')
+            //->where('is_deleted',0)
+            //->where('status',1)
+            ->whereIn('type','1,3')
+            ->where('id',$collect_id)
+            ->field('id,cover,name,label,price,inventory,now_inventory,state,sell_time,describe,buy_count,instructions,buy_notice,warm_prompt,auth_img,auth_name,share_img,intro,detail_img,format,sold_out')
+            ->find();
+        if (!$info) $this->error('藏品不存在');
+        $info['describe'] = explode('|',$info['describe']);
+        $info['is_remind'] = getRemind($this->uid,$info['id']) ? true : false;
+        $now_inventory = getCollectionInventory($info['id']);
+        $info['now_inventory'] = $now_inventory<=0 ? 0 : $now_inventory;
+        if (!$info['sold_out']){
+            $info['state'] =3;
+            $info['now_inventory'] = 0;
+        }
+
+        $info['is_card'] = 0;
+
+        if ($info['state']==2){
+            $cards  = checkUseSnapcard($this->uid,$collect_id,$info['sell_time']);
+            if ($cards){
+                $info['is_card']  = 1;
+            }
+        }
+
+        $info['id'] = (string)$info['id'];
+        $this->success('成功',$info);
+    }
+
+
+    /**
+     * @title 设置提醒
+     * @desc 设置提醒
+     * @author  Gavin
+     * @url /api/Index/setRemind
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:string require:1 default:-- desc:藏品ID
+     */
+    public function setRemind(){
+        parent::check_login();
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $remind = Db::name('store_collection_remind')
+            ->where('mid',$this->uid)
+            ->where('c_id',$id)
+            ->count();
+        if ($remind) $this->error('已设置过');
+        $data = [
+            'c_id'=>$id,
+            'mid'=>$this->uid,
+            'mobile'=>Db::name('store_member')->where('id',$this->uid)->value('phone')
+        ];
+        if (Db::name('store_collection_remind')->insert($data)) $this->success('设置成功');
+        $this->error('设置失败');
+    }
+
+    public function jytest(){
+        $lot_number = input('lot_number');
+        $captcha_output = input('captcha_output');
+        $pass_token = input('pass_token');
+        $gen_time = input('gen_time');
+        $jy = new Jiyan();
+        $jy->jy($lot_number,$captcha_output,$pass_token,$gen_time,2);
+    }
+
+
+}

+ 291 - 0
application/api/controller/Login.php

@@ -0,0 +1,291 @@
+<?php
+
+// +----------------------------------------------------------------------
+// | ThinkAdmin
+// +----------------------------------------------------------------------
+// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
+// +----------------------------------------------------------------------
+// | 官方网站: http://demo.thinkadmin.top
+// +----------------------------------------------------------------------
+// | 开源协议 ( https://mit-license.org )
+// +----------------------------------------------------------------------
+// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
+// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
+// +----------------------------------------------------------------------
+
+namespace app\api\controller;
+
+use app\common\library\AliPay3;
+use app\common\library\HuijuH5Pay;
+use app\common\library\HuijuKjPay;
+use think\cache\driver\Redis;
+use think\Db;
+use Firebase\JWT\JWT;
+use think\facade\Validate;
+
+/**
+ * @title 用户登录
+ * @controller Login
+ * @group worker
+ */
+class Login extends Base
+{
+    /**
+     * @title 注册
+     * @desc 注册
+     * @url /api/Login/Register
+     * @method POST
+     * @tag 基础
+     * @header
+     * @param  name:phone type:string require:1 desc:手机号
+     * @param  name:ver_code type:string require:1 desc:验证码
+     * @param  name:password type:string require:1 desc:密码
+     * @param  name:confirm_password type:string require:1 desc:确认密码
+     * @param  name:second_password type:string require:1 desc:二级密码
+     * @param  name:confirm_second_password type:string require:1 desc:二级确认密码
+     * @param  name:invite_code type:string require:0 desc:邀请码
+     *
+     */
+    public function Register(){
+        $phone = input('phone');
+        $ver_code = input('ver_code');
+        //$password = input('password');
+      //  $confirm_password = input('confirm_password');
+        $second_password = input('second_password');
+        $confirm_second_password = input('confirm_second_password');
+        $invite_code = input('invite_code');
+        if (!$phone || !$ver_code  || !$second_password || !$confirm_second_password){
+            $this->error('参数错误');
+        }
+
+        if (!Validate::regex($phone, "^1\d{10}$")) {
+            $this->error('手机号格式错误');
+        }
+
+        //验证短信验证码
+        $time = time()-60;
+        $sms = Db::name('store_sms')->where(['mobile' => $phone, 'event' => 'register'])
+            ->where('createtime','>',$time)
+            ->order('id', 'DESC')
+            ->find();
+       // if (!$sms || $sms['code'] != $ver_code) $this->error('短信验证码不正确!');
+
+        $user = Db::name('store_member')
+            ->where('is_deleted',0)
+            ->where('phone',$phone)
+            ->find();
+        if ($user) $this->error('手机号已注册');
+       // if (!preg_match('/^[0-9a-z]{6,12}$/i',$password)) $this->error('密码格式错误,请输入6-12位数字+字母');
+       // if ($password!=$confirm_password) $this->error('密码与确认密码不一致');
+        if (!preg_match('/^[0-9]{6}$/i',$second_password)) $this->error('二级密码格式错误,请输入6位纯数字');
+        if ($second_password!=$confirm_second_password) $this->error('二级密码与确认密码不一致');
+        if ($invite_code){
+            $isset = Db::name('store_member')->where('is_deleted',0)->where('invite_code',$invite_code)->find();
+            if (!$isset) $this->error('邀请码不存在');
+            $invitecode = $isset['id'];
+        }else{
+            $invitecode = 0;
+        }
+        $wallet_address = '';
+        $accountName = '';
+        $data = [
+            'phone'=>$phone,
+            'pid'=>$invitecode,
+            //'password'=>md5($password),
+            'second_password'=>md5($second_password),
+            'wallet_address'=>$wallet_address,
+            'accountName'=>$accountName,
+            'reg_ip'=>request()->ip()
+        ];
+        $member_id = Db::name('store_member')->insertGetId($data);
+        if ($member_id){
+            $code = get32Str(8);
+            $invite_img = setintivecode($code);
+            $invite_address = getintiveaddress($code);
+            Db::name('store_member')->where('id',$member_id)->update(['name'=>'收藏者'.$member_id,'invite_img'=>$invite_img,'invite_address'=>$invite_address,'invite_code'=>$code]);
+            //邀请好友送积分
+            if ($invite_code>0){
+                $invite_friends_integral = getConfigValue('invite_friends_integral');
+                memberMoneyChange($invite_friends_integral,1,$invite_code,'邀请好友',1,$member_id);
+            }
+            $this->success('注册成功');
+        }
+        $this->error('注册失败');
+    }
+
+    /**
+     * @title 登录
+     * @desc 登录
+     * @url /api/Login/passwordLogin
+     * @method POST
+     * @tag 基础
+     * @header
+     * @param name:phone type:int require:1 default:-- desc:手机号
+     * @param name:password type:string require:1 default:-- desc:密码
+     * @param  name:ver_code type:string require:1 desc:验证码
+     * @return name:token type:string default:-- desc:用户登录成功后的token值
+     */
+    public function passwordLogin()
+    {
+        $type = input('type',1);   //1:验证码登录 2:密码登录
+        $phone = input('phone');
+
+        $password = input('password');
+
+        $ver_code = input('ver_code');
+        if ($type==1){
+            if (empty($ver_code) || empty($phone)) {
+                $this->error('参数错误');
+            }
+        }else{
+            if (empty($password) || empty($phone)) {
+                $this->error('参数错误');
+            }
+        }
+
+        $member = Db::name('store_member')
+            ->where('phone', $phone)
+            ->where('is_deleted',0)
+            ->find();
+        if (!$member) $this->error('手机号未注册');
+
+        //验证短信验证码
+        if ($phone!='15801075991'){
+            if ($type==1){
+                $time = time()-60;
+                $sms = Db::name('store_sms')->where(['mobile' => $phone, 'event' => 'login'])
+                    ->where('createtime','>',$time)
+                    ->order('id', 'DESC')
+                    ->find();
+                // if (!$sms || $sms['code'] != $ver_code) $this->error('短信验证码不正确!');
+            }else{
+                 if ($member['password']!=md5($password)) $this->error('密码错误');
+            }
+        }
+        $token = self::create_jwt($member['id']);
+        setMemberInfoHash($member['id']);
+
+        Db::name('store_member')->where('id',$member['id'])->update(['ip'=>request()->ip()]);
+        LoginLog($member['id']);
+
+        $this->success('登录成功', $token);
+    }
+
+    //token加密
+    public function create_jwt($uid)
+    {
+        $key = md5(config('app.jwt')); //jwt的签发密钥,验证token的时候需要用到
+        $time = time(); //签发时间
+        $expire = $time + config('app.jwt_time'); //过期时间
+        $token = array(
+            "uid" => $uid,
+            "iss" => "https://zain.com",//签发组织
+            "aud" => "https://zain.com", //签发作者
+            "iat" => $time,
+            "nbf" => $time,
+            "exp" => $expire
+        );
+        $jwt = JWT::encode($token, $key);
+        return $jwt;
+    }
+
+
+    /**
+     * @title 找回密码
+     * @desc 找回密码
+     * @url /api/Login/ForgetPassword
+     * @method POST
+     * @tag 基础
+     * @header
+     * @param name:phone type:int require:1 default:-- desc:手机号
+     * @param name:ver_code type:string require:1 desc:验证码
+     * @param name:password type:string require:1 default:-- desc:密码
+     * @param  name:confirm_password type:string require:1 desc:确认密码
+     */
+    public function ForgetPassword(){
+
+        $phone = input('phone');
+        $ver_code = input('ver_code');
+        $password = input('password');
+        $confirm_password = input('confirm_password');
+        if (!$phone || !$ver_code || !$password || !$confirm_password) $this->error('参数错误');
+        $member = Db::name('store_member')
+            ->where('phone', $phone)
+            ->where('is_deleted',0)
+            ->find();
+        if (!$member) $this->error('手机号未注册');
+        //验证短信验证码
+        $time = time()-60;
+        $sms = Db::name('store_sms')->where(['mobile' => $phone, 'event' => 'forgetpwd'])
+            ->where('createtime','>',$time)
+            ->order('id', 'DESC')
+            ->find();
+        if (!$sms || $sms['code'] != $ver_code) $this->error('短信验证码不正确!');
+
+        if (!preg_match('/^[0-9a-z]{6,12}$/i',$password)) $this->error('密码格式错误,请输入6-12位数字+字母');
+        if ($password!=$confirm_password) $this->error('密码与确认密码不一致');
+        $data = [
+            'password'=>md5($password),
+            'update_at'=>date('Y-m-d H:i:s')
+        ];
+        if (Db::name('store_member')->where('id',$member['id'])->update($data)) $this->success('修改成功');
+        $this->error('修改失败');
+    }
+
+
+    public function test(){
+
+//        $huiju = new HuijuH5Pay();
+//        $result = $huiju->we_pay();
+//        die();
+//
+//
+//        $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/mine/order-list?num=1';
+//        $huiju = new HuijuH5Pay();
+//        $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylOrderNotify';//回调地址
+//        $huiju->ylh5Pay(get_order_sn(),$callback_url,$notify_url,"测试藏品","0.01",'100000',1);
+
+        $huiju = new HuijuKjPay();
+       // $huiju->sms();
+       // $huiju->signing();
+         $huiju->paySms();
+         $huiju->kuaijiePay();
+        die;
+
+
+
+
+        require_once env('root_path').'application/common/library/lib/YopRequest.php';
+        require_once env('root_path').'application/common/library/lib/YopClient3.php';
+        require_once env('root_path').'application/common/library/lib/YopRsaClient.php';
+        $private_key ="MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQCSBUBO99o0ZCGp8/6qbaSW2MIeHKg/4RZEIexS11rW5wxFH9arHM16yZXObtm1qO1xNnRyofBkKvcOes63yNGqiulDjJbx2gxXzQdq6g0ckO06YZsb6v4AmkCLwu/kFPYTd2zu3nM6VCBsppkjXe2kcdgeKH55PSqTLcqe4TCeCQxxTjmixX/qFdKva7uJxXbdNjE+GERMTPsn9FBGjrv2/6RCz62eZljRcQf2zY9aooXnYaYOOkeYxJaFu4M8k1kEfL/A2ftDGa+9zyWUUp/PskIcSUCLtdd+yR1ZvNldBW7iaeTuZ4YQXg6N3BqFBacBmkHztpqbrg1+C46Hp9C5AgMBAAECggEAHxpGPevWSzWO5BwIQTYUilsGUhKpld20d/O/2k9r1n2lY+3ptE3wyLg6lBuRfPvMA8LlH+u1gCiUsj1fXc9FNa3cDNrb5g0bdvJBo7jR7SVDmp1ixBoUW31/xSVTIsKCPdhA2EqCwchclbgQw3NE5LOpJJdchSeNs4bb82alC+wwDaYnrag/OuehknazOQWqWQVaPI9zJ+eXPNW3etWte1OLDyVoNiSrAdr/BWnBxSxcoANuzypkKN/Sw5W3vgkIHyylReiOLP2JLDNpnDlwgNIaxYOF7p0VGLC1bAqZTjuR4nMKZhohC4FtNC1cy9ZV2c2KV3bD8OXoMPIQB1a8OQKBgQDJnVxK2Mwk/2jitP7mA6JpBFeioJLRzj4gKK56M4InYaRRVZj6UWJd5Jt0uEs6XMlus8T24cfSoglCnXIvTMx8okTbusxn72Ff+/+f0GRLI/VO/cyhSWFl+0P3XTCT+zUVI3+kWvZQ6Tj7l5BWubqLObBdrNDhpbYhQptnm+KHJQKBgQC5aMwHmtE5Cs6+z2zGkIvVw4dOQcvo9E+gGCsjbSdeFvcrOpg/NNQ0dmu54F8CF6z9VkoitL8NTI63V4LqLnZ9BK+K/0L8UMzbbhCqT9+JTrjNraby77akxldKTUZZHiC1edWT/Ll8t6xT6oVw02yZz8Flpc7DnNXuJGGxQIlpBQKBgHinh3fzThhGEUq67mBQhH08cAWO+s48EsnEDsZq1NUkheV2yFnlImz2RmIPeFWw+YsaYiEs9RyEg+4tswvDdgyQP40PuQvqIegDMvM9Dwg4uLFRCYjsFTGmsQHz7wJgL2A7IkQFe6roJ4XJKkP8Wjq/h2BH76laNiipZ78VdvdRAoGAKqgnuOAvnrYT1uaWGiELawTvTlvo9BGxeB3XXvEwC3+xIir6+B/JasCMDFnIwgx8alS9/boj8oVYmGnz3rkN3bmEQbB00uQsHZf3EZgKz+Zisj9vl5j7JnHTh2677qj10YrUTpXofv8rEKt2Q/0AkWqxDhfVuD+PIJkSONumQZkCgYACVSnvt+kizEIwdshvGzAkzZaMBR6Xf1d//cuIWzU1bO7pkCQogq9MXv9hCZx2N1B51VSvwazJ6B029uzgYzr0iTeZBfB5ntb53/vEycLMhB9r9Bc+0x7bxrQZeWNwLlQFxhDeSO9le0ETjfGPt/nS2QbFL7bNHwRGE+AzxwH0aA==";
+        $request = new \YopRequest("app_10086863938", $private_key);
+
+        $request->addParam('requestNo',get_order_sn());
+        $request->addParam('parentMerchantNo','10086863938');
+        $request->addParam('merchantNo','10086863938');
+        $request->addParam('name','宋星伟');
+        $request->addParam('cardNo','371324199210037710');
+        $request->addParam('cardType','ID');
+        $request->addParam('returnUrl','https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/mine/order-list?num=1');
+        $request->addParam('notifyUrl','https://'.$_SERVER['SERVER_NAME'].'/api/Pay/re');
+
+        $response = \YopRsaClient::post("/rest/v1.0/auth/face-auth", $request);
+        dump(json_decode(json_encode($response,true),true));
+        die;
+
+
+
+
+
+//        require_once env('root_path').'application/common/library/lib/YopRequest.php';
+//        require_once env('root_path').'application/common/library/lib/YopClient3.php';
+//        require_once env('root_path').'application/common/library/lib/YopRsaClient.php';
+//        $private_key ="MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQCSBUBO99o0ZCGp8/6qbaSW2MIeHKg/4RZEIexS11rW5wxFH9arHM16yZXObtm1qO1xNnRyofBkKvcOes63yNGqiulDjJbx2gxXzQdq6g0ckO06YZsb6v4AmkCLwu/kFPYTd2zu3nM6VCBsppkjXe2kcdgeKH55PSqTLcqe4TCeCQxxTjmixX/qFdKva7uJxXbdNjE+GERMTPsn9FBGjrv2/6RCz62eZljRcQf2zY9aooXnYaYOOkeYxJaFu4M8k1kEfL/A2ftDGa+9zyWUUp/PskIcSUCLtdd+yR1ZvNldBW7iaeTuZ4YQXg6N3BqFBacBmkHztpqbrg1+C46Hp9C5AgMBAAECggEAHxpGPevWSzWO5BwIQTYUilsGUhKpld20d/O/2k9r1n2lY+3ptE3wyLg6lBuRfPvMA8LlH+u1gCiUsj1fXc9FNa3cDNrb5g0bdvJBo7jR7SVDmp1ixBoUW31/xSVTIsKCPdhA2EqCwchclbgQw3NE5LOpJJdchSeNs4bb82alC+wwDaYnrag/OuehknazOQWqWQVaPI9zJ+eXPNW3etWte1OLDyVoNiSrAdr/BWnBxSxcoANuzypkKN/Sw5W3vgkIHyylReiOLP2JLDNpnDlwgNIaxYOF7p0VGLC1bAqZTjuR4nMKZhohC4FtNC1cy9ZV2c2KV3bD8OXoMPIQB1a8OQKBgQDJnVxK2Mwk/2jitP7mA6JpBFeioJLRzj4gKK56M4InYaRRVZj6UWJd5Jt0uEs6XMlus8T24cfSoglCnXIvTMx8okTbusxn72Ff+/+f0GRLI/VO/cyhSWFl+0P3XTCT+zUVI3+kWvZQ6Tj7l5BWubqLObBdrNDhpbYhQptnm+KHJQKBgQC5aMwHmtE5Cs6+z2zGkIvVw4dOQcvo9E+gGCsjbSdeFvcrOpg/NNQ0dmu54F8CF6z9VkoitL8NTI63V4LqLnZ9BK+K/0L8UMzbbhCqT9+JTrjNraby77akxldKTUZZHiC1edWT/Ll8t6xT6oVw02yZz8Flpc7DnNXuJGGxQIlpBQKBgHinh3fzThhGEUq67mBQhH08cAWO+s48EsnEDsZq1NUkheV2yFnlImz2RmIPeFWw+YsaYiEs9RyEg+4tswvDdgyQP40PuQvqIegDMvM9Dwg4uLFRCYjsFTGmsQHz7wJgL2A7IkQFe6roJ4XJKkP8Wjq/h2BH76laNiipZ78VdvdRAoGAKqgnuOAvnrYT1uaWGiELawTvTlvo9BGxeB3XXvEwC3+xIir6+B/JasCMDFnIwgx8alS9/boj8oVYmGnz3rkN3bmEQbB00uQsHZf3EZgKz+Zisj9vl5j7JnHTh2677qj10YrUTpXofv8rEKt2Q/0AkWqxDhfVuD+PIJkSONumQZkCgYACVSnvt+kizEIwdshvGzAkzZaMBR6Xf1d//cuIWzU1bO7pkCQogq9MXv9hCZx2N1B51VSvwazJ6B029uzgYzr0iTeZBfB5ntb53/vEycLMhB9r9Bc+0x7bxrQZeWNwLlQFxhDeSO9le0ETjfGPt/nS2QbFL7bNHwRGE+AzxwH0aA==";
+//        $request = new \YopRequest("app_10086863938", $private_key);
+//        $request->addFile("merQual", "C:\Users\xk\Pictures\΢�Ž�ͼ_20220316150041.png");
+//        $response = \YopRsaClient::upload("/yos/v1.0/sys/merchant/qual/upload", $request);
+//        dump($response);
+    }
+
+}

+ 646 - 0
application/api/controller/Order.php

@@ -0,0 +1,646 @@
+<?php
+namespace app\api\controller;
+use app\common\library\AliPay;
+use app\common\library\AliPay3;
+use app\common\library\HuijuH5Pay;
+use app\common\library\HuijuKjPay;
+use app\common\library\Jiyan;
+use think\cache\driver\Redis;
+use think\Db;
+use think\Exception;
+use EasyWeChat\Factory;
+use think\Session;
+use function Sodium\add;
+
+
+/**
+ * @title 订单
+ * @controller GoodsOrder
+ * @group base
+ */
+class Order extends Base
+{
+    public function initialize()
+    {
+        parent::initialize();
+        parent::check_login();
+    }
+
+    /**
+     * @title 生成订单(立即购买 )
+     * @desc  生成订单(立即购买 )
+     * @author  Gavin
+     * @url /api/Order/createOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:id type:int require:1 default:-- desc:藏品id
+     * @param name:num type:int require:1 default:1 desc:数量
+     * @param name:pay_type type:string require:1 default:wx desc:wx:微信zfb:支付宝
+     * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
+     *
+     * @return  name:order_no type:int require:0 default:0 desc:订单号
+     * @return  name:pay type:string require:0 default:0 desc:支付信息
+     */
+    public function createOrder()
+    {
+        $this->error('没有支付');
+
+        $redis = new Redis([ 'select'=> 2]);
+        $redis_value = $redis->get('order_buy'.$this->uid);
+        if ($redis_value){
+            $this->error('请求过快,请稍后重试');
+        }else{
+            $redis->set('order_buy'.$this->uid,1,1);
+        }
+
+        $id = input('id');
+        $num = input('num',1);
+        $pay_type = input('pay_type','wx');
+        $from = input('from','wx');
+
+        if (!$id || !$num) $this->error('参数错误');
+        //极验数据
+//        $lot_number = input('lot_number');
+//        $captcha_output = input('captcha_output');
+//        $pass_token = input('pass_token');
+//        $gen_time = input('gen_time');
+//        if (!$lot_number || !$captcha_output || !$pass_token || !$gen_time) $this->error('参数错误');
+//
+//        $jy = new Jiyan();
+//        $result = $jy->jy($lot_number,$captcha_output,$pass_token,$gen_time,2);
+//        if ($result['result']=='fail') $this->error('校验失败,请稍后重试');
+
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        if ($user['is_auth']==0) $this->error('请实名认证后购买!');
+        $coll_info = getCollectionInfoHash($id);
+        if ($coll_info['is_deleted']==1 || $coll_info['status']==0) $this->error('藏品已下架');
+
+        $inventory = getCollectionInventory($id);
+
+        if ($inventory<$num) $this->error('库存不足');
+        if ($inventory<1) $this->error('还有未支付的订单,您有可能捡漏哦!');
+        if (!$coll_info['sold_out']) $this->error('已售罄');
+
+
+        $cards  = checkUseSnapcard($this->uid,$id,$coll_info['sell_time']);
+        if (!$cards){
+            if ($coll_info['state']!=1) $this->error('状态错误');
+
+            $sell_time = strtotime($coll_info['sell_time']);
+            if ($sell_time>time()) $this->error('商品还未开始售卖!请耐心等待');
+        }
+
+        if ($pay_type=='yhkj'){
+            if (!$user['hj_pay_status']) $this->error('请签约银行卡后购买');
+        }
+
+
+        //获取用户已经购买数量
+        $userByCount = getByCount($this->uid,$id);
+        if ($coll_info['buy_count']<($userByCount+$num)) $this->error('每人限购'.$coll_info['buy_count'].'个');
+
+        //获取是否已经铸造hash
+        //$hashCount = getLenCollection($id);
+        $hash = Db::name('hash2')->where('goods_id',$id)->where('success',1)->count();
+        if (!$hash) $this->error('nft未上架,无法购买');
+       // if (!$hashCount || $hashCount<$num) $this->error('hash未铸造,无法购买');
+
+        if ($from=='wx'){
+            if (!$user['openid']) $this->error('微信未授权,无法购买');
+        }
+
+        //先减掉库存
+        loseCollectionInventory($id,$num);
+        //先增加购买数量
+        IncrByCount($this->uid,$id,$num);
+
+
+        $order_no = get_order_sn();
+        $total_fee = bcmul($coll_info['price'],$num,2);
+
+        $redis_data = [
+            'mid'=>$this->uid,
+            'c_id'=>$id,
+            'order_no'=>$order_no,
+            'inventory'=>$coll_info['inventory'],
+            'num'=>$num,
+            'pro_info'=>json_encode($coll_info),
+            'status'=>0,
+            'create_at'=>date('Y-m-d H:i:s'),
+            'pay_price'=>$total_fee,
+            'pay_type'=>$pay_type
+        ];
+        $com = true;
+        $retrun_data['order_no'] = $order_no;
+        $retrun_data['pay'] = '';
+
+        if ($com){
+            //更新抢购卡使用记录
+            if ($cards){
+                Db::name('store_snap_card_log')->where('id',$cards[0])->update(['is_use'=>2,'update_at'=>date('Y-m-d H:i:s')]);
+            }
+            $redis = new Redis();
+            //收集下单用户id
+            setCollectionBuyUser($this->uid);
+            $redis->hMSet('order_not_pay_'.$this->uid,[$order_no=>json_encode($redis_data,true)]); //存入redis
+            $this->success('成功',$retrun_data);
+        }else{
+            //加上库存
+            addCollectionInventory($id,$num);
+            //减少用户购买数量
+            DecrByCount($this->uid,$id,$num);
+
+            $this->error('服务器繁忙,请稍后重试');
+        }
+
+    }
+
+
+    /**
+     * @title 我的订单-待支付、已购买
+     * @desc  我的订单-待支付、已购买
+     * @author  Gavin
+     * @url /api/Order/getMyOrderList
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:status type:int  : default:0 desc:0:待支付1:已购买
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:order_no type:string default:-- desc:订单号
+     * @return  name:num type:int default:-- desc:数量
+     * @return  name:pay_price type:string default:-- desc:订单金额
+     * @return  name:status type:int default:-- desc:类型0:待支付1:已支付
+     * @return  name:cancel_time type:string default:-- desc:待支付自动取消时间
+     * @return  name:pay_type type:string default:-- desc:支付方式wx:微信zfb:支付宝
+     * @return  name:create_at type:string default:-- desc:创建时间
+     * @return name:pro_info@name type:string default:-- desc:藏品名称
+     * @return name:pro_info@cover type:string default:-- desc:藏品图片
+     * @return name:pro_info@price type:string default:-- desc:藏品单价
+     * @return name:pro_info@auth_img type:string default:-- desc:藏品作者头像
+     * @return name:pro_info@auth_name type:string default:-- desc:藏品作者名称
+     */
+    public function getMyOrderList(){
+        $status = input('status',0);
+
+        if ($status==0){
+            $redis = new Redis();
+            $key = 'order_not_pay_'.$this->uid;
+            $count = $redis->hGetLen($key);
+            $list = $redis->hGetvals($key);
+            if ($list){
+                //自动取消分钟数
+                $cancel_time = getCancelTime();
+                foreach ($list as $k=>&$v){
+                    $info = json_decode($v,true);
+                    $info['pro_info'] = json_decode($info['pro_info'],true);
+                    $info['cancel_time'] = date('Y-m-d H:i:s',strtotime($info['create_at'])+($cancel_time*60));
+                    $list[$k] = $info;
+                }
+            }
+        }else{
+            $where = [
+                'is_deleted'=>0,
+                'status'=>$status,
+                'mid'=>$this->uid
+            ];
+            $count = Db::name('store_order')->where($where)->count();
+            $list = Db::name('store_order')
+                ->where($where)
+                ->field('id,c_id,order_no,num,pro_info,status,create_at,pay_price,pay_type')
+                ->order('id desc')
+                ->limit($this->off_set,$this->page_num)
+                ->select();
+            foreach ($list as &$v){
+                $v['pro_info'] = json_decode($v['pro_info'],true);
+            }
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+    /**
+     * @title 待支付、已购买订单详情
+     * @desc  待支付、已购买订单详情
+     * @author  Gavin
+     * @url /api/Order/MyOrderListDetail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:order_no type:string default:0 desc:订单号
+     * @param  name:type type:int default:1 desc:1:待支付2:已支付
+     *
+     * @return  name:order_no type:string default:-- desc:订单号
+     * @return  name:num type:int default:-- desc:数量
+     * @return  name:pay_price type:string default:-- desc:订单金额
+     * @return  name:status type:int default:-- desc:类型0:待支付1:已支付
+     * @return  name:cancel_time type:string default:-- desc:待支付自动取消时间
+     * @return  name:create_at type:string default:-- desc:创建时间
+     * @return  name:pay_at type:string default:-- desc:支付时间
+     * @return  name:pay_type type:string default:-- desc:支付方式wx:微信zfb:支付宝
+     * @return name:pro_info@name type:string default:-- desc:藏品名称
+     * @return name:pro_info@cover type:string default:-- desc:藏品图片
+     * @return name:pro_info@price type:string default:-- desc:藏品单价
+     * @return name:pro_info@auth_img type:string default:-- desc:藏品作者头像
+     * @return name:pro_info@auth_name type:string default:-- desc:藏品作者名称
+     */
+    public function MyOrderListDetail(){
+        $order_no = input('order_no');
+        $type = input('type',1);
+        if (!$order_no) $this->error('参数错误');
+        if ($type==1){
+            $redis = new Redis();
+            $key = 'order_not_pay_'.$this->uid;
+            $order = $redis->hGet($key,$order_no);
+            $order = json_decode($order,true);
+        }else{
+            $order = Db::name('store_order')
+                ->where('order_no',$order_no)
+                ->where('mid',$this->uid)
+                ->where('is_deleted',0)
+                ->find();
+        }
+        if (!$order) $this->error('订单不存在');
+        $order['pro_info'] = json_decode($order['pro_info'],true);
+        //自动取消分钟数
+        $cancel_time = getCancelTime();
+        if ($order['status']==0){
+            $order['cancel_time'] = date('Y-m-d H:i:s',strtotime($order['create_at'])+($cancel_time*60));
+        }
+        $this->success('成功',$order);
+    }
+
+
+    /**
+     * 银行卡快捷支付发送短信
+     * @url /api/Order/yhkjSms
+     */
+    public function yhkjSms(){
+        //redis原子锁
+        if (redisSetNx('yhkjSms'.$this->uid,3)){
+            $user = getMemberInfoHash($this->uid); //获取用户信息
+            $order_no = input('order_no');   //订单号
+            if (!$order_no) $this->error('参数错误');
+            $redis = new Redis();
+            $key = 'order_not_pay_'.$this->uid;
+            $order = $redis->hGet($key,$order_no);
+            if (!$order) $this->error('订单不存在');
+            $order = json_decode($order,true);
+            if ($order['status']!=0) $this->error('订单已支付或已取消');
+            $huiju = new HuijuKjPay();
+            $order_nos = get_order_sn();
+            $order['order_no'] = $order_nos;
+            $redis->hMSet($key,[$order_nos=>json_encode($order,true)]); //存入redis
+            //删除redis
+            $redis->hdel($key,$order_no);
+            $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/kuaijiePayNotify';//回调地址
+            $result = $huiju->paySms($order_nos,$order['pay_price'],'一级市场购买藏品',$this->uid,$callback_url,$user['hj_pay_bank_card']);
+            if ($result['resp_codo']=='SUCCESS'){
+                if ($result['biz_code']=='JS000000'){
+                    $data = json_decode($result['data'],true);
+                    if ($data['order_status']=='P3000'){
+                        DelRedisSetNx('yhkjSms'.$this->uid);
+                        $this->success('发送成功',$result);
+                    }else{
+                        DelRedisSetNx('yhkjSms'.$this->uid);
+                        $this->error($data['err_msg']);
+                    }
+                }else{
+                    DelRedisSetNx('yhkjSms'.$this->uid);
+                    $this->error($result['biz_msg']);
+                }
+            }else{
+                DelRedisSetNx('yhkjSms'.$this->uid);
+                $this->error('发送失败');
+            }
+        }else{
+            $this->error('请求过快');
+        }
+
+    }
+
+
+    /**
+     * @title 待支付订单支付
+     * @desc  待支付订单支付
+     * @author  Gavin
+     * @url /api/Order/payOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:order_no type:string require:1 default:-- desc:订单号
+     * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
+     *
+     * @return  name:order_no type:int require:0 default:0 desc:订单号
+     * @return  name:pay type:string require:0 default:0 desc:支付信息
+     */
+    public function payOrder(){
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        $order_no = input('order_no');   //订单号
+        $from = input('from','wx');
+        if (!$order_no) $this->error('参数错误');
+
+        $redis = new Redis();
+        $key = 'order_not_pay_'.$this->uid;
+        $order = $redis->hGet($key,$order_no);
+        if (!$order) $this->error('订单不存在');
+        $order = json_decode($order,true);
+        if ($order['status']!=0) $this->error('订单已支付或已取消');
+        $order_no = $order['order_no'];
+        $total_fee = $order['pay_price'];
+        $pay_type = $order['pay_type'];
+
+        if ($pay_type=='yhkj'){
+            $sms_code = input('sms_code');
+            if (!$sms_code) $this->error('验证码为空');
+        }
+
+        $body = 'top艺术购买藏品';
+        $com = true;
+        switch ($pay_type){
+            case 'wx':
+                $config = retrunWxConfig();
+                $total_fee = $total_fee * 100;
+                $config['notify_url'] = $this->request->root(true).'/api/Pay/WxOrderNotify';
+                $app = Factory::payment($config);
+
+                $post_data = [
+                    'body' => $body,
+                    'out_trade_no' => $order_no,
+                    'total_fee' => $total_fee,
+                    'attach'=>$this->uid,     //自定义传值
+                ];
+                //trade_type   SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
+                if ($from=='wx'){
+                    $post_data['openid'] = $user['openid'];
+                    $post_data['trade_type'] = 'JSAPI';
+                }elseif ($from=='h5'){
+                    $post_data['trade_type'] = 'MWEB';
+                }
+
+                $result = $app->order->unify($post_data);
+                if ($result['return_msg']=='OK'){
+                    if ($result['result_code']=='FAIL'){
+                        $com = false;
+                    }else{
+                        $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
+                        $redis_data['wx_order'] = $order1;
+                        $retrun_data['order_no'] = $order_no;
+                        $retrun_data['pay'] = json_decode($order1,true);
+                    }
+                }else{
+                    $com = false;
+                }
+                break;
+            case 'zfb':
+
+                $order_nos = get_order_sn();
+                $order['order_no'] = $order_nos;
+                $redis->hMSet($key,[$order_nos=>json_encode($order,true)]); //存入redis
+               //存入redis
+                $redis->hdel($key,$order_no);
+
+                $zfb = new AliPay3();
+                $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/alipayOrderNotify';//回调地址
+                $callback_url = 'https://'.$_SERVER['SERVER_NAME'].'/pages/mine/order-list?num=1';
+                $order = $zfb->aliPay($body,$total_fee,$order_nos,$notify_url,$this->uid,$callback_url);
+                $retrun_data['order_no'] = $order_nos;
+                $retrun_data['pay'] = $order['pay_url'];
+                break;
+            case 'ylh5':
+                $order_nos = get_order_sn();
+                $order['order_no'] = $order_nos;
+                $redis->hMSet($key,[$order_nos=>json_encode($order,true)]); //存入redis
+                //存入redis
+                $redis->hdel($key,$order_no);
+
+                $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/mine/order-list?num=1';
+                $huiju = new HuijuH5Pay();
+                $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylOrderNotify';//回调地址
+                $pro_info = json_decode($order['pro_info'],true);
+                $result = $huiju->ylh5Pay($order_nos,$callback_url,$notify_url,$pro_info['name'],$total_fee,$this->uid,2);
+                if ($result['ra_Code']=='100'){
+                    $retrun_data['order_no'] = $order_nos;
+                    $retrun_data['pay'] = $result['rc_Result'];
+                }else{
+                    $com = false;
+                }
+                break;
+            case 'zfbh5':
+                $order_nos = get_order_sn();
+                $order['order_no'] = $order_nos;
+                $redis->hMSet($key,[$order_nos=>json_encode($order,true)]); //存入redis
+                //存入redis
+                $redis->hdel($key,$order_no);
+
+                $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/mine/order-list?num=1';
+                $huiju = new HuijuH5Pay();
+                $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylOrderNotify';//回调地址
+                $pro_info = json_decode($order['pro_info'],true);
+                $result = $huiju->ylh5Pay($order_nos,$callback_url,$notify_url,$pro_info['name'],$total_fee,$this->uid,1);
+                if ($result['ra_Code']=='100'){
+                    $retrun_data['order_no'] = $order_nos;
+                    $retrun_data['pay'] = $result['rc_Result'];
+                }else{
+                    $com = false;
+                }
+                break;
+            case 'yhkj':
+                $huiju = new HuijuKjPay();
+                $result = $huiju->kuaijiePay($order_no,$sms_code);
+                if ($result['resp_codo']=='SUCCESS'){
+                    if ($result['biz_code']=='JS000000'){
+                        $data = json_decode($result['data'],true);
+                        if ($data['order_status']=='P1000'){
+                            $this->success('支付成功');
+                        }else{
+                            $this->error($data['err_msg']);
+                        }
+                    }else{
+                        $this->error($result['biz_msg']);
+                    }
+                }else{
+                    $this->error('支付失败');
+                }
+                break;
+        }
+        if ($com){
+            $this->success('成功',$retrun_data);
+        }else{
+            $this->error('调起支付失败,请稍后重试');
+        }
+    }
+
+
+    /**
+     * @title 取消订单
+     * @desc  取消订单
+     * @author  Gavin
+     * @url /api/Order/cancelOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:order_no type:string require:1 default:-- desc:订单号
+     */
+    public function cancelOrder(){
+        $order_no = input('order_no');
+        if (!$order_no) $this->error('参数错误');
+        $redis = new Redis();
+        $key = 'order_not_pay_'.$this->uid;
+        $order = $redis->hGet($key,$order_no);
+        if (!$order) $this->error('订单不存在');
+        $order = json_decode($order,true);
+        if ($order['status']!=0) $this->error('订单已支付或已取消');
+        $com = true;
+        Db::startTrans();
+        try {
+            $order['status'] = 2;
+            $order['cancel_at'] = date('Y-m-d H:i:s');
+            $order['cancel_state'] = 2;
+            Db::name('store_order')->insert($order);
+            Db::commit();
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            //加上库存
+            addCollectionInventory($order['c_id'],$order['num']);
+            //减少用户购买数量
+            DecrByCount($this->uid,$order['c_id'],$order['num']);
+            //删除数据
+            $redis->hdel($key,$order_no);
+
+            $this->success('取消成功');
+        }else{
+            $this->error('取消失败,请稍后重试');
+        }
+    }
+
+
+
+    /**
+     * @title 我的订单-转赠
+     * @desc  我的订单-转赠
+     * @author  Gavin
+     * @url /api/Order/getMySendList
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:order_no type:string default:-- desc:订单号
+     * @return name:over_time type:string default:-- desc:转赠时间
+     * @return name:status type:int default:-- desc:类型2:赠出3:获赠
+     * @return name:send_name type:string default:-- desc:转赠者/受赠者
+     * @return name:collectors_hash type:string default:-- desc:hash
+     * @return name:pro_info@name type:string default:-- desc:藏品名称
+     * @return name:pro_info@price type:string default:-- desc:藏品价格
+     * @return name:pro_info@price type:string default:-- desc:藏品单价
+     * @return name:pro_info@auth_img type:string default:-- desc:藏品作者头像
+     * @return name:pro_info@auth_name type:string default:-- desc:藏品作者名称
+     */
+    public function getMySendList(){
+        $where = [
+            'mid'=>$this->uid,
+        ];
+        $count = Db::name('store_order_info')->where($where)->whereIn('status','2,3')->count();
+        $list = Db::name('store_order_info')
+            ->where($where)
+            ->whereIn('status','2,3')
+            ->field('id,order_no,pro_info,over_time,create_at,status,to_mid,collectors_hash')
+            ->order('id desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            $v['pro_info'] = json_decode($v['pro_info'],true);
+            $v['send_name'] = Db::name('store_member')->where('id',$v['to_mid'])->value('name');
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+
+    /**
+     * @title 转赠订单详情
+     * @desc  转赠订单详情
+     * @author  Gavin
+     * @url /api/Order/getMySendListDetail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:order_id type:int  : default:1 desc:订单ID
+     *
+     * @return  name:order_no type:string default:-- desc:订单号
+     * @return name:over_time type:string default:-- desc:转赠时间
+     * @return name:status type:int default:-- desc:类型2:赠出3:获赠
+     * @return name:send_name type:string default:-- desc:转赠者/受赠者
+     * @return name:collectors_hash type:string default:-- desc:hash
+     * @return name:pro_info@name type:string default:-- desc:藏品名称
+     * @return name:pro_info@price type:string default:-- desc:藏品价格
+     * @return name:pro_info@price type:string default:-- desc:藏品单价
+     * @return name:pro_info@auth_img type:string default:-- desc:藏品作者头像
+     * @return name:pro_info@auth_name type:string default:-- desc:藏品作者名称
+     */
+    public function getMySendListDetail(){
+        $order_id = input('order_id');   //订单id
+        if (!$order_id) $this->error('参数错误');
+        $order = Db::name('store_order_info')
+            ->where('id',$order_id)
+            ->where('mid',$this->uid)
+            ->find();
+        if (!$order) $this->error('订单不存在');
+        $order['pro_info'] = json_decode($order['pro_info'],true);
+        $order['send_name'] = Db::name('store_member')->where('id',$order['to_mid'])->value('name');
+        $this->success('成功',$order);
+    }
+
+
+
+
+
+
+
+
+    function savetest($order_no){
+
+        $result['out_trade_no'] = $order_no;
+        $order = Db::name('store_order')->where('order_no',$result['out_trade_no'])->find();
+        if ($order['status']==0){
+            Db::startTrans();
+            try {
+                Db::name('store_order')
+                    ->where('order_no',$result['out_trade_no'])
+                    ->update(['status'=>1,'pay_at'=>date('Y-m-d H:i:s'),'return_success_info'=>json_encode($result,true)]);
+
+                $array = [];
+                for ($i=0;$i<$order['num'];$i++){
+                    //获取排名
+                    $rank = getRanking($order['c_id'])+1;
+                    $tag = getTag($order['c_id'],$rank,$order['inventory']);
+                    saveRanking($order['c_id']);
+                    $company = '象寻数字科技(上海)有限公司';
+                    $company_hash = '';
+                    $company_hash_time = '';
+                    $collectors_hash = '';
+                    $date = [
+                        'order_id'=>$order['id'],
+                        'order_no'=>get_order_sn(),
+                        'tag'=>$tag,
+                        'mid'=>$order['mid'],
+                        'pro_info'=>$order['pro_info'],
+                        'company'=>$company,
+                        'company_hash'=>$company_hash,
+                        'company_hash_time'=>$company_hash_time,
+                        'collectors_hash'=>$collectors_hash,
+                        'collectors_hash_time'=>date('Y-m-d H:i:s')
+                    ];
+                    $array[] = $date;
+                }
+                Db::name('store_order_info')->insertAll($array);
+                Db::commit();
+                // return true;
+            } catch (\Exception $e){
+                Db::rollback();
+                //return false;
+            }
+        }
+    }
+
+
+
+}

+ 547 - 0
application/api/controller/Pay.php

@@ -0,0 +1,547 @@
+<?php
+
+// +----------------------------------------------------------------------
+// | ThinkAdmin
+// +----------------------------------------------------------------------
+// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
+// +----------------------------------------------------------------------
+// | 官方网站: http://demo.thinkadmin.top
+// +----------------------------------------------------------------------
+// | 开源协议 ( https://mit-license.org )
+// +----------------------------------------------------------------------
+// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
+// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
+// +----------------------------------------------------------------------
+
+namespace app\api\controller;
+use app\common\library\AliPay3;
+use think\cache\driver\Redis;
+use EasyWeChat\Factory;
+use think\Controller;
+use think\Db;
+use think\Exception;
+
+
+use AlibabaCloud\Client\AlibabaCloud;
+use AlibabaCloud\Client\Exception\ClientException;
+use AlibabaCloud\Client\Exception\ServerException;
+use function AlibabaCloud\Client\value;
+
+/**
+ * 支付管理类
+ * Class Refund
+ * @package app\api\controller\Refund
+ */
+class Pay extends Controller
+{
+
+
+
+    public function test(){
+        $data = "{\"memberid\":220431411,\"orderid\":\"220431411OR29DAEDEEACBBF11820220602161239\",\"amount\":\"0.1000\",\"successdate\":\"2022-06-02 16:13:24\",\"bankcode\":\"953\",\"status\":\"success\",\"attach\":\"100000\",\"buyer_id\":\"2088012861214283\",\"out_trade_id\":\"202206021612394516457329\",\"notifydate\":\"2022-06-02 16:13:24\",\"num\":1,\"sign\":\"BC0568D0BA0E38DCB3246790C31570BA\"}";
+        $result= json_decode($data,true);
+        if ($result['status'] == 'success') {
+            $result['out_trade_no'] = $result['out_trade_id'];
+            $return = $this->dealData($result);
+            echo $return;
+        }
+    }
+    /**
+     * 微信支付--商品支付成功回调订单
+     */
+    public function WxOrderNotify(){
+        $payXml = file_get_contents("php://input");
+        //将xml格式转化为json格式
+        $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
+        //将json格式转成数组格式 $result['out_trade_no']
+        $result = json_decode($jsonXml, true);
+        file_put_contents("wx_pay_return.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+        if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
+            $return = $this->dealData($result);
+            if ($return){
+                $arr = array(
+                    'return_code' => 'SUCCESS',
+                    'return_msg'  => 'OK',
+                );
+                return $this->arrayToXml($arr);
+            }else{
+                file_put_contents("order_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+//    /**
+//     * 支付宝支付--支付成功回调订单
+//     */
+//    public function alipayOrderNotify(){
+//        $result    = input('post.');
+//        file_put_contents("zfb_pay_return.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+//        if ($result['trade_status'] == 'TRADE_SUCCESS' || $result['trade_status'] == 'TRADE_FINISHED') {
+//            $return = $this->dealData($result);
+//            if ($return){
+//                echo 'success';
+//            }else{
+//                file_put_contents("order_alipay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+//            }
+//        }
+//    }
+
+    /**
+     * 支付宝支付--支付成功回调订单
+     */
+    public function alipayOrderNotify(){
+        $results    = input('post.');
+        $zfb = new AliPay3();
+        $result= json_decode($zfb->sslDeReturn($results['notifyData']),true);
+        file_put_contents("zfb_pay_return.txt", json_encode($result,true) . "\n" . "\n", FILE_APPEND);
+        if ($result['status'] == 'success')
+        {
+            $result['out_trade_no'] = $result['out_trade_id'];
+            $return = $this->dealData($result);
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("order_alipay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 汇聚支付支付宝h5,银联h5
+     */
+    public function ylOrderNotify(){
+        $result  = input();
+        debug($result);
+        if ($result['r6_Status'] == '100'){
+            $result['out_trade_no'] = $result['r2_OrderNo'];
+            $result['attach'] = $result['r5_Mp'];
+            $return = $this->dealData($result);
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("order_alipay_error.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 汇聚支付快捷支付--有短支付回调
+     */
+    public function kuaijiePayNotify(){
+        $result  = input();
+        debug($result,'kuaijie');
+        if ($result['biz_code']=='JS000000'){
+            $data = json_decode($result['data'],true);
+            if ($data['order_status']=='P1000'){
+                $result['out_trade_no'] = $result['mch_order_no'];
+                $result['attach'] = $result['callback_param'];
+                $return = $this->dealData($result);
+                if ($return){
+                    echo 'success';
+                }
+            }
+        }
+    }
+
+    /**
+     * 处理数据库信息
+     * @param $result
+     * @return bool
+     */
+    function dealData($result){
+        Db::startTrans();
+        $redis = new Redis();
+        $key = 'order_not_pay_'.$result['attach'];
+        $com = true;
+        try {
+            $order = $redis->hGet($key,$result['out_trade_no']);
+            if (!$order) return false;
+            $order = json_decode($order,true);
+            $is_order = Db::name('store_order')->where('order_no',$result['out_trade_no'])->find();
+            if (isset($is_order) && $is_order['status']==1) return false;
+
+            $order['status'] = 1;
+            $order['pay_at'] = date('Y-m-d H:i:s');
+            $order['return_success_info'] = json_encode($result,true);
+            $order_id = Db::name('store_order')->insertGetId($order);
+
+            $array = [];
+            $pro_info = json_decode($order['pro_info'],true);
+            $is_nft = Db::name('hash2')->where('goods_id',$order['c_id'])->where('success',1)->find();
+            for ($i=0;$i<$order['num'];$i++){
+                //获取排名
+                $rank = getRanking($order['c_id'])+1;
+                $tag = getTag($order['c_id'],$rank,$order['inventory']);
+                saveRanking($order['c_id']);
+                $collectors_hash = '';
+                $date = [
+                    'order_id'=>$order_id,
+                    'order_no'=>get_order_sn(),
+                    'tag'=>$tag,
+                    'mid'=>$order['mid'],
+                    'c_id'=>$order['c_id'],
+                    'name'=>$pro_info['name'],
+                    'cover'=>$pro_info['cover'],
+                    'pro_info'=>$order['pro_info'],
+                    'type'=>$pro_info['type'],
+                    'tokenid'=>$is_nft['class_id'],
+                    'nfttype'=>$is_nft['operationId'],
+                    'collectors_hash'=>$collectors_hash,
+                    'collectors_hash_time'=>''
+                ];
+                $array[] = $date;
+            }
+            Db::name('store_order_info')->insertAll($array);
+            //送积分
+//            $by_collection_integral = getConfigValue('by_collection_integral');
+//            if ($by_collection_integral){
+//                $by_collection_integral = bcmul($by_collection_integral,$order['num']);
+//                memberMoneyChange($by_collection_integral,1,$order['mid'],'购买藏品',1,$order['id']);
+//            }
+            Db::commit();
+        } catch (\Exception $e){
+            $com = false;
+            Db::rollback();
+        }
+        if ($com){
+            $redis->hdel($key,$result['out_trade_no']);
+            return true;
+        }else{
+            return false;
+        }
+    }
+
+
+    /**
+     * 微信支付--充值盲盒成功回调订单
+     */
+    public function BlindRechargeNotify(){
+        $payXml = file_get_contents("php://input");
+        //将xml格式转化为json格式
+        $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
+        //将json格式转成数组格式 $result['out_trade_no']
+        $result = json_decode($jsonXml, true);
+        if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
+            $return = $this->dealBox($result);
+            if ($return){
+                $arr = array(
+                    'return_code' => 'SUCCESS',
+                    'return_msg'  => 'OK',
+                );
+                return $this->arrayToXml($arr);
+            }else{
+                file_put_contents("order_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 支付宝支付--充值盲盒回调订单
+     */
+    public function alipayBlindRechargeNotify(){
+        $result    = input('post.');
+        if ($result['trade_status'] == 'TRADE_SUCCESS' || $result['trade_status'] == 'TRADE_FINISHED') {
+            $return = $this->dealBox($result);
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("order_alipay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+
+    /**
+     * 盲盒充值次数处理数据库信息
+     * @param $result
+     * @return bool
+     */
+    function dealBox($result){
+        $com = true;
+        Db::startTrans();
+        try {
+            $order = Db::name('store_blind_recharge')->where('order_no',$result['out_trade_no'])->find();
+            if ($order['status']==1) return true;
+            Db::name('store_blind_recharge')
+                ->where('order_no',$result['out_trade_no'])
+                ->update(['status'=>1,'pay_at'=>date('Y-m-d H:i:s'),'return_success_info'=>json_encode($result,true)]);
+            //用户加次数
+            Db::name('store_member')->where('id',$order['m_id'])->setInc('lottery_number',$order['num']);
+            Db::commit();
+        } catch (\Exception $e){
+            $com = false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($order['m_id']);
+            return true;
+        }else{
+            return false;
+        }
+    }
+
+
+
+
+
+
+    /**
+     * 微信支付--二级市场回调订单
+     */
+    public function SecondaryWxOrderNotify(){
+        $payXml = file_get_contents("php://input");
+        //将xml格式转化为json格式
+        $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
+        //将json格式转成数组格式 $result['out_trade_no']
+        $result = json_decode($jsonXml, true);
+        if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
+            $return = $this->dealSecondary($result);
+            if ($return){
+                $arr = array(
+                    'return_code' => 'SUCCESS',
+                    'return_msg'  => 'OK',
+                );
+                return $this->arrayToXml($arr);
+            }else{
+                file_put_contents("order_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 支付宝支付--二级市场回调订单
+     */
+    public function alipaySecondaryNotify(){
+        $results    = input('post.');
+        $zfb = new AliPay3();
+        $result= json_decode($zfb->sslDeReturn($results['notifyData']),true);
+        file_put_contents("zfbse_pay_return.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+        if ($result['status'] == 'success') {
+            $result['out_trade_no'] = $result['out_trade_id'];
+            $return = $this->dealSecondary($result);
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("order_alipay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 杉德支付--二级市场回调订单
+     */
+    public function shandeSecondaryNotify(){
+        $data   = stripslashes($_POST['data']); //支付数据
+        file_put_contents("shandepay.txt", $data . "\n" , FILE_APPEND);
+        $data = json_decode($data,true);
+        if ($data['body']['orderStatus']==1){
+            //$data['body']['orderCode']
+            $return = $this->dealSecondary($data,'sd');
+            if ($return){
+                return 'respCode=000000';
+            }else{
+                file_put_contents("shandepay_error.txt", $data . "\n" . json_encode($data) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+
+    /**
+     * 二级市场-汇聚支付支付宝h5,银联h5
+     */
+    public function ylSecondaryNotify(){
+        $result  = input();
+        debug($result);
+        if ($result['r6_Status'] == '100'){
+            $return = $this->dealSecondary($result,'huijuh5');
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("order_error.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+    /**
+     * 二级市场-汇聚支付快捷支付--有短支付回调
+     */
+    public function kuaijieSecondPayNotify(){
+        $result  = input();
+        debug($result,'kuaijiesecond');
+        if ($result['biz_code']=='JS000000'){
+            $data = json_decode($result['data'],true);
+            if ($data['order_status']=='P1000'){
+                $return = $this->dealSecondary($result,'kuaijie');
+                if ($return){
+                    echo 'success';
+                }
+            }
+        }
+    }
+
+    /**
+     * 二级市场处理数据库信息
+     * @param $result
+     * @return bool
+     */
+    function dealSecondary($result,$from=''){
+        $com = true;
+        Db::startTrans();
+        try {
+            if ($from=='sd'){
+                $out_trade_no = $result['body']['orderCode'];
+            }elseif ($from=='sd'){
+                $out_trade_no = $result['out_trade_no'];
+            } elseif($from=='huijuh5'){
+                $out_trade_no = $result['r2_OrderNo'];
+            }elseif ($from=='kuaijie'){
+                $out_trade_no = $result['mch_order_no'];
+            }
+            $order = Db::name('store_order_info_order')->where('order_no',$out_trade_no)->find();
+            if ($order['status']==1) return false;
+
+            if (!$order) return false;
+
+            $resale_status = Db::name('store_order_info')->where('id',$order['info_id'])->value('resale_status');
+            if ($resale_status!=2){
+
+            }else{
+                Db::name('store_order_info_order')
+                    ->where('order_no',$out_trade_no)
+                    ->update(['status'=>1,'pay_at'=>date('Y-m-d H:i:s'),'return_success_info'=>json_encode($result,true)]);
+                //藏品修改状态
+                Db::name('store_order_info')->where('id',$order['info_id'])->update([
+                    'status'=>2,
+                    'resale_status'=>3,
+                    'selling_time'=>date('Y-m-d H:i:s')
+                ]);
+                $info = Db::name('store_order_info')->where('id',$order['info_id'])->find();
+                //增加一条记录
+                $to_date = [
+                    'order_id'=>$info['order_id'],
+                    'order_no'=>get_order_sn(),
+                    'tag'=>$info['tag'],
+                    'mid'=>$order['mid'],
+                    'c_id'=>$info['c_id'],
+                    'name'=>$info['name'],
+                    'cover'=>$info['cover'],
+                    'pro_info'=>$info['pro_info'],
+                    'type'=>$info['type'],
+                    'status'=>3,
+                    'from'=>2,
+                    'to_mid'=>$info['mid'],
+                    'over_time'=>date('Y-m-d H:i:s'),
+                    'tokenid'=>$info['tokenid'],
+                    'nfttype'=>$info['nfttype'],
+                    'collectors_hash'=>'',
+                    'collectors_hash_time'=>date('Y-m-d H:i:s')
+                ];
+                Db::name('store_order_info')->insert($to_date);
+                //增加用户余额
+                memberMoneyChange($order['to_account'],3,$info['mid'],'出售藏品',1,$order['id']);
+                Db::commit();
+            }
+        } catch (\Exception $e){
+            $com = false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($order['mid']);
+            return true;
+        }else{
+            return false;
+        }
+    }
+
+
+
+
+    /**
+     * 余额充值--汇聚云闪付,支付宝h5支付回调
+     */
+    public function ylrechargeNotify(){
+        $result  = input();
+        debug($result);
+        if ($result['r6_Status'] == '100'){
+            $return = $this->dealRecharge($result,'huijuh5');
+            if ($return){
+                echo 'success';
+            }else{
+                file_put_contents("recharge_error.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
+            }
+        }
+    }
+
+
+    //处理充值
+    function dealRecharge($result,$from=''){
+        $com = true;
+        Db::startTrans();
+        try {
+            if($from=='huijuh5'){
+                $out_trade_no = $result['r2_OrderNo'];;
+            }
+            $order = Db::name('store_member_recharge')->where('order_no',$out_trade_no)->find();
+            if (!$order || $order['status']==1) return false;
+
+            //增加用户余额
+            $re = memberMoneyChange($order['pay_price'],3,$order['m_id'],'余额充值',1,$order['id']);
+            if ($re){
+                Db::name('store_member_recharge')
+                    ->where('order_no',$out_trade_no)
+                    ->update(['status'=>1,'pay_at'=>date('Y-m-d H:i:s'),'return_success_info'=>json_encode($result,true)]);
+                Db::commit();
+            }else{
+                $com = false;
+                Db::rollback();
+            }
+
+        } catch (\Exception $e){
+            $com = false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($order['m_id']);
+            return true;
+        }else{
+            return false;
+        }
+    }
+
+
+
+    /**
+     * 代付回调
+     */
+    public function wePayNotify(){
+        $result  = input();
+        debug($result,'wePay');
+    }
+
+
+
+
+
+
+
+
+    /**
+     * 数组转xml
+     * @ApiInternal
+     */
+    public function arrayToXml($arr)
+    {
+        $xml = "<xml>";
+        foreach ($arr as $key => $val) {
+            if (is_numeric($val)) {
+                $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
+            } else
+                $xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">";
+        }
+        $xml .= "</xml>";
+        return $xml;
+    }
+
+
+
+}

+ 153 - 0
application/api/controller/Publics.php

@@ -0,0 +1,153 @@
+<?php
+
+namespace app\api\controller;
+
+use think\cache\driver\Redis;
+use think\Db;
+use think\facade\Validate;
+use think\Request;
+use OSS\OssClient;
+use OSS\Core\OssException;
+use Zxing\Qrcode\Decoder\DataBlock;
+
+/**
+ * @title 公共类
+ * @controller Publics
+ */
+class Publics extends Base
+{
+    /**
+     * @title 上传图片
+     * @desc 上传图片
+     * @url /api/Publics/uploadLocality
+     * @method POST
+     * @tag 基础
+     *
+     * @param  name:file type:file require:1 desc:上传
+     * @return name:data@url type:string desc:图片地址
+     **/
+    public function uploadLocality(){
+        $file = request()->file('file');
+        $dir = dirname(realpath(dirname($_SERVER['SCRIPT_FILENAME']))) . '/public/upload';
+        if(!file_exists($dir)){
+            //检查是否有该文件夹,如果没有就创建,并给予最高权限
+            mkdir($dir, 0777,true);
+        }
+        $ossClient = new OssClient(getConfigValue('storage_oss_keyid'), getConfigValue('storage_oss_secret'), getConfigValue('storage_oss_endpoint'));
+        $info = $file->move($dir);
+        if($info){
+            $newName = $info->getSaveName();
+            $storage_type = getConfigValue('storage_type');
+            if ($storage_type == 'oss'){
+                $file_path = dirname($_SERVER['SCRIPT_FILENAME']) . "/upload/".$newName;
+                $result = $ossClient->uploadFile(getConfigValue('storage_oss_bucket'), $newName, $file_path);
+                $data['url'] =  $result['info']['url'];
+                unlink($file_path);
+            }elseif ($storage_type=='local'){
+                //压缩图片
+                //image_png_size_add(ROOT_PATH . 'public' . DS . 'uploads/images/'.$newName,ROOT_PATH . 'public' . DS . 'uploads/images/'.$newName);
+                $data['url'] =  'http://'.$_SERVER['SERVER_NAME']."/upload/{$newName}";
+            }
+        }else{
+            $this->error($file->getError());
+        }
+        $this->success('上传成功',$data);
+    }
+
+
+    /**
+     * @title 获取系统配置
+     * @desc 获取系统配置
+     * @url /api/Publics/config_info
+     * @method POST
+     * @tag 基础
+     *
+     * @return name:agreement type:string desc:使用条款
+     * @return name:privacy type:string desc:隐私协议
+     * @return name:platform_agreement type:string desc:违规处罚
+     * @return name:about_us type:string desc:关于我们
+     * @return name:registration_agreement type:string desc:用户注册协议
+     * @return name:terms_service type:string desc:联通统一认证服务条款
+     * @return name:chain_on_query type:string desc:链上查询
+     * @return name:examples_illustrate type:string desc:转赠说明
+     * @return name:android_version type:string desc:安卓当前版本号
+     * @return name:android_package type:string desc:安卓包地址
+     * @return name:ios_package type:string desc:ios地址
+     **/
+    public function config_info(){
+        $nameArray = [
+            'agreement',
+            'privacy',
+            'platform_agreement',
+            'about_us',
+            'registration_agreement',
+            'terms_service',
+            'chain_on_query',
+            'examples_illustrate',
+            'android_version',
+            'android_package',
+            'ios_package',
+            'free_lucky_number',
+            'lucky_recharge_price',
+            'service_fee',
+            'royalties',
+            'consignment',
+            'statement',
+            'invitation_background_img',
+            'share_background_img',
+            'new_instructions',
+            'share_switch'
+        ];
+        $array = getConfig($nameArray);
+        $this->success('成功',$array);
+    }
+
+
+    /**
+     * @title 获取分享的签名
+     * @desc 获取分享的签名
+     * @url /api/Publics/getWechatSign
+     * @method POST
+     * @tag 基础
+     *
+     **/
+    public function getWechatSign(){
+        $urls = input('url');
+        $appid = getConfigValue('wechat_appid');
+        $secret = getConfigValue('wechat_appsecret');
+
+        $redis = new Redis();
+        $access_token = $redis->get('access_token');
+        if (!$access_token){
+            $access_token = '';
+            $url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=$appid&secret=$secret";
+            $res=curlRequest($url);
+            $res = json_decode($res,true);
+            if (isset($res['access_token'])){
+                $redis->set('access_token',$res['access_token'],'7000');
+                $access_token = $res['access_token'];
+            }
+        }
+        $url2 ="https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=".$access_token."&type=jsapi";
+        $res2=curlRequest($url2);
+
+        $res2 = json_decode($res2,true);
+        if (!isset($res2) || $res2['errcode']!=0){
+            $this->error('获取ticket失败');
+        }
+        $timestamp = time();
+        $noncestr = get32Str(15);
+//        $urls = 'http://jybl.hdlkeji.com/web/h5/';
+        $string = "jsapi_ticket=".$res2['ticket']."&noncestr=$noncestr&timestamp=$timestamp&url=".$urls;
+        $sign = sha1($string);
+        $return = [
+            'appid'=>$appid,
+            'noncestr'=>$noncestr,
+            'timestamp'=>$timestamp,
+            'url'=>$urls,
+            'sign'=>$sign,
+            'ticket'=>$res2['ticket']
+        ];
+        $this->success('成功',$return);
+    }
+}

+ 103 - 0
application/api/controller/Recharge.php

@@ -0,0 +1,103 @@
+<?php
+namespace app\api\controller;
+use AlibabaCloud\Client\AlibabaCloud;
+use AlibabaCloud\Client\Exception\ClientException;
+use AlibabaCloud\Client\Exception\ServerException;
+use app\common\library\AliPay2;
+use app\common\library\HuijuH5Pay;
+use think\cache\driver\Redis;
+use think\Db;
+/**
+ * @title 充值余额
+ * @controller UserCenter
+ * @group base
+ */
+class Recharge extends Base
+{
+    public function initialize()
+    {
+        parent::initialize();
+        parent::check_login();
+    }
+
+    /**
+     * @title 余额充值
+     * @desc 余额充值
+     * @author Gavin
+     * @url /api/Recharge/recharge
+     * @method POST
+     * @tag 余额充值
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:price type:string  : default: desc:充值金额
+     * @param  name:pay_type type:string require:1 default:wx desc:wx:微信zfb:支付宝
+     */
+    public function recharge()
+    {
+        //redis原子锁
+        if (redisSetNx('recharge'.$this->uid,2)){
+
+            $v = getConfigValue('recharge_switch');
+            if (!$v) $this->error('维护中,暂时关闭');
+
+            $price = input('price');
+            $pay_type = input('pay_type');
+            if (!$price || !$pay_type) $this->error('参数错误');
+            if (!isAmount($price)) $this->error('金额错误');
+            $order_no = get_order_sn();
+            $data = [
+                'order_no'=>$order_no,
+                'm_id'=>$this->uid,
+                'price'=>$price,
+                'pay_price'=>$price,
+                'pay_type'=>$pay_type
+            ];
+            if (Db::name('store_member_recharge')->insert($data)){
+                $com = true;
+                switch ($pay_type){
+                    case 'ylh5':
+                        $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/mine/qianbao';
+                        $huiju = new HuijuH5Pay();
+                        $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylrechargeNotify';//回调地址
+                        $result = $huiju->ylh5Pay($order_no,$callback_url,$notify_url,'余额充值',$price,$this->uid,2);
+                        if ($result['ra_Code']=='100'){
+                            $retrun_data['order_no'] = $order_no;
+                            $retrun_data['pay'] = $result['rc_Result'];
+                        }else{
+                            $com = false;
+                        }
+                        break;
+                    case 'zfbh5':
+                        $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/mine/qianbao';
+                        $huiju = new HuijuH5Pay();
+                        $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylrechargeNotify';//回调地址
+                        $result = $huiju->ylh5Pay($order_no,$callback_url,$notify_url,'余额充值',$price,$this->uid,1);
+                        if ($result['ra_Code']=='100'){
+                            $retrun_data['order_no'] = $order_no;
+                            $retrun_data['pay'] = $result['rc_Result'];
+                        }else{
+                            $com = false;
+                        }
+                        break;
+                    default:
+                        $com = false;
+                        break;
+                }
+                if ($com){
+                    $this->success('成功',$retrun_data);
+                }else{
+                    $this->error('调起支付失败,请稍后重试');
+                }
+            }else{
+                $this->error('失败,请稍后重试');
+            }
+        }else{
+            $this->error('请求过快');
+        }
+    }
+
+
+
+
+
+
+}

+ 944 - 0
application/api/controller/Secondary.php

@@ -0,0 +1,944 @@
+<?php
+
+
+namespace app\api\controller;
+
+use Alipay\EasySDK\Kernel\Util\ResponseChecker;
+use app\common\library\AliPay;
+use app\common\library\AliPay3;
+use app\common\library\HuijuH5Pay;
+use app\common\library\HuijuKjPay;
+use app\common\library\Shande;
+use EasyWeChat\Factory;
+use think\cache\driver\Redis;
+use think\Db;
+use function EasyWeChat\Kernel\Support\get_client_ip;
+use function Symfony\Component\String\length;
+
+
+/**
+ * @title 二级市场
+ * @controller secondary
+ * @package app\api\controller
+ */
+class Secondary extends Base
+{
+
+    public function initialize(){
+        parent::initialize();
+        parent::check_login();
+    }
+
+    /**
+     * @title 藏品出售
+     * @desc  藏品出售
+     * @author  Gavin
+     * @url /api/Secondary/sell
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:int require:1 default:1 desc:藏品ID
+     * @param name:resale_price type:decimal require:1 default:1 desc:出售价格
+     * @param name:second_password type:int require:1 default:-- desc:二级密码
+     *
+     */
+    public function sell(){
+        $this->checkSwitch(1);
+        $id = input('id');
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        if ($user['is_auth']==0) $this->error('请先实名认证!');
+        $resale_price = input('resale_price');
+        $second_password = input('second_password');
+        if (!$id || !$resale_price) $this->error('参数错误');
+        //if ($resale_price<'1') $this->error('最低1元');
+        if ($resale_price<'0.1') $this->error('最低0.1元');
+        if ($resale_price>'99999') $this->error('最高99999元');
+        $info = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('id',$id)
+            ->find();
+        if (!$info || $info['status']==2) $this->error('藏品不存在');
+        if ($info['resale_status']!=1) $this->error('挂售状态错误');
+
+        $is_sell = Db::name('store_collection')->where('id',$info['c_id'])->value('is_sell');
+        if (!$is_sell) $this->error('该藏品不支持寄售');
+
+        if ($user['second_password']!=md5($second_password)) $this->error('密码错误');
+        $update_data = [
+            'resale_status'=>2,
+            'resale_time'=>date('Y-m-d H:i:s'),
+            'resale_price'=>$resale_price
+        ];
+        if (Db::name('store_order_info')->where('id',$id)->update($update_data)){
+            $this->success('成功');
+        }
+        $this->error('失败');
+    }
+
+    /**
+     * @title 藏品取消出售
+     * @desc  藏品取消出售
+     * @author  Gavin
+     * @url /api/Secondary/cancel_sell
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:int require:1 default:1 desc:藏品ID
+     */
+    public function cancel_sell(){
+        $this->checkSwitch(1);
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $info = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('id',$id)
+            ->find();
+        if (!$info || $info['status']==2) $this->error('藏品不存在');
+        if ($info['resale_status']!=2) $this->error('挂售状态错误');
+        //判断是否有待支付订单
+        $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
+        if ($count) $this->error('支付中,无法取消');
+
+        $update_data = [
+            'resale_status'=>1,
+        ];
+        if (Db::name('store_order_info')->where('id',$id)->update($update_data)){
+            $this->success('成功');
+        }
+        $this->error('失败');
+    }
+
+
+    /**
+     * @title 标签列表
+     * @desc  标签列表
+     * @author  Gavin
+     * @url /api/Secondary/label_list
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     */
+    public function label_list(){
+        $list = Db::name('store_collection')->whereNotIn('label','测试,测试勿拍')->group('label')->column('label');
+        $list = array_merge(['全部'],$list);
+        $this->success('成功',['label_list'=>$list]);
+    }
+
+
+    /**
+     * @title 二级市场列表
+     * @desc  二级市场列表
+     * @author  Gavin
+     * @url /api/Secondary/sell_list
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @param name:keyword type:string require:0 default: desc:关键词
+     * @param name:label type:string require:0 default: desc:标签
+     * @param name:time_order type:string require:0 default: desc:时间排序(asc:正序desc:倒序)
+     * @param name:price_order type:string require:0 default: desc:价格排序(asc:正序desc:倒序)
+     *
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:图片
+     * @return  name:member_name type:string require:0 default:0 desc:出售人名称
+     * @return  name:member_headimg type:string require:0 default:0 desc:出售人头像
+     * @return  name:resale_price type:string require:0 default:0 desc:出售价格
+     */
+    public function sell_list(){
+        $keyword = input('keyword');
+        $label = input('label');
+        $time_order = input('time_order');
+        $price_order = input('price_order');
+        $count = Db::name('store_order_info')
+            ->where('status','neq',2)
+            ->where('resale_status',2)
+            ->where('is_destruction',1)
+            ->when($keyword,function ($query) use ($keyword){
+                $query->whereLike('name','%'.$keyword.'%');
+            })
+            ->when($label,function ($query) use ($label){
+                if ($label!='全部'){
+                    $ids = Db::name('store_collection')->where('label',$label)->column('id');
+                    $query->whereIn('c_id',$ids);
+                }
+            })
+            ->count();
+        $list = Db::name('store_order_info')
+            ->where('status','neq',2)
+            ->where('resale_status',2)
+            ->where('is_destruction',1)
+            ->when($keyword,function ($query) use ($keyword){
+                $query->whereLike('name','%'.$keyword.'%');
+            })
+            ->when($label,function ($query) use ($label){
+                if ($label!='全部'){
+                    $ids = Db::name('store_collection')->where('label',$label)->column('id');
+                    $query->whereIn('c_id',$ids);
+                }
+            })
+            ->when($price_order,function ($query) use ($price_order){
+                $query->order('resale_price '.$price_order);
+            })
+            ->when($time_order,function ($query) use ($time_order){
+                $query->order('resale_time '.$time_order);
+            })
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            $member = getMemberInfoHash($v['mid']); //获取用户信息
+            $v['member_name'] = $member['name'];
+            $v['member_headimg'] = $member['headimg'];
+            $v['pro_info'] = json_decode($v['pro_info'],true);
+            $count = Db::name('store_order_info_order')->where('info_id',$v['id'])->where('status',0)->count();
+            $v['is_buy'] = $count>0 ? 1 : 0;
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+
+    /**
+     * @title 二级市场详情
+     * @desc  二级市场详情
+     * @author  Gavin
+     * @url /api/Secondary/sell_list_detail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:id type:int  : default: desc:id
+     */
+    public function sell_list_detail(){
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $info = Db::name('store_order_info')
+            ->where('status','neq',2)
+            ->where('resale_status',2)
+            ->where('is_destruction',1)
+            ->where('id',$id)
+            ->find();
+        if (!$info) $this->error('藏品不存在');
+        $member = getMemberInfoHash($info['mid']); //获取用户信息
+        $info['member_name'] = $member['name'];
+        $info['member_headimg'] = $member['headimg'];
+        $info['pro_info'] = json_decode($info['pro_info'],true);
+        //判断是否有待支付订单
+        $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
+        $info['is_buy'] = $count>0 ? 1 : 0;
+        $this->success('成功',$info);
+    }
+
+
+
+    /**
+     * @title 购买
+     * @desc  购买
+     * @author  Gavin
+     * @url /api/Secondary/createOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:id type:int require:1 default: desc:主键ID
+     * @param name:pay_type type:string require:1 default:wx desc:wx:微信zfb:支付宝sd:杉德h5
+     * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
+     *
+     * @return  name:order_no type:int require:0 default:0 desc:订单号
+     * @return  name:pay type:string require:0 default:0 desc:支付信息
+     */
+    public function createOrder(){
+        $this->checkSwitch(3);
+        $redis = new Redis([ 'select'=> 2]);
+        $redis_value = $redis->get('secondary_buy'.$this->uid);
+        if ($redis_value){
+            $this->error('请求过快,请稍后重试');
+        }else{
+            $redis->set('secondary_buy'.$this->uid,1,1);
+        }
+
+        $id = input('id');
+        $pay_type = input('pay_type','wx');
+
+        $this->checkSwitch(2,$pay_type);
+
+        $from = input('from','wx');
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        if (!$id) $this->error('参数错误');
+        if ($user['is_auth']==0) $this->error('请先实名认证!');
+        $info = Db::name('store_order_info')->where('id',$id)->find();
+        if (!$info) $this->error('藏品不存在');
+        if ($info['resale_status']==3) $this->error('藏品已出售');
+        if ($info['resale_status']==1) $this->error('藏品已撤销出售');
+        if ($info['mid']==$this->uid) $this->error('不能购买自己出售的藏品');
+
+        if ($pay_type=='yhkj'){
+            if (!$user['hj_pay_status']) $this->error('请签约银行卡后购买');
+        }
+
+       // if (isset($user['buy_time']) && $user['buy_time']>date('Y-m-d H:i:s')) $this->error('一小时取消3次以上,24小时内禁止下单');
+
+        //判断是否有未支付订单
+        $order_count = Db::name('store_order_info_order')->where('mid',$this->uid)->where('status',0)->count();
+        if ($order_count) $this->error('有未支付订单,无法下单');
+
+        //判断是否有待支付订单
+        $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
+        if ($count) $this->error('支付中,无法下单');
+
+        $service_fee = getConfigValue('service_fee');
+        $royalties = getConfigValue('royalties');
+        $com = true;
+        Db::startTrans();
+        try {
+            $order_no = get_order_sn();
+            //获取价格
+            $price = $info['resale_price'];
+            $num = 1;
+            $proportion = sprintf("%.2f",$price *($service_fee/100));    //四舍五入保留两位小数点
+            $roya = sprintf("%.2f",$price *($royalties/100));    //四舍五入保留两位小数点
+            $total_fee = bcmul($price,$num,2);
+            $real_money = $price-$proportion-$roya;
+            $data = [
+                'order_no'=>$order_no,
+                'mid'=>$this->uid,
+                'info_id'=>$id,
+                'num'=>$num,
+                'pro_info'=>json_encode($info,true),
+                'pay_price'=>$total_fee,
+                'service_fee'=>$service_fee,
+                'royalties'=>$royalties,
+                'to_account'=>$real_money,
+                'pay_type'=>$pay_type
+            ];
+            $id = Db::name('store_order_info_order')->insertGetId($data);
+
+            $retrun_data['order_no'] = $order_no;
+            $retrun_data['id'] = $id;
+            $retrun_data['pay'] = '';
+            Db::commit();
+//            $body = 'top艺术购买二级市场藏品';
+//            switch ($pay_type){
+//                case 'wx':
+//                    $config = retrunWxConfig();
+//                    $total_fee = $total_fee * 100;
+//                    $config['notify_url'] = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/SecondaryWxOrderNotify';
+//                    $app = Factory::payment($config);
+//
+//                    $post_data = [
+//                        'body' => $body,
+//                        'out_trade_no' => $order_no,
+//                        'total_fee' => $total_fee,
+//                        'attach'=>$this->uid,     //自定义传值
+//                    ];
+//                    //trade_type   SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
+//                    if ($from=='wx'){
+//                        $post_data['openid'] = $user['openid'];
+//                        $post_data['trade_type'] = 'JSAPI';
+//                    }elseif ($from=='h5'){
+//                        $post_data['trade_type'] = 'MWEB';
+//                    }
+//                    $result = $app->order->unify($post_data);
+//
+//                    if ($result['return_msg']=='OK'){
+//                        if ($result['result_code']=='FAIL'){
+//                            $com = false;
+//                            Db::rollback();
+//                        }else{
+//                            $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
+//                            $retrun_data['order_no'] = $order_no;
+//                            $retrun_data['id'] = $id;
+//                            $retrun_data['pay'] = json_decode($order1,true);
+//                            Db::commit();
+//                        }
+//                    }else{
+//                        $com = false;
+//                        Db::rollback();
+//                    }
+//                    break;
+//                case 'zfb':
+//
+//                    $zfb = new AliPay3();
+//                    $notify_url = $this->request->root(true).'/index.php/api/Pay/alipaySecondaryNotify';//回调地址
+//                    $callback_url = 'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order';
+//                    $order = $zfb->aliPay($body,$total_fee,$order_no,$notify_url,$this->uid,$callback_url);
+//                    $retrun_data['order_no'] = $order_no;
+//                    $retrun_data['id'] = $id;
+//                    $retrun_data['pay'] = $order['pay_url'];
+//                    Db::commit();
+//                    break;
+//                case 'sd':
+//                    $client = new Shande();
+//                    $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/shandeSecondaryNotify';//回调地址
+//                    $total_fee = $total_fee*100;
+//                    $lenth = strlen($total_fee);
+//                    $total_fee = get0number($lenth).$total_fee;
+//                    $result = $client->orderPay($order_no,$total_fee,$body,$notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');
+//                    $retrun_data['order_no'] = $order_no;
+//                    $retrun_data['id'] = $id;
+//                    $retrun_data['pay'] = json_decode($result['data'],true);
+//                    Db::commit();
+//                    break;
+//                default:
+//                    $retrun_data['order_no'] = $order_no;
+//                    $retrun_data['id'] = $id;
+//                    $retrun_data['pay'] = '';
+//                    Db::commit();
+//                    break;
+//
+//            }
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            $this->success('成功',$retrun_data);
+        }
+        $this->error('失败,请稍后重试');
+    }
+
+
+        /**
+     * @title 二级市场我的转售列表
+     * @desc  二级市场我的转售列表
+     * @author  Gavin
+     * @url /api/Secondary/my_sell_list
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @param name:keyword type:string require:0 default: desc:关键词
+     * @param name:time_order type:string require:0 default: desc:时间排序(asc:正序desc:倒序)
+     * @param name:price_order type:string require:0 default: desc:价格排序(asc:正序desc:倒序)
+     *
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:图片
+     * @return  name:member_name type:string require:0 default:0 desc:出售人名称
+     * @return  name:member_headimg type:string require:0 default:0 desc:出售人头像
+     * @return  name:resale_price type:string require:0 default:0 desc:出售价格
+     */
+    public function my_sell_list(){
+        $keyword = input('keyword');
+        $time_order = input('time_order');
+        $price_order = input('price_order');
+        $count = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('resale_status','neq',1)
+            ->when($keyword,function ($query) use ($keyword){
+                $query->whereLike('name','%'.$keyword.'%');
+            })
+            ->count();
+        $list = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('resale_status','neq',1)
+            ->when($keyword,function ($query) use ($keyword){
+                $query->whereLike('name','%'.$keyword.'%');
+            })
+
+            ->when($price_order,function ($query) use ($price_order){
+                $query->order('resale_price '.$price_order);
+            })
+            ->when($time_order,function ($query) use ($time_order){
+                $query->order('resale_time '.$time_order);
+            })
+            ->order('resale_status asc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            $member = getMemberInfoHash($v['mid']); //获取用户信息
+            $v['member_name'] = $member['name'];
+            $v['member_headimg'] = $member['headimg'];
+            $v['pro_info'] = json_decode($v['pro_info'],true);
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+
+
+
+
+    /**
+     * @title 二级市场订单
+     * @desc  二级市场订单
+     * @author  Gavin
+     * @url /api/Secondary/secondaryOrderList
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @param name:status type:string require:0 default: desc:0:待支付1:已购买2:转售中3:已转售
+     *
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:图片
+     * @return  name:member_name type:string require:0 default:0 desc:出售人名称
+     * @return  name:member_headimg type:string require:0 default:0 desc:出售人头像
+     * @return  name:pay_price type:string require:0 default:0 desc:出售价格(已购买、待支付使用)
+     * @return  name:resale_price type:string require:0 default:0 desc:出售价格(转售中、已转售使用)
+     */
+    public function secondaryOrderList(){
+        $status = input('status',0);
+        switch ($status){
+            case 0:case 1:
+                $count =  Db::name('store_order_info_order')
+                    ->where('mid',$this->uid)
+                    ->where('status',$status)
+                    ->count();
+                $list = Db::name('store_order_info_order')
+                    ->where('mid',$this->uid)
+                    ->where('status',$status)
+                    ->limit($this->off_set,$this->page_num)
+                    ->select();
+                //自动取消分钟数
+                $cancel_time = getCancelTime();
+                foreach ($list as &$v){
+                    $v['pro_info'] = json_decode($v['pro_info'],true);
+                    $member = getMemberInfoHash($v['mid']); //获取用户信息
+                    $v['member_name'] = $member['name'];
+                    $v['member_headimg'] = $member['headimg'];
+
+                    if ($v['status']==0){
+                        $v['cancel_time'] = date('Y-m-d H:i:s',strtotime($v['create_at'])+($cancel_time*60));
+                    }
+                }
+                break;
+            case 2:case 3:
+                $count = Db::name('store_order_info')
+                    ->where('mid',$this->uid)
+                    ->where('resale_status',$status)
+                    ->where('is_destruction',1)
+                    ->count();
+                $list = Db::name('store_order_info')
+                    ->where('mid',$this->uid)
+                    ->where('resale_status',$status)
+                    ->where('is_destruction',1)
+                    ->limit($this->off_set,$this->page_num)
+                    ->select();
+                foreach ($list as &$v){
+                    $v['pro_info'] = json_decode($v['pro_info'],true);
+                    $member = getMemberInfoHash($v['mid']); //获取用户信息
+                    $v['member_name'] = $member['name'];
+                    $v['member_headimg'] = $member['headimg'];
+                }
+                break;
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+    /**
+     * @title 订单详情
+     * @desc  订单详情
+     * @author  Gavin
+     * @url /api/Secondary/orderDetail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:status type:string require:0 default: desc:0:待支付1:已购买2:转售中3:已转售
+     * @param  name:order_no type:string default:0 desc:订单号
+     */
+    public function orderDetail(){
+        $order_no = input('order_no');
+        $status = input('status',0);
+        if (!$order_no) $this->error('参数错误');
+        switch ($status){
+            case 0:case 1:
+                 $order = Db::name('store_order_info_order')
+                    ->where('mid',$this->uid)
+                    ->where('status',$status)
+                    ->where('order_no',$order_no)
+                    ->find();
+                //自动取消分钟数
+                $cancel_time = getCancelTime();
+                $order['pro_info'] = json_decode($order['pro_info'],true);
+                $member = getMemberInfoHash($order['mid']); //获取用户信息
+                $order['member_name'] = $member['name'];
+                $order['member_headimg'] = $member['headimg'];
+                if ($order['status']==0){
+                    $order['cancel_time'] = date('Y-m-d H:i:s',strtotime($order['create_at'])+($cancel_time*60));
+                }
+                break;
+            case 2:case 3:
+                $order = Db::name('store_order_info')
+                    ->where('mid',$this->uid)
+                    ->where('order_no',$order_no)
+                    ->where('resale_status',$status)
+                    ->where('is_destruction',1)
+                    ->find();
+                $order['pro_info'] = json_decode($order['pro_info'],true);
+                $member = getMemberInfoHash($order['mid']); //获取用户信息
+                $v['member_name'] = $member['name'];
+                $v['member_headimg'] = $member['headimg'];
+                break;
+        }
+        $this->success('成功',$order);
+    }
+
+
+    /**
+     * @title 取消订单
+     * @desc  取消订单
+     * @author  Gavin
+     * @url /api/Secondary/cancelOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:order_no type:string require:1 default:-- desc:订单号
+     */
+    public function cancelOrder(){
+        $order_no = input('order_no');
+        if (!$order_no) $this->error('参数错误');
+        $order = Db::name('store_order_info_order')
+            ->where('order_no',$order_no)
+            ->where('mid',$this->uid)
+            ->find();
+        if (!$order) $this->error('订单不存在');
+        if ($order['status']!=0) $this->error('订单已支付或已取消');
+        $com = true;
+        Db::startTrans();
+        try {
+            $up_data = [
+                'status'=>2,
+                'cancel_at'=>date('Y-m-d H:i:s'),
+                'cancel_state'=>2
+            ];
+            Db::name('store_order_info_order')->where('order_no',$order_no)->update($up_data);
+
+            $cancle = [
+                'mid'=>$this->uid,
+                'order_id'=>$order['id']
+            ];
+            Db::name('store_order_info_cancel_log')->insert($cancle);
+
+            $time = date('Y-m-d H:i:s',time()-(60*60));
+            $count = Db::name('store_order_info_cancel_log')->where('mid',$this->uid)->where('create_at','gt',$time)->count();
+            if ($count>2){
+                $buy_time = date('Y-m-d H:i:s',time()+(24*60*60));
+                Db::name('store_member')->where('id',$this->uid)->update(['buy_time'=>$buy_time]);
+            }
+
+            Db::commit();
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($this->uid);
+            $this->success('取消成功');
+        }else{
+            $this->error('取消失败,请稍后重试');
+        }
+    }
+
+    /**
+     * 银行卡快捷支付发送短信
+     * @url /api/Secondary/yhkjSms
+     */
+    public function yhkjSms(){
+        //redis原子锁
+        if (redisSetNx('SecondaryyhkjSms'.$this->uid,2)){
+            $this->checkSwitch(1);
+            $user = getMemberInfoHash($this->uid); //获取用户信息
+            $order_no = input('order_no');   //订单号
+            $id = input('id');
+            $from = input('from','wx');
+            if (!$order_no) $this->error('参数错误');
+            if (!$id) $this->error('参数错误');
+            $order = Db::name('store_order_info_order')
+                ->where('id',$id)
+                ->where('mid',$this->uid)
+                ->find();
+            $order_no = $order['order_no'];
+            $pay_type = $order['pay_type'];
+            $this->checkSwitch(2,$pay_type);
+            if (!$order) $this->error('订单不存在');
+            if ($order['status']!=0) $this->error('订单已支付或已取消');
+            $info = Db::name('store_order_info')->where('id',$order['info_id'])->find();
+            if ($info['resale_status']!=2) $this->error('藏品已出售或已撤销出售');
+            $order_nos = get_order_sn();
+            if (Db::name('store_order_info_order')
+                ->where('order_no',$order_no)
+                ->where('mid',$this->uid)
+                ->update(['order_no'=>$order_nos]))
+            {
+                $huiju = new HuijuKjPay();
+                $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/kuaijieSecondPayNotify';//回调地址
+                $result = $huiju->paySms($order_nos,$order['pay_price'],'二级市场购买藏品',$this->uid,$callback_url,$user['hj_pay_bank_card']);
+                if ($result['resp_codo']=='SUCCESS'){
+                    if ($result['biz_code']=='JS000000'){
+                        $data = json_decode($result['data'],true);
+                        if ($data['order_status']=='P3000'){
+                            DelRedisSetNx('SecondaryyhkjSms'.$this->uid);
+                            $this->success('发送成功',$result);
+                        }else{
+                            DelRedisSetNx('SecondaryyhkjSms'.$this->uid);
+                            $this->error($data['err_msg']);
+                        }
+                    }else{
+                        DelRedisSetNx('SecondaryyhkjSms'.$this->uid);
+                        $this->error($result['biz_msg']);
+                    }
+                }else{
+                    DelRedisSetNx('SecondaryyhkjSms'.$this->uid);
+                    $this->error('发送失败');
+                }
+            }else{
+                DelRedisSetNx('SecondaryyhkjSms'.$this->uid);
+                $this->error('失败');
+            }
+        }else{
+            $this->error('请求过快');
+        }
+
+    }
+
+    /**
+     * @title 待支付订单支付
+     * @desc  待支付订单支付
+     * @author  Gavin
+     * @url /api/Secondary/payOrder
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     * @param name:order_no type:string require:1 default:-- desc:订单号
+     * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
+     *
+     * @return  name:order_no type:int require:0 default:0 desc:订单号
+     * @return  name:pay type:string require:0 default:0 desc:支付信息
+     */
+    public function payOrder(){
+        $this->checkSwitch(1);
+        $user = getMemberInfoHash($this->uid); //获取用户信息
+        $order_no = input('order_no');   //订单号
+        $id = input('id');
+        $from = input('from','wx');
+        if (!$order_no) $this->error('参数错误');
+        if (!$id) $this->error('参数错误');
+        $order = Db::name('store_order_info_order')
+           // ->where('order_no',$order_no)
+           ->where('id',$id)
+            ->where('mid',$this->uid)
+            ->find();
+        $order_no = $order['order_no'];
+        $pay_type = $order['pay_type'];
+
+        if ($pay_type=='yhkj'){
+            $sms_code = input('sms_code');
+            if (!$sms_code) $this->error('验证码为空');
+        }
+        $this->checkSwitch(2,$pay_type);
+        if (!$order) $this->error('订单不存在');
+        if ($order['status']!=0) $this->error('订单已支付或已取消');
+        $info = Db::name('store_order_info')->where('id',$order['info_id'])->find();
+        if ($info['resale_status']!=2) $this->error('藏品已出售或已撤销出售');
+        $com = true;
+        Db::startTrans();
+        try {
+            //获取价格
+            $total_fee = $order['pay_price'];
+            $body = 'top艺术购买二级市场藏品';
+            switch ($pay_type){
+                case 'wx':
+                    $config = retrunWxConfig();
+                    $total_fee = $total_fee * 100;
+                    $config['notify_url'] = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/SecondaryWxOrderNotify';
+                    $app = Factory::payment($config);
+
+                    $post_data = [
+                        'body' => $body,
+                        'out_trade_no' => $order_no,
+                        'total_fee' => $total_fee,
+                        'attach'=>$this->uid,     //自定义传值
+                    ];
+                    //trade_type   SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
+                    if ($from=='wx'){
+                        $post_data['openid'] = $user['openid'];
+                        $post_data['trade_type'] = 'JSAPI';
+                    }elseif ($from=='h5'){
+                        $post_data['trade_type'] = 'MWEB';
+                    }
+                    $result = $app->order->unify($post_data);
+
+                    if ($result['return_msg']=='OK'){
+                        if ($result['result_code']=='FAIL'){
+                            $com = false;
+                            Db::rollback();
+                        }else{
+                            $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
+                            $retrun_data['order_no'] = $order_no;
+                            $retrun_data['id'] = $order['id'];
+                            $retrun_data['pay'] = json_decode($order1,true);
+                            Db::commit();
+                        }
+                    }else{
+                        $com = false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'zfb':
+                    $order_nos = get_order_sn();
+                    if (Db::name('store_order_info_order')
+                        ->where('order_no',$order_no)
+                        ->where('mid',$this->uid)
+                        ->update(['order_no'=>$order_nos])){
+                        $zfb = new AliPay3();
+                        $notify_url = $this->request->root(true).'/index.php/api/Pay/alipaySecondaryNotify';//回调地址
+                        $callback_url = 'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order';
+                        $order_result = $zfb->aliPay($body,$total_fee,$order_nos,$notify_url,$this->uid,$callback_url);
+                        $retrun_data['order_no'] = $order_nos;
+                        $retrun_data['id'] = $order['id'];
+                        $retrun_data['pay'] = $order_result['pay_url'];
+                        Db::commit();
+                    }else{
+                        $com=false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'sd':
+                    $client = new Shande();
+                    $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/shandeSecondaryNotify';//回调地址
+                    $total_fee = $total_fee*100;
+                    $lenth = strlen($total_fee);
+                    $total_fee = get0number($lenth).$total_fee;
+                    $order_nos = get_order_sn();
+                    if (Db::name('store_order_info_order')
+                        ->where('order_no',$order_no)
+                        ->where('mid',$this->uid)
+                        ->update(['order_no'=>$order_nos])){
+                        $result = $client->orderPay($order_nos,$total_fee,$body,$notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');
+                        $retrun_data['order_no'] = $order_nos;
+                        $retrun_data['id'] = $order['id'];
+                        $retrun_data['pay'] = json_decode($result['data'],true);
+                        Db::commit();
+                    }else{
+                        $com=false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'ylh5':
+                    $order_nos = get_order_sn();
+                    if (Db::name('store_order_info_order')->where('order_no',$order_no)->where('mid',$this->uid)->update(['order_no'=>$order_nos])){
+                            $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/shop/order';
+                            $huiju = new HuijuH5Pay();
+                            $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylSecondaryNotify';//回调地址
+                            $result = $huiju->ylh5Pay($order_nos,$callback_url,$notify_url,$info['name'],$total_fee,$this->uid,2);
+                            if ($result['ra_Code']=='100'){
+                                $retrun_data['order_no'] = $order_nos;
+                                $retrun_data['id'] = $order['id'];
+                                $retrun_data['pay'] = $result['rc_Result'];
+                                Db::commit();
+                            }else{
+                                $com = false;
+                                Db::rollback();
+                            }
+                    }else{
+                        $com = false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'zfbh5':
+                    $order_nos = get_order_sn();
+                    if (Db::name('store_order_info_order')
+                        ->where('order_no',$order_no)
+                        ->where('mid',$this->uid)
+                        ->update(['order_no'=>$order_nos])){
+
+                        $callback_url = 'http://'.$_SERVER['SERVER_NAME'].'/pages/shop/order';
+                        $huiju = new HuijuH5Pay();
+                        $notify_url = 'http://'.$_SERVER['SERVER_NAME'].'/index.php/api/Pay/ylSecondaryNotify';//回调地址
+                        $result = $huiju->ylh5Pay($order_nos,$callback_url,$notify_url,$info['name'],$total_fee,$this->uid,1);
+                        if ($result['ra_Code']=='100'){
+                            $retrun_data['order_no'] = $order_nos;
+                            $retrun_data['id'] = $order['id'];
+                            $retrun_data['pay'] = $result['rc_Result'];
+                            Db::commit();
+                        }else{
+                            $com = false;
+                            Db::rollback();
+                        }
+                    }else{
+                        $com = false;
+                        Db::rollback();
+                    }
+                    break;
+                case 'yhkj':
+                    $huiju = new HuijuKjPay();
+                    $result = $huiju->kuaijiePay($order_no,$sms_code);
+                    if ($result['resp_codo']=='SUCCESS'){
+                        if ($result['biz_code']=='JS000000'){
+                            $data = json_decode($result['data'],true);
+                            if ($data['order_status']=='P1000'){
+                                Db::commit();
+                                $this->success('支付成功');
+                            }else{
+                                Db::rollback();
+                                $this->error($data['err_msg']);
+                            }
+                        }else{
+                            Db::rollback();
+                            $this->error($result['biz_msg']);
+                        }
+                    }else{
+                        Db::rollback();
+                        $this->error('支付失败');
+                    }
+                    break;
+
+            }
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            $this->success('成功',$retrun_data);
+        }
+        $this->error('失败,请稍后重试');
+    }
+
+
+    /**
+     * @param 判断开关
+     * @param string $pay_type
+     * @return bool
+     * @throws \think\db\exception\DataNotFoundException
+     * @throws \think\db\exception\ModelNotFoundException
+     * @throws \think\exception\DbException
+     */
+    public function checkSwitch($type,$pay_type=''){
+        if ($type==1){
+            $v = getConfigValue('secondary_sell_switch');
+            if (!$v){
+                $this->error('功能暂时关闭');
+            }else{
+                $start = getConfigValue('shelves_start_time');
+                $end = getConfigValue('shelves_end_time');
+                if (time()>strtotime($start) && time()<=strtotime($end)){
+
+                }else{
+                    $this->error('出售时间为:'.$start.'-'.$end);
+                }
+            }
+        }elseif ($type==2){
+            $nameArray = ['secondary_wx_switch','secondary_zfb_switch','secondary_sd_switch'];
+            $values = getConfig($nameArray);
+            if ($pay_type=='wx'){
+                if (!$values['secondary_wx_switch']) $this->error('微信支付暂时关闭');
+            }elseif ($pay_type=='zfb'){
+                if (!$values['secondary_zfb_switch']) $this->error('支付宝支付暂时关闭');
+            }elseif ($pay_type=='sd'){
+                if (!$values['secondary_sd_switch']) $this->error('杉德支付暂时关闭');
+            }
+        }elseif ($type==3){
+            $v = getConfigValue('secondary_buy_switch');
+            if (!$v){
+                $this->error('功能暂时关闭');
+            }else{
+                $start = getConfigValue('buy_start_time');
+                $end = getConfigValue('buy_end_time');
+                if (time()>strtotime($start) && time()<=strtotime($end)){
+
+                }else{
+                    $this->error('时间时间为:'.$start.'-'.$end);
+                }
+            }
+        }
+        return true;
+    }
+
+
+
+
+
+}

+ 130 - 0
application/api/controller/Sms.php

@@ -0,0 +1,130 @@
+<?php
+
+namespace app\api\controller;
+
+use app\common\library\Common;
+use app\common\library\Jiyan;
+use think\Db;
+use think\facade\Validate;
+use think\Request;
+use AlibabaCloud\Client\AlibabaCloud;
+
+/**
+ * @title 发送短信
+ * @controller Sms
+
+ */
+class Sms extends Base
+{
+    /**
+     * @title 发送短信
+     * @desc 发送短信
+     * @url /api/Sms/send
+     * @method POST
+     * @tag 基础
+     * @header
+     * @param  name:phone type:string require:1 default:-- desc:手机号
+     * @param  name:event type:string require:0 default:register desc:发送类型register:注册forgetpwd:找回密码login:找回密码
+     */
+    public function send()
+    {
+        $this->error('没有短信支付');
+
+        $phone = input("phone");
+        $event = input("event",'register');
+        if (!$phone || !Validate::regex($phone, "^1\d{10}$")) {
+            $this->error('手机号不正确');
+        }
+
+//        $lot_number = input('lot_number');
+//        $captcha_output = input('captcha_output');
+//        $pass_token = input('pass_token');
+//        $gen_time = input('gen_time');
+//        if (!$lot_number || !$captcha_output || !$pass_token || !$gen_time) $this->error('参数错误');
+//
+//        $jy = new Jiyan();
+//        $result = $jy->jy($lot_number,$captcha_output,$pass_token,$gen_time,1);
+//        if ($result['result']=='fail') $this->error('校验失败,请稍后重试');
+
+        $last = Db::name('store_sms')->where(['mobile' => $phone, 'event' => $event])
+            ->order('id', 'DESC')
+            ->find();
+        if ($last && time() - $last['createtime']< 60) {
+            $this->error('发送频繁!');
+        }
+//        $ipSendTotal = Db::name('store_sms')->where(['ip' => request()->ip()])->whereTime('createtime', '-1 hours')->count();
+//        if ($ipSendTotal >= 5) {
+//            $this->error('发送频繁!');
+//        }
+
+        $member = Db::name('store_member')
+            ->where('phone',$phone)
+            ->where('is_deleted',0)
+            ->count();
+        switch ($event){
+            case 'register':
+                if ($member) $this->error('手机号已注册');
+                break;
+            case 'forgetpwd': case 'login':
+                if (!$member) $this->error('手机号未注册');
+                break;
+        }
+
+        //发送阿里云短信
+        $ret = $this->accessKeyClient($event, $phone, mt_rand(100000, 999999));
+        if ($ret['Code'] === 'OK') {
+            $this->success('发送成功!');
+        } elseif ($ret['Code'] === 'isv.BUSINESS_LIMIT_CONTROL') {
+            $this->error('发送太过频繁!');
+        } else {
+            $this->error($ret['msg']);
+        }
+    }
+
+    function accessKeyClient($event, $mobile, $num)
+    {
+        $sett = [
+            'ali_accesskey',
+            'ali_accesskey_secret',
+            'templateCode',
+            'sign_name'
+        ];
+        $array = getConfig($sett);
+
+        $ali_accesskey = $array['ali_accesskey'];
+        $ali_accesskey_secret = $array['ali_accesskey_secret'];
+        $templateCode = $array['templateCode'];
+        AlibabaCloud::accessKeyClient($ali_accesskey, $ali_accesskey_secret)
+            ->regionId('cn-hangzhou')
+            ->asDefaultClient();
+
+        try {
+            $result = AlibabaCloud::rpc()
+                ->product('Dysmsapi')
+                // ->scheme('https') // https | http
+                ->version('2017-05-25')
+                ->action('SendSms')
+                ->method('POST')
+                ->host('dysmsapi.aliyuncs.com')
+                ->options([
+                    'query' => [
+                        'PhoneNumbers' => $mobile,
+                        'SignName' => $array['sign_name'],
+                        'TemplateCode' => $templateCode,
+                        'TemplateParam' => '{"code":' . $num . '}',
+                    ],
+                ])
+                ->request();
+            $info = $result->toArray();
+            if ($info['Code'] == 'OK') {
+                $ip = request()->ip();
+                Db::name('store_sms')->insert(['event' => $event, 'mobile' => $mobile, 'createtime'=>time(),'code' => $num, 'ip' => $ip]);
+            }
+            return $info;
+        } catch (ClientException $e) {
+            echo $e->getErrorMessage() . PHP_EOL;
+        } catch (ServerException $e) {
+            echo $e->getErrorMessage() . PHP_EOL;
+        }
+    }
+}

+ 248 - 0
application/api/controller/Synthetic.php

@@ -0,0 +1,248 @@
+<?php
+
+
+namespace app\api\controller;
+use app\common\library\AliPay;
+use EasyWeChat\Factory;
+use think\cache\driver\Redis;
+use think\Db;
+/**
+ * @title 合成藏品
+ * @controller secondary
+ * @package app\api\controller
+ */
+class Synthetic extends Base
+{
+
+    public function initialize(){
+        parent::initialize();
+        parent::check_login();
+    }
+
+
+    /**
+     * @title 合成藏品列表
+     * @desc  合成藏品列表
+     * @author  Gavin
+     * @url /api/Synthetic/synList
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     *
+     * @return name:name type:string default:-- desc:藏品名称
+     * @return name:cover type:string default:-- desc:藏品图
+     * @return name:label type:string default:-- desc:标签
+     * @return name:price type:DECIMAL default:-- desc:价格
+     * @return name:inventory type:int default:-- desc:库存
+     * @return name:now_inventory type:int default:-- desc:剩余库存
+     * @return name:sy_state type:float default:-- desc:藏品状态(1:进行中2:即将开售3:已结束4:已售罄)
+     * @return name:sell_time type:string default:-- desc:发行时间
+     * @return name:end_time type:string default:-- desc:结束时间
+     */
+    public function synList(){
+        checkSynCollectionState();
+        $list = Db::name('store_collection')
+            ->where('is_deleted',0)
+            ->where('status',1)
+            ->where('type',2)
+            ->field('id,cover,name,label,price,inventory,now_inventory,sy_state,sell_time,end_time')
+            ->order('sy_state asc,sell_time asc')
+            ->select();
+        foreach ($list as &$v) {
+            $v['now_inventory'] = getCollectionInventory($v['id']);
+            $v['id'] = (string)$v['id'];
+        }
+        $this->success('成功',$list);
+    }
+
+    /**
+     * @title 藏品详情
+     * @desc 藏品详情
+     * @author  Gavin
+     * @url /api/Synthetic/CollectionDetail
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:string require:1 default:-- desc:藏品ID
+     *
+     * @return name:name type:string default:-- desc:藏品名称
+     * @return name:cover type:string default:-- desc:藏品图
+     * @return name:label type:string default:-- desc:标签
+     * @return name:price type:DECIMAL default:-- desc:价格
+     * @return name:inventory type:int default:-- desc:库存
+     * @return name:now_inventory type:int default:-- desc:剩余库存
+     * @return name:sy_state type:float default:-- desc:藏品状态(1:进行中2:即将开售3:已结束4:已售罄)
+     * @return name:sell_time type:string default:-- desc:发行时间
+     * @return name:end_time type:string default:-- desc:结束时间
+     * @return name:describe type:string default:-- desc:商品描述
+     * @return name:instructions type:int default:1 desc:权益说明
+     * @return name:buy_notice type:int default:1 desc:购买须知
+     * @return name:warm_prompt type:int default:1 desc:温馨提示
+     * @return name:auth_img type:int default:1 desc:作者头像
+     * @return name:auth_name type:int default:1 desc:作者姓名
+     * @return name:is_exchange type:int default:1 desc:材料是否够(1:够,可以合成0:不够)
+     * @return name:material@name type:string default:-- desc:材料名称
+     * @return name:material@cover type:string default:-- desc:材料图片
+     * @return name:material@num type:string default:-- desc:所需数量
+     * @return name:material@now_num type:string default:-- desc:当前所有数量
+     */
+    public function CollectionDetail(){
+        $collect_id = input('id');
+        if (!$collect_id) $this->error('参数错误');
+        checkSynCollectionState($collect_id);
+        $info = Db::name('store_collection')
+            ->where('is_deleted',0)
+            ->where('status',1)
+            ->where('type',2)
+            ->where('id',$collect_id)
+            ->field('id,cover,name,label,price,inventory,now_inventory,sy_state,sell_time,end_time,describe,buy_count,instructions,buy_notice,warm_prompt,auth_img,auth_name,share_img,intro')
+            ->find();
+        if (!$info) $this->error('藏品不存在');
+        $info['describe'] = explode('|',$info['describe']);
+        $info['now_inventory'] = getCollectionInventory($info['id']);
+        $material = Db::name('store_collection_material')
+            ->alias('a')
+            ->join('store_collection b','a.c_id=b.id')
+            ->where('a.cid',$collect_id)
+            ->field('a.id,a.c_id,a.num,b.name,b.cover')
+            ->select();
+        $info['is_exchange'] = $this->check_exchange($collect_id,$this->uid);
+        foreach ($material as &$v){
+            $v['now_num'] = Db::name('store_order_info')
+                ->where('c_id',$v['c_id'])
+                ->where('mid',$this->uid)
+                ->where('status','neq',2)
+                ->where('is_destruction',1)
+                ->count();
+        }
+        $info['material'] = $material;
+        $info['id'] = (string)$info['id'];
+        $this->success('成功',$info);
+    }
+
+
+
+
+    /**
+     * 判断材料是否足够可以兑换
+     */
+    public function check_exchange($id,$mid){
+        $list = Db::name('store_collection_material')
+            ->alias('a')
+            ->join('store_collection b','a.c_id=b.id')
+            ->where('a.cid',$id)
+            ->field('a.id,a.c_id,a.num,b.name,b.cover')
+            ->select();
+        $is_exchange = 1;
+        foreach ($list as &$v){
+            $user_count = Db::name('store_order_info')
+                ->where('c_id',$v['c_id'])
+                ->where('mid',$mid)
+                ->where('status','neq',2)
+                ->where('is_destruction',1)
+                ->count();
+            if ($user_count<$v['num']){
+                $is_exchange = 0;
+                break;
+            }
+        }
+        return $is_exchange;
+
+    }
+
+    /**
+     * @title 合成藏品
+     * @desc  合成藏品
+     * @author  Gavin
+     * @url /api/Synthetic/synCollection
+     * @method POST
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:id type:int require:1 default:1 desc:藏品ID
+     */
+    public function synCollection(){
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $info = Db::name('store_collection')
+            ->where('is_deleted',0)
+            ->where('status',1)
+            ->where('type',2)
+            ->where('id',$id)
+            ->find();
+        if (!$info) $this->error('藏品不存在');
+        $now_inventory = getCollectionInventory($id);
+        if ($now_inventory<=0) $this->error('已售罄');
+        if (strtotime($info['sell_time'])>time()) $this->error('未开始,无法合成');
+        if (strtotime($info['end_time'])<time())  $this->error('已结束,无法合成');
+
+//        $info_count = Db::name('store_order_info')->where('mid',$this->uid)->where('c_id',$id)->count();
+//        if ($info_count) $this->error('已合成过,只能合成一次');
+
+        //获取是否已经铸造hash
+        $is_nft = Db::name('hash2')->where('goods_id',$id)->where('success',1)->find();
+        if (!$is_nft) $this->error('nft未上架,无法合成');
+        $is_exchange = $this->check_exchange($id,$this->uid);
+        if (!$is_exchange) $this->error('材料不足,无法合成');
+        $com = true;
+        Db::startTrans();
+        try {
+            $material = Db::name('store_collection_material')
+                ->where('cid',$id)
+                ->select();
+            foreach ($material as &$v){
+                Db::name('store_order_info')
+                    ->where('c_id',$v['c_id'])
+                    ->where('mid',$this->uid)
+                    ->where('status','neq',2)
+                    ->where('is_destruction',1)
+                    ->limit($v['num'])
+                    ->order('id asc')
+                    ->update(['is_destruction'=>0]);
+            }
+
+            //获取排名
+            $rank = getRanking($id)+1;
+            $tag = getTag($id,$rank,$info['inventory']);
+            saveRanking($id);
+            $collectors_hash = '';
+            $date = [
+                'order_id'=>0,
+                'order_no'=>get_order_sn(),
+                'tag'=>$tag,
+                'mid'=>$this->uid,
+                'c_id'=>$id,
+                'name'=>$info['name'],
+                'cover'=>$info['cover'],
+                'pro_info'=>json_encode($info,true),
+                'type'=>$info['type'],
+                'tokenid'=>$is_nft['class_id'],
+                'nfttype'=>$is_nft['operationId'],
+                'collectors_hash'=>$collectors_hash,
+                'collectors_hash_time'=>'',
+                'status'=>5
+            ];
+            Db::name('store_order_info')->insert($date);
+
+            Db::commit();
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            setMemberInfoHash($this->uid);
+            //减掉库存
+            loseCollectionInventory($id,1);
+
+            $this->success('合成成功');
+        }
+        $this->error('合成失败,请稍后重试');
+
+
+
+    }
+
+
+
+
+
+}

+ 411 - 0
application/api/controller/Timedtask.php

@@ -0,0 +1,411 @@
+<?php
+
+
+namespace app\api\controller;
+use AlibabaCloud\Client\AlibabaCloud;
+use AlibabaCloud\Client\Exception\ClientException;
+use AlibabaCloud\Client\Exception\ServerException;
+use think\cache\driver\Redis;
+use think\Db;
+use think\Exception;
+use function AlibabaCloud\Client\value;
+
+/**
+ * @title 定时任务
+ * Class Timedtask
+ * @controller Timedtask
+ * @group base
+ */
+class Timedtask
+{
+    /**
+     * @title 取消订单定时任务
+     * @desc  未支付的自动取消
+     * @author  Gavin
+     * @url /api/Timedtask/cancelGoodsOrder
+     * @method GET
+     */
+    public function cancelGoodsOrder(){
+        $CancelTime = getCancelTime();
+        if ($CancelTime<=0){
+            die;
+        }
+        $redis = new Redis();
+        $users = $redis->hkeys('buyUserInfo');
+        if ($users){
+            foreach ($users as &$value){
+                $key = 'order_not_pay_'.$value;
+                $len = $redis->hGetLen($key);
+                if ($len){
+                    $list = $redis->hGetvals($key);
+                    foreach ($list as &$a){
+                        $info = json_decode($a,true);
+
+                        $cancel_time = strtotime($info['create_at'])+($CancelTime*60);
+                        if ($cancel_time<time()){
+                            $info['status'] = 2;
+                            $info['cancel_at'] = date('Y-m-d H:i:s');
+                            Db::name('store_order')->insert($info);
+                            //加上库存
+                            addCollectionInventory($info['c_id'],$info['num']);
+                            //减少用户购买数量
+                            DecrByCount($info['mid'],$info['c_id'],$info['num']);
+                            //删除数据
+                            $redis->hdel($key,$info['order_no']);
+                        }
+                    }
+                }else{
+                    $redis->hdel('buyUserInfo',$value);
+                }
+            }
+        }
+    }
+
+
+    /**
+     * @title 二级市场未支付的自动取消
+     * @desc  二级市场未支付的自动取消
+     * @author  Gavin
+     * @url /api/Timedtask/SecondaryancelGoodsOrder
+     * @method GET
+     */
+    public function SecondaryancelGoodsOrder(){
+        $CancelTime = getConfigValue('secondary_cancel_time');
+        if ($CancelTime<=0){
+            die;
+        }
+        $list = Db::name('store_order_info_order')->where('status',0)->select();
+        foreach ($list as &$v){
+            $cancel_time = strtotime($v['create_at'])+($CancelTime*60);
+            if ($cancel_time<time()){
+                $info['status'] = 2;
+                $info['cancel_at'] = date('Y-m-d H:i:s');
+                Db::name('store_order_info_order')->where('id',$v['id'])->update($info);
+
+                $cancle = [
+                    'mid'=>$v['mid'],
+                    'order_id'=>$v['id']
+                ];
+                Db::name('store_order_info_cancel_log')->insert($cancle);
+
+                $time = date('Y-m-d H:i:s',time()-(60*60));
+                $count = Db::name('store_order_info_cancel_log')->where('mid',$v['mid'])->where('create_at','gt',$time)->count();
+                if ($count>2){
+                    $buy_time = date('Y-m-d H:i:s',time()+(24*60*60));
+                    Db::name('store_member')->where('id',$v['mid'])->update(['buy_time'=>$buy_time]);
+                }
+            }
+        }
+    }
+
+    /**
+     * @title 创建链账户
+     * @desc  创建链账户
+     * @author  Gavin
+     * @url /api/Timedtask/createAddress
+     * @method GET
+     */
+    public function createAddress(){
+        set_time_limit(0);
+        $member = Db::name('store_member')
+            ->where('wallet_address','eq','')
+            //->whereNull('wallet_address')
+            ->field('id,offline_account,phone,wallet_address')
+            ->order('id asc')
+            ->limit(30)
+            ->select();
+        foreach ($member as &$v){
+            $name = $v['phone'];
+            $operationId = $v['phone'];
+            $operationId .= rand(10000,99999);
+            $url = getIpAddress()."customNFT/createAccount?name=".$name."&operationId=".$operationId;
+            $offlineaccount = file_get_contents($url);
+            $offline_account = json_decode($offlineaccount,true);
+            if (isset($offline_account) && isset($offline_account['account'])){
+                Db::name('store_member')->where('id',$v['id'])
+                    ->update(['accountName'=>$operationId,'wallet_address'=>$offline_account['account'],'offline_account'=>$offlineaccount]);
+            }
+        }
+    }
+
+
+
+    /**
+     * 查询链上架回执
+     * @url /api/Timedtask/setNftCheck
+     */
+    public function setNftCheck(){
+        Db::name('hash2')->where('success',0)->chunk(30,function ($list){
+            foreach ($list as &$v){
+                $result =$this->checkhashSuccess($v['operationId']);
+                $data['result'] = json_encode($result,true);
+                if (isset($result) && $result['status']==1){
+                    $data['success'] = 1;
+                    $data['class_id'] =$result['class_id'];
+                }
+                Db::name('hash2')->where('id',$v['id'])->update($data);
+            }
+        },'id','asc');
+    }
+
+
+    /**
+     * 创建nft
+     * @url /api/Timedtask/createNft
+     */
+    public function createNft(){
+        Db::name('store_order_info')
+            ->whereIn('status','1,4,5')
+            ->where('collectors_hash','eq','')
+            ->whereNull('collectors_hash2')
+            ->chunk(30,function ($list){
+                foreach ($list as &$v){
+                    $name = get32Str(10);
+                    $operationId = $v['id'];
+                    $operationId .= rand(1000000,9999999);
+                    $to =  Db::name('store_member')->where('id',$v['mid'])->value('wallet_address');
+                    $url = getIpAddress().'customNFT/createNft?classId='.$v['tokenid'].'&name='.$name.'&operationId='.$operationId.'&recipient='.$to;
+                    $res=curlRequest($url);
+                    $result=json_decode($res,true);
+                    if (isset($result) && isset($result['task_id'])){
+                        Db::name('store_order_info')->where('id',$v['id'])->update(['collectors_hash2'=>$result['task_id'],'nfttype'=>$operationId]);
+                    }
+                }
+            },'id','asc');
+    }
+
+    /**
+     * 判断创建的nft是否成功
+     * @url /api/Timedtask/checkcreateNft
+     */
+    public function checkcreateNft(){
+        Db::name('store_order_info')
+            ->whereIn('status','1,4,5')
+            ->where('collectors_hash','eq','')
+            ->whereNotNull('collectors_hash2')
+            ->chunk(30,function ($list){
+                foreach ($list as &$v){
+                    $result = $this->checkhashSuccess($v['collectors_hash2']);
+                    $data['result'] = json_encode($result,true);
+                    if (isset($result) && $result['status']==1){
+                        $data['tokenid'] = $result['class_id'];
+                        $data['collectors_hash'] = $result['tx_hash'];
+                        $data['nftid'] = $result['nft_id'];
+                        $data['collectors_hash_time'] = date('Y-m-d H:i:s');
+                    }
+                    Db::name('store_order_info')->where('id',$v['id'])->update($data);
+                }
+            },'id','asc');
+    }
+
+
+    /**
+     * 转移nft
+     * @url /api/Timedtask/judgeHash
+     */
+    public function judgeHash(){
+        Db::name('store_order_info')
+            ->where('status',3)
+            ->where('collectors_hash','eq','')
+            ->whereNull('collectors_hash2')
+            ->chunk(30,function ($list){
+                foreach ($list as &$v){
+                    $operationId = $v['id'];
+                    $operationId .= rand(10000000,99999999);
+                    $add = Db::name('store_member')->where('id',$v['to_mid'])->value('wallet_address');
+                    $to =  Db::name('store_member')->where('id',$v['mid'])->value('wallet_address');
+                    $url = getIpAddress().'customNFT/transfer?classId='.$v['tokenid'].'&NFTId='.$v['nftid'].'&operationId='.$operationId.'&owner='.$add.'&recipient='.$to;
+                    $res=curlRequest($url);
+                    $result=json_decode($res,true);
+                    dump($result);
+                    if (isset($result) && isset($result['task_id'])){
+                        Db::name('store_order_info')->where('id',$v['id'])->update(['collectors_hash2'=>$result['task_id'],'nfttype'=>$operationId]);
+                    }
+                }
+            },'id','asc');
+    }
+
+
+    /**
+     * 判断转移的nft是否成功
+     * @url /api/Timedtask/checkjudgeNft
+     */
+    public function checkjudgeNft(){
+        Db::name('store_order_info')
+            ->where('status',3)
+            ->where('collectors_hash','eq','')
+            ->whereNotNull('collectors_hash2')
+            ->chunk(30,function ($list){
+                foreach ($list as &$v){
+                    $result = $this->checkhashSuccess($v['collectors_hash2']);
+                    $data['result'] = json_encode($result,true);
+                    dump($result);
+                    if (isset($result) && $result['status']==1){
+                        $data['tokenid'] = $result['class_id'];
+                        $data['collectors_hash'] = $result['tx_hash'];
+                        $data['nftid'] = $result['nft_id'];
+                    }
+                    Db::name('store_order_info')->where('id',$v['id'])->update($data);
+                }
+            },'id','asc');
+    }
+
+
+
+
+
+
+
+
+    /**
+     * 查询交易是否成功
+     * @url /api/Timedtask/checkhashSuccess
+     */
+    public function checkhashSuccess($operationId){
+        $url = getIpAddress().'customNFT/queryDealResult?operationId='.$operationId;
+        $res=curlRequest($url);
+        $result=json_decode($res,true);
+        return $result;
+    }
+
+
+    /**
+     * 判断转移的数据是否属于当前转移用户
+     * @url /api/Timedtask/checkjudgeHashOwer
+     */
+    public function checkjudgeHashOwer(){
+        $list = Db::name('store_order_info')
+            ->where('status',3)
+            ->where('collectors_hash','eq','')
+            ->whereNull('collectors_hash2')
+            ->order('id asc')
+            ->limit(30)
+            ->select();
+        foreach ($list as &$v){
+            if (time()>strtotime($v['collectors_hash_time'])+(8*60*60)){
+                $url = getIpAddress().'customNFT/sNFT?classId='.$v['tokenid'].'&NFTId='.$v['nftid'];
+                $res=json_decode(curlRequest($url),true);
+                dump($res);
+                if (isset($res)){
+                    $add = Db::name('store_member')->where('id',$v['to_mid'])->value('wallet_address');
+                    if ($add != $res['owner']){
+                        $operationId = $v['id'];
+                        $operationId .= rand(10000000,99999999);
+                        $add = $res['owner'];
+                        $to =  $add;
+                        $url = getIpAddress().'customNFT/transfer?classId='.$v['tokenid'].'&NFTId='.$v['nftid'].'&operationId='.$operationId.'&owner='.$add.'&recipient='.$to;
+                        curlRequest($url);
+                    }
+                }
+            }
+        }
+    }
+
+
+
+    /**
+     * redis 加锁
+     */
+    function redisCreateSetNx($id){
+        $redis = new Redis();
+        $key = 'hash_'.$id;
+        $exptime = 450;
+        $is_lock = $redis->setnx($key,time()+$exptime);
+        if ($is_lock){
+            return true;
+        }else{
+            //加锁失败的情况下,判断锁是否已经存在,如果存在切已经过期,删除锁,重新加锁
+            $val = $redis->get($key);
+            if ($val && $val<time()){
+                $redis->del($key);
+            }
+            return $redis->setnx($key,time()+$exptime);
+        }
+    }
+
+
+    /**
+     * redis  nonce加锁
+     */
+    function redisNonceSetNx(){
+        $redis = new Redis();
+        $key = 'noncenx';
+        $exptime = 10;
+        $is_lock = $redis->setnx($key,time()+$exptime);
+        if ($is_lock){
+            return true;
+        }else{
+            //加锁失败的情况下,判断锁是否已经存在,如果存在切已经过期,删除锁,重新加锁
+            $val = $redis->get($key);
+            if ($val && $val<time()){
+                $redis->del($key);
+            }
+            return $redis->setnx($key,time()+$exptime);
+        }
+    }
+
+
+    /**
+     * 预约藏品前十五分钟发送短信
+     * /api/Timedtask/sendSms
+     */
+    public function sendSms(){
+        Db::name('store_collection_remind')->where('is_send',0)->chunk(50,function ($list){
+            foreach ($list as &$v){
+                $sell_time = Db::name('store_collection')->where('id',$v['c_id'])->value('sell_time');
+                $t = (strtotime($sell_time)-time())/60;
+                if ($t<=15 && $t>0){
+                    $result = $this->accessKeyClient($v['mobile'],$v['mid'],$v['c_id']);
+                    if ($result['Code'] === 'OK') {
+                        Db::name('store_collection_remind')->where('id',$v['id'])->update(['is_send'=>1]);
+                    }
+                }
+            }
+        },'id','asc');
+    }
+
+    function accessKeyClient($mobile,$mid,$cid)
+    {
+        $ali_accesskey = 'LTAI5tSTBuRP5AnPBHDz8gTF';
+        $ali_accesskey_secret = '7RVjRKv8cCaKW4hMMVZ1SFPkqeIbn4';
+        $templateCode = 'SMS_243370550';
+        AlibabaCloud::accessKeyClient($ali_accesskey, $ali_accesskey_secret)
+            ->regionId('cn-hangzhou')
+            ->asDefaultClient();
+
+        $user = getMemberInfoHash($mid); //获取用户信息
+        $coll_info = getCollectionInfoHash($cid);
+        $post = [
+            'name'=>$user['name'],
+            'goodname'=>$coll_info['name'],
+            'pubtime'=>$coll_info['sell_time']
+        ];
+        $TemplateParam = json_encode($post,true);
+
+        try {
+            $result = AlibabaCloud::rpc()
+                ->product('Dysmsapi')
+                // ->scheme('https') // https | http
+                ->version('2017-05-25')
+                ->action('SendSms')
+                ->method('POST')
+                ->host('dysmsapi.aliyuncs.com')
+                ->options([
+                    'query' => [
+                        'PhoneNumbers' => $mobile,
+                        'SignName' => '同质科技',
+                        'TemplateCode' => $templateCode,
+                        'TemplateParam' => $TemplateParam
+                    ],
+                ])
+                ->request();
+            $info = $result->toArray();
+            return $info;
+        } catch (ClientException $e) {
+            echo $e->getErrorMessage() . PHP_EOL;
+        } catch (ServerException $e) {
+            echo $e->getErrorMessage() . PHP_EOL;
+        }
+    }
+
+}

+ 298 - 0
application/api/controller/Timedtask1.php

@@ -0,0 +1,298 @@
+<?php
+
+
+namespace app\api\controller;
+use AlibabaCloud\Client\AlibabaCloud;
+use AlibabaCloud\Client\Exception\ClientException;
+use AlibabaCloud\Client\Exception\ServerException;
+use think\cache\driver\Redis;
+use think\Db;
+use think\Exception;
+
+/**
+ * @title 定时任务
+ * Class Timedtask
+ * @controller Timedtask
+ * @group base
+ */
+class Timedtask1
+{
+    /**
+     * @title 取消订单定时任务
+     * @desc  未支付的自动取消
+     * @author  Gavin
+     * @url /api/Timedtask/cancelGoodsOrder
+     * @method GET
+     */
+    public function cancelGoodsOrder(){
+        $CancelTime = getCancelTime();
+        if ($CancelTime<=0){
+            die;
+        }
+        $redis = new Redis();
+        $users = $redis->hkeys('buyUserInfo');
+        if ($users){
+            foreach ($users as &$value){
+                $key = 'order_not_pay_'.$value;
+                $len = $redis->hGetLen($key);
+                if ($len){
+                    $list = $redis->hGetvals($key);
+                    foreach ($list as &$a){
+                        $info = json_decode($a,true);
+                        $cancel_time = strtotime($info['create_at'])+($CancelTime*60);
+                        if ($cancel_time<time()){
+                            $info['status'] = 2;
+                            $info['cancel_at'] = date('Y-m-d H:i:s');
+                            Db::name('store_order')->insert($info);
+                            //加上库存
+                            addCollectionInventory($info['c_id'],$info['num']);
+                            //减少用户购买数量
+                            DecrByCount($info['mid'],$info['c_id'],$info['num']);
+                            //删除数据
+                            $redis->hdel($key,$info['order_no']);
+                        }
+                    }
+                }else{
+                    $redis->hdel('buyUserInfo',$value);
+                }
+            }
+        }
+    }
+
+
+    /**
+     * @title 二级市场未支付的自动取消
+     * @desc  二级市场未支付的自动取消
+     * @author  Gavin
+     * @url /api/Timedtask/SecondaryancelGoodsOrder
+     * @method GET
+     */
+    public function SecondaryancelGoodsOrder(){
+        $CancelTime = getConfigValue('secondary_cancel_time');
+        if ($CancelTime<=0){
+            die;
+        }
+        $list = Db::name('store_order_info_order')->where('status',0)->select();
+        foreach ($list as &$v){
+            $cancel_time = strtotime($v['create_at'])+($CancelTime*60);
+            if ($cancel_time<time()){
+                $info['status'] = 2;
+                $info['cancel_at'] = date('Y-m-d H:i:s');
+                Db::name('store_order_info_order')->where('id',$v['id'])->update($info);
+
+                $cancle = [
+                    'mid'=>$v['mid'],
+                    'order_id'=>$v['id']
+                ];
+                Db::name('store_order_info_cancel_log')->insert($cancle);
+
+                $time = date('Y-m-d H:i:s',time()-(60*60));
+                $count = Db::name('store_order_info_cancel_log')->where('mid',$v['mid'])->where('create_at','gt',$time)->count();
+                if ($count>2){
+                    $buy_time = date('Y-m-d H:i:s',time()+(24*60*60));
+                    Db::name('store_member')->where('id',$v['mid'])->update(['buy_time'=>$buy_time]);
+                }
+            }
+        }
+    }
+
+    /**
+     * 藏品铸造hash 定时任务
+     * @url /api/Timedtask/castingHash
+     */
+    public function castingHash(){
+        set_time_limit(0);
+        $redis = new Redis();
+        $list = Db::name('store_collection')->where('is_deleted',0)->select();
+        $id = 0;
+        $set_count = 0;
+        foreach ($list as &$v){
+            $count = $redis->get('castingHash_'.$v['id']);
+            if ($count && $count>0){
+                $id = $v['id'];
+                $set_count = $count;
+                break;
+            }
+        }
+        if ($id){
+            $address = '0x3153052307c15c46abb7b3667cdbcd48a7e2a341';
+            for ($i=0;$i<$set_count;$i++){
+                if ($this->redisNonceSetNx()){
+                    $str=$id.'-'.rand(100000000,999999999);
+                    $url='http://47.111.246.47:8083/ddc/createHashAutoNonce?address='.$address.'&ddcURI='.$str;
+                    $res=curlRequest($url);
+                    $result=json_decode($res,true);
+                    if($result['code']){
+                        continue;
+                    }else{
+                        $data['goods_id']=$id;
+                        $data['hash']=$res;
+                        $data['ddcURI'] = $str;
+
+                        if (Db::name('hash')->insert($data)){
+                            $redis->del('noncenx');
+                            $redis->Decr('castingHash_'.$id);  //减一
+                        }
+                    }
+                }
+            }
+        }
+
+    }
+
+    /**
+     * 铸造hash判断回执 定时任务
+     * @url /api/Timedtask/transactionReceipt
+     */
+    public function transactionReceipt(){
+        set_time_limit(0);
+        Db::name('hash')->where('success',0)
+            ->chunk(50,function ($list){
+            $redis = new Redis();
+            foreach ($list as &$v){
+                $url = 'http://47.111.246.47:8083/ddc/getTransactionReceipt?hash='.$v['hash'];
+                $res=curlRequest($url);
+                Db::name('hash')->where('id',$v['id'])->update(['result'=>$res]);
+                $result3=json_decode($res,true);
+                if (isset($result3['status']) && $result3['status']=='0x1'){
+                    $url4='http://47.111.246.47:8083/ddc/createDdcid?hash='.$v['hash'];
+                    $ddcid=curlRequest($url4);
+                    $result4=json_decode($ddcid,true);
+                    if($result4['code']){
+
+                    }else{
+                        $update_data['success'] = 1;
+                        $update_data['ddcid'] = $ddcid;
+                        if (Db::name('hash')->where('id',$v['id'])->update($update_data)){
+                            //存入reids list
+                            $redis_data = ['hash'=>$v['hash'],'ddcid'=>$ddcid,'create_at'=>date('Y-m-d H:i:s')];
+                            $redis->rPush('collectionHash_'.$v['goods_id'],json_encode($redis_data));
+                        }
+                    }
+                }
+            }
+        },'id', 'asc');
+
+    }
+
+    /**
+     * 发放hash
+     * @url /api/Timedtask/sendHash
+     */
+    public function sendHash(){
+        set_time_limit(0);
+        Db::name('store_order_info')
+            ->whereNotNull('company_hash')
+            ->whereIn('status','1,3')
+            ->where('company_hash','neq','')
+            ->where('type',1)
+            ->where('collectors_hash','eq','')
+            ->chunk('20',function ($list){
+                $from = '0x3153052307c15c46abb7b3667cdbcd48a7e2a341';
+                $redis = new Redis();
+                foreach ($list as &$v){
+                    if ($v['status']==1){
+                        $mid = $v['mid'];
+                    }elseif ($v['status']==3){
+                        $from = Db::name('store_member')->where('id',$v['to_mid'])->value('wallet_address');
+                        $mid = $v['to_mid'];
+                    }
+
+                    $to = Db::name('store_member')->where('id',$mid)->value('wallet_address');
+                    if (empty($to) || $to == ''){
+                        continue;
+                    }
+                    if ($this->redisNonceSetNx()){
+                        $ddcid = $v['ddcid'];
+                        $url = "http://47.111.246.47:8083/ddc/transferAutoNonce?from=$from&to=$to&ddcid=".$ddcid;
+                        $res=curlRequest($url);
+                        $result=json_decode($res,true);
+                        if($result['code']){
+                            continue;
+                        }else{
+                            Db::name('store_order_info')
+                                ->where('id',$v['id'])
+                                ->update(['collectors_hash'=>$res,'collectors_hash_time'=>date('Y-m-d H:i:s')]);
+                            $redis->del('noncenx');
+                        }
+                    }
+                }
+            },'id','desc');
+    }
+
+
+    /**
+     * @title 创建链账户
+     * @desc  创建链账户
+     * @author  Gavin
+     * @url /api/Timedtask/createAddress
+     * @method GET
+     */
+    public function createAddress(){
+        set_time_limit(0);
+        $member = Db::name('store_member')
+            ->whereNull('wallet_address')
+            ->field('id,offline_account,phone,wallet_address')->order('id asc')->limit(30)->select();
+        foreach ($member as &$v){
+            $url = 'http://47.111.246.47:8083/ddc/createAccount';
+            $offlineaccount = file_get_contents($url);
+            echo $offlineaccount;
+            $offline_account = json_decode($offlineaccount,true);
+            $address = $offline_account['address'];
+            $phone =$v['phone'];
+            //$phone .= rand(10000,99999);
+            $url = "http://47.111.246.47:8083/ddc/createAddress?name=".$phone."&account=".$address;
+            $res=curlRequest($url);
+            $laddress = json_decode($res,true);
+            dump($laddress);
+            if ($laddress['code']==0){
+                $wallet_address = $laddress['data']['opbChainClientAddress'];
+                Db::name('store_member')->where('id',$v['id'])
+                    ->update(['accountName'=>$phone,'wallet_address'=>$wallet_address,'offline_account'=>$offlineaccount]);
+            }
+        }
+
+    }
+
+    /**
+     * redis 加锁
+     */
+    function redisCreateSetNx($id){
+        $redis = new Redis();
+        $key = 'hash_'.$id;
+        $exptime = 450;
+        $is_lock = $redis->setnx($key,time()+$exptime);
+        if ($is_lock){
+            return true;
+        }else{
+            //加锁失败的情况下,判断锁是否已经存在,如果存在切已经过期,删除锁,重新加锁
+            $val = $redis->get($key);
+            if ($val && $val<time()){
+                $redis->del($key);
+            }
+            return $redis->setnx($key,time()+$exptime);
+        }
+    }
+
+
+    /**
+     * redis  nonce加锁
+     */
+    function redisNonceSetNx(){
+        $redis = new Redis();
+        $key = 'noncenx';
+        $exptime = 10;
+        $is_lock = $redis->setnx($key,time()+$exptime);
+        if ($is_lock){
+            return true;
+        }else{
+            //加锁失败的情况下,判断锁是否已经存在,如果存在切已经过期,删除锁,重新加锁
+            $val = $redis->get($key);
+            if ($val && $val<time()){
+                $redis->del($key);
+            }
+            return $redis->setnx($key,time()+$exptime);
+        }
+    }
+
+}

+ 151 - 0
application/api/controller/Upload.php

@@ -0,0 +1,151 @@
+<?php
+
+// +----------------------------------------------------------------------
+// | ThinkAdmin
+// +----------------------------------------------------------------------
+// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
+// +----------------------------------------------------------------------
+// | 官方网站: http://demo.thinkadmin.top
+// +----------------------------------------------------------------------
+// | 开源协议 ( https://mit-license.org )
+// +----------------------------------------------------------------------
+// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
+// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
+// +----------------------------------------------------------------------
+
+namespace app\api\controller;
+
+use library\File;
+
+/**
+ * @title OSS对象存储
+ * @controller Upload
+ * @group common
+ */
+class Upload extends Base
+{
+    protected $id= 'LTAI5t76oDeTRvdNWZjs4oGd';
+    protected $key= '8Z1zyFjtaF4v9kdjJeWnlFVx3xhtEF';
+    protected $host = 'https://shuzisc.oss-cn-beijing.aliyuncs.com';
+
+    /**
+     * @title 文件上传
+     * @desc 文件上传
+     * @author qc
+     * @url /api/Upload/upload
+     * @method POST
+     * @tag 文件上传
+     */
+
+    public function upload()
+    {
+        if (!($file = $this->getUploadFile()) || empty($file)) {
+            $this->error('文件上传异常,文件可能过大或未上传!');
+        }
+        if (!$file->checkExt(strtolower(sysconf('storage_local_exts')))) {
+            $this->error('文件上传类型受限,请在后台配置!');
+        }
+        if ($file->checkExt('php,sh')) {
+            $this->error('可执行文件禁止上传到本地服务器!');
+        }
+        $this->safe = boolval(input('safe'));
+        $this->uptype = $this->getUploadType();
+        $this->extend = pathinfo($file->getInfo('name'), PATHINFO_EXTENSION);
+        $name = File::name($file->getPathname(), $this->extend, '', 'md5_file');
+        $info = File::instance($this->uptype)->save($name, file_get_contents($file->getRealPath()), $this->safe);
+        if (is_array($info) && isset($info['url'])) {
+            $url = $this->safe ? $name : $info['url'];
+            $this->success('上传成功',$url);
+        } else {
+            $this->error('文件处理失败,请稍候再试!');
+        }
+    }
+    /**
+     * 获取本地文件对象
+     * @return \think\File
+     */
+    private function getUploadFile()
+    {
+        try {
+            return $this->request->file('file');
+        } catch (\Exception $e) {
+            $this->error(lang($e->getMessage()));
+        }
+    }
+
+    /**
+     * 获取文件上传方式
+     * @return string
+     * @throws \think\Exception
+     * @throws \think\exception\PDOException
+     */
+    private function getUploadType()
+    {
+        $this->uptype = input('uptype');
+        if (!in_array($this->uptype, ['local', 'oss', 'qiniu'])) {
+            $this->uptype = sysconf('storage_type');
+        }
+        return $this->uptype;
+    }
+    /**
+     * @title 获取配置
+     * @desc 获取配置
+     * @author QGF
+     * @url /api/Upload/getSignedUrl
+     * @method GET
+     * @tag 配置信息
+     * @param name:type type:string require:1 default:-- desc:
+     */
+    public function getSignedUrl(){
+
+        $type=$this->request->get('type');
+        $data=$this->getPolicy($type.'/');
+        $this->success('操作成功',$data);
+    }
+
+    public function getPolicy($path,$maxSize=5000000){
+        $now = time();
+        $expire = 600; //设置该policy超时时间是10s. 即这个policy过了这个有效时间,将不能访问
+        $end = $now + $expire;
+        $expiration = $this->_gmt_iso8601($end);
+        $policy = [
+            'expiration'=>$expiration,
+            'conditions'=>[
+                ['content-length-range', 1, $maxSize],
+                ['starts-with', '$key', $path],
+            ]
+        ];
+        $policy = json_encode($policy);
+        $policy = base64_encode($policy);
+        $signature = base64_encode(hash_hmac('sha1', $policy, $this->key, true));
+
+        $response = array();
+        $response['OSSAccessKeyId'] = $this->id;
+        $response['policy'] = $policy;
+        $response['Signature'] = $signature;
+        $response['host'] = $this->host;
+        $response['expire'] = $end;
+        $response['success_action_status'] = 200;
+        //这个参数是设置用户上传指定的前缀
+        $response['key'] = $path;
+        return $response;
+    }
+    protected function _gmt_iso8601($time) {
+        $dtStr = date("c", $time);
+        $mydatetime = new \DateTime($dtStr);
+        $expiration = $mydatetime->format(\DateTime::ISO8601);
+        $pos = strpos($expiration, '+');
+        $expiration = substr($expiration, 0, $pos);
+        return $expiration."Z";
+    }
+
+
+    public function getSysDirection()
+    {
+        $data = [
+            'film_notice'=>htmlspecialchars_decode(sysconf('film_notice')),
+            'film_direction'=>htmlspecialchars_decode(sysconf('film_direction'))
+        ];
+        $this->success('获取成功',$data);
+    }
+}

+ 1354 - 0
application/api/controller/UserCenter.php

@@ -0,0 +1,1354 @@
+<?php
+namespace app\api\controller;
+use AlibabaCloud\Client\AlibabaCloud;
+use AlibabaCloud\Client\Exception\ClientException;
+use AlibabaCloud\Client\Exception\ServerException;
+use app\common\library\AliPay2;
+use app\common\library\HuijuKjPay;
+use think\cache\driver\Redis;
+use think\Db;
+use think\facade\Validate;
+
+/**
+ * @title 会员个人中心
+ * @controller UserCenter
+ * @group base
+ */
+class UserCenter extends Base
+{
+    public function initialize()
+    {
+        parent::initialize();
+//        parent::check_login();
+    }
+
+    /**
+     * @title 获取个人信息
+     * @desc 个人信息
+     * @author Gavin
+     * @url /api/User_center/getUserInfo
+     * @method POST
+     * @tag 个人信息
+     * @header name:Authorization require:1 desc:Token
+     * @return name:id type:int default:-- desc:ID(邀请ID)
+     * @return name:phone type:string default:-- desc:手机号
+     * @return name:name type:string default:-- desc:昵称
+     * @return name:headimg type:string default:-- desc:头像地址
+     * @return name:integral type:int default:-- desc:积分
+     * @return name:snap_card type:int default:-- desc:抢购卡
+     * @return name:wallet_address type:int default:-- desc:钱包地址
+     * @return name:true_name type:string default:-- desc:真实姓名
+     * @return name:id_card type:string default:-- desc:身份证号
+     * @return name:is_auth type:int default:-- desc:是否认证(0否1是)
+     * @return name:auth_at type:string default:-- desc:认证时间
+     * @return name:vip_name type:int default:-- desc:身份
+     * @return name:is_order_no_paid type:string default:-- desc:是否有未支付订单true:有false:没有
+     * @return name:is_sign type:string default:-- desc:今日是否已经签到true:已签到false:未签到
+     * @return name:invite_img type:string default:-- desc:邀请二维码
+     * @return name:invite_address type:string default:-- desc:邀请地址
+     * @return name:lucky_number type:int default:-- desc:剩余抽奖次数
+     * @return name:money type:string default:-- desc:余额
+     */
+    public function getUserInfo()
+    {
+        parent::check_login();
+        $uid = $this->uid;
+        $user_info = Db::name('store_member')
+            ->field('password,second_password',true)
+            ->where('id',$uid)
+            ->find();
+        if(empty($user_info)) $this->error('用户信息不正确');
+        $user_info['vip_name'] = $user_info['vip']==1 ? "藏友" : "创世勋章";
+        //是否有待支付订单
+        $order = Db::name('store_order')->where('status',0)->where('is_deleted',0)->where('mid',$uid)->count();
+        $user_info['is_order_no_paid'] = $order ? true : false;
+        //今日是否签到
+        $date = date('Y-m-d');
+        $sign = Db::name('store_member_sign')->where('mid',$uid)->where('date',$date)->count();
+        $user_info['is_sign'] = $sign ? true : false;
+        //获取每日免费次数
+        $free_number = getConfigValue('free_lucky_number');
+        //获取当日免费抽奖记录
+        $date = date('Y-m-d');
+        $now_date_count = Db::name('store_blind_box_log')->where('m_id',$this->uid)->where('date',$date)->where('is_free',0)->count();
+        $lucky_number = $user_info['lottery_number'] + ($free_number-$now_date_count);
+        $user_info['lucky_number'] =$lucky_number<=0 ? 0 : $lucky_number;
+        $user_info['id'] = (string)$user_info['id'];
+
+        //生成邀请码
+
+        if (!$user_info['invite_img'] || !$user_info['invite_address']){
+            $code = $user_info['invite_code'];
+            $invite_img = setintivecode($code);
+            $invite_address = getintiveaddress($code);
+            Db::name('store_member')->where('id',$this->uid)->update(['invite_img'=>$invite_img,'invite_address'=>$invite_address]);
+        }
+
+        $user_info['invite_img'] = str_replace('http:','https:',$user_info['invite_img']);
+
+        $this->success('获取成功',$user_info);
+    }
+
+    /**
+     * @title 绑定opendid
+     * @desc 绑定opendid
+     * @author Gavin
+     * @url /api/User_center/bindOpenid
+     * @method POST
+     * @tag 编辑信息
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:code type:string require:0 default:-- desc:code
+     */
+    public function bindOpenid()
+    {
+        parent::check_login();
+        $code = input('code');
+        if(empty($code)) $this->error('参数错误');
+        $appid = getConfigValue('wechat_appid');
+        $secret = getConfigValue('wechat_appsecret');
+        $res = http_get('https://api.weixin.qq.com/sns/oauth2/access_token?appid='.$appid.'&secret='.$secret.'&code='.$code.'&grant_type=authorization_code');
+        $res = json_decode($res,true);
+        if (isset($res['openid'])){
+            $update_data['openid'] = $res['openid'];
+            $update_data['update_at'] = date('Y-m-d H:i:s');
+            if (Db::name('store_member')->where('id',$this->uid)->update($update_data)){
+                setMemberInfoHash($this->uid);
+                $this->success('绑定成功');
+            }
+            $this->error('绑定失败');
+        }else{
+            $this->error('获取openid失败');
+        }
+    }
+
+
+    /**
+     * @title 编辑个人信息
+     * @desc 编辑个人信息
+     * @author Gavin
+     * @url /api/User_center/updateUserInfo
+     * @method POST
+     * @tag 编辑信息
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:name type:string require:0 default:-- desc:姓名
+     * @param name:headimg type:string require:0 default:-- desc:头像地址
+     */
+    public function updateUserInfo()
+    {
+        parent::check_login();
+        $headimg  = input('post.headimg');
+        $name     = trim(input('post.name',''));
+        if(!$headimg && !$name)   $this->error('参数错误');
+        $update_data= [];
+        if($name) $update_data['name'] = $name;
+        if ($headimg) $update_data['headimg'] = $headimg;
+        if($name){
+            $check_member =  Db::name('store_member')
+                ->where('name',$name)
+                ->where('id','<>',$this->uid)
+                ->count();
+            if($check_member) $this->error('该用户名已被占用');
+        }
+        $update_data['update_at'] = date('Y-m-d H:i:s');
+        if (Db::name('store_member')->where('id',$this->uid)->update($update_data)){
+            setMemberInfoHash($this->uid);
+            $this->success('编辑成功');
+        }
+        $this->error('编辑失败');
+    }
+
+
+    /**
+     * @title 绑定支付宝
+     * @desc 绑定支付宝
+     * @author Gavin
+     * @url /api/User_center/bind_zfb
+     * @method POST
+     * @tag 编辑信息
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:account type:string require:0 default:-- desc:支付宝账号
+     * @param name:zfb_real_name type:string require:0 default:-- desc:真实姓名
+     */
+    public function bind_zfb(){
+        parent::check_login();
+        $this->check_login();
+        $account = input('account');    //账号
+        $real_name = input('real_name'); //真实姓名
+        if (!$account) $this->error('支付宝账号为空');
+        if (!$real_name) $this->error('真实姓名为空');
+        $update_data['zfb_account'] = $account;
+        $update_data['zfb_real_name'] = $real_name;
+        $update_data['update_at'] = date('Y-m-d H:i:s');
+        if (Db::name('store_member')->where('id',$this->uid)->update($update_data)){
+            setMemberInfoHash($this->uid);
+            $this->success('绑定成功');
+        }
+        $this->error('绑定失败');
+    }
+
+    /**
+     * @title 实名认证
+     * @desc 实名认证
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/userCertification
+     * @header name:Authorization require:1 desc:Token
+     * @param name:true_name type:string require:1 default:-- desc:真实姓名
+     * @param name:id_card type:string require:1 default:-- desc:身份证号
+     * @param name:bank_card type:string require:1 default:-- desc:银行卡号
+     * @param name:bank_mobile type:string require:0 default:-- desc:银行卡预留手机号
+     */
+    public function userCertification(){
+        parent::check_login();
+        //redis原子锁
+        if (redisSetNx('userCertification'.$this->uid,2)){
+            $true_name = input('post.true_name');
+            $id_card = input('post.id_card');
+            $bank_card = input('post.bank_card');
+            $bank_mobile = input('post.bank_mobile');
+            if (!$true_name || !$id_card || !$bank_card) $this->error('参数错误');
+            $check_id_card = isCreditNo($id_card);
+            //获取年龄
+            $age = getAge($id_card);
+//            if (!$check_id_card) $this->error('身份证号格式错误');
+//            if (!checkbank($bank_card)) $this->error('银行卡号格式错误');
+            if ($age<18 || $age>65) $this->error('年龄不合规');
+            $is_auth = Db::table('store_member')->where('id',$this->uid)->value('is_auth');
+            if($is_auth) $this->error('已认证');
+            $check = Db::name('store_member')->where('id_card',$id_card)->count();
+            if ($check) $this->error('身份证号已经认证过');
+            $bank_check = Db::name('store_member')->where('bank_card',$bank_card)->count();
+            if ($bank_check) $this->error('银行卡号已经使用过');
+
+        //二要素
+        //if (!idcardlingyu($id_card,$true_name)) $this->error('身份证与姓名验证失败');
+        //四要素
+            $res = idcardlingyufour($id_card,$true_name,$bank_card,$bank_mobile);
+            if (!$res['success']) $this->error($res['remark']);
+
+            if (Db::table('store_member')->where('id',$this->uid)
+                ->update(['is_auth'=>1,'true_name'=>$true_name,'id_card'=>$id_card,'bank_card'=>$bank_card,'bank_mobile'=>$bank_mobile,'auth_at'=>date('Y-m-d H:i:s')])){
+
+                    setMemberInfoHash($this->uid);
+
+                    $user1 = getMemberInfoHash($this->uid); //获取用户信息
+                if ($user1['pid']){
+                    $data = [
+                        'mid'=>$user1['pid'],
+                        'to_mid'=>$this->uid,
+                        'create_at'=>date('Y-m-d H:i:s')
+                    ];
+                    Db::name('store_activities_invite_log')->insert($data);
+                }
+
+                pull_new($this->uid);
+                DelRedisSetNx('userCertification'.$this->uid);
+                $this->success('认证成功');
+            }
+            DelRedisSetNx('userCertification'.$this->uid);
+            $this->error('认证失败');
+        }else{
+            $this->error('请求过快');
+        }
+
+    }
+
+    /**
+     * @title 修改密码
+     * @desc 修改密码
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/updatePass
+     * @header name:Authorization require:1 desc:Token
+     * @param name:old_password type:string require:1 default:-- desc:旧密码
+     * @param name:new_password type:string require:1 default:-- desc:新密码
+     * @param name:confirm_password type:string require:1 default:-- desc:确认密码
+     */
+    public function updatePass()
+    {
+        parent::check_login();
+        $old_password = input('post.old_password');
+        $new_password = input('post.new_password');
+        $confirm_password = input('post.confirm_password');
+       // $yzm = input('post.yzm');
+        if (!$old_password || !$new_password || !$confirm_password) $this->error('参数错误');
+        if ($new_password!=$confirm_password) $this->error('密码与确认密码不一致');
+        if (!preg_match('/^[0-9a-z]{6,12}$/i',$new_password)) $this->error('密码格式错误,请输入6-12位数字+字母');
+        $member = Db::name('store_member')->where('id',$this->uid)->find();
+
+        //验证短信验证码
+//        $time = time()-60;
+//        $sms = Db::name('store_sms')->where(['mobile' => $member['phone'], 'event' => 'forgetpwd'])
+//            ->where('createtime','>',$time)
+//            ->order('id', 'DESC')
+//            ->find();
+//        if (!$sms || $sms['code'] != $yzm) $this->error('短信验证码不正确!');
+
+        if ($member['password']!=md5($old_password)) $this->error('旧密码错误');
+        $data['password'] = md5($new_password);
+        $data['update_at'] = date('Y-m-d H:i:s');
+        if (Db::name('store_member')->where('id',$this->uid)->update($data)){
+            setMemberInfoHash($this->uid);
+            $this->success('修改成功');
+        }
+        $this->error('修改失败');
+    }
+
+    /**
+     * @title 修改二级密码
+     * @desc 修改二级密码
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/updateSecondPass
+     * @header name:Authorization require:1 desc:Token
+     * @param name:second_password type:string require:1 default:-- desc:二级密码
+     * @param name:confirm_second_password type:string require:1 default:-- desc:确认密码
+     */
+    public function updateSecondPass()
+    {
+        parent::check_login();
+        $second_password = input('post.second_password');
+        $confirm_second_password = input('post.confirm_second_password');
+        $yzm = input('post.yzm');
+        if (!$second_password || !$confirm_second_password) $this->error('参数错误');
+        if ($second_password!=$confirm_second_password) $this->error('密码与确认密码不一致');
+        if (!preg_match('/^[0-9]{6}$/i',$second_password)) $this->error('二级密码格式错误,请输入6位纯数字');
+
+        $member = getMemberInfoHash($this->uid); //获取用户信息
+        //验证短信验证码
+        $time = time()-60;
+        $sms = Db::name('store_sms')->where(['mobile' => $member['phone'], 'event' => 'forgetpwd'])
+            ->where('createtime','>',$time)
+            ->order('id', 'DESC')
+            ->find();
+        if (!$sms || $sms['code'] != $yzm) $this->error('短信验证码不正确!');
+
+        $data['second_password'] = md5($second_password);
+        $data['update_at'] = date('Y-m-d H:i:s');
+        if (Db::name('store_member')->where('id',$this->uid)->update($data)){
+            setMemberInfoHash($this->uid);
+            $this->success('修改成功');
+        }
+        $this->error('修改失败');
+    }
+
+    /**
+     * @title 我的藏品
+     * @desc 我的藏品
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/myCollection
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @return  name:id type:int require:0 default:0 desc:藏品ID
+     * @return  name:tag type:string require:0 default:0 desc:唯一标签
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:藏品图片
+     * @return  name:auth_img type:string require:0 default:0 desc:作者头像
+     * @return  name:auth_name type:string require:0 default:0 desc:作者名称
+     */
+    public function myCollection(){
+        parent::check_login();
+
+        $type = input('type');
+        $where = [
+            'mid'=>$this->uid,
+            'is_destruction'=>1,
+            'resale_status'=>1
+        ];
+        $cids = Db::name('store_order_info')->where($where)
+            ->where('status','neq','2')
+            ->when($type,function ($query) use ($type){
+                if ($type==1){
+                    $query->where('type','neq',3);
+                }else{
+                    $query->where('type',3);
+                }
+            })
+            ->group('c_id')
+            ->column('c_id');
+        $arr = [];
+        foreach ($cids as &$v){
+            $array['c_id'] = $v;
+            $array['count'] = Db::name('store_order_info')
+                ->where($where)
+                ->when($type,function ($query) use ($type){
+                    if ($type==1){
+                        $query->where('type','neq',3);
+                    }else{
+                        $query->where('type',3);
+                    }
+                })
+                ->where('c_id',$v)
+                ->where('status','neq','2')
+                ->count();
+            $info = Db::name('store_order_info')
+                ->where($where)->where('c_id',$v)
+                ->limit(1)
+                ->field('id,name,cover,tag,pro_info')
+                ->find();
+            $array['cover'] = $info['cover'];
+            $array['name'] = $info['name'];
+            $pro_info = json_decode($info['pro_info'],true);
+            $array['auth_img'] = $pro_info['auth_img'];
+            $array['auth_name'] = $pro_info['auth_name'];
+            $tag = explode('#',$info['tag']);
+            $tag2 = explode('/',$tag[1]);
+            $array['tag'] = $tag[0].'#???'.'/'.$tag2[1];
+            array_push($arr,$array);
+        }
+        $this->success('成功',$arr);
+
+
+        $list = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->when($type,function ($query) use ($type){
+                if ($type==1){
+                    $query->where('type','neq',3);
+                }else{
+                    $query->where('type',3);
+                }
+            })
+            ->where('status','neq','2')
+            ->where('is_destruction',1)
+            ->where('resale_status',1)
+            ->field('id,tag,pro_info,c_id')
+            ->order('id desc')
+            ->select();
+        foreach ($list as &$v){
+            $pro_info = json_decode($v['pro_info'],true);
+            $v['name'] = $pro_info['name'];
+            $v['cover'] = $pro_info['cover'];
+            $v['auth_img'] = $pro_info['auth_img']? $pro_info['auth_img'] : Db::name('store_collection')->where('id',$v['c_id'])->value('auth_img');
+            $v['auth_name'] = $pro_info['auth_name'] ? $pro_info['auth_name'] : Db::name('store_collection')->where('id',$v['c_id'])->value('auth_name');
+            if ($v['tag']){
+                $tag = explode('#',$v['tag']);
+                $tag1 = substr($tag[0],0,6);
+                $v['tag'] = $tag1.'#'.$tag[1];
+            }
+            unset($v['pro_info']);
+        }
+        $this->success('成功',$list);
+    }
+
+
+
+
+    /**
+     * @title 藏品列表
+     * @desc 藏品列表
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/myCollectionList
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:c_id type:string require:1 default:-- desc:藏品ID
+     *
+     * @return  name:id type:int require:0 default:0 desc:藏品ID
+     * @return  name:tag type:string require:0 default:0 desc:唯一标签
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:藏品图片
+     * @return  name:auth_img type:string require:0 default:0 desc:作者头像
+     * @return  name:auth_name type:string require:0 default:0 desc:作者名称
+     */
+    public function myCollectionList(){
+        parent::check_login();
+
+        $c_id = input('c_id');
+        if (!$c_id) $this->error('参数错误');
+        $list = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('status','neq','2')
+            ->whereIn('type','1,3')
+            ->where('c_id',$c_id)
+            ->where('is_destruction',1)
+            ->where('resale_status',1)
+            ->field('id,tag,pro_info')
+            ->order('id desc')
+            ->select();
+        foreach ($list as &$v){
+            $pro_info = json_decode($v['pro_info'],true);
+            $v['name'] = $pro_info['name'];
+            $v['cover'] = $pro_info['cover'];
+            $v['auth_img'] = $pro_info['auth_img'];
+            $v['auth_name'] = $pro_info['auth_name'];
+            unset($v['pro_info']);
+        }
+        $this->success('成功',$list);
+    }
+
+    /**
+     * @title 我的盲盒
+     * @desc 我的盲盒
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/myBoxCollection
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @return  name:id type:int require:0 default:0 desc:藏品ID
+     * @return  name:tag type:string require:0 default:0 desc:唯一标签
+     * @return  name:name type:string require:0 default:0 desc:藏品名称
+     * @return  name:cover type:string require:0 default:0 desc:藏品图片
+     * @return  name:auth_img type:string require:0 default:0 desc:作者头像
+     * @return  name:auth_name type:string require:0 default:0 desc:作者名称
+     */
+    public function myBoxCollection(){
+        parent::check_login();
+        $list = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('status','neq','2')
+            ->where('type',3)
+            ->where('is_destruction',1)
+            ->where('resale_status',1)
+            ->field('id,tag,pro_info')
+            ->order('id desc')
+            ->select();
+        foreach ($list as &$v){
+            $pro_info = json_decode($v['pro_info'],true);
+            $v['name'] = $pro_info['name'];
+            $v['cover'] = $pro_info['cover'];
+            $v['auth_img'] = $pro_info['auth_img'];
+            $v['auth_name'] = $pro_info['auth_name'];
+            unset($v['pro_info']);
+        }
+        $this->success('成功',$list);
+    }
+
+    /**
+     * @title 藏品详情
+     * @desc 藏品详情
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/myCollectionDetail
+     * @header name:Authorization require:1 desc:Token
+     * @param name:id type:int require:1 default:-- desc:藏品ID
+     *
+     * @return  name:id type:int require:0 default:0 desc:藏品ID
+     * @return  name:tag type:string require:0 default:0 desc:唯一标签
+     * @return  name:collectors_name type:string require:0 default:0 desc:收藏者
+     * @return  name:collectors_hash type:string require:0 default:0 desc:收藏者hash
+     * @return  name:create_at type:string require:0 default:0 desc:收藏时间
+     * @return  name:company type:string require:0 default:0 desc:流转公司
+     * @return  name:company_hash type:string require:0 default:0 desc:流转公司hash
+     * @return  name:examples_illustrate type:string require:0 default:0 desc:转赠说明
+     * @return  name:pro_info@name type:string require:0 default:0 desc:藏品名称
+     * @return  name:pro_info@price type:string require:0 default:0 desc:藏品价格
+     * @return  name:pro_info@label type:string require:0 default:0 desc:藏品标签
+     * @return  name:pro_info@cover type:string require:0 default:0 desc:藏品图片
+     * @return  name:pro_info@auth_img type:string require:0 default:0 desc:作者头像
+     * @return  name:pro_info@auth_name type:string require:0 default:0 desc:作者姓名
+     * @return  name:pro_info@warm_prompt type:string require:0 default:0 desc:温馨提示
+     * @return  name:pro_info@share_img type:string require:0 default:0 desc:分享二维码
+     */
+    public function myCollectionDetail(){
+        parent::check_login();
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $info = Db::name('store_order_info')
+            ->where('mid',$this->uid)
+            ->where('id',$id)
+            ->find();
+        if (!$info) $this->error('藏品不存在');
+        $info['pro_info'] = json_decode($info['pro_info'],true);
+        if (!isset($info['pro_info']['share_img'])){
+            $info['pro_info']['share_img'] = Db::name('store_collection')->where('id',$info['pro_info']['id'])->value('share_img');
+        }
+
+        $issuer = Db::name('store_collection')->where('id',$info['c_id'])->value('issuer');
+        if (!empty($issuer)){
+            $info['company'] = $issuer;
+        }
+
+
+        $info['detail_img'] = $info['pro_info']['detail_img'];
+        $info['format'] = $info['pro_info']['format'];
+        $info['collectors_name'] = Db::name('store_member')->where('id',$this->uid)->value('name');
+        //$info['collectors_hash'] = $info['collectors_hash']=='' ? "发放中" : $info['collectors_hash'];
+        $info['collectors_hash_time'] = $info['collectors_hash_time']=='' ? "发放中" : $info['collectors_hash_time'];
+        $info['examples_illustrate'] = Db::table('system_config')->where('name','examples_illustrate')->value('value');
+        $info['c_id'] = (string)$info['c_id'];
+
+        if (!empty($info['tag'])){
+            $tag = explode('#',$info['tag']);
+            $tag1 = substr($tag[0],0,6);
+            $info['tag'] = $tag1.'#'.$tag[1];
+        }
+
+        $info['pro_info']['auth_name'] = $info['pro_info']['auth_name'] ? $info['pro_info']['auth_name'] : Db::name('store_collection')->where('id',$info['c_id'])->value('auth_name');
+        $info['pro_info']['auth_img'] = $info['pro_info']['auth_img'] ? $info['pro_info']['auth_name'] : Db::name('store_collection')->where('id',$info['c_id'])->value('auth_img');
+
+
+        if ($info['collectors_hash_time']!='0000-00-00 00:00:00'){
+            $info['create_at'] = $info['collectors_hash_time'];
+        }
+
+
+        $this->success('成功',$info);
+    }
+
+
+    /**
+     * @title 转赠
+     * @desc 转赠
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/examples
+     * @header name:Authorization require:1 desc:Token
+
+     * @param name:id type:int require:1 default:-- desc:藏品ID
+     * @param name:phone type:string require:1 default:-- desc:手机号
+     * @param name:wallet_address type:int require:1 default:-- desc:钱包地址
+     * @param name:second_password type:int require:1 default:-- desc:二级密码
+     *
+     */
+    public function examples(){
+        parent::check_login();
+
+        //redis原子锁
+        if (redisSetNx('examples'.$this->uid,2)){
+            $v = getConfigValue('examples_switch');
+            if (!$v) $this->error('维护中,暂时关闭');
+
+            $id = input('id');
+            $phone = input('phone');
+            $wallet_address = input('wallet_address');
+            $second_password = input('second_password');
+            if (!$id || !$phone || !$wallet_address) $this->error('参数错误');
+            $info = Db::name('store_order_info')->where('id',$id)->where('mid',$this->uid)->find();
+            if (!$info) $this->error('藏品不存在');
+            if ($info['status']==2) $this->error('藏品已转赠');
+            $member = Db::name('store_member')->where('phone',$phone)->where('wallet_address',$wallet_address)->find();
+            if (!$member) $this->error('转赠用户不存在');
+            if ($member['id']==$this->uid) $this->error('不能转赠给自己');
+
+            if (!$info['collectors_hash']) $this->error('发放中,无法转赠');
+
+            $mem = getMemberInfoHash($this->uid);
+            if ($mem['second_password']!=md5($second_password)) $this->error('密码错误');
+
+            $pro_info = Db::name('store_collection')->where('id',$info['c_id'])->find();
+            $log = Db::name('store_collect_examples_log')
+                ->where('order_info_id',$id)
+                ->find();
+            if (!$log){
+                if ($pro_info['one_given_day']!=0){
+                    $exam_time = strtotime($info['create_at'])+($pro_info['one_given_day']*24*60*60);
+                    if ($exam_time>time()) $this->error('持有时间限制,无法转赠');
+                }
+            }else{
+                if ($pro_info['other_given_day']!=0){
+                    $exam_time = strtotime($log['create_at'])+($pro_info['other_given_day']*24*60*60);
+                    if ($exam_time>time()) $this->error('持有时间限制,无法转赠');
+                }
+            }
+            $com = true;
+            Db::startTrans();
+            try {
+                Db::name('store_order_info')
+                    ->where('id',$id)
+                    ->update(['status'=>2,'over_time'=>date('Y-m-d H:i:s'),'to_mid'=>$member['id']]);
+                $to_date = [
+                    'order_id'=>$info['order_id'],
+                    'order_no'=>get_order_sn(),
+                    'tag'=>$info['tag'],
+                    'mid'=>$member['id'],
+                    'c_id'=>$info['c_id'],
+                    'name'=>$pro_info['name'],
+                    'cover'=>$pro_info['cover'],
+                    'pro_info'=>$info['pro_info'],
+                    'type'=>$pro_info['type'],
+                    'status'=>3,
+                    'to_mid'=>$this->uid,
+                    'over_time'=>date('Y-m-d H:i:s'),
+                    'tokenid'=>$info['tokenid'],
+                    'nfttype'=>$info['nfttype'],
+                    'nftid'=>$info['nftid'],
+                    'collectors_hash'=>'',
+                    'collectors_hash_time'=>date('Y-m-d H:i:s')
+                ];
+                $new_id = Db::name('store_order_info')->insertGetId($to_date);
+                $log_date = [
+                    'order_info_id'=>$new_id,
+                    'mid'=>$this->uid,
+                    'to_mid'=>$member['id'],
+                    'date'=>date('Y-m')
+                ];
+                Db::name('store_collect_examples_log')->insert($log_date);
+                Db::commit();
+            }catch (\Exception $e){
+                $com = false;
+                Db::rollback();
+            }
+            if ($com){
+                DelRedisSetNx('examples'.$this->uid);
+                $this->success('转赠成功');
+            }
+            DelRedisSetNx('examples'.$this->uid);
+            $this->error('转赠失败');
+        }else{
+            $this->error('请求过快');
+        }
+
+    }
+
+    /**
+     * @title 消息列表
+     * @desc 消息列表
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/messageList
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:id type:int require:0 default:0 desc:消息ID
+     * @return  name:title type:string require:0 default:0 desc:标题
+     * @return  name:content type:string require:0 default:0 desc:内容
+     * @return  name:create_at type:string require:0 default:0 desc:发布时间
+     * @return  name:is_read type:string require:0 default:0 desc:是否查看true:已查看false:未查看
+     */
+    public function messageList(){
+        $count = Db::name('store_message')->where('is_deleted',0)->count();
+        $list = Db::name('store_message')
+            ->field('is_deleted,update_at',true)
+            ->where('is_deleted',0)
+            ->order('id desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+//        echo Db::name('store_message')->getLastSql();die;
+        foreach ($list as &$v){
+            $is_read = Db::name('store_message_read_log')->where('mid',$this->uid)->where('message_id',$v['id'])->count();
+            $v['is_read'] = $is_read ? true : false;
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+    /**
+     * @title 点击消息
+     * @desc 点击消息,查看消息
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/readMessage
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:id type:int  : default:1 desc:消息ID
+     *
+     */
+    public function readMessage(){
+        $id = input('id');
+        if (!$id) $this->error('参数错误');
+        $log = Db::name('store_message_read_log')->where('message_id',$id)->where('mid',$this->uid)->count();
+        if (!$log){
+            $data = [
+                'mid'=>$this->uid,
+                'message_id'=>$id
+            ];
+            Db::name('store_message_read_log')->insert($data);
+        }
+        $this->success('成功');
+    }
+
+    /**
+     * @title 人脸认证sign
+     * @desc 人脸认证sign
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/getFaceSign
+     * @header name:Authorization require:1 desc:Token
+     *
+     *
+     * @param  name:name type:string  default:1 desc:姓名
+     * @param  name:idNo type:string  default:1 desc:身份证号
+     *
+     * @return  name:sign type:string require: default:-- desc:签名
+     *
+     */
+    public function getFaceSign(){
+        parent::check_login();
+        $name = input('name');
+        $idNo = input('idNo');
+        if (!$name || !$idNo) $this->error('参数错误');
+        $userId =  $this->uid;
+        //$userId = get32Str();
+        $data = getfaceid($name,$idNo,$userId);
+        $this->success('成功',$data);
+    }
+
+    /**
+     * @title 邀请记录
+     * @desc 邀请记录
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/invitedRecord
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:id type:int require:0 default:0 desc:用户ID
+     * @return  name:headimg type:string require:0 default:0 desc:头像
+     * @return  name:name type:string require:0 default:0 desc:名称
+     * @return  name:create_at type:string require:0 default:0 desc:时间
+     */
+    public function invitedRecord(){
+        parent::check_login();
+        $count = Db::name('store_member')
+            ->where('is_auth',1)
+            ->where('pid',$this->uid)->count();
+        $list = Db::name('store_member')
+            ->field('id,headimg,name,create_at,phone,is_auth')
+            ->where('pid',$this->uid)
+            //->where('is_auth',1)
+            ->order('id desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            $v['phone'] = substr_replace($v['phone'],'****',3,4);
+            $v['name'] =  substr($v['name'],0,16);
+            $v['create_at'] = date('Y-m-d',strtotime($v['create_at']));
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+    /**
+     * @title 邀请排行榜
+     * @desc 邀请排行榜
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/invitedList
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:id type:int require:0 default:0 desc:用户ID
+     * @return  name:headimg type:string require:0 default:0 desc:头像
+     * @return  name:name type:string require:0 default:0 desc:名称
+     * @return  name:phone type:string require:0 default:0 desc:手机号
+     * @return  name:create_at type:string require:0 default:0 desc:时间
+     */
+    public function invitedList(){
+
+        $count = Db::name('store_member as a')
+            ->field('a.id,a.name,a.headimg,a.phone,a.pid,(select count(b.id) from store_member as b where b.pid=a.id) as count')
+            ->group('a.id')
+            ->having('count>0')
+            ->count();
+        $list = Db::name('store_member as a')
+            ->field('a.id,a.name,a.headimg,a.phone,a.pid,(select count(b.id) from store_member as b where b.pid=a.id) as count')
+            ->group('a.id')
+            ->having('count>0')
+            ->order('count desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        $this->success('成功',compact('count','list'));
+    }
+
+    /**
+     * @title 签到
+     * @desc 签到
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/sign
+     * @header name:Authorization require:1 desc:Token
+     *
+     */
+    public function sign(){
+        parent::check_login();
+        $date = date('Y-m-d');
+        $log = Db::name('store_member_sign')->where('mid',$this->uid)->where('date',$date)->count();
+        if ($log) $this->error('今日已签到');
+        $data = [
+            'mid'=>$this->uid,
+            'date'=>$date
+        ];
+        $com = true;
+        Db::startTrans();
+        try {
+            $id = Db::name('store_member_sign')->insertGetId($data);
+            $sign_integral = getConfigValue('sign_integral');
+            if ($sign_integral>0){
+                memberMoneyChange($sign_integral,1,$this->uid,'签到',1,$id);
+            }
+            Db::commit();
+        }catch (\Exception $e){
+            $com=false;
+            Db::rollback();
+        }
+        if ($com){
+            $this->success('签到成功');
+        }
+        $this->error('签到失败');
+    }
+
+
+
+    /**
+     * @title 积分记录
+     * @desc 积分记录
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/integralLog
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:id type:int require:0 default:0 desc:id
+     * @return  name:change type:string require:0 default:0 desc:变动数值
+     * @return  name:title type:string require:0 default:0 desc:标题
+     * @return  name:create_at type:string require:0 default:0 desc:时间
+     */
+    public function integralLog(){
+        parent::check_login();
+        $where = [
+            'm_id'=>$this->uid,
+            'type'=>1
+        ];
+        $status = input('status');
+        $count = Db::name('store_member_log')
+            ->where($where)
+            ->when($status,function ($query) use ($status){
+                $pm = $status==1 ? 1 : 0;
+                $query->where('pm',$pm);
+            })
+            ->count();
+        $list = Db::name('store_member_log')
+            ->field('id,pm,change,title,create_at')
+            ->where($where)
+            ->when($status,function ($query) use ($status){
+                $pm = $status==1 ? 1 : 0;
+                $query->where('pm',$pm);
+            })
+            ->order('id desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            if ($v['pm']==1){
+                $v['change'] = '+'.$v['change'];
+            }else{
+                $v['change'] = '-'.$v['change'];
+            }
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+
+    /**
+     * @title 余额记录
+     * @desc 余额记录
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/moneyLog
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+     * @return  name:id type:int require:0 default:0 desc:id
+     * @return  name:change type:string require:0 default:0 desc:变动数值
+     * @return  name:title type:string require:0 default:0 desc:标题
+     * @return  name:create_at type:string require:0 default:0 desc:时间
+     */
+    public function moneyLog(){
+        parent::check_login();
+        $where = [
+            'm_id'=>$this->uid,
+            'type'=>3
+        ];
+        $count = Db::name('store_member_log')->where($where)->count();
+        $list = Db::name('store_member_log')
+            ->field('id,pm,change,title,create_at')
+            ->where($where)
+            ->order('id desc')
+            ->limit($this->off_set,$this->page_num)
+            ->select();
+        foreach ($list as &$v){
+            if ($v['pm']==1){
+                $v['change'] = '+'.$v['change'];
+            }else{
+                $v['change'] = '-'.$v['change'];
+            }
+        }
+        $this->success('成功',compact('count','list'));
+    }
+
+
+
+    /**
+     * @title 提现
+     * @desc 提现
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/withdraw
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:money type:int  : default: desc:提现金额
+     * @param  name:withdraw_type type:int  : default: desc:提现方式
+     *
+     */
+    public function withdraw(){
+        parent::check_login();
+        //redis原子锁
+        if (redisSetNx('withdraw'.$this->uid,3)){
+            $v = getConfigValue('withdraw_switch');
+            if (!$v) $this->error('维护中,暂时关闭');
+            $money = input('money');
+            $withdraw_type = input('withdraw_type','bank');
+            if (!$money) $this->error('参数错误');
+            $user = Db::name('store_member')->where('id',$this->uid)->find();
+            if ($user['money']<$money) $this->error('余额不足');
+            $withdraw_min_price = getConfigValue('withdraw_min_price');  //最小提现金额
+            $withdraw_max_price = getConfigValue('withdraw_max_price');  //最大提现金额
+            $poundage_proportion = getConfigValue('poundage_proportion');    //手续费百分比
+
+            if ($money < $withdraw_min_price)
+                $this->error('最低提现'.$withdraw_min_price.'元');
+            if ($money > $withdraw_max_price)
+                $this->error('最大提现'.$withdraw_min_price.'元');
+
+            if ($withdraw_type=='zfb'){
+                if (!$user['zfb_account'] || !$user['zfb_real_name']) $this->error('请先绑定支付宝');
+            }
+
+            if ($withdraw_type=='bank'){
+                if (!$user['withdraw_name'] || !$user['withdraw_bank_card']) $this->error('请先绑定提现银行卡');
+            }
+
+            $data['mid'] = $this->uid;
+            $order_no = get_order_sn();
+            $data['order_no'] = $order_no;
+            $data['price'] = $money;
+            $data['name'] = $user['withdraw_name'];
+            $data['bank_id_card'] = $user['withdraw_bank_card'];
+            $data['poundage_proportion'] = $poundage_proportion;
+            $proportion = sprintf("%.2f", $money*($poundage_proportion/100));    //四舍五入保留两位小数点
+            $data['proportion'] = $proportion;
+            $real_money = $money-$proportion;
+            $data['real_money'] = $real_money;
+            $data['withdraw_type'] = $withdraw_type;
+            $id =Db::name('store_member_withdraw')->insertGetId($data);
+            if ($id){
+                Db::startTrans();
+                try {
+                    //减少用户余额
+                    $res = memberMoneyChange($money,3,$this->uid,'余额提现',0,$id);
+                    if ($res){
+                        Db::commit();
+                        DelRedisSetNx('withdraw'.$this->uid);
+                        $this->success('提交成功,等待审核');
+                    }else{
+                        Db::rollback();
+                        DelRedisSetNx('withdraw'.$this->uid);
+                        $this->error('提现失败,请稍后重试');
+                    }
+                }catch (Exception $e) {
+                    Db::rollback();
+                    DelRedisSetNx('withdraw'.$this->uid);
+                    $this->error('提现失败,请稍后重试1');
+                }
+            }else{
+                DelRedisSetNx('withdraw'.$this->uid);
+                $this->error('提现失败,请稍后重试2');
+            }
+        }else{
+            $this->error('请求过快,请稍后重试');
+        }
+
+
+
+    }
+
+    /**
+     * @title 打开盲盒
+     * @desc 打开盲盒
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/openBoxCollection
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:id type:int  : default: desc:藏品ID
+     *
+     */
+    public function openBoxCollection(){
+        parent::check_login();
+        //redis原子锁
+        if (redisSetNx('openBox'.$this->uid,3)){
+            $id = input('id');
+            if (!$id) $this->error('参数错误');
+            $info = Db::name('store_order_info')
+                ->where('id',$id)
+                ->where('status','neq',2)
+                ->where('mid',$this->uid)
+                ->where('is_destruction',1)
+                ->find();
+            if (!$info) $this->error('盲盒不存在');
+            if ($info['resale_status']!=1) $this->error('盲盒挂售中或已出售');
+            $com = true;
+            Db::startTrans();
+            try {
+                //销毁
+                Db::name('store_order_info')->where('id',$id)->update(['is_destruction'=>0]);
+                checkBoxCollection($info['c_id']);
+                $prize_arr = Db::name('store_box')
+                    ->where('status',1)
+                    ->where('coll_id',$info['c_id'])
+                    ->where('is_del',1)
+                    ->field('id,rate')
+                    ->select();
+                foreach ($prize_arr as $key => $val) {
+                    $arr[$val['id']] = $val['rate'];//概率数组
+                }
+                $rid = get_rand($arr); //根据概率获取奖项id
+                $pr_info = Db::name('store_box')->where('id',$rid)->field('id,title,prize,is_prize')->find();
+                $return = [
+                    'is_prize'=>$pr_info['is_prize']
+                ];
+                $coll_info =  getCollectionInfoHash($pr_info['prize']);
+                if ($pr_info['is_prize']==1){
+                    $json_info = json_encode($coll_info,true);
+                    $return['name'] = $pr_info['title'];
+                    $return['cover'] = $coll_info['cover'];
+                }else{
+                    $json_info = '';
+                    $return['name'] = '未中奖';
+                    $return['cover'] = '';
+                }
+                $data = [
+                    'm_id'=>$this->uid,
+                    'b_id'=>$rid,
+                    'is_prize'=>$pr_info['is_prize'],
+                    'name'=>$return['name'],
+                    'cover'=>$return['cover'],
+                    'info'=>$json_info,
+                    'date'=>date('Y-m-d'),
+                    'is_free'=>1
+                ];
+                Db::name('store_blind_box_log')->insert($data);
+                if ($pr_info['is_prize']==1){
+                    //获取排名
+                    $rank = getRanking($pr_info['prize'])+1;
+                    $tag = getTag($pr_info['prize'],$rank,$coll_info['inventory']);
+                    saveRanking($pr_info['prize']);
+                    $is_nft = Db::name('hash2')->where('goods_id',$pr_info['prize'])->where('success',1)->find();
+                    $collectors_hash = '';
+                    $date = [
+                        'order_id'=>0,
+                        'order_no'=>get_order_sn(),
+                        'tag'=>$tag,
+                        'mid'=>$this->uid,
+                        'c_id'=>$pr_info['prize'],
+                        'name'=>$coll_info['name'],
+                        'cover'=>$coll_info['cover'],
+                        'pro_info'=>$json_info,
+                        'tokenid'=>$is_nft['class_id'],
+                        'nfttype'=>$is_nft['operationId'],
+                        'collectors_hash'=>$collectors_hash,
+                        'collectors_hash_time'=>'',
+                        'status'=>4
+                    ];
+                    Db::name('store_order_info')->insert($date);
+                }
+                Db::commit();
+            }catch (Exception $e) {
+                $com = false;
+                Db::rollback();
+            }
+            if ($com){
+                setMemberInfoHash($this->uid);
+                if ($pr_info['is_prize']==1){
+                    //减掉库存
+                    loseCollectionInventory($pr_info['prize'],1);
+                }
+                DelRedisSetNx('openBox'.$this->uid);
+                $this->success('成功',$return);
+            }else{
+                DelRedisSetNx('openBox'.$this->uid);
+                $this->error('开启失败,请稍后重试');
+            }
+        }else{
+            $this->error('请求过快');
+        }
+    }
+
+
+    /**
+     * @title 抢购卡列表
+     * @desc 抢购卡列表
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/snapLog
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param  name:page type:int  : default:1 desc:页数
+     * @param  name:page_num type:int  : default:20 desc:每页数
+     *
+
+     */
+    public function snapLog(){
+        parent::check_login();
+        checksnapcard($this->uid);
+        $status = input('status',1);
+        $list = Db::name('store_snap_card_log')
+            ->where('m_id',$this->uid)
+            ->where('is_use',$status)
+            ->select();
+        foreach ($list as &$v){
+            if ($v['type']==2){
+                $v['collection_name'] = Db::name('store_collection')->where('id',$v['collection_id'])->value('name');
+            }
+        }
+        $this->success('成功',$list);
+    }
+
+
+    /**
+     * @title 签约银行卡发送短信
+     * @desc 签约银行卡发送短信
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/bindCardsms
+     * @header name:Authorization require:1 desc:Token
+     */
+    public function bindCardsms(){
+        parent::check_login();
+        $name = input('name');
+        $id_card_no = input('id_card_no');
+        $bank_card_no = input('bank_card_no');
+        $mobile = input('mobile');
+        if (!$name || !$id_card_no || !$bank_card_no || !$mobile) $this->error('参数错误');
+        $check_id_card = isCreditNo($id_card_no);
+        if (!$check_id_card) $this->error('身份证号格式错误');
+        if (!checkbank($bank_card_no)) $this->error('银行卡号格式错误');
+        if (!Validate::regex($mobile, "^1\d{10}$")) $this->error('手机号格式错误');
+        $huiju = new HuijuKjPay();
+        $result = $huiju->sms($name,$id_card_no,$bank_card_no,$mobile);
+        if ($result['resp_codo']=='SUCCESS'){
+            if ($result['biz_code']=='JS000000'){
+                $data = json_decode($result['data'],true);
+                if ($data['order_status']=='P3000'){
+                    $this->success('发送成功',$result);
+                }else{
+                    $this->error($data['err_msg']);
+                }
+            }else{
+                $this->error($result['biz_msg']);
+            }
+        }else{
+            $this->error('发送失败');
+        }
+    }
+
+
+    /**
+     * @title 签约银行卡
+     * @desc 签约银行卡
+     * @author Gavin
+     * @method POST
+     * @tag 编辑信息
+     * @url /api/User_center/bindCard
+     * @header name:Authorization require:1 desc:Token
+     */
+    public function bindCard(){
+        parent::check_login();
+        $name = input('name');
+        $id_card_no = input('id_card_no');
+        $bank_card_no = input('bank_card_no');
+        $mobile = input('mobile');
+        $seckey = input('seckey');
+        $mch_order_no = input('mch_order_no');
+        $sms_code = input('sms_code');
+        if (!$name || !$id_card_no || !$bank_card_no || !$mobile || !$seckey || !$mch_order_no || !$sms_code) $this->error('参数错误');
+        $check_id_card = isCreditNo($id_card_no);
+        if (!$check_id_card) $this->error('身份证号格式错误');
+        if (!checkbank($bank_card_no)) $this->error('银行卡号格式错误');
+        if (!Validate::regex($mobile, "^1\d{10}$")) $this->error('手机号格式错误');
+        $huiju = new HuijuKjPay();
+        $result = $huiju->signing($seckey,$mch_order_no,$sms_code);
+        if ($result['resp_codo']=='SUCCESS'){
+            if ($result['biz_code']=='JS000000'){
+                $data = json_decode($result['data'],true);
+                if ($data['order_status']=='P1000'){
+                    $date = [
+                        'hj_pay_name'=>$name,
+                        'hj_pay_id_card'=>$id_card_no,
+                        'hj_pay_bank_card'=>$bank_card_no,
+                        'hj_pay_mobile'=>$mobile,
+                        'hj_pay_status'=>1,
+                        'hj_pay_time'=>date('Y-m-d H:i:s')
+                    ];
+                    if (Db::name('store_member')->where('id',$this->uid)->update($date)){
+                        setMemberInfoHash($this->uid);
+                        $this->success('签约成功',$result);
+                    }else{
+                        $this->error('签约失败');
+                    }
+                }else{
+                    $this->error($data['err_msg']);
+                }
+            }else{
+                $this->error($result['biz_msg']);
+            }
+        }else{
+            $this->error('签约失败');
+        }
+    }
+    /**
+     * @title 绑定提现银行卡
+     * @desc 绑定提现银行卡
+     * @author Gavin
+     * @url /api/User_center/bind_withdraw_card
+     * @method POST
+     * @tag 编辑信息
+     * @header name:Authorization require:1 desc:Token
+     *
+     * @param name:withdraw_name type:string require:0 default:-- desc:姓名
+     * @param name:withdraw_bank_card type:string require:0 default:-- desc:卡号
+     */
+    public function bind_withdraw_card(){
+        parent::check_login();
+        $this->check_login();
+        $withdraw_name = input('withdraw_name');
+        $withdraw_bank_card = input('withdraw_bank_card');
+        if (!$withdraw_name) $this->error('姓名为空');
+        if (!$withdraw_bank_card) $this->error('卡号为空');
+        if (!checkbank($withdraw_bank_card)) $this->error('银行卡号格式错误');
+        $update_data['withdraw_name'] = $withdraw_name;
+        $update_data['withdraw_bank_card'] = $withdraw_bank_card;
+        $update_data['update_at'] = date('Y-m-d H:i:s');
+        if (Db::name('store_member')->where('id',$this->uid)->update($update_data)){
+            setMemberInfoHash($this->uid);
+            $this->success('绑定成功');
+        }
+        $this->error('绑定失败');
+    }
+
+
+
+
+
+}

+ 74 - 0
application/api/controller/WeChart.php

@@ -0,0 +1,74 @@
+<?php
+namespace app\api\controller;
+use EasyWeChat\Factory;
+use  think\cache\driver\Redis;
+use  think\Session;
+class WeChart {
+
+    protected $app_id ='wx7a5f15ddeda9ef5a';
+    protected $app_secret = 'a4c33b42917949652c31447e42132de0';
+
+
+    /**
+     * 获取token
+     * @return mixed|string
+     */
+    public function  getAccessToken()
+    {
+        $session = new Session();
+        $over_time = $session->get('token_over_time');
+        if($over_time < time()) {
+            $url=  "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={$this->app_id}&secret={$this->app_secret}";
+            $ret = http_curl($url);
+            $session->set('access_token',$ret['access_token']);
+            $session->set('token_over_time',time() +$ret['expires_in'] );
+        }
+        return  $session->get('access_token');
+    }
+
+
+    public function send_message($order_sn=''){
+
+        $url='https://api.weixin.qq.com/cgi-bin/message/wxopen/template/uniform_send?access_token='.$access_token;
+        $order = Db::name('StoreOrder')->field('id,order_sn,create_at,num,take_time,user_id,goods_id')->where('order_sn',$order_sn)->find();
+        $member = Db::name('storeMember')->field('id,vip_level,openid,username,nickname')->where('id',$order['user_id'])->find();
+        $book_book = Db::name('book_book')->field('id,title')->where('id',$order['goods_id'])->find();
+        $data=[
+            'touser'=>$member['openid'],
+            'mp_template_msg'=>[
+                "appid"=>"wx3346d52f010b611d",
+                "template_id"=>"SH6KcRTVDiU0-7cSDujHsCg8MhbgzGpHb8US1QiPYW0",
+                "url"=>"http://www.baidu.com",
+                'miniprogram'=>[
+                    "appid"=>"wx2b03c73ff0547cc3",
+                ],
+                'data'=>[
+                    'first'=>[
+                        "value"=>'恭喜你已借阅到该图书',
+                        "color"=>"#173177"
+                    ],
+                    'keyword1'=>[
+                        "value"=>$book_book['title'],
+                        "color"=>"#173177"
+                    ],
+                    'keyword2'=>[
+                        "value"=>$member['username']?$member['username']:$member['nickname'],
+                        "color"=>"#173177"
+                    ],
+                    'keyword3'=>[
+                        "value"=>date('Y-m-d H:i:s',time()),
+                        "color"=>"#173177"
+                    ],
+                    'remark'=>[
+                        "value"=>'谢谢你对哩哈教育的支持!',
+                        "color"=>"#173177"
+                    ]
+                ]
+            ],
+        ];
+        $res = requestPost($url,json_encode($data));
+    }
+
+
+
+}

+ 2 - 0
config/app.php

@@ -22,4 +22,6 @@ return [
     'url_param_type' => 1,
     // 当前 ThinkAdmin 版本号
     'thinkadmin_ver' => 'v5',
+    'jwt'            => 'collect',
+    'jwt_time'       => 311040000,
 ];