Wallet.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. <?php
  2. namespace App\Http\Controllers\admin;
  3. use App\Constants\HttpStatus;
  4. use App\Http\Controllers\Controller;
  5. use App\Models\Recharge;
  6. use App\Services\BalanceLogService;
  7. use App\Services\MenuService;
  8. use App\Services\RechargeService;
  9. use App\Services\TopUpService;
  10. use Exception;
  11. use Illuminate\Support\Facades\Cache;
  12. use Illuminate\Support\Facades\DB;
  13. use Illuminate\Validation\ValidationException;
  14. use App\Models\Wallet as WalletModel;
  15. use App\Models\Withdraw;
  16. use App\Models\Config;
  17. use App\Models\PaymentOrder;
  18. use App\Models\Order;
  19. use App\Models\LhcOrder;
  20. use App\Models\Bank;
  21. use App\Models\Address;
  22. use App\Services\Payment\QianBaoService;
  23. class Wallet extends Controller
  24. {
  25. /**
  26. * 获取提现通道
  27. */
  28. public function withdrawChannel()
  29. {
  30. $list = QianBaoService::withdrawChannel();
  31. $data[] = ['label' => 'USDT', 'value' => 'USDT'];
  32. foreach ($list as $key => $item) {
  33. $data[] = ['label' => $item, 'value' => $key];
  34. }
  35. return $this->success($data);
  36. }
  37. //银行卡/支付宝列表
  38. public function bankList()
  39. {
  40. try {
  41. $params = request()->validate([
  42. 'channel' => 'nullable',
  43. 'member_id' => 'nullable',
  44. ]);
  45. $where = [];
  46. if (!empty($params['channel'])) {
  47. $where[] = ['channel', '=', $params['channel']];
  48. }
  49. if (!empty($params['member_id'])) {
  50. $where[] = ['member_id', '=', $params['member_id']];
  51. }
  52. $list = Bank::where($where)->get()->toArray();
  53. return $this->success([
  54. 'list' => $list,
  55. ]);
  56. } catch (\Exception $e) {
  57. return $this->error($e->getMessage());
  58. }
  59. }
  60. //地址列表
  61. public function address()
  62. {
  63. try {
  64. $params = request()->validate([
  65. 'member_id' => 'nullable',
  66. ]);
  67. $where = [];
  68. if (!empty($params['member_id'])) {
  69. $where[] = ['member_id', '=', $params['member_id']];
  70. }
  71. $list = Address::where($where)->get()->toArray();
  72. return $this->success([
  73. 'list' => $list,
  74. ]);
  75. } catch (\Exception $e) {
  76. return $this->error($e->getMessage());
  77. }
  78. }
  79. public function getChangeTypes()
  80. {
  81. return $this->success(BalanceLogService::$manualRecharge);
  82. }
  83. /**
  84. * @api {get} /admin/wallet/getPendingTasks 待处理任务
  85. * @apiDescription 后台每隔10秒,轮询请求这个接口获取待处理的任务数,如果有待处理的任务,则播放音乐
  86. * @apiGroup 充值管理
  87. * @apiUse result
  88. * @apiUse header
  89. * @apiVersion 1.0.0
  90. *
  91. * @apiSuccess (data) {Object} data
  92. * @apiSuccess (data) {int} data.recharge_task 待处理充值任务
  93. * @apiSuccess (data) {int} data.withdraw_task 待处理提现任务
  94. */
  95. function getPendingTasks()
  96. {
  97. $data = [
  98. 'recharge_task' => 0,
  99. 'withdraw_task' => 0,
  100. 'rmb_withdraw_task' => 0,
  101. ];
  102. if (MenuService::checkMenu(request()->user->id, 'withdraw/topUp')) {
  103. $data['recharge_task'] = Recharge::where('type', 2)->where('status', 0)->count();
  104. }
  105. if (MenuService::checkMenu(request()->user->id, 'listUSDT')) {
  106. $data['withdraw_task'] = Withdraw::where('status', 0)->count();
  107. }
  108. if (MenuService::checkMenu(request()->user->id, 'listRMB')) {
  109. $data['rmb_withdraw_task'] = PaymentOrder::where('type', 2)->where('status', 0)->count();
  110. }
  111. return $this->success($data);
  112. }
  113. //扣款
  114. public function debiting()
  115. {
  116. DB::beginTransaction();
  117. try {
  118. request()->validate([
  119. 'amount' => ['required', 'numeric', 'min:0.01'],
  120. 'member_id' => ['required', 'string', 'min:1'],
  121. 'remark' => ['required', 'string', 'min:1'],
  122. ]);
  123. $memberId = request()->input('member_id');
  124. $amount = request()->input('amount');
  125. $amount = $amount * -1;
  126. $remark = request()->input('remark');
  127. $key = 'api_request_' . md5($memberId . json_encode([
  128. 'amount' => $amount,
  129. 'remark' => $remark,
  130. 'action' => "wallet/debiting"
  131. ]));
  132. if (Cache::has($key)) throw new Exception("请求太频繁,请稍后再试。", HttpStatus::CUSTOM_ERROR);
  133. Cache::put($key, true, 3);
  134. $wallet = WalletModel::where('member_id', $memberId)->first();
  135. if (!$wallet) throw new Exception('用户不存在', HttpStatus::CUSTOM_ERROR);
  136. $availableBalance = bcadd($wallet->available_balance, $amount, 10);
  137. if ($availableBalance < 0) throw new Exception('可用余额不足', HttpStatus::CUSTOM_ERROR);
  138. BalanceLogService::addLog($memberId, $amount, $wallet->available_balance, $availableBalance, "人工扣款", null, $remark);
  139. $wallet->available_balance = $availableBalance;
  140. $wallet->save();
  141. DB::commit();
  142. } catch (ValidationException $e) {
  143. DB::rollBack();
  144. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  145. } catch (Exception $e) {
  146. DB::rollBack();
  147. return $this->error(intval($e->getCode()), $e->getMessage());
  148. }
  149. $availableBalance = floatval($availableBalance);
  150. // 去除多余0后,再用 sprintf 补足两位
  151. $availableBalance = sprintf('%.2f', $availableBalance);
  152. TopUpService::notifyTransferSuccess($memberId, "您的账户余额更新:" . ($amount > 0 ? '+' : '') . "{$amount} \n总余额为:{$availableBalance}");
  153. return $this->success();
  154. }
  155. //会员账号充值
  156. public function topUp()
  157. {
  158. DB::beginTransaction();
  159. try {
  160. request()->validate([
  161. 'amount' => ['required', 'numeric', 'min:0.01'],
  162. 'member_id' => ['required', 'string', 'min:1'],
  163. 'remark' => ['required', 'string', 'min:1'],
  164. 'change_type' => ['required', 'string', 'in:' . implode(',', BalanceLogService::$manualRecharge)],
  165. ]);
  166. $memberId = request()->input('member_id');
  167. $amount = request()->input('amount');
  168. $remark = request()->input('remark');
  169. $changeType = request()->input('change_type');
  170. $key = 'api_request_' . md5($memberId . json_encode([
  171. 'amount' => $amount,
  172. 'remark' => $remark,
  173. 'action' => "wallet/topUp"
  174. ]));
  175. if (Cache::has($key)) throw new Exception("请求太频繁,请稍后再试。", HttpStatus::CUSTOM_ERROR);
  176. Cache::put($key, true, 3);
  177. $wallet = WalletModel::where('member_id', $memberId)->first();
  178. if (!$wallet) throw new Exception('用户不存在', HttpStatus::CUSTOM_ERROR);
  179. $availableBalance = bcadd($wallet->available_balance, $amount, 10);
  180. BalanceLogService::addLog($memberId, $amount, $wallet->available_balance, $availableBalance, $changeType, null, $remark);
  181. $wallet->available_balance = $availableBalance;
  182. $wallet->save();
  183. DB::commit();
  184. } catch (ValidationException $e) {
  185. DB::rollBack();
  186. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  187. } catch (Exception $e) {
  188. DB::rollBack();
  189. return $this->error(intval($e->getCode()), $e->getMessage());
  190. }
  191. $availableBalance = floatval($availableBalance);
  192. // 去除多余0后,再用 sprintf 补足两位
  193. $availableBalance = sprintf('%.2f', $availableBalance);
  194. TopUpService::notifyTransferSuccess($memberId, "您的账户余额更新:" . ($amount > 0 ? '+' : '') . "{$amount} \n总余额为:{$availableBalance}");
  195. return $this->success();
  196. }
  197. //订单充值
  198. public function orderTopUp()
  199. {
  200. DB::beginTransaction();
  201. try {
  202. request()->validate([
  203. 'amount' => ['required', 'numeric', 'min:0.01'],
  204. 'remark' => ['required', 'string', 'min:1'],
  205. 'order_type' => ['nullable'],
  206. 'order_id' => ['nullable'],
  207. ]);
  208. $order_type = request()->input('order_type');
  209. $order_id = request()->input('order_id');
  210. $amount = request()->input('amount');
  211. $remark = request()->input('remark');
  212. $changeType = "人工充值";
  213. $key = 'api_request_' . md5($order_id . json_encode([
  214. 'amount' => $amount,
  215. 'remark' => $remark,
  216. 'action' => "wallet/orderTopUp"
  217. ]));
  218. if (Cache::has($key)) throw new Exception("请求太频繁,请稍后再试。", HttpStatus::CUSTOM_ERROR);
  219. Cache::put($key, true, 3);
  220. if ($order_type == 'sport') {
  221. $info = Order::where('id', $order_id)->first();
  222. } elseif ($order_type == 'lhc') {
  223. $info = LhcOrder::where('id', $order_id)->first();
  224. }
  225. if (!$info) {
  226. throw new \Exception('订单不存在');
  227. } else {
  228. $info->remark = $info->remark ? $info->remark. '|' . $remark : $remark;
  229. $info->save();
  230. }
  231. $memberId = $order_type == 'sport' ? $info->user_id : $info->member_id;
  232. $wallet = WalletModel::where('member_id', $memberId)->first();
  233. if (!$wallet) throw new Exception('用户不存在', HttpStatus::CUSTOM_ERROR);
  234. $availableBalance = bcadd($wallet->available_balance, $amount, 10);
  235. BalanceLogService::addLog($memberId, $amount, $wallet->available_balance, $availableBalance, $changeType, null, $remark);
  236. $wallet->available_balance = $availableBalance;
  237. $wallet->save();
  238. DB::commit();
  239. } catch (ValidationException $e) {
  240. DB::rollBack();
  241. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  242. } catch (Exception $e) {
  243. DB::rollBack();
  244. return $this->error(intval($e->getCode()), $e->getMessage());
  245. }
  246. return $this->success();
  247. }
  248. //订单扣款
  249. public function orderDebiting()
  250. {
  251. DB::beginTransaction();
  252. try {
  253. request()->validate([
  254. 'amount' => ['required', 'numeric', 'min:0.01'],
  255. 'remark' => ['required', 'string', 'min:1'],
  256. 'order_type' => ['required'],
  257. 'order_id' => ['required'],
  258. ]);
  259. $order_type = request()->input('order_type');
  260. $order_id = request()->input('order_id');
  261. $amount = request()->input('amount');
  262. $amount = $amount * -1;
  263. $remark = request()->input('remark');
  264. $key = 'api_request_' . md5($order_id . json_encode([
  265. 'amount' => $amount,
  266. 'remark' => $remark,
  267. 'action' => "wallet/orderDebiting"
  268. ]));
  269. if (Cache::has($key)) throw new Exception("请求太频繁,请稍后再试。", HttpStatus::CUSTOM_ERROR);
  270. Cache::put($key, true, 3);
  271. if ($order_type == 'sport') {
  272. $info = Order::where('id', $order_id)->first();
  273. } elseif ($order_type == 'lhc') {
  274. $info = LhcOrder::where('id', $order_id)->first();
  275. }
  276. if (!$info) {
  277. throw new \Exception('订单不存在');
  278. } else {
  279. $info->remark = $info->remark ? $info->remark. '|' . $remark : $remark;
  280. $info->save();
  281. }
  282. $memberId = $order_type == 'sport' ? $info->user_id : $info->member_id;
  283. $wallet = WalletModel::where('member_id', $memberId)->first();
  284. if (!$wallet) throw new Exception('用户不存在', HttpStatus::CUSTOM_ERROR);
  285. $availableBalance = bcadd($wallet->available_balance, $amount, 10);
  286. if ($availableBalance < 0) throw new Exception('可用余额不足', HttpStatus::CUSTOM_ERROR);
  287. BalanceLogService::addLog($memberId, $amount, $wallet->available_balance, $availableBalance, "人工扣款", null, $remark);
  288. $wallet->available_balance = $availableBalance;
  289. $wallet->save();
  290. DB::commit();
  291. } catch (ValidationException $e) {
  292. DB::rollBack();
  293. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  294. } catch (Exception $e) {
  295. DB::rollBack();
  296. return $this->error(intval($e->getCode()), $e->getMessage());
  297. }
  298. return $this->success();
  299. }
  300. /**
  301. * @api {post} /admin/wallet/verifyRecharge 审核
  302. * @apiGroup 充值管理
  303. * @apiUse result
  304. * @apiUse header
  305. * @apiVersion 1.0.0
  306. * @apiParam {int} id 充值表的ID
  307. * @apiParam {int} status 状态
  308. * - 1 通过
  309. * - 2 拒绝
  310. * @apiParam {string} [remark] 说明
  311. * - 当status =2 时,此参数必填
  312. */
  313. public
  314. function verifyRecharge()
  315. {
  316. DB::beginTransaction();
  317. try {
  318. request()->validate([
  319. 'id' => ['required', 'integer', 'min:1'],
  320. 'status' => ['required', 'integer', 'in:1,2'],
  321. 'remark' => ['required_if:status,2', 'string', 'min:1']
  322. ]);
  323. $id = request()->input('id');
  324. $status = request()->input('status');
  325. $remark = request()->input('remark', '');
  326. $recharge = Recharge::where('id', $id)
  327. ->where('type', 2)
  328. ->where('status', 0)->first();
  329. if (!$recharge) throw new Exception("数据不存在", HttpStatus::CUSTOM_ERROR);
  330. $amount = floatval($recharge->amount);
  331. if ($status == 2) {
  332. $recharge->status = 2;
  333. $recharge->save();
  334. $text = "充值结果通知\n";
  335. $text .= "充值数量:{$amount} USDT\n";
  336. $text .= "充值地址:{$recharge->to_address}\n";
  337. $text .= "状态:失败\n";
  338. $text .= "原因:{$remark}\n";
  339. TopUpService::notifyTransferSuccess($recharge->member_id, $text);
  340. } else {
  341. // 汇率
  342. $rate = Config::where('field', 'exchange_rate_rmb')->first()->val ?? 1;
  343. $recharge->status = 1;
  344. $recharge->exchange_rate = $rate;
  345. $recharge->save();
  346. $wallet = WalletModel::where('member_id', $recharge->member_id)
  347. ->first();
  348. $rate_amount = bcmul($amount, $rate, 10); // 汇率转换后分数
  349. $available_balance = bcadd($wallet->available_balance, $rate_amount, 10);
  350. BalanceLogService::addLog($recharge->member_id, $rate_amount, $wallet->available_balance, $available_balance, "充值", $recharge->id, "用户充值后台审核到账");
  351. $wallet->available_balance = $available_balance;
  352. $wallet->save();
  353. $text = "充值结果通知\n";
  354. $text .= "充值数量:{$amount} USDT\n";
  355. $text .= "充值地址:{$recharge->to_address}\n";
  356. $text .= "汇率:1 USDT = {$rate} RMB\n";
  357. $text .= "折合金额:" . number_format($rate_amount, 2) . " RMB\n";
  358. $text .= "状态:成功\n";
  359. TopUpService::notifyTransferSuccess($recharge->member_id, $text);
  360. }
  361. DB::commit();
  362. } catch (ValidationException $e) {
  363. DB::rollBack();
  364. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  365. } catch (Exception $e) {
  366. DB::rollBack();
  367. return $this->error(intval($e->getCode()), $e->getMessage());
  368. }
  369. return $this->success();
  370. }
  371. /**
  372. * @api {get} /admin/wallet 充值列表
  373. * @apiGroup 充值管理
  374. * @apiDescription 如果列表的type=2 并且 status = 0 <br/>那么代表这条数据是需要后台手动审核的
  375. * 需要有审核按钮,审核通过或者拒绝,其它情况则不需要审核按钮
  376. *
  377. * @apiUse result
  378. * @apiUse header
  379. * @apiVersion 1.0.0
  380. *
  381. * @apiParam {int} [page=1]
  382. * @apiParam {int} [limit=10]
  383. * @apiParam {string} [member_id] 房主 tg会员ID
  384. * @apiParam {int} [status] 状态
  385. * - 0 待确认
  386. * - 1 已确认
  387. * - 2 失败
  388. * - 3 已忽略
  389. * @apiSuccess (data) {Object} data
  390. * @apiSuccess (data) {int} data.total 数量
  391. * @apiSuccess (data) {Object[]} data.data 列表
  392. * @apiSuccess (data) {int} data.data.id
  393. * @apiSuccess (data) {int} data.data.member_id tg会员id
  394. * @apiSuccess (data) {string} data.data.net 链接类型
  395. * @apiSuccess (data) {string} data.data.coin 币种
  396. * @apiSuccess (data) {string} data.data.amount 充值数量
  397. * @apiSuccess (data) {string} data.data.to_address 充值地址(平台地址)
  398. * @apiSuccess (data) {string} data.data.from_address 转出地址(用户发起地址)
  399. * @apiSuccess (data) {string} data.data.txid 链上交易哈希
  400. * @apiSuccess (data) {string} data.data.block_time 区块时间
  401. * @apiSuccess (data) {string} data.data.block_height
  402. * @apiSuccess (data) {int} data.data.confirmations 确认数
  403. * @apiSuccess (data) {string} data.data.updated_at
  404. * @apiSuccess (data) {string} data.data.created_at
  405. * @apiSuccess (data) {int} data.data.status 状态
  406. * - 0 待确认
  407. * - 1 已确认
  408. * - 2 失败
  409. * - 3 已忽略
  410. * @apiSuccess (data) {int} data.data.type 充值类型
  411. * - 1 自动
  412. * - 2 手动
  413. * @apiSuccess (data) {string} data.data.image 充值转账凭证图片
  414. */
  415. public
  416. function index()
  417. {
  418. try {
  419. request()->validate([
  420. 'member_id' => ['nullable', 'string', 'min:1'],
  421. 'status' => ['nullable', 'string', 'min:0', 'max:3'],
  422. 'type' => ['nullable', 'integer', 'in:1,2'],
  423. ]);
  424. $search = request()->all();
  425. $result = RechargeService::paginate($search);
  426. foreach($result['data'] as &$item) {
  427. if ($item['from'] == 2) {
  428. $item['image'] = str_replace(env('APP_URL'), env('FOOTBALL_APP_URL').'/', $item['image']);
  429. }
  430. }
  431. } catch (ValidationException $e) {
  432. return $this->error(HttpStatus::CUSTOM_ERROR, $e->validator->errors()->first());
  433. } catch (Exception $e) {
  434. return $this->error(intval($e->getCode()));
  435. }
  436. return $this->success($result);
  437. }
  438. public
  439. function test()
  440. {
  441. // $url = "https://ydpc28.co/api/pc28/list";
  442. // $result = file_get_contents($url);
  443. // $result = json_decode($result, true);
  444. $input = request()->get('input', '');
  445. // var_dump($input);
  446. // $contractAddress = TronHelper::getContractAddress('USDT');
  447. // $result = TronHelper::getTrc20Balance('TTJ1vH18Q4K3seDcjD4912KDHHzm327rtL',$contractAddress);
  448. // $result = TronHelper::getBalance('TDeGNiweUm86JBJHQ7kXwQ8XQKtrKorHad','USDT');
  449. // dump($result);
  450. // echo WalletService::createRechargeQrCode('TDeGNiweUm86JBJHQ7kXwQ8XQKtrKorHad');
  451. // $result = TronHelper::getTrc20UsdtRecharges('TGQaMxtyWeGowy8xqwh98JNNLtc77nzZ8M');
  452. // $result = TronHelper::getTransactionConfirmations('06407fa9a2ba51c88f1ed01c2296f0069bf305477d3847d41bc3f35cf9190f74');
  453. // var_dump($result);
  454. $memberId = '6325700519';
  455. // $result = BetService::bet($memberId,$input);
  456. // $info = IssueService::findOne(['issue_no'=>'202508271120']);
  457. // $winning_numbers = explode(',',$info['winning_numbers']);
  458. // $result = IssueService::award($winning_numbers);
  459. // var_dump($result);
  460. // RechargeService::handleRechargeConfirmation('45f313ccc3a2f4113f6cc9a7511e8b5096daa1de76cb57397e152a491c17249f');
  461. // WalletService::getUserWallet('1777');
  462. // $result = GameplayRuleService::validateInput('大100');
  463. // $result = IssueService::currentLotteryResults($memberId);
  464. // var_dump($result);
  465. // $result = IssueService::getLatestIssue();
  466. // $result = ConfigService::syncExchangeRate();
  467. // $url = "https://ydpc28.co/api/pc28/list";
  468. // $result = file_get_contents($url);
  469. // $result = json_decode($result, true);
  470. // $awards = IssueService::award([7,7,7]);
  471. // $result = BetService::betSettled2('3356003',$awards);
  472. // $result = IssueService::sendLotteryImage($memberId, 3356000);
  473. // $order_no = QianBaoService::createOrderNo();
  474. // echo $order_no;
  475. // $result = QianBaoService::pay(100,$order_no);
  476. // $result = QianBaoService::payout(100,$order_no,'厦门银行','张三',1245679451259741541,QianBaoService::ALIPAY_TO_ALIPAY);
  477. // $result = PaymentOrderService::createPayout($memberId,100,QianBaoService::ALIPAY_TO_ALIPAY,'支付宝','张三','17605957777');
  478. // PaymentOrderService::sendMessage($result['chat_id'],$result['text']);
  479. // $result = PaymentOrderService::createPay($memberId,100,QianBaoService::PAY_ZFB1,'张三');
  480. // $result = mb_strlen('哈哈哈123ABC');
  481. // $order_no = SanJinService::createOrderNo();
  482. // $result = SanJinService::pay(10000,$order_no);
  483. // $result = PaymentOrderService::createPay($memberId,200,'test');
  484. // PaymentOrderService::sendMessage($result['chat_id'],$result['text'],[],$result['image']??'');
  485. // $result = KeyboardService::checkStart($memberId,'Select Language');
  486. // $awards = IssueService::award([7,2,1]);
  487. // $result = BetService::betSettled('3367329', $awards);
  488. echo "<pre>";
  489. // $result = PaymentOrderService::syncPayOrder();
  490. // $result = WalletService::violationWallet();
  491. $result = '';
  492. $result = RechargeService::syncUsdtRechargeRecords($memberId);
  493. var_dump($result);
  494. }
  495. }