Transaction.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. <?php
  2. namespace app\common\library;
  3. use think\Db;
  4. /**
  5. * 苹果server to server
  6. */
  7. class Transaction
  8. {
  9. protected $table = 'transaction';
  10. protected $primaryKey = 'id';
  11. protected $guarded = [];
  12. protected function serializeDate(DateTimeInterface $date)
  13. {
  14. return $date->format('Y-m-d H:i:s');
  15. }
  16. /**
  17. * 续费or升级
  18. *
  19. * @param $transaction_info
  20. * @param bool $is_upgrade
  21. * @return bool
  22. */
  23. public static function autoRenew($transaction_info, $is_upgrade = false)
  24. {
  25. $original_transaction_id = isset($transaction_info['originalTransactionId']) ? $transaction_info['originalTransactionId'] : ''; // 订阅原始ID
  26. $transaction_id = isset($transaction_info['transactionId']) ? $transaction_info['transactionId'] : ''; // 苹果订单号
  27. $product_id = isset($transaction_info['productId']) ? $transaction_info['productId'] : ''; // 产品id
  28. if (!$original_transaction_id || !$transaction_id || !$product_id) {
  29. return '参数有误';
  30. }
  31. filePut("\r\n\r\n".'新的S2S');
  32. $prefix = '[S2S]originalTransactionId:'.$original_transaction_id.',transactionId:'.$transaction_id.',productId:'.$product_id.'。';
  33. //检查重复订单
  34. $check_map = [
  35. 'original_transaction_id' => $original_transaction_id,
  36. 'transaction_id' => $transaction_id,
  37. ];
  38. $check_order = Db::name('user_vipxufei_task')->where($check_map)->field('id')->find();
  39. if($check_order){
  40. filePut($prefix.'续费早已完成');
  41. return true;
  42. }
  43. Db::startTrans();
  44. //查找订单,可能找到以前的,非当前用户的。根据原始id 和 用户id不是终生绑定
  45. $order_map = [
  46. 'original_transaction_id' => $original_transaction_id,
  47. ];
  48. $order_info = Db::name('user_vipxufei_task')->where($order_map)->order('expires_date_ms desc')->find();
  49. if (!$order_info) {
  50. Db::rollback();
  51. filePut($prefix.'不存在的订单');
  52. return '不存在的订单';
  53. }
  54. //续订,但是换了产品了,重新定义order_info
  55. //原始id换给别人用了
  56. $order_info_bundle_id = $order_info['bundle_id'];
  57. $order_info_user_id = $order_info['user_id'];
  58. if($product_id != $order_info['bundle_id']){
  59. $pay_order_map = [
  60. 'user_id' => $order_info['user_id'],
  61. 'bundle_id' => $product_id,
  62. 'order_status' => 0,
  63. 'table_name' => 'vip_recharge',
  64. ];
  65. $pay_order = Db::name('pay_order')->where($pay_order_map)->order('id desc')->lock(true)->find();
  66. if(!$pay_order){
  67. Db::rollback();
  68. filePut($prefix.'未找到匹配的交易,新订单找不到');
  69. return '未找到匹配的新订单';
  70. }
  71. // 修改订单状态
  72. $update_order = [
  73. 'notifytime'=>time(),
  74. 'order_status'=>1,
  75. 'original_transaction_id' => $original_transaction_id,
  76. 'transaction_info' => json_encode($transaction_info),
  77. ];
  78. $ros = Db::name('pay_order')->where(['id' => $pay_order['id']])->update($update_order);
  79. if($ros === false) {
  80. filePut($prefix.'修改订单状态失败');
  81. Db::rollback();
  82. return '充值失败';
  83. }
  84. $args = json_decode($pay_order['args'],true);
  85. //修改order_info
  86. $order_info['order_id'] = $pay_order['id'];
  87. $order_info['user_id'] = $pay_order['user_id'];
  88. $order_info['bundle_id'] = $pay_order['bundle_id'];
  89. $order_info['days'] = $args['days'];
  90. $order_info['original_transaction_id'] = $original_transaction_id;//多余
  91. }
  92. //验证时间,不得小于最新的一条预定信息
  93. if($transaction_info['purchaseDate'] <= $order_info['purchase_date_ms'] || $transaction_info['expiresDate'] <= $order_info['expires_date_ms']){
  94. Db::rollback();
  95. filePut($prefix.'时间对不上,返回成功,finish掉');
  96. return true;
  97. }
  98. //逻辑开始
  99. //先充值
  100. $user_info = Db::name('user_wallet')->where('user_id',$order_info['user_id'])->lock(true)->find();
  101. if($user_info['vip_endtime'] < time()){
  102. //过期了
  103. $vip_endtime = time() + (intval($order_info['days']) * 86400);
  104. }else{
  105. //追加vip
  106. $vip_endtime = $user_info['vip_endtime'] + (intval($order_info['days']) * 86400);
  107. }
  108. $update_data = [
  109. 'vip_endtime'=>$vip_endtime,
  110. ];
  111. $result = Db::name('user_wallet')->where('user_id',$order_info['user_id'])->update($update_data);
  112. if($result === false)
  113. {
  114. filePut($prefix.'逻辑续费vip时间失败');
  115. Db::rollback();
  116. return '充值失败';
  117. }
  118. //逻辑结束
  119. //添加新的一个task
  120. $task_data = $order_info;
  121. unset($task_data['id']);
  122. $task_data['createtime'] = time();
  123. $task_data['apple_receipt'] = '';
  124. $task_data['in_app_one'] = '';
  125. $task_data['transaction_id'] = $transaction_id;
  126. $task_data['times'] = $order_info['times'] + 1;
  127. $task_data['original_purchase_date_ms'] = $transaction_info['originalPurchaseDate'];
  128. $task_data['purchase_date_ms'] = $transaction_info['purchaseDate'];
  129. $task_data['expires_date_ms'] = $transaction_info['expiresDate'];
  130. $task_data['transaction_info'] = json_encode($transaction_info);
  131. if($product_id != $order_info_bundle_id){
  132. $task_data['times'] = 1;//回归1
  133. }
  134. $task_id = Db::name('user_vipxufei_task')->insertGetId($task_data);
  135. if(!$task_id)
  136. {
  137. filePut($prefix.'用户添加vipxufei_task失败');
  138. Db::rollback();
  139. return '充值失败';
  140. }
  141. Db::commit();
  142. filePut($prefix.'充值成功');
  143. return true;
  144. //逻辑结束
  145. }
  146. /**
  147. * 退款
  148. *
  149. * @param $transaction_info
  150. * @return bool
  151. */
  152. public static function refund($transaction_info)
  153. {
  154. return true;
  155. /*$original_transaction_id = $transaction_info['originalTransactionId']; // 订阅ID
  156. $web_order_line_item_id = $transaction_info['webOrderLineItemId']; // 交易ID
  157. $base_msg = self::_getBaseMsg($original_transaction_id, $web_order_line_item_id) . '- 退款:';
  158. $transaction_info_data = Transaction::query()
  159. ->where('web_order_line_item_id', '=', $transaction_info['webOrderLineItemId'])
  160. ->where('original_transaction_id', '=', $transaction_info['originalTransactionId'])
  161. ->first();
  162. if (!$transaction_info_data) {
  163. Log::channel('transaction')->warning($base_msg . ' not find transaction info');
  164. return false;
  165. }
  166. // 修改退款时间以及状态
  167. Transaction::query()
  168. ->where('web_order_line_item_id', '=', $transaction_info['webOrderLineItemId'])
  169. ->where('original_transaction_id', '=', $transaction_info['originalTransactionId'])
  170. ->update([
  171. 'cancellation_date_ms' => date('Y-m-d H:i:s', $transaction_info['revocationDate'] / 1000),
  172. 'is_cancellation' => 1,
  173. ]);
  174. // 修改用户会员为到期
  175. User::saveExpireVip($transaction_info_data->user_id);
  176. return true;*/
  177. }
  178. /**
  179. * 订阅
  180. *
  181. * @param $transaction_info
  182. * @param $subtype
  183. * @return bool
  184. */
  185. public static function updateSubscribed($transaction_info, $subtype)
  186. {
  187. return true;
  188. }
  189. /**
  190. * 续订失败
  191. *
  192. * @param $transaction_info
  193. * @return bool
  194. */
  195. public static function updateExpiredTrans($transaction_info)
  196. {
  197. return true;
  198. }
  199. /**
  200. * 续订失败
  201. *
  202. * @param $transaction_info
  203. * @return bool
  204. */
  205. public static function updateRenewFailTrans($transaction_info)
  206. {
  207. return true;
  208. }
  209. /**
  210. * 自动续费状态修改
  211. *
  212. * @param $transaction_info
  213. * @param $subtype
  214. * @return bool
  215. */
  216. public static function changeRenewStatus($transaction_info, $subtype)
  217. {
  218. return true;
  219. }
  220. /**
  221. * 检测交易
  222. *
  223. * @param $transaction_info
  224. * @param $base_msg
  225. * @return array
  226. */
  227. private static function _checkTransaction($transaction_info)
  228. {
  229. $ret = [
  230. 'check' => false,
  231. 'data' => [],
  232. ];
  233. // 2.判断交易ID是否已经入库过
  234. $is_exist = Db::name('user_vipxufei_task')
  235. ->where('original_transaction_id', $transaction_info['originalTransactionId'])
  236. ->where('transaction_id', $transaction_info['transactionId'])
  237. ->find();
  238. if ($is_exist) {
  239. return $ret;
  240. }
  241. //查找订单,可能找到以前的,非当前用户的。根据原始id 和 用户id不是终生绑定
  242. $order_map = [
  243. 'original_transaction_id' => $transaction_info['originalTransactionId'],
  244. ];
  245. $order_info = Db::name('user_vipxufei_task')->where($order_map)->order('expires_date_ms desc')->find();
  246. if (!$order_info) {
  247. return $ret;
  248. }
  249. //
  250. $ret['check'] = true;
  251. $ret['data'] = $order_info;
  252. return $ret;
  253. }
  254. /**
  255. * 获取日志
  256. *
  257. * @param $original_transaction_id
  258. * @param $web_order_line_item_id
  259. * @return string
  260. */
  261. private static function _getBaseMsg($original_transaction_id, $web_order_line_item_id)
  262. {
  263. return "notify-{$original_transaction_id}-{$web_order_line_item_id} ";
  264. }
  265. /**
  266. * 获取入库信息
  267. *
  268. * @param $user_id
  269. * @param $transaction_info
  270. * @return array
  271. */
  272. private static function _getAddData($user_id, $transaction_info)
  273. {
  274. return [
  275. 'user_id' => $user_id, // fixme 此处不考虑用户ID切换
  276. 'transaction_id' => $transaction_info['transactionId'],
  277. 'product_id' => $transaction_info['productId'],
  278. 'web_order_line_item_id' => $transaction_info['webOrderLineItemId'],
  279. 'original_transaction_id' => $transaction_info['originalTransactionId'],
  280. 'original_purchase_date_ms' => date('Y-m-d H:i:s', $transaction_info['originalPurchaseDate'] / 1000), //首次订阅时间
  281. 'purchase_date_ms' => date('Y-m-d H:i:s', $transaction_info['purchaseDate'] / 1000), // 购买时间
  282. 'expires_date_ms' => date('Y-m-d H:i:s', $transaction_info['expiresDate'] / 1000), // 过期时间
  283. 'subscription_group_identifier' => $transaction_info['subscriptionGroupIdentifier'] ?? '',
  284. 'in_app_ownership_type' => $transaction_info['inAppOwnershipType'],
  285. 'environment' => $transaction_info['environment'] ?? '',
  286. 'sub_type' => 'DID_RENEW', // DID_RENEW UPGRADE SUBSCRIBED-INITIAL_BUY SUBSCRIBED-RESUBSCRIBE
  287. 'auto_renew_result' => 1, // 自动续费结果 0默认 1成功 2失败 3过期
  288. 'auto_renew_status' => 1, // 自动订阅状态 1开启 2-关闭
  289. ];
  290. }
  291. }