Nav apraksta
Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

TikTok.php 21KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海口快推科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace app\common\model;
  14. use think\Db;
  15. use think\Config;
  16. /**
  17. * 抖音公共模型
  18. */
  19. load_trait('controller/Jump');
  20. class TikTok
  21. {
  22. use \traits\controller\Jump;
  23. // 构造函数
  24. public function __construct()
  25. {
  26. // 统一接收参数处理
  27. $this->times = getTime();
  28. // 抖音信息
  29. $this->tikTokConfig = tpSetting("OpenMinicode.conf_toutiao");
  30. $this->tikTokConfig = !empty($this->tikTokConfig) ? json_decode($this->tikTokConfig, true) : [];
  31. // dump($this->tikTokConfig);
  32. // 接口请求头
  33. $this->headers = [
  34. 'Content-Type: application/json',
  35. ];
  36. }
  37. // 获取抖音支付订单号(order_id)、订单token(order_token)
  38. public function getTikTokAppletsPay($orderID = '', $orderCode = '', $orderAmount = 0, $orderUnpayCloseTime = 2880, $table = 'shop_order')
  39. {
  40. // 接口参数
  41. $postData = [
  42. 'app_id' => $this->tikTokConfig['appid'],
  43. 'out_order_no' => strval($orderCode),
  44. 'total_amount' => floatval($orderAmount * 100),
  45. 'subject' => '抖音支付',
  46. 'body' => '商品购买',
  47. 'valid_time' => intval(intval($orderUnpayCloseTime) * 60),
  48. 'cp_extra' => "tikTok|,|" . session('users_id') . "|,|" . $orderID . "|,|" . $orderCode . "|,|" . $table,
  49. 'notify_url' => request()->domain() . ROOT_DIR . '/index.php',
  50. ];
  51. // MD5加密签名
  52. $postData['sign'] = $this->getSignParam($postData);
  53. // 接口API
  54. $url = 'https://developer.toutiao.com/api/apps/ecpay/v1/create_order';
  55. $jsonData = json_decode(httpRequest($url, 'POST', json_encode($postData), $this->headers, 300000), true);
  56. // 返回订单创建信息
  57. if (isset($jsonData['err_no']) && 0 === intval($jsonData['err_no']) && isset($jsonData['err_tips']) && 'success' == strval($jsonData['err_tips'])) {
  58. return [
  59. 'success' => true,
  60. 'order_id' => $jsonData['data']['order_id'],
  61. 'order_token' => $jsonData['data']['order_token'],
  62. ];
  63. } else {
  64. return [
  65. 'success' => false,
  66. 'err_tips' => $jsonData['err_tips'],
  67. ];
  68. }
  69. }
  70. // 抖音小程序商品购买支付后续处理
  71. public function tikTokAppletsPayDealWith($post = [], $notify = false, $table = 'shop_order')
  72. {
  73. // 异步回调时执行
  74. $cpExtra = !empty($post['msg']['cp_extra']) ? explode('|,|', $post['msg']['cp_extra']) : [];
  75. if (true === $notify && !empty($cpExtra[0]) && 'tikTok' == $cpExtra[0]) {
  76. $post['users_id'] = !empty($cpExtra[1]) ? $cpExtra[1] : 0;
  77. $post['order_id'] = !empty($cpExtra[2]) ? $cpExtra[2] : 0;
  78. $post['order_code'] = !empty($cpExtra[3]) ? $cpExtra[3] : '';
  79. $table = !empty($cpExtra[4]) ? $cpExtra[4] : $table;
  80. }
  81. if (!empty($post['users_id'])) {
  82. // 获取系统订单
  83. $order = $this->getSystemOrder($post, $notify, $table);
  84. // 系统订单号
  85. if ('shop_order' == $table) {
  86. $orderCode = strval($order['order_code']);
  87. } else if ('users_recharge_pack_order' == $table) {
  88. $orderCode = true === $notify ? strval($post['order_code']) : strval($order['order_pay_code']);
  89. } else if ('users_money' == $table) {
  90. $orderCode = true === $notify ? strval($post['order_code']) : strval($order['order_number']);
  91. }
  92. // 查询抖音支付订单是否真实完成支付
  93. $jsonData = $this->queryOrderPayResult($orderCode);
  94. if (isset($jsonData['err_no']) && 0 === intval($jsonData['err_no']) && !empty($jsonData['payment_info']['order_status'])) {
  95. // 支付成功
  96. if ('SUCCESS' == $jsonData['payment_info']['order_status']) {
  97. // 处理系统订单
  98. $this->handleSystemOrder($post, $notify, $table, $order, $jsonData);
  99. }
  100. // 正在支付中
  101. else if ('PROCESSING' == $jsonData['payment_info']['order_status']) {
  102. if (true !== $notify) $this->success('正在支付中');
  103. }
  104. // 超时未支付
  105. else if ('TIMEOUT' == $jsonData['payment_info']['order_status']) {
  106. if (true !== $notify) $this->error($jsonData['err_tips']);
  107. }
  108. // 支付失败
  109. else if ('FAIL' == $jsonData['payment_info']['order_status']) {
  110. if (true !== $notify) $this->error($jsonData['err_tips']);
  111. }
  112. } else {
  113. if (true !== $notify) $this->error($jsonData['err_tips']);
  114. }
  115. }
  116. }
  117. // 查询系统订单
  118. private function getSystemOrder($post = [], $notify = false, $table = 'shop_order')
  119. {
  120. // 商城商品订单
  121. if ('shop_order' == $table) {
  122. // 查询系统订单信息
  123. $where = [
  124. 'users_id' => intval($post['users_id']),
  125. 'order_id' => intval($post['order_id']),
  126. 'order_code' => strval($post['order_code']),
  127. ];
  128. $order = Db::name('shop_order')->where($where)->find();
  129. if (empty($order)) {
  130. // 同步
  131. if (true !== $notify) $this->error('无效订单');
  132. } else if (0 < $order['order_status']) {
  133. // 异步
  134. if (true === $notify) {
  135. exit(json_encode(['err_no' => 0, 'err_tips' => 'success']));
  136. }
  137. // 同步
  138. else {
  139. $usersData = Db::name('users')->where('users_id', $post['users_id'])->find();
  140. // 邮箱发送
  141. $resultData['email'] = GetEamilSendData(tpCache('smtp'), $usersData, $order, 1, 'wechat');
  142. // 短信发送
  143. $resultData['mobile'] = GetMobileSendData(tpCache('sms'), $usersData, $order, 1, 'wechat');
  144. // 跳转链接
  145. $url = 1 == input('param.fenbao/d') ? '' : '/pages/order/index';
  146. $resultData['url'] = $url;
  147. $this->success('支付完成', $url, $resultData);
  148. }
  149. }
  150. }
  151. // 会员充值套餐订单
  152. else if ('users_recharge_pack_order' == $table) {
  153. $where = [
  154. 'users_id' => intval($post['users_id']),
  155. 'order_id' => intval($post['order_id']),
  156. 'order_code' => strval($post['order_code']),
  157. 'order_pay_code' => strval($post['order_pay_code']),
  158. ];
  159. $order = Db::name('users_recharge_pack_order')->where($where)->find();
  160. if (empty($order)) {
  161. // 同步
  162. if (true !== $notify) $this->error('无效订单');
  163. } else if (1 < $order['order_status']) {
  164. // 异步
  165. if (true === $notify) {
  166. exit(json_encode(['err_no' => 0, 'err_tips' => 'success']));
  167. }
  168. // 同步
  169. else {
  170. $this->success('支付完成');
  171. }
  172. }
  173. }
  174. // 会员充值余额订单
  175. else if ('users_money' == $table) {
  176. $post['moneyid'] = !empty($post['order_id']) ? $post['order_id'] : $post['moneyid'];
  177. $post['order_number'] = !empty($post['order_code']) ? $post['order_code'] : $post['order_number'];
  178. $where = [
  179. 'moneyid' => intval($post['moneyid']),
  180. 'users_id' => intval($post['users_id']),
  181. 'order_number' => strval($post['order_number']),
  182. ];
  183. $order = Db::name('users_money')->where($where)->find();
  184. if (empty($order)) {
  185. // 同步
  186. if (true !== $notify) $this->error('无效订单');
  187. } else if (1 < $order['status']) {
  188. // 异步
  189. if (true === $notify) {
  190. exit(json_encode(['err_no' => 0, 'err_tips' => 'success']));
  191. }
  192. // 同步
  193. else {
  194. $this->success('支付完成');
  195. }
  196. }
  197. }
  198. return $order;
  199. }
  200. // 处理系统订单
  201. private function handleSystemOrder($post = [], $notify = false, $table = 'shop_order', $order = [], $jsonData = [])
  202. {
  203. $moneyID = !empty($order['moneyid']) ? intval($order['moneyid']) : 0;
  204. $usersID = !empty($order['users_id']) ? intval($order['users_id']) : 0;
  205. $orderID = !empty($order['order_id']) ? intval($order['order_id']) : 0;
  206. $orderCode = !empty($order['order_code']) ? strval($order['order_code']) : '';
  207. $orderNumber = !empty($order['order_number']) ? strval($order['order_number']) : '';
  208. $orderPayCode = !empty($order['order_pay_code']) ? strval($order['order_pay_code']) : '';
  209. $cpExtra = !empty($jsonData['payment_info']['cp_extra']) ? explode('|,|', $jsonData['payment_info']['cp_extra']) : [];
  210. $cpExtra[0] = !empty($cpExtra[0]) ? $cpExtra[0] : '';
  211. $cpExtra[1] = !empty($cpExtra[1]) ? $cpExtra[1] : 0;
  212. $cpExtra[2] = !empty($cpExtra[2]) ? $cpExtra[2] : 0;
  213. $cpExtra[3] = !empty($cpExtra[3]) ? $cpExtra[3] : 0;
  214. if ('tikTok' == $cpExtra[0] && $usersID == $cpExtra[1]) {
  215. // 商城商品订单
  216. if ('shop_order' == $table && $orderID == $cpExtra[2]) {
  217. // 订单已支付,处理订单流程
  218. $where = [
  219. 'order_id' => $orderID,
  220. 'users_id' => $usersID
  221. ];
  222. $update = [
  223. 'order_status' => 1,
  224. 'pay_details' => serialize($jsonData),
  225. 'pay_time' => getTime(),
  226. 'update_time' => getTime()
  227. ];
  228. $resultID = Db::name('shop_order')->where($where)->update($update);
  229. if (!empty($resultID)) {
  230. // 添加订单操作记录
  231. AddOrderAction($orderID, $usersID, 0, 1, 0, 1, '支付成功', '会员使用抖音小程序完成支付');
  232. // 虚拟自动发货
  233. $shopModel = new \app\user\model\Pay();
  234. $where = [
  235. 'lang' => get_home_lang(),
  236. 'users_id' => $usersID,
  237. 'order_id' => $orderID
  238. ];
  239. $shopModel->afterVirtualProductPay($where);
  240. // 订单支付通知
  241. $params = [
  242. 'users_id' => $usersID,
  243. 'result_id' => $orderID,
  244. ];
  245. eyou_send_notice(9, $params);
  246. // 站内信通知
  247. $order['pay_method'] = '微信支付';
  248. SendNotifyMessage($order, 5, 1, 0);
  249. // 添加会员积分记录
  250. $where = [
  251. 'users_id' => $usersID,
  252. 'order_id' => $orderID
  253. ];
  254. $data = Db::name('shop_order_details')->where($where)->getField('data');
  255. $data = !empty($data) ? unserialize($data) : [];
  256. $pointsGoodsBuyField = !empty($data['pointsGoodsBuyField']) ? json_decode($data['pointsGoodsBuyField'], true) : [];
  257. if (!empty($pointsGoodsBuyField['goodsTotalPoints'])) {
  258. $insert = [
  259. 'type' => 11, // 积分商城订单支付
  260. 'users_id' => $usersID,
  261. 'score' => $pointsGoodsBuyField['goodsTotalPoints'],
  262. 'info' => '积分商城订单支付',
  263. 'remark' => '积分商城订单支付',
  264. ];
  265. addConsumObtainScores($insert, 1, false);
  266. }
  267. // 异步
  268. if (true === $notify) {
  269. exit(json_encode(['err_no' => 0, 'err_tips' => 'success']));
  270. }
  271. // 同步
  272. else {
  273. // 统计销售额
  274. eyou_statistics_data(2);
  275. eyou_statistics_data(3, $order['order_amount']);
  276. $usersData = Db::name('users')->where('users_id', $usersID)->find();
  277. // 邮箱发送
  278. $resultData['email'] = GetEamilSendData(tpCache('smtp'), $usersData, $order, 1, 'wechat');
  279. // 短信发送
  280. $resultData['mobile'] = GetMobileSendData(tpCache('sms'), $usersData, $order, 1, 'wechat');
  281. // 跳转链接
  282. $url = 1 == input('param.fenbao/d') ? '' : '/pages/order/index';
  283. $resultData['url'] = $url;
  284. $this->success('支付完成', $url, $resultData);
  285. }
  286. }
  287. }
  288. // 会员充值套餐订单
  289. else if ('users_recharge_pack_order' == $table && $orderID == $cpExtra[2]) {
  290. // 更新订单为已支付
  291. $where = [
  292. 'users_id' => $usersID,
  293. 'order_id' => $orderID,
  294. 'order_code' => $orderCode,
  295. 'order_pay_code' => $orderPayCode,
  296. ];
  297. $update = [
  298. 'order_status' => 2,
  299. 'order_pay_time' => $this->times,
  300. 'order_pay_name' => 'tikTokPay',
  301. 'order_pay_terminal' => 4,
  302. 'order_pay_details' => serialize($jsonData),
  303. 'update_time' => $this->times,
  304. ];
  305. $result_1 = Db::name('users_recharge_pack_order')->where($where)->update($update);
  306. if (!empty($result_1)) {
  307. // 根据会员充值套餐给会员充储值余额
  308. $result_2 = Db::name('users')->where(['users_id' => $usersID])->setInc('users_money', $order['order_face_value']);
  309. if (!empty($result_2)) {
  310. // 更新订单为已充值
  311. $update = [
  312. 'order_status' => 3,
  313. 'update_time' => $this->times,
  314. ];
  315. Db::name('users_recharge_pack_order')->where($where)->update($update);
  316. // 增加充值套餐的销售量
  317. $where = [
  318. 'pack_id' => $order['pack_id'],
  319. ];
  320. Db::name('users_recharge_pack')->where($where)->setInc('pack_sales_num');
  321. // 会员当前余额
  322. $usersMoney = Db::name('users')->where('users_id', $usersID)->getField('users_money');
  323. $moneyData = [
  324. 'users_id' => $usersID,
  325. 'money' => unifyPriceHandle($order['order_face_value']),
  326. 'users_money' => unifyPriceHandle($usersMoney),
  327. 'cause' => $order['order_pack_names'] . '(充值套餐)',
  328. 'cause_type' => 1,
  329. 'status' => 3,
  330. 'pay_method' => 'tikTokPay',
  331. 'pay_details' => serialize($jsonData),
  332. 'order_number' => $orderPayCode,
  333. 'lang' => get_home_lang(),
  334. 'add_time' => $this->times,
  335. 'update_time' => $this->times,
  336. ];
  337. Db::name('users_money')->insertGetId($moneyData);
  338. // 异步
  339. if (true === $notify) {
  340. exit(json_encode(['err_no' => 0, 'err_tips' => 'success']));
  341. }
  342. // 同步
  343. else {
  344. $this->success('充值成功');
  345. }
  346. }
  347. }
  348. }
  349. // 会员充值余额订单
  350. else if ('users_money' == $table && $moneyID == $cpExtra[2]) {
  351. // 更新订单为已支付
  352. $where = [
  353. 'moneyid' => intval($moneyID),
  354. 'users_id' => intval($usersID),
  355. 'order_number' => strval($orderNumber),
  356. ];
  357. $update = [
  358. 'status' => 2,
  359. 'pay_method' => 'tikTokPay',
  360. 'wechat_pay_type' => '',
  361. 'pay_details' => serialize($jsonData),
  362. 'update_time' => getTime()
  363. ];
  364. $result_1 = Db::name('users_money')->where($where)->update($update);
  365. if (!empty($result_1)) {
  366. // 更新增加会员余额
  367. $result_2 = Db::name('users')->where(['users_id' => intval($usersID)])->setInc('users_money', $order['money']);
  368. if (!empty($result_2)) {
  369. // 更新订单为已充值
  370. $update = [
  371. 'status' => 3,
  372. 'update_time' => getTime()
  373. ];
  374. Db::name('users_money')->where($where)->update($update);
  375. $this->success('充值成功');
  376. }
  377. }
  378. $this->error('付款成功,充值失败,请联系管理员');
  379. }
  380. }
  381. }
  382. // 查询抖音支付订单是否真实完成支付
  383. public function queryOrderPayResult($orderCode = '')
  384. {
  385. // 接口参数
  386. $postData = [
  387. 'app_id' => $this->tikTokConfig['appid'],
  388. 'out_order_no' => strval($orderCode),
  389. ];
  390. // MD5加密签名
  391. $postData['sign'] = $this->getSignParam($postData);
  392. // 接口API
  393. $url = 'https://developer.toutiao.com/api/apps/ecpay/v1/query_order';
  394. // 返回结果
  395. return json_decode(httpRequest($url, 'POST', json_encode($postData), $this->headers, 300000), true);
  396. }
  397. private function getSignParam($map = [])
  398. {
  399. $param = [];
  400. foreach($map as $key => $value) {
  401. // 排除不参与加密的字段
  402. if ($key == "other_settle_params" || $key == "app_id" || $key == "sign" || $key == "thirdparty_id") continue;
  403. // 参数处理
  404. $valueNew = trim(strval($value));
  405. if (is_array($value)) $valueNew = arrayToStr($value);
  406. $len = strlen($valueNew);
  407. if ($len > 1 && substr($valueNew, 0,1)=="\"" && substr($valueNew, $len-1)=="\"") $valueNew = substr($valueNew,1, $len-1);
  408. $valueNew = trim($valueNew);
  409. if ($valueNew == "" || $valueNew == "null") continue;
  410. $param[] = $valueNew;
  411. }
  412. $param[] = $this->tikTokConfig['salt'];
  413. sort($param, SORT_STRING);
  414. return md5(trim(implode('&', $param)));
  415. }
  416. private function arrayToStr($map = [])
  417. {
  418. $isMap = isArrMap($map);
  419. $result = "";
  420. if (!empty($isMap)) $result = "map[";
  421. $keyArr = array_keys($map);
  422. if (!empty($isMap)) sort($keyArr);
  423. $paramsArr = array();
  424. foreach($keyArr as $key) {
  425. $v = $map[$key];
  426. if (!empty($isMap)) {
  427. if (is_array($v)) {
  428. $paramsArr[] = sprintf("%s:%s", $key, arrayToStr($v));
  429. } else {
  430. $paramsArr[] = sprintf("%s:%s", $key, trim(strval($v)));
  431. }
  432. } else {
  433. if (is_array($v)) {
  434. $paramsArr[] = arrayToStr($v);
  435. } else {
  436. $paramsArr[] = trim(strval($v));
  437. }
  438. }
  439. }
  440. $result = sprintf("%s%s", $result, join(" ", $paramsArr));
  441. if (empty($isMap)) {
  442. $result = sprintf("[%s]", $result);
  443. } else {
  444. $result = sprintf("%s]", $result);
  445. }
  446. return $result;
  447. }
  448. private function isArrMap($map = [])
  449. {
  450. foreach($map as $k =>$v) {
  451. if (is_string($k)) return true;
  452. }
  453. return false;
  454. }
  455. }