Wallet.php 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610
  1. <?php
  2. namespace App\Http\Controllers\api;
  3. use App\Constants\Util;
  4. use App\Constants\HttpStatus;
  5. use App\Services\WalletService;
  6. use App\Services\ConfigService;
  7. use App\Models\Config;
  8. use App\Models\Recharge;
  9. use App\Models\Wallet as WalletModel;
  10. use App\Models\Withdraw;
  11. use App\Models\User;
  12. use App\Models\Bank;
  13. use App\Models\Address;
  14. use App\Models\RechargeChannel;
  15. use App\Models\PaymentOrder;
  16. use App\Services\BalanceLogService;
  17. use App\Services\PaymentOrderService;
  18. use App\Services\Payment\JdPayService;
  19. use App\Services\QianBaoWithdrawService;
  20. use App\Services\WithdrawService;
  21. use Illuminate\Support\Facades\DB;
  22. use Illuminate\Validation\ValidationException;
  23. use Exception;
  24. /**
  25. * 充值提现接口
  26. */
  27. class Wallet extends BaseController
  28. {
  29. //获取三斤充值通道(微信、支付宝、扫码充值)
  30. public function getChannel()
  31. {
  32. $member_id = request()->user->member_id;
  33. if (empty(request()->user->recharge_channel_group_id)) {
  34. $recharge_channel_group_id = User::where('member_id', $member_id)->value('recharge_channel_group_id');
  35. } else {
  36. $recharge_channel_group_id = request()->user->recharge_channel_group_id;
  37. }
  38. $list = RechargeChannel::getFormatChannel(1, $recharge_channel_group_id);
  39. if (!JdPayService::canUserRecharge(request()->user->id)) {
  40. $list = array_values(array_filter($list, function ($item) {
  41. return !JdPayService::isChannel($item['value'] ?? '');
  42. }));
  43. }
  44. return $this->success([
  45. 'list' => $list,
  46. ]);
  47. }
  48. /**
  49. * 创建代收订单(自动扫码充值)
  50. */
  51. public function createPay()
  52. {
  53. try {
  54. $params = request()->validate([
  55. 'amount' => ['required', 'numeric', 'min:0.01'],
  56. 'payment_type' => ['required', 'string'],
  57. ]);
  58. if ($params['payment_type'] == 'rgcz') {
  59. return $this->error('参数有误');
  60. }
  61. $member_id = request()->user->member_id;
  62. $user = User::where('member_id', $member_id)->first();
  63. if (JdPayService::isChannel($params['payment_type']) && !JdPayService::canUserRecharge($user->id)) {
  64. throw new Exception(lang("不支持此充值方式"));
  65. }
  66. //校验是否支持此充值方式
  67. $check = RechargeChannel::checkRechargeChannel($params['payment_type'], $user->recharge_channel_group_id);
  68. if ($check === false) {
  69. throw new Exception(lang("不支持此充值方式").$user->recharge_channel_group_id);
  70. }
  71. $res = PaymentOrderService::createPay($member_id, $params['amount'], $params['payment_type']);
  72. if ($res['code'] == 0) {
  73. return $this->success($res);
  74. }
  75. return $this->error($res['text']);
  76. } catch (ValidationException $e) {
  77. return $this->error($e->validator->errors()->first());
  78. } catch (\Exception $e) {
  79. return $this->error($e->getMessage());
  80. }
  81. }
  82. /**
  83. * 获取充值二维码(USDT充值)
  84. */
  85. public function scan()
  86. {
  87. try {
  88. $member_id = request()->user->member_id;
  89. $params = request()->validate([
  90. 'type' => ['required', 'string'],
  91. ]);
  92. if (strtolower($params['type']) === "trc20") {
  93. $address = Config::where('field', 'receiving_address')->first()->val;
  94. } elseif (strtolower($params['type']) === "erc20") {
  95. $address = Config::where('field', 'receiving_address_erc20')->first()->val;
  96. } else {
  97. return $this->error(lang('充值类型错误'));
  98. }
  99. $receivingType = ConfigService::getVal("receiving_type");
  100. //自动充值
  101. if ($receivingType == 1) {
  102. $res = WalletService::getRechargeImageAddress($member_id);
  103. $address = $res['address'];
  104. $qrCode = $res['full_path'];
  105. } else {
  106. //手动充值
  107. $res = WalletService::getPlatformImageAddress($address);
  108. $res['net'] = $params['type'];
  109. $qrCode = $res['full_path'];
  110. }
  111. return $this->success([
  112. 'qrcode' => $qrCode,
  113. 'address' => $address,
  114. // 'photo' => InputFile::create($qrCode),
  115. ]);
  116. } catch (ValidationException $e) {
  117. return $this->error($e->validator->errors()->first());
  118. } catch (\Exception $e) {
  119. return $this->error($e->getMessage());
  120. }
  121. }
  122. /**
  123. * 提交充值凭证(USDT手动充值)
  124. */
  125. public function recharge()
  126. {
  127. try {
  128. $params = request()->validate([
  129. 'net' => ['required', 'string'],
  130. 'amount' => ['required', 'numeric', 'min:0.01'],
  131. 'toAddress' => ['required', 'string'],
  132. 'image' => ['required', 'url'],
  133. ]);
  134. $member_id = request()->user->member_id;
  135. $user = User::where('member_id', $member_id)->first();
  136. //校验是否支持此充值方式
  137. $check = RechargeChannel::checkRechargeChannel('usdt', $user->recharge_channel_group_id);
  138. if ($check === false) {
  139. throw new Exception(lang("不支持此充值方式"));
  140. }
  141. $recharge = new Recharge();
  142. $recharge->member_id = $member_id;
  143. $recharge->net = $params['net'];
  144. $recharge->coin = "USDT";
  145. $recharge->amount = $params['amount'];
  146. $recharge->to_address = $params['toAddress'];
  147. $recharge->status = 0;
  148. $recharge->type = 2;
  149. $recharge->image = Util::replacePartInUrl($params['image']);
  150. $recharge->save();
  151. return $this->success($recharge,'提交成功');
  152. } catch (ValidationException $e) {
  153. return $this->error($e->validator->errors()->first());
  154. } catch (Exception $e) {
  155. return $this->error($e->getMessage());
  156. }
  157. }
  158. //支付宝、微信、银行卡手动充值
  159. public function paymentOrder()
  160. {
  161. try {
  162. $member_id = request()->user->member_id;
  163. $params = request()->validate([
  164. 'amount' => ['required', 'numeric', 'min:0.01'],
  165. 'payment_type' => ['required', 'integer'],
  166. ]);
  167. $member_id = request()->user->member_id;
  168. $user = User::where('member_id', $member_id)->first();
  169. //校验是否支持此充值方式
  170. $check = RechargeChannel::checkRechargeChannel('rgcz', $user->recharge_channel_group_id);
  171. if ($check === false) {
  172. throw new Exception(lang("不支持此充值方式"));
  173. }
  174. $data = [];
  175. $data['type'] = PaymentOrderService::TYPE_SELF_PAY;
  176. $data['order_no'] = PaymentOrderService::createOrderNo('rgcz_', $member_id);
  177. $data['member_id'] = $member_id;
  178. $data['fee'] = 0;
  179. $data['amount'] = number_format($params['amount'], 2, '.', '');
  180. $data['payment_type'] = $params['payment_type'];
  181. $data['channel'] = 'rgcz';
  182. $data['status'] = PaymentOrderService::STATUS_STAY;
  183. $data['remark'] = '';
  184. // 创建待处理状态的提现记录
  185. $info = PaymentOrder::create($data);
  186. return $this->success($info,'提交成功,请等待人工回复');
  187. } catch (ValidationException $e) {
  188. return $this->error($e->validator->errors()->first());
  189. } catch (Exception $e) {
  190. return $this->error($e->getMessage());
  191. }
  192. }
  193. //支付宝、微信、银行卡手动充值提交图片
  194. public function submitImage()
  195. {
  196. try {
  197. $member_id = request()->user->member_id;
  198. $params = request()->validate([
  199. 'id' => ['required', 'integer'],
  200. 'image' => ['required', 'url'],
  201. ]);
  202. // 创建待处理状态的提现记录
  203. $info = PaymentOrder::where('id', $params['id'])->where('member_id', $member_id)->first();
  204. if (!$info) {
  205. return $this->error(lang('找不到此记录'));
  206. }
  207. if ($info->status != PaymentOrderService::STATUS_USER) {
  208. return $this->error(lang('待处理中,请稍后'));
  209. }
  210. $info->image = $params['image'];
  211. $info->status = PaymentOrderService::STATUS_AUDIT;
  212. $info->save();
  213. return $this->success($info,'提交成功,请等待人工审核');
  214. } catch (ValidationException $e) {
  215. return $this->error($e->validator->errors()->first());
  216. } catch (Exception $e) {
  217. return $this->error($e->getMessage());
  218. }
  219. }
  220. /**
  221. * 获取提现通道
  222. */
  223. public function withdrawChannel()
  224. {
  225. $member_id = request()->user->member_id;
  226. if (empty(request()->user->recharge_channel_group_id)) {
  227. $recharge_channel_group_id = User::where('member_id', $member_id)->value('recharge_channel_group_id');
  228. } else {
  229. $recharge_channel_group_id = request()->user->recharge_channel_group_id;
  230. }
  231. $list = RechargeChannel::getFormatChannel(2, $recharge_channel_group_id);
  232. return $this->success($list);
  233. }
  234. //USDT提现
  235. public function withdraw()
  236. {
  237. $error_code = -1;
  238. try {
  239. $member_id = request()->user->member_id;
  240. $params = request()->validate([
  241. 'amount' => ['required', 'numeric', 'min:0.01'],
  242. 'address' => ['required', 'string'],
  243. 'safe_word' => ['required'],
  244. ]);
  245. $user = User::where('member_id', $member_id)->first();
  246. if (empty($user->payment_password)) {
  247. $error_code = 10011;
  248. throw new Exception(lang("请先设置资金密码"));
  249. }
  250. //校验资金密码
  251. if (!password_verify($params['safe_word'], $user->payment_password)) {
  252. throw new Exception(lang('资金密码错误'));
  253. }
  254. //校验是否支持此提现方式
  255. $check = RechargeChannel::checkWithdrawChannel('usdt', $user->recharge_channel_group_id);
  256. if ($check === false) {
  257. throw new Exception(lang("不支持此提现方式"));
  258. }
  259. $amount = $params['amount'];
  260. $address = $params['address'];
  261. //校验提现金额的限制
  262. if ($amount < $check->min || $amount > $check->max) {
  263. throw new Exception(lang("提现金额超出限制"));
  264. }
  265. $serviceCharge = (new WithdrawService())->serviceCharge;
  266. $real = bcsub($amount, $serviceCharge, 10);
  267. $real = floatval($real);
  268. if ($amount <= $serviceCharge) {
  269. throw new Exception(lang("提现不能少于") . "{$serviceCharge} USDT");
  270. }
  271. $wallet = WalletModel::where('member_id', $member_id)->first();
  272. $temp = floatval($wallet->available_balance);
  273. // 汇率
  274. $rate = Config::where('field', 'exchange_rate_rmb')->first()->val ?? 1;
  275. $exchange_rate_difference = Config::where('field', 'exchange_rate_difference')->first()->val ?? 0;
  276. $rate = bcadd($rate, $exchange_rate_difference, 2);
  277. $rate_usdt_amount = bcdiv($temp, $rate, 2); // 钱包可用余额 折合USDT
  278. $rate_rmb_amount = bcmul($amount, $rate, 2); // 提现金额 折合RMB
  279. if ($amount > $rate_usdt_amount) {
  280. throw new Exception(lang("余额不足") . "{$serviceCharge} USDT");
  281. }
  282. $wallet = WalletModel::where('member_id', $member_id)->first();
  283. $changeAmount = bcmul(($amount * -1), $rate, 2);
  284. $beforeBalance = $wallet->available_balance;
  285. $afterBalance = bcsub($wallet->available_balance, $rate_rmb_amount, 2);
  286. $wallet->available_balance = $afterBalance;
  287. $wallet->save();
  288. $withdraw = Withdraw::create([
  289. 'member_id' => $member_id,
  290. 'amount' => $amount,
  291. 'service_charge' => $serviceCharge,
  292. 'to_account' => $real,
  293. 'address' => $address,
  294. 'exchange_rate' => $rate,
  295. 'status' => 0,
  296. 'after_balance' => $afterBalance
  297. ]);
  298. BalanceLogService::addLog($member_id, $changeAmount, $beforeBalance, $afterBalance, '提现', $withdraw->id, '');
  299. return $this->success($withdraw,'提交成功');
  300. } catch (ValidationException $e) {
  301. return $this->error($e->validator->errors()->first());
  302. } catch (\Exception $e) {
  303. return $this->error($e->getMessage(), [], $error_code);
  304. }
  305. }
  306. /**
  307. * 提现(手动到账): DF001 支付宝转卡; DF002 支付宝转支付宝; DF005数字人民币; JDpay JD钱包; rgtx(人工提现,手动打款)
  308. */
  309. public function payout() {
  310. $error_code = -1;
  311. try {
  312. $params = request()->validate([
  313. 'amount' => ['required', 'numeric', 'min:0.01'],
  314. 'channel' => ['required', 'string', 'in:DF001,DF002,DF005,JDpay,jdpay,rgtx'],
  315. 'bank_name' => ['required', 'string'],
  316. 'account' => ['required', 'string'],
  317. 'card_no' => ['required', 'string'],
  318. 'safe_word' => ['required'],
  319. ]);
  320. $member_id = request()->user->member_id;
  321. $user = User::where('member_id', $member_id)->first();
  322. if (empty($user->payment_password)) {
  323. $error_code = 10011;
  324. throw new Exception(lang("请先设置资金密码"));
  325. }
  326. //校验资金密码
  327. if (!password_verify($params['safe_word'], $user->payment_password)) {
  328. throw new Exception(lang('资金密码错误'));
  329. }
  330. //校验是否支持此提现方式
  331. $check = RechargeChannel::checkWithdrawChannel($params['channel'], $user->recharge_channel_group_id);
  332. if ($check === false) {
  333. throw new Exception(lang("不支持此提现方式"));
  334. }
  335. $amount = $params['amount'];
  336. //校验提现金额的限制
  337. if ($amount < $check->min || $amount > $check->max) {
  338. throw new Exception(lang("提现金额超出限制"));
  339. }
  340. if ($params['channel'] == 'rgtx') {
  341. DB::beginTransaction();
  342. try {
  343. $wallet = WalletService::findOne(['member_id' => $member_id]);
  344. if (!$wallet) throw new Exception('钱包不存在', HttpStatus::CUSTOM_ERROR);
  345. $balance = $wallet->available_balance;
  346. if (bccomp($balance, $amount, 2) < 0) {
  347. throw new Exception("您的钱包余额不足!", HttpStatus::CUSTOM_ERROR);
  348. }
  349. $available_balance = bcsub($balance, $amount, 10);
  350. // 先预扣款(锁定资金)
  351. $wallet->available_balance = $available_balance;
  352. if (!$wallet->save()) throw new Exception('钱包更新失败!', HttpStatus::CUSTOM_ERROR);
  353. $rate = RechargeChannel::where('type', 'rgtx')->value('rate');
  354. $rate = $rate ?? 1;
  355. $data = [];
  356. $data['type'] = PaymentOrderService::TYPE_SELF_PAYOUT;
  357. $data['order_no'] = PaymentOrderService::createOrderNo('rgtx' . $data['type'] . '_', $member_id);
  358. $data['member_id'] = $member_id;
  359. $data['fee'] = $amount * $rate;
  360. $data['amount'] = number_format($amount, 2, '.', '');
  361. $data['channel'] = $params['channel'];
  362. $data['bank_name'] = $params['bank_name'];
  363. $data['account'] = $params['account'];
  364. $data['card_no'] = $params['card_no'];
  365. $data['status'] = PaymentOrderService::STATUS_STAY;
  366. // 创建待处理状态的提现记录
  367. $info = PaymentOrder::create($data);
  368. // 记录余额变动日志
  369. BalanceLogService::addLog($member_id, $amount * -1, $balance, $available_balance, '人工提现', $info->id, '钱宝提现费率:'.($rate * 100) . '%');
  370. $balance = bcadd($available_balance, 0, 2);
  371. DB::commit();
  372. } catch (Exception $e) {
  373. DB::rollBack();
  374. if ($e->getCode() === HttpStatus::CUSTOM_ERROR) {
  375. return $this->error($e->getMessage());
  376. }
  377. }
  378. return $this->success($info,'提交成功');
  379. } else {
  380. $res = QianBaoWithdrawService::createOrder($member_id, $params['amount'], $params['channel'], $params['bank_name'], $params['account'], $params['card_no']);
  381. if ($res['code'] == 0) {
  382. return $this->success($res,'提交成功');
  383. }
  384. return $this->error($res['text']);
  385. }
  386. } catch (ValidationException $e) {
  387. return $this->error($e->validator->errors()->first());
  388. } catch (\Exception $e) {
  389. return $this->error($e->getMessage(), [], $error_code);
  390. }
  391. }
  392. /**
  393. * 提现(自动到账): DF001 支付宝转卡/DF002 支付宝转支付宝
  394. */
  395. public function autoPayout()
  396. {
  397. try {
  398. $params = request()->validate([
  399. 'amount' => ['required', 'numeric', 'min:0.01'],
  400. 'channel' => ['required', 'string'],
  401. 'bank_name' => ['required', 'string'],
  402. 'account' => ['required', 'string'],
  403. 'card_no' => ['required', 'string'],
  404. ]);
  405. $member_id = request()->user->member_id;
  406. $res = PaymentOrderService::autoCreatePayout($member_id, $params['amount'], $params['channel'], $params['bank_name'], $params['account'], $params['card_no']);
  407. if (empty($res['text'])) {
  408. return $this->success($res,'提交成功');
  409. }
  410. return $this->error($res['text']);
  411. } catch (ValidationException $e) {
  412. return $this->error($e->validator->errors()->first());
  413. } catch (Exception $e) {
  414. return $this->error($e->getMessage());
  415. }
  416. }
  417. public function addBank()
  418. {
  419. try {
  420. $params = request()->validate([
  421. 'id' => 'nullable|integer',
  422. 'channel' => 'required',
  423. 'account' => 'required',
  424. 'card_no' => 'required',
  425. 'bank_name' => 'required',
  426. 'alias' => 'nullable',
  427. ]);
  428. $member_id = request()->user->member_id;
  429. if (!empty($params['id'])) {
  430. $info = Bank::where('id', $params['id'])->where('member_id', $member_id)->first();
  431. if (empty($info)) throw new Exception(lang('找不到此记录'));
  432. $info->channel = $params['channel'];
  433. $info->account = $params['account'];
  434. $info->card_no = $params['card_no'];
  435. $info->bank_name = $params['bank_name'];
  436. $info->alias = $params['alias'] ?? '';
  437. $info->save();
  438. } else {
  439. $count = Bank::where('member_id', $member_id)->where('channel', $params['channel'])->count();
  440. if ($count >= 5) throw new Exception(lang('已达添加上限'));
  441. Bank::create([
  442. 'member_id' => $member_id,
  443. 'channel' => $params['channel'],
  444. 'account' => $params['account'],
  445. 'card_no' => $params['card_no'],
  446. 'bank_name' => $params['bank_name'],
  447. 'alias' => $params['alias'] ?? '',
  448. ]);
  449. }
  450. return $this->success([],'提交成功');
  451. } catch (ValidationException $e) {
  452. return $this->error($e->validator->errors()->first());
  453. } catch (\Exception $e) {
  454. return $this->error($e->getMessage());
  455. }
  456. }
  457. public function delBank()
  458. {
  459. try {
  460. $params = request()->validate([
  461. 'id' => 'required|integer',
  462. ]);
  463. $member_id = request()->user->member_id;
  464. $info = Bank::where('id', $params['id'])->where('member_id', $member_id)->first();
  465. if (empty($info)) throw new Exception(lang('找不到此记录'));
  466. $info->delete();
  467. return $this->success([],'删除成功');
  468. } catch (ValidationException $e) {
  469. return $this->error($e->validator->errors()->first());
  470. } catch (\Exception $e) {
  471. return $this->error($e->getMessage());
  472. }
  473. }
  474. public function bankList()
  475. {
  476. try {
  477. $params = request()->validate([
  478. 'channel' => 'nullable',
  479. ]);
  480. $member_id = request()->user->member_id;
  481. $where = !empty($params['channel']) ? ['channel' => $params['channel']] : [];
  482. $list = Bank::where('member_id', $member_id)->where($where)->get()->toArray();
  483. return $this->success([
  484. 'list' => $list,
  485. ]);
  486. } catch (\Exception $e) {
  487. return $this->error($e->getMessage());
  488. }
  489. }
  490. public function addAddress()
  491. {
  492. try {
  493. $params = request()->validate([
  494. 'id' => 'nullable|integer',
  495. 'address' => 'required',
  496. 'alias' => 'nullable',
  497. ]);
  498. $member_id = request()->user->member_id;
  499. if (!empty($params['id'])) {
  500. $info = Address::where('id', $params['id'])->where('member_id', $member_id)->first();
  501. if (empty($info)) throw new Exception(lang('找不到此记录'));
  502. $info->address = $params['address'];
  503. $info->alias = $params['alias'] ?? '';
  504. $info->save();
  505. } else {
  506. $count = Address::where('member_id', $member_id)->where('address', $params['address'])->count();
  507. if ($count >= 5) throw new Exception(lang('已达添加上限'));
  508. Address::create([
  509. 'member_id' => $member_id,
  510. 'address' => $params['address'],
  511. 'alias' => $params['alias'] ?? '',
  512. ]);
  513. }
  514. return $this->success([],'提交成功');
  515. } catch (ValidationException $e) {
  516. return $this->error($e->validator->errors()->first());
  517. } catch (\Exception $e) {
  518. return $this->error($e->getMessage());
  519. }
  520. }
  521. public function delAddress()
  522. {
  523. try {
  524. $params = request()->validate([
  525. 'id' => 'required|integer',
  526. ]);
  527. $member_id = request()->user->member_id;
  528. $info = Address::where('id', $params['id'])->where('member_id', $member_id)->first();
  529. if (empty($info)) throw new Exception(lang('找不到此记录'));
  530. $info->delete();
  531. return $this->success([],'删除成功');
  532. } catch (ValidationException $e) {
  533. return $this->error($e->validator->errors()->first());
  534. } catch (\Exception $e) {
  535. return $this->error($e->getMessage());
  536. }
  537. }
  538. public function address()
  539. {
  540. try {
  541. $member_id = request()->user->member_id;
  542. $list = Address::where('member_id', $member_id)->get()->toArray();
  543. return $this->success([
  544. 'list' => $list,
  545. ]);
  546. } catch (\Exception $e) {
  547. return $this->error($e->getMessage());
  548. }
  549. }
  550. }