Secondary.php 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720
  1. <?php
  2. namespace app\api\controller;
  3. use think\Db;
  4. use EasyWeChat\Factory;
  5. use think\cache\driver\Redis;
  6. use app\common\library\AliPay;
  7. use app\common\library\Shande;
  8. use app\api\service\CacheService;
  9. use app\index\enum\SecondaryEnum;
  10. use Alipay\EasySDK\Kernel\Util\ResponseChecker;
  11. use function EasyWeChat\Kernel\Support\get_client_ip;
  12. /**
  13. * @title 二级市场
  14. * @controller secondary
  15. * @package app\api\controller
  16. */
  17. class Secondary extends Base
  18. {
  19. public function initialize(){
  20. parent::initialize();
  21. parent::check_login();
  22. }
  23. /**
  24. * @title 藏品出售
  25. * @desc 藏品出售
  26. * @author Gavin
  27. * @url /api/Secondary/sell
  28. * @method POST
  29. * @header name:Authorization require:1 desc:Token
  30. *
  31. * @param name:id type:int require:1 default:1 desc:藏品ID
  32. * @param name:resale_price type:decimal require:1 default:1 desc:出售价格
  33. * @param name:second_password type:int require:1 default:-- desc:二级密码
  34. *
  35. */
  36. public function sell(){
  37. $this->checkSwitch(1);
  38. $id = input('id');
  39. $user = getMemberInfoHash($this->uid); //获取用户信息
  40. if ($user['is_auth']==0) $this->error('请先实名认证!');
  41. $resale_price = input('resale_price');
  42. $second_password = input('second_password');
  43. if (!$id || !$resale_price) $this->error('参数错误');
  44. if ($resale_price<'1') $this->error('最低1元');
  45. if ($resale_price>'99999') $this->error('最高99999元');
  46. $info = Db::name('store_order_info')
  47. ->where('mid',$this->uid)
  48. ->where('id',$id)
  49. ->find();
  50. if (!$info || $info['status']==2) $this->error('藏品不存在');
  51. if ($info['resale_status']!=1) $this->error('挂售状态错误');
  52. if ($user['second_password']!=md5($second_password)) $this->error('密码错误');
  53. $update_data = [
  54. 'resale_status'=>2,
  55. 'resale_time'=>date('Y-m-d H:i:s'),
  56. 'resale_price'=>$resale_price
  57. ];
  58. if (Db::name('store_order_info')->where('id',$id)->update($update_data)){
  59. $this->success('成功');
  60. }
  61. $this->error('失败');
  62. }
  63. /**
  64. * @title 藏品取消出售
  65. * @desc 藏品取消出售
  66. * @author Gavin
  67. * @url /api/Secondary/cancel_sell
  68. * @method POST
  69. * @header name:Authorization require:1 desc:Token
  70. *
  71. * @param name:id type:int require:1 default:1 desc:藏品ID
  72. */
  73. public function cancel_sell(){
  74. $this->checkSwitch(1);
  75. $id = input('id');
  76. if (!$id) $this->error('参数错误');
  77. $info = Db::name('store_order_info')
  78. ->where('mid',$this->uid)
  79. ->where('id',$id)
  80. ->find();
  81. if (!$info || $info['status']==2) $this->error('藏品不存在');
  82. if ($info['resale_status']!=2) $this->error('挂售状态错误');
  83. //判断是否有待支付订单
  84. $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
  85. if ($count) $this->error('支付中,无法取消');
  86. $update_data = [
  87. 'resale_status'=>1,
  88. ];
  89. if (Db::name('store_order_info')->where('id',$id)->update($update_data)){
  90. $this->success('成功');
  91. }
  92. $this->error('失败');
  93. }
  94. /**
  95. * @title 标签列表
  96. * @desc 标签列表
  97. * @author Gavin
  98. * @url /api/Secondary/label_list
  99. * @method POST
  100. * @header name:Authorization require:1 desc:Token
  101. */
  102. public function label_list(){
  103. $list = Db::name('store_collection')->whereNotIn('label','测试,测试勿拍')->group('label')->column('label');
  104. $list = array_merge(['全部'],$list);
  105. $this->success('成功',['label_list'=>$list]);
  106. }
  107. /**
  108. * @title 二级市场列表
  109. * @desc 二级市场列表
  110. * @author Gavin
  111. * @url /api/Secondary/sell_list
  112. * @method POST
  113. * @header name:Authorization require:1 desc:Token
  114. * @param name:page type:int : default:1 desc:页数
  115. * @param name:page_num type:int : default:20 desc:每页数
  116. *
  117. * @param name:keyword type:string require:0 default: desc:关键词
  118. * @param name:label type:string require:0 default: desc:标签
  119. * @param name:time_order type:string require:0 default: desc:时间排序(asc:正序desc:倒序)
  120. * @param name:price_order type:string require:0 default: desc:价格排序(asc:正序desc:倒序)
  121. *
  122. * @return name:name type:string require:0 default:0 desc:藏品名称
  123. * @return name:cover type:string require:0 default:0 desc:图片
  124. * @return name:member_name type:string require:0 default:0 desc:出售人名称
  125. * @return name:member_headimg type:string require:0 default:0 desc:出售人头像
  126. * @return name:resale_price type:string require:0 default:0 desc:出售价格
  127. */
  128. public function sell_list(){
  129. $keyword = input('keyword');
  130. $label = input('label');
  131. $time_order = input('time_order');
  132. $price_order = input('price_order');
  133. $count = Db::name('store_order_info')
  134. ->where('status','neq',2)
  135. ->where('resale_status',2)
  136. ->where('is_destruction',1)
  137. ->when($keyword,function ($query) use ($keyword){
  138. $query->whereLike('name','%'.$keyword.'%');
  139. })
  140. ->when($label,function ($query) use ($label){
  141. if ($label!='全部'){
  142. $ids = Db::name('store_collection')->where('label',$label)->column('id');
  143. $query->whereIn('c_id',$ids);
  144. }
  145. })
  146. ->count();
  147. $list = Db::name('store_order_info')
  148. ->where('status','neq',2)
  149. ->where('resale_status',2)
  150. ->where('is_destruction',1)
  151. ->when($keyword,function ($query) use ($keyword){
  152. $query->whereLike('name','%'.$keyword.'%');
  153. })
  154. ->when($label,function ($query) use ($label){
  155. if ($label!='全部'){
  156. $ids = Db::name('store_collection')->where('label',$label)->column('id');
  157. $query->whereIn('c_id',$ids);
  158. }
  159. })
  160. ->when($price_order,function ($query) use ($price_order){
  161. $query->order('resale_price '.$price_order);
  162. })
  163. ->when($time_order,function ($query) use ($time_order){
  164. $query->order('resale_time '.$time_order);
  165. })
  166. ->limit($this->off_set,$this->page_num)
  167. ->select();
  168. foreach ($list as &$v){
  169. $member = getMemberInfoHash($v['mid']); //获取用户信息
  170. $v['member_name'] = $member['name'];
  171. $v['member_headimg'] = $member['headimg'];
  172. $v['pro_info'] = json_decode($v['pro_info'],true);
  173. $count = Db::name('store_order_info_order')->where('info_id',$v['id'])->where('status',0)->count();
  174. $v['is_buy'] = $count>0 ? 1 : 0;
  175. }
  176. $this->success('成功',compact('count','list'));
  177. }
  178. /**
  179. * @title 二级市场详情
  180. * @desc 二级市场详情
  181. * @author Gavin
  182. * @url /api/Secondary/sell_list_detail
  183. * @method POST
  184. * @header name:Authorization require:1 desc:Token
  185. * @param name:id type:int : default: desc:id
  186. */
  187. public function sell_list_detail(){
  188. $id = input('id');
  189. if (!$id) $this->error('参数错误');
  190. $info = Db::name('store_order_info')
  191. ->where('status','neq',2)
  192. ->where('resale_status',2)
  193. ->where('is_destruction',1)
  194. ->where('id',$id)
  195. ->find();
  196. if (!$info) $this->error('藏品不存在');
  197. $member = getMemberInfoHash($info['mid']); //获取用户信息
  198. $info['member_name'] = $member['name'];
  199. $info['member_headimg'] = $member['headimg'];
  200. $info['pro_info'] = json_decode($info['pro_info'],true);
  201. //判断是否有待支付订单
  202. $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
  203. $info['is_buy'] = $count>0 ? 1 : 0;
  204. $this->success('成功',$info);
  205. }
  206. /**
  207. * @title 购买
  208. * @desc 购买
  209. * @author Gavin
  210. * @url /api/Secondary/createOrder
  211. * @method POST
  212. * @header name:Authorization require:1 desc:Token
  213. * @param name:id type:int require:1 default: desc:主键ID
  214. * @param name:pay_type type:string require:1 default:wx desc:wx:微信zfb:支付宝sd:杉德h5
  215. * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
  216. *
  217. * @return name:order_no type:int require:0 default:0 desc:订单号
  218. * @return name:pay type:string require:0 default:0 desc:支付信息
  219. */
  220. public function createOrder(){
  221. $this->checkSwitch(1);
  222. $redis = new Redis([ 'select'=> 2]);
  223. $redis_value = $redis->get('secondary_buy'.$this->uid);
  224. if ($redis_value){
  225. $this->error('请求过快,请稍后重试');
  226. }else{
  227. $redis->set('secondary_buy'.$this->uid,1,1);
  228. }
  229. $id = input('id');
  230. $pay_type = input('pay_type','wx');
  231. $from = input('from','wx');
  232. $this->checkSwitch(2,$pay_type);
  233. $user = getMemberInfoHash($this->uid); //获取用户信息
  234. if (!$id) $this->error('参数错误');
  235. if ($user['is_auth']==0) $this->error('请先实名认证!');
  236. $info = Db::name('store_order_info')->where('id',$id)->find();
  237. if (!$info) $this->error('藏品不存在');
  238. if ($info['resale_status']==3) $this->error('藏品已出售');
  239. if ($info['resale_status']==1) $this->error('藏品已撤销出售');
  240. if ($info['mid']==$this->uid) $this->error('不能购买自己出售的藏品');
  241. if (isset($user['buy_time']) && $user['buy_time']>date('Y-m-d H:i:s')) $this->error('一小时取消3次以上,4小时内禁止下单');
  242. //判断是否有未支付订单
  243. $order_count = Db::name('store_order_info_order')->where('mid',$this->uid)->where('status',0)->count();
  244. if ($order_count) $this->error('有未支付订单,无法下单');
  245. //判断是否有待支付订单
  246. $count = Db::name('store_order_info_order')->where('info_id',$id)->where('status',0)->count();
  247. if ($count) $this->error('支付中,无法下单');
  248. // 拿锁,防止超卖
  249. $lockKey = SecondaryEnum::CREATE_ORDER_KEY;
  250. $lock = CacheService::incr($lockKey);
  251. if ($lock > 1) {//没有拿到锁
  252. $this->error('藏品已出售');
  253. }
  254. CacheService::expire($lockKey, 5);
  255. $service_fee = getConfigValue('service_fee');
  256. $royalties = getConfigValue('royalties');
  257. $com = true;
  258. Db::startTrans();
  259. try {
  260. $order_no = get_order_sn();
  261. //获取价格
  262. $price = $info['resale_price'];
  263. $num = 1;
  264. $proportion = sprintf("%.2f",$price *($service_fee/100)); //四舍五入保留两位小数点
  265. $roya = sprintf("%.2f",$price *($royalties/100)); //四舍五入保留两位小数点
  266. $total_fee = bcmul($price,$num,2);
  267. $real_money = $price-$proportion-$roya;
  268. $data = [
  269. 'order_no'=>$order_no,
  270. 'mid'=>$this->uid,
  271. 'info_id'=>$id,
  272. 'num'=>$num,
  273. 'pro_info'=>json_encode($info,true),
  274. 'pay_price'=>$total_fee,
  275. 'service_fee'=>$service_fee,
  276. 'royalties'=>$royalties,
  277. 'to_account'=>$real_money,
  278. 'pay_type'=>$pay_type
  279. ];
  280. $id = Db::name('store_order_info_order')->insertGetId($data);
  281. $body = '纪元部落购买二级市场藏品';
  282. switch ($pay_type){
  283. case 'wx':
  284. $config = retrunWxConfig();
  285. $total_fee = $total_fee * 100;
  286. $config['notify_url'] = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/SecondaryWxOrderNotify';
  287. $app = Factory::payment($config);
  288. $post_data = [
  289. 'body' => $body,
  290. 'out_trade_no' => $order_no,
  291. 'total_fee' => $total_fee,
  292. 'attach'=>$this->uid, //自定义传值
  293. ];
  294. //trade_type SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
  295. if ($from=='wx'){
  296. $post_data['openid'] = $user['openid'];
  297. $post_data['trade_type'] = 'JSAPI';
  298. }elseif ($from=='h5'){
  299. $post_data['trade_type'] = 'MWEB';
  300. }
  301. $result = $app->order->unify($post_data);
  302. if ($result['return_msg']=='OK'){
  303. if ($result['result_code']=='FAIL'){
  304. $com = false;
  305. Db::rollback();
  306. }else{
  307. $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
  308. $retrun_data['order_no'] = $order_no;
  309. $retrun_data['id'] = $id;
  310. $retrun_data['pay'] = json_decode($order1,true);
  311. Db::commit();
  312. }
  313. }else{
  314. $com = false;
  315. Db::rollback();
  316. }
  317. break;
  318. case 'zfb':
  319. $zfb = new AliPay();
  320. $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/alipaySecondaryNotify';//回调地址
  321. $order = $zfb->ali_pay_pc($body, $total_fee, $order_no, $notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');//调用支付宝支付的方法
  322. $retrun_data['order_no'] = $order_no;
  323. $retrun_data['id'] = $id;
  324. $retrun_data['pay'] = $order;
  325. Db::commit();
  326. break;
  327. case 'sd':
  328. $client = new Shande();
  329. $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/shandeSecondaryNotify';//回调地址
  330. $total_fee = $total_fee*100;
  331. $lenth = strlen($total_fee);
  332. $total_fee = get0number($lenth).$total_fee;
  333. $result = $client->orderPay($order_no,$total_fee,$body,$notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');
  334. $retrun_data['order_no'] = $order_no;
  335. $retrun_data['id'] = $id;
  336. $retrun_data['pay'] = json_decode($result['data'],true);
  337. Db::commit();
  338. break;
  339. }
  340. }catch (\Exception $e){
  341. $com=false;
  342. Db::rollback();
  343. }
  344. if ($com){
  345. $this->success('成功',$retrun_data);
  346. }
  347. $this->error('失败,请稍后重试');
  348. }
  349. /**
  350. * @title 二级市场我的转售列表
  351. * @desc 二级市场我的转售列表
  352. * @author Gavin
  353. * @url /api/Secondary/my_sell_list
  354. * @method POST
  355. * @header name:Authorization require:1 desc:Token
  356. * @param name:page type:int : default:1 desc:页数
  357. * @param name:page_num type:int : default:20 desc:每页数
  358. *
  359. * @param name:keyword type:string require:0 default: desc:关键词
  360. * @param name:time_order type:string require:0 default: desc:时间排序(asc:正序desc:倒序)
  361. * @param name:price_order type:string require:0 default: desc:价格排序(asc:正序desc:倒序)
  362. *
  363. * @return name:name type:string require:0 default:0 desc:藏品名称
  364. * @return name:cover type:string require:0 default:0 desc:图片
  365. * @return name:member_name type:string require:0 default:0 desc:出售人名称
  366. * @return name:member_headimg type:string require:0 default:0 desc:出售人头像
  367. * @return name:resale_price type:string require:0 default:0 desc:出售价格
  368. */
  369. public function my_sell_list(){
  370. $keyword = input('keyword');
  371. $time_order = input('time_order');
  372. $price_order = input('price_order');
  373. $count = Db::name('store_order_info')
  374. ->where('mid',$this->uid)
  375. ->where('resale_status','neq',1)
  376. ->when($keyword,function ($query) use ($keyword){
  377. $query->whereLike('name','%'.$keyword.'%');
  378. })
  379. ->count();
  380. $list = Db::name('store_order_info')
  381. ->where('mid',$this->uid)
  382. ->where('resale_status','neq',1)
  383. ->when($keyword,function ($query) use ($keyword){
  384. $query->whereLike('name','%'.$keyword.'%');
  385. })
  386. ->when($price_order,function ($query) use ($price_order){
  387. $query->order('resale_price '.$price_order);
  388. })
  389. ->when($time_order,function ($query) use ($time_order){
  390. $query->order('resale_time '.$time_order);
  391. })
  392. ->order('resale_status asc')
  393. ->limit($this->off_set,$this->page_num)
  394. ->select();
  395. foreach ($list as &$v){
  396. $member = getMemberInfoHash($v['mid']); //获取用户信息
  397. $v['member_name'] = $member['name'];
  398. $v['member_headimg'] = $member['headimg'];
  399. $v['pro_info'] = json_decode($v['pro_info'],true);
  400. }
  401. $this->success('成功',compact('count','list'));
  402. }
  403. /**
  404. * @title 二级市场订单
  405. * @desc 二级市场订单
  406. * @author Gavin
  407. * @url /api/Secondary/secondaryOrderList
  408. * @method POST
  409. * @header name:Authorization require:1 desc:Token
  410. * @param name:page type:int : default:1 desc:页数
  411. * @param name:page_num type:int : default:20 desc:每页数
  412. *
  413. * @param name:status type:string require:0 default: desc:0:待支付1:已购买2:转售中3:已转售
  414. *
  415. * @return name:name type:string require:0 default:0 desc:藏品名称
  416. * @return name:cover type:string require:0 default:0 desc:图片
  417. * @return name:member_name type:string require:0 default:0 desc:出售人名称
  418. * @return name:member_headimg type:string require:0 default:0 desc:出售人头像
  419. * @return name:pay_price type:string require:0 default:0 desc:出售价格(已购买、待支付使用)
  420. * @return name:resale_price type:string require:0 default:0 desc:出售价格(转售中、已转售使用)
  421. */
  422. public function secondaryOrderList(){
  423. $status = input('status',0);
  424. switch ($status){
  425. case 0:case 1:
  426. $count = Db::name('store_order_info_order')
  427. ->where('mid',$this->uid)
  428. ->where('status',$status)
  429. ->count();
  430. $list = Db::name('store_order_info_order')
  431. ->where('mid',$this->uid)
  432. ->where('status',$status)
  433. ->limit($this->off_set,$this->page_num)
  434. ->select();
  435. //自动取消分钟数
  436. $cancel_time = getCancelTime();
  437. foreach ($list as &$v){
  438. $v['pro_info'] = json_decode($v['pro_info'],true);
  439. $member = getMemberInfoHash($v['mid']); //获取用户信息
  440. $v['member_name'] = $member['name'];
  441. $v['member_headimg'] = $member['headimg'];
  442. if ($v['status']==0){
  443. $v['cancel_time'] = date('Y-m-d H:i:s',strtotime($v['create_at'])+($cancel_time*60));
  444. }
  445. }
  446. break;
  447. case 2:case 3:
  448. $count = Db::name('store_order_info')
  449. ->where('mid',$this->uid)
  450. ->where('resale_status',$status)
  451. ->where('is_destruction',1)
  452. ->count();
  453. $list = Db::name('store_order_info')
  454. ->where('mid',$this->uid)
  455. ->where('resale_status',$status)
  456. ->where('is_destruction',1)
  457. ->limit($this->off_set,$this->page_num)
  458. ->select();
  459. foreach ($list as &$v){
  460. $v['pro_info'] = json_decode($v['pro_info'],true);
  461. $member = getMemberInfoHash($v['mid']); //获取用户信息
  462. $v['member_name'] = $member['name'];
  463. $v['member_headimg'] = $member['headimg'];
  464. }
  465. break;
  466. }
  467. $this->success('成功',compact('count','list'));
  468. }
  469. /**
  470. * @title 取消订单
  471. * @desc 取消订单
  472. * @author Gavin
  473. * @url /api/Secondary/cancelOrder
  474. * @method POST
  475. * @header name:Authorization require:1 desc:Token
  476. * @param name:order_no type:string require:1 default:-- desc:订单号
  477. */
  478. public function cancelOrder(){
  479. $order_no = input('order_no');
  480. if (!$order_no) $this->error('参数错误');
  481. $order = Db::name('store_order_info_order')
  482. ->where('order_no',$order_no)
  483. ->where('mid',$this->uid)
  484. ->find();
  485. if (!$order) $this->error('订单不存在');
  486. if ($order['status']!=0) $this->error('订单已支付或已取消');
  487. $com = true;
  488. Db::startTrans();
  489. try {
  490. $up_data = [
  491. 'status'=>2,
  492. 'cancel_at'=>date('Y-m-d H:i:s'),
  493. 'cancel_state'=>2
  494. ];
  495. Db::name('store_order_info_order')->where('order_no',$order_no)->update($up_data);
  496. $cancle = [
  497. 'mid'=>$this->uid,
  498. 'order_id'=>$order['id']
  499. ];
  500. Db::name('store_order_info_cancel_log')->insert($cancle);
  501. $time = date('Y-m-d H:i:s',time()-(60*60));
  502. $count = Db::name('store_order_info_cancel_log')->where('mid',$this->uid)->where('create_at','gt',$time)->count();
  503. if ($count>2){
  504. $buy_time = date('Y-m-d H:i:s',time()+(4*60*60));
  505. Db::name('store_member')->where('id',$this->uid)->update(['buy_time'=>$buy_time]);
  506. }
  507. Db::commit();
  508. }catch (\Exception $e){
  509. $com=false;
  510. Db::rollback();
  511. }
  512. if ($com){
  513. setMemberInfoHash($this->uid);
  514. $this->success('取消成功');
  515. }else{
  516. $this->error('取消失败,请稍后重试');
  517. }
  518. }
  519. /**
  520. * @title 待支付订单支付
  521. * @desc 待支付订单支付
  522. * @author Gavin
  523. * @url /api/Secondary/payOrder
  524. * @method POST
  525. * @header name:Authorization require:1 desc:Token
  526. * @param name:order_no type:string require:1 default:-- desc:订单号
  527. * @param name:from type:string require:1 default:wx desc:wx:微信公众号h5:网页
  528. *
  529. * @return name:order_no type:int require:0 default:0 desc:订单号
  530. * @return name:pay type:string require:0 default:0 desc:支付信息
  531. */
  532. public function payOrder(){
  533. $this->checkSwitch(1);
  534. $user = getMemberInfoHash($this->uid); //获取用户信息
  535. $order_no = input('order_no'); //订单号
  536. $id = input('id');
  537. $from = input('from','wx');
  538. if (!$order_no) $this->error('参数错误');
  539. if (!$id) $this->error('参数错误');
  540. $order = Db::name('store_order_info_order')
  541. ->where('id',$id)
  542. ->where('mid',$this->uid)
  543. ->find();
  544. $order_no = $order['order_no'];
  545. $pay_type = $order['pay_type'];
  546. $this->checkSwitch(2,$pay_type);
  547. if (!$order) $this->error('订单不存在');
  548. if ($order['status']!=0) $this->error('订单已支付或已取消');
  549. $info = Db::name('store_order_info')->where('id',$order['info_id'])->find();
  550. if ($info['resale_status']!=2) $this->error('藏品已出售或已撤销出售');
  551. $com = true;
  552. Db::startTrans();
  553. try {
  554. //获取价格
  555. $total_fee = $order['pay_price'];
  556. $body = '纪元部落购买二级市场藏品';
  557. switch ($pay_type){
  558. case 'wx':
  559. $config = retrunWxConfig();
  560. $total_fee = $total_fee * 100;
  561. $config['notify_url'] = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/SecondaryWxOrderNotify';
  562. $app = Factory::payment($config);
  563. $post_data = [
  564. 'body' => $body,
  565. 'out_trade_no' => $order_no,
  566. 'total_fee' => $total_fee,
  567. 'attach'=>$this->uid, //自定义传值
  568. ];
  569. //trade_type SAPI--JSAPI支付(或小程序支付)、NATIVE--Native支付、APP--app支付,MWEB--H5支付
  570. if ($from=='wx'){
  571. $post_data['openid'] = $user['openid'];
  572. $post_data['trade_type'] = 'JSAPI';
  573. }elseif ($from=='h5'){
  574. $post_data['trade_type'] = 'MWEB';
  575. }
  576. $result = $app->order->unify($post_data);
  577. if ($result['return_msg']=='OK'){
  578. if ($result['result_code']=='FAIL'){
  579. $com = false;
  580. Db::rollback();
  581. }else{
  582. $order1 = $app->jssdk->bridgeConfig($result['prepay_id']);//执行二次签名返回参数
  583. $retrun_data['order_no'] = $order_no;
  584. $retrun_data['id'] = $order['id'];
  585. $retrun_data['pay'] = json_decode($order1,true);
  586. Db::commit();
  587. }
  588. }else{
  589. $com = false;
  590. Db::rollback();
  591. }
  592. break;
  593. case 'zfb':
  594. $zfb = new AliPay();
  595. $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/alipaySecondaryNotify';//回调地址
  596. $order = $zfb->ali_pay_pc($body, $total_fee, $order_no, $notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');//调用支付宝支付的方法
  597. $retrun_data['order_no'] = $order_no;
  598. $retrun_data['id'] = $order['id'];
  599. $retrun_data['pay'] = $order;
  600. Db::commit();
  601. break;
  602. case 'sd':
  603. $client = new Shande();
  604. $notify_url = 'https://'.$_SERVER['SERVER_NAME'].'/api/Pay/shandeSecondaryNotify';//回调地址
  605. $total_fee = $total_fee*100;
  606. $lenth = strlen($total_fee);
  607. $total_fee = get0number($lenth).$total_fee;
  608. $order_nos = get_order_sn();
  609. if (Db::name('store_order_info_order')
  610. ->where('order_no',$order_no)
  611. ->where('mid',$this->uid)
  612. ->update(['order_no'=>$order_nos])){
  613. $result = $client->orderPay($order_nos,$total_fee,$body,$notify_url,'https://'.$_SERVER['SERVER_NAME'].'/web/h5/pages/shop/order');
  614. $retrun_data['order_no'] = $order_nos;
  615. $retrun_data['id'] = $order['id'];
  616. $retrun_data['pay'] = json_decode($result['data'],true);
  617. Db::commit();
  618. }else{
  619. $com=false;
  620. Db::rollback();
  621. }
  622. break;
  623. }
  624. }catch (\Exception $e){
  625. $com=false;
  626. Db::rollback();
  627. }
  628. if ($com){
  629. $this->success('成功',$retrun_data);
  630. }
  631. $this->error('失败,请稍后重试');
  632. }
  633. /**
  634. * @param 判断开关
  635. * @param string $pay_type
  636. * @return bool
  637. * @throws \think\db\exception\DataNotFoundException
  638. * @throws \think\db\exception\ModelNotFoundException
  639. * @throws \think\exception\DbException
  640. */
  641. public function checkSwitch($type,$pay_type=''){
  642. if ($type==1){
  643. $v = getConfigValue('secondary_sell_switch');
  644. if (!$v) $this->error('维护中,暂时关闭');
  645. }elseif ($type==2){
  646. $nameArray = ['secondary_wx_switch','secondary_zfb_switch','secondary_sd_switch'];
  647. $values = getConfig($nameArray);
  648. if ($pay_type=='wx'){
  649. if (!$values['secondary_wx_switch']) $this->error('微信支付暂时关闭');
  650. }elseif ($pay_type=='zfb'){
  651. if (!$values['secondary_zfb_switch']) $this->error('支付宝支付暂时关闭');
  652. }elseif ($pay_type=='sd'){
  653. if (!$values['secondary_sd_switch']) $this->error('杉德支付暂时关闭');
  654. }
  655. }
  656. return true;
  657. }
  658. }