UserExtractController.php 8.9 KB

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