UserExtractController.php 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. <?php
  2. namespace app\api\controller\user;
  3. use app\models\user\UserBill;
  4. use app\models\user\UserExtract;
  5. use app\Request;
  6. use crmeb\payment\MachantPay;
  7. use crmeb\services\payment\PaymentService;
  8. use crmeb\services\UtilService;
  9. use think\facade\Log;
  10. use tw\redis\UserRds;
  11. /**
  12. * 提现类
  13. * Class UserExtractController
  14. * @package app\api\controller\user
  15. */
  16. class UserExtractController
  17. {
  18. /**
  19. * @api {get} /extract/bank 提现银行
  20. * @apiName ExtractBank
  21. * @apiGroup Brokerage
  22. *
  23. * @apiSuccess {float} broken_commission 冻结佣金
  24. * @apiSuccess {float} brokerage_price 总佣金
  25. * @apiSuccess {float} commissionCount 可提现佣金
  26. * @apiSuccess {string[]} extractBank 银行列表
  27. * @apiSuccess {float} minPrice 最小提现金额
  28. */
  29. public function bank(Request $request)
  30. {
  31. $user = $request->user();
  32. // 佣金冻结时间
  33. $frozen_days = intval(sys_config('extract_time'));
  34. $frozen_from = time() - 86400 * $frozen_days;
  35. //可提现佣金
  36. $brokerage_commission = UserBill::where(['uid' => $user['uid'], 'category' => 'now_money', 'type' => 'brokerage'])
  37. ->where('add_time', '>', $frozen_from)
  38. ->where('pm', 1)
  39. ->sum('number');
  40. $refund_commission = UserBill::where(['uid' => $user['uid'], 'category' => 'now_money', 'type' => 'brokerage'])
  41. ->where('add_time', '>', $frozen_from)
  42. ->where('pm', 0)
  43. ->sum('number');
  44. $data['broken_commission'] = floatval(bcsub($brokerage_commission, $refund_commission, 2));
  45. if ($data['broken_commission'] < 0) {
  46. $data['broken_commission'] = 0;
  47. }
  48. $data['brokerage_price'] = $user['brokerage_price'];
  49. //可提现佣金
  50. $data['commissionCount'] = floatval(bcsub($data['brokerage_price'], $data['broken_commission'], 2));
  51. $extractBank = sys_config('user_extract_bank') ?? []; //提现银行
  52. $extractBank = str_replace("\r\n", "\n", $extractBank); //防止不兼容
  53. $data['extractBank'] = explode("\n", is_array($extractBank) ? (isset($extractBank[0]) ? $extractBank[0] : $extractBank) : $extractBank);
  54. $data['minPrice'] = sys_config('user_extract_min_price'); //提现最低金额
  55. // 输入记忆
  56. $ur = new UserRds();
  57. $cach = $ur->gets($user['uid'], ['wxpayName', 'bankCardNo', 'bankUser', 'bankName']);
  58. $data['wxpayName'] = $cach['wxpayName'] ? $cach['wxpayName'] : '';
  59. $data['bankCardNo'] = $cach['bankCardNo'] ? $cach['bankCardNo'] : '';
  60. $data['bankUser'] = $cach['bankUser'] ? $cach['bankUser'] : '';
  61. $data['bankName'] = $cach['bankName'] ? $cach['bankName'] : '';
  62. return app('json')->successful($data);
  63. }
  64. /**
  65. * @api {post} /extract/bank_fee 查询银行提现手续费
  66. * @apiName PostExtractBankFee
  67. * @apiGroup Brokerage
  68. *
  69. * @apiBody {int} extract_type 提现类型/通道
  70. * @apiBody {float} money 提现金额
  71. *
  72. * @apiSuccess {float} rate 手续费率
  73. * @apiSuccess {float} min 最少收取
  74. * @apiSuccess {float} max 最多收取
  75. * @apiSuccess {float} fee 当前收取
  76. * @apiSuccess {float} valid 实际到帐金额
  77. */
  78. public function bank_fee(Request $request)
  79. {
  80. $extractInfo = UtilService::postMore([
  81. ['alipay_code', ''],
  82. ['extract_type', ''],
  83. ['money', 0],
  84. ['name', ''],
  85. ['bankname', ''],
  86. ['cardnum', ''],
  87. ['weixin', ''],
  88. ], $request);
  89. $user = $request->user();
  90. if ($extractInfo['extract_type'] != 'bank') {
  91. return app('json')->fail('不是银行卡提现');
  92. }
  93. if ($extractInfo['money'] < sys_config('user_extract_min_price')) {
  94. return app('json')->fail('金额小于最低提现金额');
  95. }
  96. $bankMax = 1000; // 银行渠道限额
  97. if ($extractInfo['money'] > $bankMax) {
  98. return app('json')->fail("金额不能超过 $bankMax 元");
  99. }
  100. list($ok, $fee, $rate, $min, $max) = MachantPay::toBankByWeixinFee($extractInfo['money']);
  101. if (!$ok) {
  102. return app('json')->fail('金额错误');
  103. }
  104. // 实际到帐金额
  105. $valid = floatval(bcsub($extractInfo['money'], $fee, 2));
  106. return app('json')->successful([
  107. 'rate' => $rate, // 费率
  108. 'min' => $min, // 最少收费
  109. 'max' => $max, // 最大收费
  110. 'fee' => $fee, // 当前收费
  111. 'valid' => $valid, // 实际到帐
  112. ]);
  113. }
  114. /**
  115. * @api {post} /extract/cash 提现申请
  116. * @apiName ExtractCash
  117. * @apiGroup Brokerage
  118. *
  119. * @apiBody {string} alipay_code 支付宝号码
  120. * @apiBody {string="alipay","bank","weixin"} extract_type 提现类型
  121. * @apiBody {float} money 提现金额
  122. * @apiBody {string} bankName 开户行
  123. * @apiBody {string} cardnum 卡号
  124. * @apiBody {string} name 微信实名/银行户名/支付宝帐号
  125. * @apiBody {string} weixin 微信实名
  126. *
  127. * @apiSuccessExample:
  128. * {
  129. *
  130. * }
  131. * @apiErrorExample:
  132. * {
  133. *
  134. * }
  135. *
  136. */
  137. public function cash(Request $request)
  138. {
  139. $extractInfo = UtilService::postMore([
  140. ['alipay_code', ''],
  141. ['extract_type', ''],
  142. ['money', 0],
  143. ['name', ''],
  144. ['bankname', ''],
  145. ['cardnum', ''],
  146. ['weixin', ''],
  147. ], $request);
  148. $user = $request->user();
  149. list($suc, $msg) = PaymentService::user_request_extract($user, $extractInfo);
  150. if ($suc) {
  151. return app('json')->successful('申请提现成功!');
  152. } else {
  153. return app('json')->fail($msg);
  154. }
  155. }
  156. /**
  157. * 即时提現,參數返回值同 cash
  158. * 本接口支持即時到帐,调用三方支付提现 API
  159. *
  160. * 用户账户必须满足以下规则。
  161. * 1. 满足对应提现通道要求,微信支付:每个用户1天限1次,每次限200, 平台每日限额 200,000
  162. * 2. 15 日内必须有退款订单。且30日内退款订单金额 > 提现金额
  163. * 3. 其他基本需求(最低提现限额等等)
  164. *
  165. * 不满足体现条件,则交给人工处理
  166. * API 执行失败,则必须重新提交提现请求,因为人工操作也是失败。
  167. * 如果API 返回 SYSTEMERROR, 则按照原订单参数重新提交。
  168. */
  169. public function flash_cash(Request $request)
  170. {
  171. // 提交申请
  172. $extractInfo = UtilService::postMore([
  173. ['alipay_code', ''],
  174. ['extract_type', ''],
  175. ['money', 0],
  176. ['name', ''],
  177. ['bankname', ''],
  178. ['cardnum', ''],
  179. ['weixin', ''],
  180. ], $request);
  181. $user = $request->user();
  182. list($suc, $msg) = PaymentService::user_request_extract($user, $extractInfo);
  183. if (!$suc) {
  184. return app('json')->fail($msg);
  185. }
  186. $row = $msg;
  187. $extractRow = UserExtract::getUserExtractInfo($user['uid'], $row['extract_type'], $row['add_time']);
  188. if (!$extractRow) {
  189. Log::error('Unbelievable error getUserExtractInfo(' . $user['uid'] . ' ' . $row['add_time'] . ')');
  190. return app('json')->fail('提交申请失败');
  191. }
  192. // 检查条件
  193. list($suc, $msg) = PaymentService::user_extract_auto_conditions($user);
  194. if (!$suc) {
  195. // 条件不满足,不进一步处理,相当与交给人工处理
  196. return app('json')->fail($msg);
  197. }
  198. // API
  199. list($suc, $ec, $es) = PaymentService::extract_by_api($extractRow);
  200. if (!$suc) {
  201. // API 失败 直接拒绝,因为重试也会大概率失败
  202. list($ok, $msg) = PaymentService::user_extract_reject($extractRow['id'], $es);
  203. if (!$ok) {
  204. errlog('user_extract_reject()' . $extractRow['id']);
  205. }
  206. return app('json')->fail($es);
  207. } else {
  208. list($ok, $msg) = PaymentService::user_extract_passed($extractRow);
  209. if (!$ok) {
  210. errlog('user_extract_passed()' . $extractRow['id']);
  211. }
  212. return app('json')->successful('提现成功');
  213. }
  214. }
  215. }