Wallet.php 24 KB

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