DouYinService.php 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001
  1. <?php
  2. namespace app\api\service;
  3. use app\adminapi\logic\external\ExternalConsultationLogic;
  4. use app\api\logic\ServiceOrderLogic;
  5. use app\common\enum\PayEnum;
  6. use app\common\logic\PayNotifyLogic;
  7. use app\common\model\Config;
  8. use app\common\model\external\DouyinOrder;
  9. use app\common\model\external\DouyinRefundOrder;
  10. use app\common\model\external\DouyinUserAuth;
  11. use app\common\model\external\ExternalConsultation;
  12. use app\common\model\external\ExternalConsultationOrder;
  13. use app\common\model\external\ExternalPlatformGoods;
  14. use app\common\model\goods\Goods;
  15. use app\common\model\goods_category\GoodsCategory;
  16. use app\common\model\recharge\RechargeOrder;
  17. use app\common\model\user\User;
  18. use app\common\model\user\UserAuth;
  19. use app\common\model\works\ServiceWork;
  20. use app\common\service\ConfigService;
  21. use app\common\service\FileService;
  22. use GuzzleHttp\Client;
  23. use think\facade\Db;
  24. use think\facade\Log;
  25. class DouYinService
  26. {
  27. protected static int $terminal = \app\common\enum\user\UserTerminalEnum::DOUYIN;
  28. protected static int $external_platform_id = 6;
  29. protected CONST EXTERNAL_PLATFORM_ID = 6;
  30. // ********************************* 注册登录
  31. public static function register(array $params)
  32. {
  33. $userSn = User::createUserSn();
  34. $params['password'] = !empty($params['password'])?$params['password']:rand(100000,999999);
  35. $passwordSalt = \think\facade\Config::get('project.unique_identification');
  36. $password = create_password($params['password'], $passwordSalt);
  37. $avatar = ConfigService::get('default_image', 'user_avatar');
  38. $user = User::create([
  39. 'sn' => $userSn,
  40. 'avatar' => $avatar,
  41. 'nickname' => '用户' . $userSn,
  42. 'account' => $params['account'],
  43. 'mobile' => !empty($params['mobile'])?$params['mobile']:'',
  44. 'password' => $password,
  45. 'channel' => self::$terminal,
  46. 'user_type' => $params['user_type']??0,
  47. ]);
  48. return $user;
  49. }
  50. public static function phoneLogin(array $params)
  51. {
  52. try {
  53. $where = ['mobile' => $params['mobile']];
  54. $params['account'] = $params['mobile'];
  55. $user = User::where($where)->findOrEmpty();
  56. if ($user->isEmpty()) {
  57. //直接注册用户
  58. $params['channel'] = self::$terminal;
  59. $user = self::register($params);
  60. }
  61. //更新登录信息
  62. $user->login_time = time();
  63. $user->login_ip = request()->ip();
  64. $user->save();
  65. $userInfo = UserTokenService::setToken($user->id, self::$terminal);
  66. //返回登录信息
  67. $avatar = $user->avatar ?: Config::get('project.default_image.user_avatar');
  68. $avatar = FileService::getFileUrl($avatar);
  69. return [
  70. 'nickname' => $userInfo['nickname'],
  71. 'sn' => $userInfo['sn'],
  72. 'mobile' => $userInfo['mobile'],
  73. 'avatar' => $avatar,
  74. 'token' => $userInfo['token'],
  75. ];
  76. } catch (\Exception $e) {
  77. throw new \Exception($e->getMessage());
  78. }
  79. }
  80. public static function getDouyinUserByOpenId(array $openId)
  81. {
  82. try {
  83. $user = DouyinUserAuth::where('openid',$openId)->findOrEmpty();
  84. if ($user->isEmpty()) {
  85. //直接注册用户
  86. $params['channel'] = self::$terminal;
  87. $user = self::register($params);
  88. }
  89. //更新登录信息
  90. $user->login_time = time();
  91. $user->login_ip = request()->ip();
  92. $user->save();
  93. $userInfo = UserTokenService::setToken($user->id, self::$terminal);
  94. //返回登录信息
  95. $avatar = $user->avatar ?: Config::get('project.default_image.user_avatar');
  96. $avatar = FileService::getFileUrl($avatar);
  97. return [
  98. 'nickname' => $userInfo['nickname'],
  99. 'sn' => $userInfo['sn'],
  100. 'mobile' => $userInfo['mobile'],
  101. 'avatar' => $avatar,
  102. 'token' => $userInfo['token'],
  103. ];
  104. } catch (\Exception $e) {
  105. throw new \Exception($e->getMessage());
  106. }
  107. }
  108. // **************************** 商品管理 goods_category_id goods_id external_platform_id
  109. public static function addProduct($params)
  110. {
  111. $send_url = env('internal_api.api_url_host').'platf/dou_yin/addGoods';
  112. $res = http_request($send_url,http_build_query($params));
  113. Log::info('addProduct:'
  114. .'url:'.$send_url
  115. .'|data:'.json_encode($params,JSON_UNESCAPED_UNICODE)
  116. .'|res:'.json_encode([$res],JSON_UNESCAPED_UNICODE)
  117. );
  118. return $res?:[];
  119. }
  120. // ******************************** 订单业务
  121. public static function getOrderDetail($params)
  122. {
  123. //抖音订单信息/商品信息/预约信息(地址、时间、履约状态与信息)
  124. // $params['order_number'] user_id
  125. $order = DouyinOrder::with(['goods','serviceWork','douyinRefundOrder'])->where('order_number', $params['order_number'])->where('user_id', $params['user_id'])->findOrEmpty();
  126. if($order->isEmpty()){
  127. return [];
  128. }
  129. $orderInfo = $order->toArray();
  130. empty($orderInfo['goods']) && $orderInfo['goods'] = [];
  131. empty($orderInfo['serviceWork']) && $orderInfo['serviceWork'] = [];
  132. if($orderInfo['serviceWork'] && $orderInfo['serviceWork']['service_status'] == 3 && in_array($orderInfo['order_status'],[1,2])){
  133. $order->order_status = 4;
  134. $order->pay_status = 1;
  135. $order->save();
  136. }
  137. empty($orderInfo['douyinRefundOrder']) && $orderInfo['douyinRefundOrder'] = [];
  138. $orderInfo['book_info'] = json_decode($orderInfo['book_info']?:'{}',true);
  139. $rechargeOrder = RechargeOrder::where(['work_id'=>$orderInfo['work_id']?:0,'payment_type'=>2])->findOrEmpty();
  140. if($rechargeOrder->isEmpty()){
  141. $orderInfo['tail_order'] = [];
  142. }else{
  143. $orderInfo['tail_order'] = $rechargeOrder->toArray();
  144. // 尾款未支付时 展示尾单信息加入临时总金额用于展示
  145. if(isset($orderInfo['tail_order']['pay_status']) && $orderInfo['tail_order']['pay_status'] == 0){
  146. $orderInfo['total_amount'] += $orderInfo['tail_order']['order_amount'];
  147. }
  148. }
  149. $work_status = $orderInfo['serviceWork']['work_status']??0;
  150. $performance = [];
  151. // tmp
  152. switch ($work_status){
  153. case 0:
  154. $performance[] = ['status' => '待派单','title' => '待派单','time' => date('Y-m-d H:i:s',time())];
  155. break;
  156. case 1:
  157. case 2:
  158. case 3:
  159. $performance[] = ['status' => '待派单','title' => '待派单','time' => date('Y-m-d H:i:s',time())];
  160. $performance[] = ['status' => '已派单','title' => '已派单','time' => date('Y-m-d H:i:s',time())];
  161. break;
  162. case 4:
  163. case 5:
  164. case 6:
  165. $performance[] = ['status' => '待派单','title' => '待派单','time' => date('Y-m-d H:i:s',time())];
  166. $performance[] = ['status' => '已派单','title' => '已派单','time' => date('Y-m-d H:i:s',time())];
  167. $performance[] = ['status' => '服务中','title' => '服务中','time' => date('Y-m-d H:i:s',time())];
  168. break;
  169. case 7:
  170. case 8:
  171. $performance[] = ['status' => '待派单','title' => '待派单','time' => date('Y-m-d H:i:s',time())];
  172. $performance[] = ['status' => '已派单','title' => '已派单','time' => date('Y-m-d H:i:s',time())];
  173. $performance[] = ['status' => '服务中','title' => '服务中','time' => date('Y-m-d H:i:s',time())];
  174. $performance[] = ['status' => '已完结','title' => '已完结','time' => date('Y-m-d H:i:s',time())];
  175. break;
  176. }
  177. $orderInfo['performance'] = $performance;
  178. return $orderInfo;
  179. }
  180. public static function createOrder($params)
  181. {
  182. try {
  183. $goods_id = $params['goods_id']??0;
  184. $open_id = '_000o7ntqTR--_hCTBOBCSR_NJkyp_hiqlEK';
  185. $goods = Goods::where('id',$goods_id)->findOrEmpty();
  186. if($goods->isEmpty()){
  187. throw new \Exception('商品不存在!');
  188. }
  189. $goods = $goods->toArray();
  190. $platformGoods = ExternalPlatformGoods::where('goods_id', $goods_id)->where('external_platform_id', self::EXTERNAL_PLATFORM_ID)->findOrEmpty();
  191. if($platformGoods->isEmpty()){
  192. throw new \Exception('外部商品不存在!');
  193. }
  194. $platformGoods = $platformGoods->toArray();
  195. $quantity = $params['quantity']?:1;
  196. $appointment_time = strtotime($params['appointment_time']??date('Y-m-d H:i:s',time()));
  197. $bookStartTime = $appointment_time * 1000;
  198. $bookEndTime = ($appointment_time + (2 * 86400)) * 1000;
  199. $order_number = self::submitOrder([
  200. 'open_id'=>$open_id,
  201. 'goods_id'=>$goods_id,
  202. 'user_id'=>$params['user_id']??0,
  203. 'mobile'=>$params['user_info']['mobile']??'',
  204. 'quantity'=>$quantity
  205. ]);
  206. $data = [
  207. "total_amount" => $platformGoods['service_fee'] * $quantity * 100,
  208. "open_id" => $open_id,
  209. "out_order_no" => $order_number,
  210. "order_entry_schema" => ['path'=>'pages/order/detail','params'=>json_encode(['order_number'=>$order_number??0])],
  211. "cp_extra" => json_encode([
  212. "outShopId" => self::EXTERNAL_PLATFORM_ID,
  213. "skuId" => (string)$platformGoods['external_goods_sn'],
  214. "quantity" => $quantity,
  215. "user_id" => $params['user_id'],
  216. "user_address" => $params['user_address'],
  217. "lon" => $params['lon'],
  218. "lat" => $params['lat'],
  219. "appointment_time" => $params['appointment_time']
  220. ]),
  221. "goods_list" => [
  222. [
  223. "goods_id_type" => 1,
  224. "goods_id" => (string)$platformGoods['external_goods_sn'],
  225. "quantity" => 1
  226. ]
  227. ],
  228. "goods_book_info" => [
  229. "book_type" => 2,
  230. "cancel_policy" => 3,
  231. "cancel_advance_hour" => 5
  232. ],
  233. ];
  234. // 服务器向抖音发起创建订单
  235. $url = 'api/apps/trade/v2/order/create_order';
  236. $resData = self::toDyRequestUrl($url,$data);
  237. return $resData;
  238. } catch (\Exception $e) {
  239. throw new \Exception($e->getMessage());
  240. }
  241. }
  242. // 预下单接口 - 前端 首次/尾款
  243. public static function getPluginCreateOrderData($goods_id, $quantity, $douyinOrderId,$params)
  244. {
  245. try {
  246. $goods = Goods::where('id',$goods_id)->findOrEmpty();
  247. if($goods->isEmpty()){
  248. throw new \Exception('商品不存在!');
  249. }
  250. $goods = $goods->toArray();
  251. $platformGoods = ExternalPlatformGoods::where('goods_id', $goods_id)->where('external_platform_id', self::EXTERNAL_PLATFORM_ID)->findOrEmpty();
  252. if($platformGoods->isEmpty()){
  253. throw new \Exception('外部商品不存在!');
  254. }
  255. $platformGoods = $platformGoods->toArray();
  256. $quantity = $quantity?:1;
  257. $appointment_time = strtotime($params['appointment_time']??date('Y-m-d H:i:s',time()));
  258. $bookStartTime = $appointment_time * 1000;
  259. $bookEndTime = ($appointment_time + (2 * 86400)) * 1000;
  260. $data = [
  261. /*"goodsList" => [
  262. "quantity" => $quantity,
  263. "price" => $platformGoods['service_fee'] * 100,
  264. "goodsName" => $goods['goods_name'],
  265. "goodsPhoto" => $goods['goods_image']??'',
  266. "goodsId" => '',
  267. "goodsType" => 1
  268. ],*/
  269. "skuList" => [
  270. [
  271. "quantity" => (int)$quantity,
  272. "skuId" => (string)$platformGoods['external_goods_sn'],
  273. "skuType" => 1, // 1:商品库商品 2:非商品库商品(融合预约品走加价时,固定传2)
  274. "price" => $platformGoods['service_fee'] * 100,
  275. ]
  276. ],
  277. "bookInfo" => [
  278. "itemBookInfoList"=>[
  279. [
  280. "poiId" => '7511543640776017961',
  281. "shopName" => '亿蜂快修·武汉市',
  282. "outShopId" => (string)self::EXTERNAL_PLATFORM_ID,
  283. "skuId" => (string)$platformGoods['external_goods_sn'],
  284. "bookStartTime" => $bookStartTime?:'',
  285. "bookEndTime" => $bookEndTime?:'',
  286. ]
  287. ]
  288. ],
  289. "payment" => [
  290. "totalAmount" => $quantity * $platformGoods['service_fee'] * 100,
  291. ],
  292. //"callbackUrl" => $callbackUrl?:'',
  293. "callbackData" => [
  294. "outShopId" => self::EXTERNAL_PLATFORM_ID,
  295. "skuId" => (string)$platformGoods['external_goods_sn'],
  296. "quantity" => $quantity,
  297. "user_id" => $params['user_id'],
  298. "douyinOrderId" => (string)$douyinOrderId?:'',
  299. "user_name" => $params['user_name']??'',
  300. "mobile" => $params['mobile']??'',
  301. "user_address" => $params['user_address']??'',
  302. "lon" => $params['lon']??'',
  303. "lat" => $params['lat']??'',
  304. "appointment_time" => $params['appointment_time']??''
  305. ],
  306. /*"tradeOption" => json_encode([
  307. "life_trade_flag" => 1,
  308. "order_relation_info" => [
  309. "related_order_id" => 0, // 加价时上个订单号
  310. "relation_type" => 'multi_buy_as_one'
  311. ]
  312. ])*/
  313. ];
  314. if($douyinOrderId){ // 说明是来自首单订单即要创建尾款
  315. unset($data['skuList']);
  316. unset($data['bookInfo']);
  317. unset($data['payment']);
  318. $douyinOrder = DouyinOrder::where(['user_id'=>$params['user_id'],'dy_order_id'=>$douyinOrderId])->findOrEmpty();
  319. $rechargeOrder = RechargeOrder::where(['work_id'=>$douyinOrder->work_id,'payment_type'=>2])->findOrEmpty();
  320. if(empty($rechargeOrder->order_amount)) throw new \Exception('尾款报价不存在!');
  321. $data['skuList'] = [
  322. [
  323. "quantity" => (int)$quantity,
  324. "skuId" => (string)$goods_id,
  325. "skuType" => 2, // 1:商品库商品 2:非商品库商品(融合预约品走加价时,固定传2)
  326. "price" => $rechargeOrder->order_amount * 100,
  327. "goodsInfo" => [
  328. "goodsName"=>$goods['goods_name'],
  329. "goodsPhoto"=>$goods['goods_image'], // 商品图片链接 必填
  330. "goodsId"=> (string)$goods_id,
  331. "goodsType"=>2
  332. ],
  333. "extraInfo" =>["feeType"=>14]
  334. ]
  335. ];
  336. $data['payment'] = [
  337. "totalAmount" => $rechargeOrder->order_amount * 100,
  338. ];
  339. //$data['callbackUrl'] = env('douyin.pay_tail_notify_url')??'';
  340. $bookInfo = json_decode($douyinOrder->book_info,true);
  341. $item_order_id = $bookInfo['item_order_info_list'][0]['item_order_id']??$douyinOrderId;
  342. $data['tradeOption'] = [
  343. "life_trade_flag" => 1,
  344. "trade_mode" => '3',
  345. "order_relation_info" => [
  346. "related_order_id" => (string)$item_order_id, // 加价时上个订单号
  347. "relation_type" => 'multi_buy_as_one'
  348. ]
  349. ];
  350. }
  351. /*else{ // 创建首单订单
  352. $data['callbackUrl'] = '';
  353. }*/
  354. return $data;
  355. } catch (\Exception $e) {
  356. throw new \Exception($e->getMessage());
  357. }
  358. }
  359. // ++++++++++++++++++++ 首次/尾款 扩展点
  360. /**
  361. * 预下单扩展点
  362. * @param $params
  363. * @throws \Exception
  364. * @author liugc <466014217@qq.com>
  365. * @date 2025/6/4 14:03
  366. */
  367. public static function submitOrderNotify($params = [])
  368. {
  369. try {
  370. $params['external_platform_id'] = self::EXTERNAL_PLATFORM_ID;
  371. // order_id goods total_amount discount cp_extra create_order_time phone_num contact_name open_id
  372. $params['cp_extra'] = json_decode($params['cp_extra'], true);
  373. $user_id = $params['cp_extra']['user_id'];
  374. $user = User::where('id',$user_id)->findOrEmpty();
  375. if($params['cp_extra']['douyinOrderId']>0){ // 说明是尾款单
  376. // 创建尾款单
  377. $order_number = self::tailOrder([
  378. 'order_id'=> $params['order_id'],
  379. 'douyinOrderId'=>$params['cp_extra']['douyinOrderId'],
  380. 'user_id'=>$user_id??0,
  381. 'total_amount'=>$params['total_amount'],
  382. 'discount_amount'=>$params['discount_amount']??0,
  383. ]);
  384. $path_order_number = DouyinOrder::where(['dy_order_id'=>$params['cp_extra']['douyinOrderId']])->value('order_number')??0;
  385. $payNotifyUrl = 'https://weixiudev.kyjlkj.com/api/dou_yin/payTailNotify';
  386. }else{
  387. // 创建首单 goods_id user_info.mobile user_id quantity
  388. $params['cp_extra']['open_book_info'] = $params['open_book_info']??[];
  389. $params['cp_extra']['item_order_info_list'] = $params['item_order_info_list']??[];
  390. $order_number = self::submitOrder([
  391. 'open_id'=>$params['open_id'],
  392. 'order_id'=>$params['order_id'], // 抖音订单号
  393. 'goods_id'=>$params['cp_extra']['skuId'],
  394. 'user_id'=>$user_id??0,
  395. 'mobile'=>$user['mobile']??'',
  396. 'quantity'=>$params['cp_extra']['quantity'],
  397. 'book_info'=>$params['cp_extra'],
  398. ]);
  399. $path_order_number = $order_number;
  400. $payNotifyUrl = 'https://weixiudev.kyjlkj.com/api/dou_yin/payNotify';
  401. }
  402. return [
  403. "out_order_no" => $order_number,
  404. "order_entry_schema" => [
  405. "path" => "pages/order/detail",
  406. "params" => json_encode(['order_number' => $path_order_number])
  407. ],
  408. "pay_notify_url" => $payNotifyUrl
  409. ];
  410. } catch (\Exception $e) {
  411. throw new \Exception($e->getMessage());
  412. }
  413. }
  414. // ++++++++++++++++++++ 首次创单
  415. /**
  416. * 预下单扩展点-子
  417. * @param array $params goods_id user_info.mobile user_id quantity
  418. * @return array|false
  419. */
  420. public static function submitOrder($params)
  421. {
  422. Db::startTrans();
  423. try {
  424. $platformGoods = ExternalPlatformGoods::where('external_goods_sn', $params['goods_id'])->where('external_platform_id', self::EXTERNAL_PLATFORM_ID)->findOrEmpty();
  425. if($platformGoods->isEmpty()){
  426. throw new \Exception('产品不存在!');
  427. }
  428. $goods = Goods::findOrEmpty($platformGoods['goods_id']);
  429. if($goods->isEmpty()){
  430. throw new \Exception('产品不存在!');
  431. }
  432. if(empty($params['mobile'])){
  433. throw new \Exception('请先补充您的联系方式后在提交订单');
  434. }
  435. // TODO tmp防抖1m
  436. $isExist = DouyinOrder::where(['user_id'=>$params['user_id'],'goods_id'=>$goods['id']])->where('create_time','>',(time() - 60))->findOrEmpty();
  437. if(!$isExist->isEmpty()){
  438. //throw new \Exception('请勿重复下单!');
  439. }
  440. $quantity = $params['quantity']??1;
  441. //生成订单
  442. $create_data = [
  443. 'user_id' => $params['user_id'],
  444. 'mobile' => $params['mobile'],
  445. 'open_id' => $params['open_id'],
  446. 'goods_id'=>$goods['id'],
  447. 'title' => $goods['goods_name'],
  448. 'book_info' => json_encode($params['book_info']??[]),
  449. 'unit_price' => $platformGoods['service_fee'],
  450. 'quantity' => $quantity,
  451. 'total_amount' => $platformGoods['service_fee'] * $quantity,
  452. 'dy_order_id' => $params['order_id']??'',
  453. 'order_number' => generate_sn(DouyinOrder::class, 'order_number'),
  454. ];
  455. $order = DouyinOrder::create($create_data);
  456. Db::commit();
  457. return $create_data['order_number'];
  458. } catch (\Exception $e) {
  459. Db::rollback();
  460. throw new \Exception($e->getMessage());
  461. }
  462. }
  463. // 支付成功回调
  464. public static function payNotify($params)
  465. {
  466. Log::write(json_encode($params));
  467. $transaction_id = $params['order_id']??'';
  468. $rechargeOrder = RechargeOrder::where(['transaction_id'=>$transaction_id,'payment_type'=>2])->findOrEmpty();
  469. if(!$rechargeOrder->isEmpty()){
  470. return self::payTailNotify($params);
  471. }
  472. // 查询抖音订单是否完成支付
  473. if ($params['status'] === 'SUCCESS') {
  474. $params['cp_extra'] = json_decode($params['cp_extra'], true);
  475. $paid_amount = bcdiv(bcsub($params['total_amount'] ,$params['discount_amount']), '100', 2)??0;
  476. $out_order_no = $params['out_order_no'];
  477. $pay_time = time();
  478. $order = DouyinOrder::where('order_number', $out_order_no)->findOrEmpty();
  479. if(!$order->isEmpty() && empty($order->consultation_id)){
  480. // 更新充值订单状态
  481. $order->transaction_id = $transaction_id;
  482. $order->order_status = 2;
  483. $order->pay_time = $pay_time;
  484. $order->paid_amount = $paid_amount;
  485. $user = User::where('id',$order->user_id)->findOrEmpty()->toArray();
  486. $form_detail = [
  487. 'user_name' => $user['real_name']??'',
  488. 'mobile' => $user['mobile'],
  489. 'transaction_id' => $transaction_id,
  490. 'out_trade_no' => $out_order_no,
  491. 'paid_amount' => $paid_amount,
  492. 'params' => $params,
  493. ];
  494. $consultation = ExternalConsultation::create([
  495. 'external_platform_id' => self::EXTERNAL_PLATFORM_ID,
  496. 'form_detail' => json_encode($form_detail),
  497. 'user_name' => $user['real_name']??'',
  498. 'mobile' => $user['mobile'],
  499. 'goods_id' => $order->goods_id,
  500. 'amount' => $paid_amount
  501. ]);
  502. $order->consultation_id = $consultation->id;
  503. $order->save();
  504. //order_number user_address lon lat appointment_time
  505. self::reservation([
  506. 'order_number' => $out_order_no,
  507. 'user_address' => $params['cp_extra']['user_address'],
  508. 'lon' => $params['cp_extra']['lon'],
  509. 'lat' => $params['cp_extra']['lat'],
  510. 'appointment_time' => $params['cp_extra']['appointment_time'],
  511. ]);
  512. return true;
  513. }
  514. }elseif ($params['status'] === 'CANCEL' && $params['message'] == 'TIME_OUT'){
  515. // 超时取消支付
  516. Log::info('支付超时取消:order_id'.$params['order_id']."--out_order_no:" .$params['out_order_no']);
  517. return true;
  518. }
  519. return false;
  520. }
  521. // ++++++++++++++++++++ 首次创单 end
  522. // ++++++++++++++++++++ 尾款创单
  523. /**
  524. * 预下单扩展点-子
  525. * @param array $params goods_id user_info.mobile user_id quantity
  526. * @return array|false
  527. */
  528. public static function tailOrder($params)
  529. {
  530. Db::startTrans();
  531. try {
  532. $amount = bcdiv(bcsub($params['total_amount'] ,$params['discount_amount']), '100', 2)??0;
  533. $work_id = DouyinOrder::where(['dy_order_id'=>$params['douyinOrderId']])->value('work_id')??0;
  534. $sn = '';
  535. $rechargeOrder = RechargeOrder::where(['work_id'=>$work_id,'payment_type'=>2])->findOrEmpty();
  536. if($work_id && $rechargeOrder->isEmpty()){
  537. //新增待支付尾款
  538. $sn = generate_sn(RechargeOrder::class, 'sn');
  539. $order_data = [
  540. 'order_type' => 0,
  541. 'sn' => $sn,
  542. 'order_terminal' => 1,
  543. 'work_id' => $work_id,
  544. 'user_id' => $params['user_id'],
  545. 'payment_type' => 2,
  546. 'order_total' => $amount,
  547. 'order_amount' => $amount,
  548. 'pay_status' => 0,
  549. 'paid_amount' => 0,
  550. 'pay_way' => 4
  551. ];
  552. RechargeOrder::create($order_data);
  553. }else{
  554. $sn = $rechargeOrder->sn;
  555. $rechargeOrder->transaction_id = $params['order_id']??'';
  556. $rechargeOrder->save();
  557. }
  558. Db::commit();
  559. return $sn;
  560. } catch (\Exception $e) {
  561. Db::rollback();
  562. throw new \Exception($e->getMessage());
  563. }
  564. }
  565. // 尾款支付成功回调
  566. public static function payTailNotify($params)
  567. {
  568. Log::write(json_encode($params));
  569. // 查询抖音订单是否完成支付
  570. if ($params['status'] === 'SUCCESS') {
  571. $transaction_id = $params['order_id']??'';
  572. $paid_amount = bcdiv(bcsub($params['total_amount'] ,$params['discount_amount']), '100', 2)??0;
  573. $pay_time = time();
  574. /*$out_order_no = $params['out_order_no'];
  575. $params['cp_extra'] = json_decode($params['cp_extra'], true);
  576. $work_id = DouyinOrder::where(['id'=>$params['cp_extra']['douyinOrderId']])->value('work_id')??0;
  577. $rechargeOrder = RechargeOrder::where(['work_id'=>$work_id,'sn'=>$out_order_no,'payment_type'=>2])->findOrEmpty();*/
  578. $rechargeOrder = RechargeOrder::where(['transaction_id'=>$transaction_id,'payment_type'=>2])->findOrEmpty();
  579. if(!$rechargeOrder->isEmpty()){
  580. // 更新充值订单状态
  581. $rechargeOrder->transaction_id = $transaction_id;
  582. $rechargeOrder->pay_status = 1;
  583. $rechargeOrder->pay_time = $pay_time;
  584. $rechargeOrder->paid_amount = $paid_amount;
  585. $rechargeOrder->save();
  586. // 尾款订单支付成功后续操作 fun
  587. //抖音订单服务完成
  588. $order = DouyinOrder::where(['work_id'=>$rechargeOrder->work_id])->findOrEmpty();
  589. $order->order_status = 4;
  590. $order->pay_status = 1;
  591. $order->total_amount += $paid_amount;
  592. $order->paid_amount += $paid_amount;
  593. $order->save();
  594. //工单完结
  595. self::paymentSuccessful(['sn'=>$rechargeOrder->sn,'pay_way'=>4]);
  596. return true;
  597. }
  598. }elseif ($params['status'] === 'CANCEL' && $params['message'] == 'TIME_OUT'){
  599. // 超时取消支付
  600. Log::info('支付超时取消:transaction_id'.$params['order_id']."--out_order_no:" .($params['out_order_no']??''));
  601. return true;
  602. }
  603. return false;
  604. }
  605. public static function paymentSuccessful($data = [])
  606. {
  607. try {
  608. $params = $data;
  609. $params['sn'] = mb_substr($params['sn'], 0, 18);
  610. $order = RechargeOrder::where(['sn' => $params['sn']])->findOrEmpty();
  611. if($order->isEmpty()) {
  612. throw new \Exception('内部订单不存在:'.$params['sn'],404);
  613. }
  614. if($order->pay_status == PayEnum::ISPAID) {
  615. return true;
  616. }
  617. if(!empty($params['pay_way']??'')) $params['extra']['pay_way'] = $params['pay_way'];
  618. $payNotifyLogic = PayNotifyLogic::handle('goods', $params['sn'], $params['extra']??[]);
  619. if($payNotifyLogic === true){
  620. // 用户下单后,给订单运营专员(配置固定ID)发送公众号提醒(订单信息)
  621. $order = RechargeOrder::where('sn', $params['sn'])
  622. ->where('payment_type','IN',[0,1])
  623. ->where('pay_status','=',1)
  624. ->findOrEmpty();
  625. if(!$order->isEmpty()){
  626. $workDetail = ServiceWork::findOrEmpty($order->work_id);
  627. if(!$workDetail->isEmpty()){
  628. event('Notice', [
  629. 'scene_id' => 100,
  630. 'params' => [
  631. 'user_id' => 0,
  632. 'order_id' => $workDetail['id'],
  633. 'thing3' => $workDetail['title'],
  634. 'time6' => $workDetail['appointment_time'],
  635. 'phone_number8' => asteriskString($workDetail['mobile']),
  636. 'thing5' => (iconv_strlen($workDetail['address'])>15)?(mb_substr($workDetail['address'],0,15,'UTF-8').'...'):$workDetail['address'],
  637. ]
  638. ]);
  639. }
  640. }
  641. return true;
  642. }
  643. throw new \Exception($payNotifyLogic,404);
  644. }catch(\Exception $e){
  645. throw new \Exception($e->getMessage());
  646. }
  647. }
  648. // ++++++++++++++++++++ 尾款创单 end
  649. // ******************************** 订单预约/改约
  650. // 创建预约单
  651. public static function reservation($params)
  652. {
  653. /*$lon_lat = get_address_lat_lng($params['user_address']);
  654. $params['lon'] = $lon_lat['lon'];
  655. $params['lat'] = $lon_lat['lat'];*/
  656. // $params['order_number'] user_address lon lat appointment_time
  657. Db::startTrans();
  658. try {
  659. $order = DouyinOrder::where('order_number', $params['order_number'])->findOrEmpty();
  660. if(!$order->isEmpty() && $order->consultation_id && empty($order->work_id)){
  661. $consultation = ExternalConsultation::where('id', $order->consultation_id)->findOrEmpty()->toArray();
  662. $consultation['user_name'] = $params['user_name']??$consultation['user_name'];
  663. $consultation['mobile'] = $params['mobile']??$consultation['mobile'];
  664. $consultation['user_address'] = $params['user_address'];
  665. $consultation['lon'] = $params['lon'];
  666. $consultation['lat'] = $params['lat'];
  667. $consultation['appointment_time'] = $params['appointment_time'];
  668. $consultationOrderId = ExternalConsultationLogic::order($consultation);
  669. if (false === $consultationOrderId) {
  670. throw new \Exception('预约失败');
  671. }
  672. if(!empty($consultationOrderId)){
  673. $consultationOrder = ExternalConsultationOrder::where('id', $consultationOrderId)->findOrEmpty()->toArray();
  674. $work_status = ServiceWork::where('id', $consultationOrder['work_id'])->value('work_status');
  675. $order->work_id = $consultationOrder['work_id'];
  676. $order->fulfillment_status = $work_status;
  677. $order->save();
  678. }
  679. $params['dy_order_id'] = $order->dy_order_id;
  680. $params['consultationOrderId'] = $consultationOrderId;
  681. $params['open_id'] = $order->open_id;
  682. $params['goods_id'] = $order->goods_id;
  683. }elseif ($order->consultation_id && $order->work_id){
  684. $consultationOrderId = ExternalConsultationOrder::where('work_id', $order->work_id)->where('consultation_id', $order->consultation_id)->value('id')??0;
  685. $params['dy_order_id'] = $order->dy_order_id;
  686. $params['consultationOrderId'] = $consultationOrderId;
  687. $params['open_id'] = $order->open_id;
  688. $params['goods_id'] = $order->goods_id;
  689. }else{
  690. throw new \Exception('预约失败');
  691. }
  692. Db::commit();
  693. // 抖音创建预约单
  694. //$url = 'api/apps/trade/v2/book/create_book';
  695. //$resData = self::toDyRequestUrl($url,self::getCreateBookParams($params));
  696. //book_id result
  697. // 抖音预约接单结果回调
  698. $bookInfo = json_decode($order->book_info, true);
  699. if(isset($bookInfo['open_book_info']) && $bookInfo['open_book_info']){
  700. $bookurl = 'api/apps/trade/v2/book/book_result_callback';
  701. $res = self::toDyRequestUrl($bookurl,[
  702. 'book_id' => $bookInfo['open_book_info']['book_id']??$params['dy_order_id'],
  703. 'result' => 1,
  704. ]);
  705. }
  706. return $order['id']??0;
  707. } catch (\Exception $e) {
  708. Db::rollback();
  709. throw new \Exception($e->getMessage());
  710. }
  711. }
  712. public static function getCreateBookParams($params)
  713. {
  714. try {
  715. $platformGoods = ExternalPlatformGoods::where('goods_id', $params['goods_id'])->where('external_platform_id', self::EXTERNAL_PLATFORM_ID)->findOrEmpty();
  716. if($platformGoods->isEmpty()){
  717. throw new \Exception('产品不存在!');
  718. }
  719. $appointment_time = strtotime($params['appointment_time']);
  720. $book_start_time = $appointment_time * 1000;
  721. $book_end_time = ($appointment_time + (2 * 86400)) * 1000;
  722. $data = [
  723. "order_id"=> (string)$params['dy_order_id']??'',
  724. "out_book_no"=> (string)$params['consultationOrderId']??'',
  725. "open_id"=> (string)$params['open_id']??'',
  726. "item_book_info_list" => [
  727. [
  728. "poi_id" => '7511543640776017961',
  729. "shop_name" => '亿蜂快修·武汉市',
  730. "ext_shop_id" => (string)self::EXTERNAL_PLATFORM_ID,
  731. "goods_id" => (string)$platformGoods->external_goods_sn,
  732. "book_start_time" => (int)$book_start_time??0,
  733. "book_end_time" => (int)$book_end_time??0,
  734. ]
  735. ]
  736. ];
  737. return $data;
  738. } catch (\Exception $e) {
  739. return [];
  740. }
  741. }
  742. public static function upReservation($params)
  743. {
  744. // $params['order_number']
  745. Db::startTrans();
  746. try {
  747. $order = DouyinOrder::where('order_number', $params['order_number'])->findOrEmpty();
  748. if(!$order->isEmpty()){
  749. // sn appointment_time
  750. $result = ServiceOrderLogic::approvalChangeAppointment(['sn'=>RechargeOrder::where('work_id', $order->work_id)->value('sn'),'appointment_time'=>$params['appointment_time']]);
  751. if (false === $result) {
  752. throw new \Exception(ServiceOrderLogic::getError());
  753. }
  754. $order->fulfillment_status = ServiceWork::where('id', $order->work_id)->value('work_status');
  755. $order->save();
  756. }
  757. Db::commit();
  758. return $order['id'];
  759. } catch (\Exception $e) {
  760. Db::rollback();
  761. throw new \Exception($e->getMessage());
  762. }
  763. }
  764. // ******************************** 订单退款
  765. public static function cancelOrder($params)
  766. {
  767. // $params['order_number']
  768. Db::startTrans();
  769. try {
  770. $order = DouyinOrder::where('order_number', $params['order_number'])->findOrEmpty();
  771. if(!$order->isEmpty()){
  772. if($order->order_status == 1 && $order->pay_status == 0){
  773. $order->order_status = 4;
  774. $order->save();
  775. }else{
  776. throw new \Exception('订单状态不可取消!');
  777. }
  778. }
  779. Db::commit();
  780. return $order['id'];
  781. } catch (\Exception $e) {
  782. Db::rollback();
  783. throw new \Exception($e->getMessage());
  784. }
  785. }
  786. public static function refund($params)
  787. {
  788. Db::startTrans();
  789. try {
  790. // $params['order_number'] user_id
  791. $order = DouyinOrder::with(['goods','serviceWork'])->where('order_number', $params['order_number'])->where('user_id', $params['user_id'])->findOrEmpty();
  792. if($order->isEmpty()){
  793. throw new \Exception('订单不存在');
  794. }
  795. $orderInfo = $order->toArray();
  796. $work_status = $orderInfo['serviceWork']['work_status']??0;
  797. if(3 < $work_status){
  798. throw new \Exception('该订单禁止退款');
  799. }
  800. DouyinRefundOrder::create([
  801. 'refund_number' => generate_sn(DouyinRefundOrder::class, 'refund_number'),
  802. 'order_number' => $orderInfo['order_number'],
  803. 'transaction_id' => $orderInfo['transaction_id'],
  804. 'reason' => $params['reason']??'',
  805. 'refund_status' => 0,
  806. 'user_id' => $orderInfo['user_id'],
  807. 'refund_amount' => $orderInfo['paid_amount'],
  808. ]);
  809. Db::commit();
  810. return true;
  811. } catch (\Exception $e) {
  812. Db::rollback();
  813. throw new \Exception($e->getMessage());
  814. }
  815. }
  816. public static function refundExamine($params)
  817. {
  818. Db::startTrans();
  819. try {
  820. // $params['order_number']
  821. $order = DouyinOrder::with(['goods','serviceWork'])->where('order_number', $params['order_number'])->findOrEmpty();
  822. if($order->isEmpty()){
  823. throw new \Exception('订单不存在');
  824. }
  825. $orderInfo = $order->toArray();
  826. //$refund_number = $params['refund_number']??'';
  827. $douyinRefundOrder = DouyinRefundOrder::where('order_number', $params['order_number'])->order('id', 'desc')->findOrEmpty();
  828. if($params['is_examine_ok'] === 'pass'){
  829. $douyinRefundOrder->refund_status = 2;
  830. RechargeOrder::where('work_id', $orderInfo['work_id'])->update([
  831. 'pay_status' => 2,
  832. 'pay_time' => 0,
  833. 'paid_amount' => 0,
  834. ]);
  835. ServiceWork::where('id', $orderInfo['work_id'])->update([
  836. 'work_status' => 0,
  837. 'user_confirm_status' => 0,
  838. 'service_status' => 4,
  839. 'work_pay_status' => 0
  840. ]);
  841. }else{
  842. $douyinRefundOrder->refund_status = 1;
  843. }
  844. $douyinRefundOrder->save();
  845. Db::commit();
  846. if($params['is_examine_ok'] === 'pass'){
  847. //通过后向抖音申请退款
  848. self::sendRefundCreate($params['order_number']);
  849. }
  850. return true;
  851. } catch (\Exception $e) {
  852. Db::rollback();
  853. throw new \Exception($e->getMessage());
  854. }
  855. }
  856. public static function sendRefundCreate($order_number)
  857. {
  858. try {
  859. // $params['order_number']
  860. $order = DouyinOrder::with(['goods','serviceWork'])->where('order_number', $order_number)->findOrEmpty();
  861. if($order->isEmpty()){
  862. throw new \Exception('订单不存在');
  863. }
  864. $orderInfo = $order->toArray();
  865. $douyinRefundOrder = DouyinRefundOrder::where('order_number', $order_number)->order('id', 'desc')->findOrEmpty();
  866. //通过后向抖音申请退款
  867. $url = 'api/trade_basic/v1/developer/refund_create/';
  868. $data = [
  869. "order_id" => $orderInfo['transaction_id'],
  870. "out_refund_no" => $douyinRefundOrder->refund_number,
  871. "cp_extra" => $orderInfo['id'].'|'.$douyinRefundOrder->id,
  872. "order_entry_schema" => [
  873. /*"path" => "page/index/index",
  874. "params" => json_encode(['refund_number'=>$douyinRefundOrder->refund_number])*/
  875. "path" => "pages/order/detail",
  876. "params" => json_encode(['order_number' => $order_number])
  877. ],
  878. "refund_total_amount " => $douyinRefundOrder->refund_amount * 100,
  879. //"notify_url" => config('douyin.refundNotifyUrl'),
  880. "refund_reason" => [
  881. [
  882. "code" => 101,
  883. "text" => "不想要了"
  884. ]
  885. ]
  886. ];
  887. $resData = self::toDyRequestUrl($url,$data);
  888. if(isset($resData['data']) && $resData['data']){
  889. $douyinRefundOrder->transaction_id = $resData['data']['refund_id'];
  890. $douyinRefundOrder->save();
  891. }
  892. return true;
  893. } catch (\Exception $e) {
  894. Log::info($e->getMessage());
  895. return false;
  896. }
  897. }
  898. public static function refundNotify($params)
  899. {
  900. Db::startTrans();
  901. try {
  902. $douyinRefundOrder = DouyinRefundOrder::where('refund_number', $params['out_refund_no'])->findOrEmpty();
  903. if($douyinRefundOrder->isEmpty()){
  904. throw new \Exception('退款订单不存在');
  905. }
  906. if($douyinRefundOrder->refund_status == 0){
  907. if($params['status'] === 'SUCCESS'){
  908. $douyinRefundOrder->refund_status = 3;
  909. DouyinOrder::where('order_number', $douyinRefundOrder->order_number)->update([
  910. 'order_status' => 4,
  911. 'pay_status' => 3,
  912. ]);
  913. }elseif($params['status'] === 'FAIL'){
  914. $douyinRefundOrder->refund_status = 4;
  915. }else{
  916. throw new \Exception('退款状态未知');
  917. }
  918. $douyinRefundOrder->save();
  919. }
  920. Db::commit();
  921. return true;
  922. } catch (\Exception $e) {
  923. Db::rollback();
  924. throw new \Exception($e->getMessage());
  925. }
  926. }
  927. public static function toDyRequestUrl($url,$data,$headers = [],$resFunction = 'extraErrorCodeReturn',$isHost = 0)
  928. {
  929. $toData = [
  930. 'url' => $url,
  931. 'data' => $data,
  932. 'headers' => $headers,
  933. 'resFunction' => $resFunction,
  934. 'isHost' => $isHost
  935. ];
  936. $res = commonHttpClient(env('internal_api.api_url_host').'platf/dou_yin/toDyRequestUrl', $toData, 'post', 'json', ['Content-Type' => 'application/json']);
  937. Log::info(json_encode($res));
  938. if(isset($res['code']) && $res['code'] === 0){
  939. Log::info("toDyRequestUrl:".json_encode($res));
  940. return $res['data'];
  941. }else{
  942. Log::info("toDyRequestUrl:".$res['msg']);
  943. throw new \Exception($res['msg']);
  944. }
  945. }
  946. }