BetbyController.php 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616
  1. <?php
  2. namespace App\Http\Controllers\Game;
  3. use App\Game\GlobalUserInfo;
  4. use App\Game\Services\BetbyService;
  5. use App\Game\Services\OuroGameService;
  6. use App\Game\Services\PlatformService;
  7. use App\Http\helper\NumConfig;
  8. use App\IpLocation;
  9. use App\Notification\TelegramBot;
  10. use App\Util;
  11. use App\Utility\SetNXLock;
  12. use Illuminate\Http\Request;
  13. use Illuminate\Http\JsonResponse;
  14. use App\Http\Controllers\Controller;
  15. use App\Game\BetBy\TransactionItem;
  16. use App\Game\BetBy\BetSlipItem;
  17. use Illuminate\Support\Facades\DB;
  18. use Illuminate\Support\Facades\Redis;
  19. class BetbyController extends Controller
  20. {
  21. protected $betbyService;
  22. public function __construct()
  23. {
  24. }
  25. /**
  26. * Ping接口
  27. * 该方法用于检查服务的连通性。它返回当前服务器时间戳。
  28. *
  29. * @return JsonResponse
  30. */
  31. public function ping()
  32. {
  33. return response()->json(['timestamp' => time()]);
  34. }
  35. private function telLog($arg)
  36. {
  37. $this->Log($arg);
  38. // (new TelegramBot())->sendMsgWithEnv(json_encode($arg));
  39. }
  40. private function Log($arg)
  41. {
  42. Util::WriteLog('betby', $arg);
  43. }
  44. private function decode(Request $request)
  45. {
  46. if(IpLocation::getIP()!="18.159.254.173"){
  47. throw new \Exception("wrong betby center ip");
  48. }
  49. return $request->payload;
  50. }
  51. private $errors = ['1005' => ['code' => 1005, 'message' => 'Player is blocked'],
  52. '1006' => ['code' => 1006, 'message' => 'Player not found'],
  53. '1007' => ['code' => 1007, 'message' => 'Session is expired'],
  54. '2001' => ['code' => 2001, 'message' => 'Not enough money'],
  55. '2002' => ['code' => 2002, 'message' => 'Invalid currency'],
  56. '2004' => ['code' => 2004, 'message' => 'Bad request'],
  57. '2005' => ['code' => 2005, 'message' => 'Invalid JWT token'],
  58. '24' => ['code' => 24, 'message' => 'Message'],
  59. '3001' => ['code' => 3001, 'message' => 'Bonus not found'],
  60. '4001' => ['code' => 4001, 'message' => 'Player limits exceeded'],
  61. '4002' => ['code' => 4002, 'message' => 'Maximum bonus bet limit exceeded']];
  62. private function getError($code)
  63. {
  64. Util::WriteLog('betbyerr', $code);
  65. return response()->json($this->errors[$code], 200);
  66. }
  67. /**
  68. * Bet Make接口
  69. * 该方法用于处理用户下注请求。
  70. *
  71. * 请求参数:
  72. * - `amount` (integer): 金额
  73. * - `currency` (string): 货币
  74. * - `player_id` (string): 玩家ID
  75. * - `session_id` (string): 会话ID
  76. * - `bonus_id` (string, 可选): 奖金ID
  77. * - `bonus_type` (string, 可选): 奖金类型
  78. * - `transaction` (array): 交易信息
  79. * - `betslip` (array): 投注信息
  80. * - `potential_win` (integer): 潜在赢利金额
  81. * - `potential_comboboost_win` (integer, 可选): 潜在奖金赢利金额
  82. *
  83. * @param Request $request
  84. * @return JsonResponse
  85. */
  86. public function betMake(Request $request)
  87. {
  88. //{"data":{"iat":1721234686,"exp":1721234986,"jti":"d245e3d473ba4393a4bbca47de0aa117","iss":"2424946715804176387","aud":"2424948766365847552","payload":{"amount":100,"currency":"BRL","player_id":"917c74999c-b53b-eb1a-0004478930","session_id":"1315946882","transaction":{"id":"2425239357104460126","betslip_id":"2425239357104460126","player_id":"2425378291990007812","operator_id":"2424946715804176387","operator_brand_id":"2424948766365847552","ext_player_id":"917c74999c-b53b-eb1a-0004478930","timestamp":1721234686.7508757,"amount":100,"currency":"BRL","operation":"bet","cross_rate_euro":"0.167429"},"betslip":{"id":"2425239357104460126","timestamp":1721234686.739,"player_id":"2425378291990007812","operator_id":"2424946715804176387","operator_brand_id":"2424948766365847552","ext_player_id":"917c74999c-b53b-eb1a-0004478930","currency":"BRL","sum":100,"type":"1\/1","k":"2.3","is_quick_bet":false,"bets":[{"id":"2425239357104460127","event_id":"2424813655611805741","sport_id":"1","tournament_id":"2291720548469837854","category_id":"1666080098480164864","live":true,"sport_name":"Soccer","category_name":"Mexico","tournament_name":"U23 Liga MX","competitor_name":["Necaxa U23","Rayados de Monterrey U23"],"market_name":"Total","outcome_name":"over 9","scheduled":1721228400,"odds":"2.3"}],"accept_odds_change":true},"potential_win":230,"potential_comboboost_win":0},"nbf":1721234681}}
  89. $data = $this->decode($request);
  90. // (new TelegramBot())->sendMsgWithEnv($data);
  91. if(!$data)return $this->getError(2005);
  92. if(!isset($data['transaction'])){
  93. return $this->getError(2004);
  94. }
  95. $transaction = new TransactionItem($data['transaction']);
  96. $betslip = new BetSlipItem($data['betslip']);
  97. //TODO: 我们开始扩第二个站的时候,这里要进行分片逻辑了。。。要根据player_id通知其他服务器
  98. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $data['player_id']);
  99. $this->telLog(compact('user', 'data', 'betslip'));
  100. // TODO: 在此处实现具体的下注逻辑
  101. //未找到用户
  102. if(!$user)return $this->getError(1006);
  103. //货币不对
  104. if($data['currency']!=env('CONFIG_24680_CURRENCY'))return $this->getError(2002);
  105. if (Redis::exists('discard_'.$transaction->id)) {
  106. return $this->getError(2004);
  107. }
  108. $res = SetNXLock::getExclusiveLock('betby_bet_'.$transaction->id, 86400);
  109. if (!$res) {
  110. return $this->getError(2004);
  111. }
  112. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  113. $score = $score*NumConfig::NUM_VALUE;
  114. $userScoreKey = 'userScore-bet-'.$user->UserID;
  115. if(Redis::exists($userScoreKey)){
  116. $score=Redis::get($userScoreKey);
  117. }else{
  118. $userScoreKey = 'userScore-bet-'.$user->UserID;
  119. Redis::set($userScoreKey,$score);
  120. }
  121. //没钱
  122. if ($score < $transaction->amount) return $this->getError(2001);
  123. $userScoreKey = 'userScore-bet-'.$user->UserID;
  124. Redis::set($userScoreKey,$score-$data['amount']);
  125. DB::connection('write')->table('QPTreasureDB.dbo.GameScoreInfo')->where('UserID', $user->UserID)->decrement('Score', $data['amount']);
  126. OuroGameService::notifyWebHall($user->UserID,"","pay_finish",["Golds"=>$score-$data['amount'],"PayNum"=>-$data['amount'],"event"=>"pay"]);
  127. //用户cancel的时候
  128. Redis::set('bet_amount_'.$transaction->id,$data['amount']?:1);
  129. Redis::expire('bet_amount_'.$transaction->id,86400*31);
  130. PlatformService::platformBet('betby','',$data['amount'],$user->UserID);
  131. Util::WriteLog('betbytt',['id' => $transaction->id,//Unique identifier for transaction assigned by Partner 我们要自己生成,可以用数据库id吧
  132. 'ext_transaction_id' => $transaction->id,
  133. 'parent_transaction_id' => null,//make是一切的开始,没有parent
  134. 'user_id' => $data['player_id'],
  135. 'operation' => 'bet',
  136. 'amount' => $data['amount'],
  137. 'currency' => $data['currency'],
  138. 'before_score' => $score,
  139. 'balance' => $score-$data['amount'], // 示例余额
  140. 'operator_bonus_amount' => 0 // 示例奖金金额
  141. ]);
  142. return response()->json([
  143. 'id' => $transaction->id,//Unique identifier for transaction assigned by Partner 我们要自己生成,可以用数据库id吧
  144. 'ext_transaction_id' => $transaction->id,
  145. 'parent_transaction_id' => null,//make是一切的开始,没有parent
  146. 'user_id' => $data['player_id'],
  147. 'operation' => 'bet',
  148. 'amount' => $data['amount'],
  149. 'currency' => $data['currency'],
  150. 'balance' => $score-$data['amount'], // 示例余额
  151. 'operator_bonus_amount' => 0 // 示例奖金金额
  152. ]);
  153. }
  154. /**
  155. * Bet Commit接口
  156. * 该方法用于确认下注交易。
  157. *
  158. * 请求参数:
  159. * - `bet_transaction_id` (string): 交易ID
  160. *
  161. * @param Request $request
  162. * @return JsonResponse
  163. */
  164. public function betCommit(Request $request)
  165. {
  166. $data = $this->decode($request);
  167. $this->telLog(compact('data'));
  168. // TODO: 在此处实现具体的确认逻辑
  169. return response()->json();
  170. }
  171. /**
  172. * Bet Settlement接口
  173. * 该方法用于结算下注交易。
  174. *
  175. * 请求参数:
  176. * - `bet_transaction_id` (string): 交易ID
  177. * - `status` (string): 状态
  178. *
  179. * @param Request $request
  180. * @return JsonResponse
  181. */
  182. public function betSettlement(Request $request)
  183. {
  184. $data = $this->decode($request);
  185. $this->telLog(compact('data'));
  186. // TODO: 在此处实现具体的结算逻辑
  187. $betLid = @$data['bet_transaction_id'];
  188. Redis::del('bet_amount_'.$betLid);
  189. return response()->json();
  190. }
  191. /**
  192. * Bet Refund接口
  193. * 该方法用于处理退款请求。
  194. *
  195. * 请求参数:
  196. * - `bet_transaction_id` (string): 交易ID
  197. * - `reason` (string): 退款原因
  198. * - `bonus_id` (string, 可选): 奖金ID
  199. * - `transaction` (array): 交易信息
  200. *
  201. * @param Request $request
  202. * @return JsonResponse
  203. */
  204. public function betRefund(Request $request)
  205. {
  206. $data = $this->decode($request);
  207. $transaction = new TransactionItem($data['transaction']);
  208. $this->telLog(compact('data'));
  209. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $transaction->ext_player_id);
  210. $bet=0;
  211. $key='bet_amount_'.$data['bet_transaction_id'];
  212. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  213. $score = $score*NumConfig::NUM_VALUE;
  214. $res = SetNXLock::getExclusiveLock('refund_betby_'.$transaction->id, 86400);
  215. if (!$res) {
  216. Util::WriteLog('betbyeee','######has refunded######_'.$transaction->id);
  217. return response()->json([
  218. 'id' => $transaction->id,
  219. 'ext_transaction_id' => $transaction->id,
  220. 'parent_transaction_id' => $transaction->parent_transaction_id,
  221. 'user_id' => $transaction->ext_player_id,
  222. 'operation' => 'refund',
  223. 'amount' => $transaction->amount,
  224. 'currency' => $transaction->currency,
  225. 'balance' => $score+$bet // 示例余额
  226. ]);
  227. // return $this->getError(2004);
  228. }
  229. if(Redis::exists($key)){
  230. $bet=$transaction->amount;
  231. if($bet>0){
  232. DB::connection('write')->table('QPTreasureDB.dbo.GameScoreInfo')->where('UserID', $user->UserID)->increment('Score', $bet);
  233. PlatformService::platformBet('betby','',-$bet,$user->UserID);
  234. $userScoreKey = 'userScore-bet-'.$user->UserID;
  235. Redis::set($userScoreKey,$score+$bet);
  236. OuroGameService::notifyWebHall($user->UserID,"","pay_finish",["Golds"=>$score+$bet,"PayNum"=>$bet,"event"=>"pay"]);
  237. Util::WriteLog('betrefund', ['before_score'=>$score,'after_score'=>$score+$bet ,'data' => $data]);
  238. }
  239. //Redis::del($key);
  240. }
  241. Util::WriteLog('betbytt',['id' => $transaction->id,
  242. 'ext_transaction_id' => $transaction->id,
  243. 'parent_transaction_id' => $transaction->parent_transaction_id,
  244. 'user_id' => $transaction->ext_player_id,
  245. 'operation' => 'refund',
  246. 'before_score' => $score,
  247. 'amount' => $transaction->amount,
  248. 'currency' => $transaction->currency,
  249. 'balance' => $score+$bet // 示例余额
  250. ]);
  251. return response()->json([
  252. 'id' => $transaction->id,
  253. 'ext_transaction_id' => $transaction->id,
  254. 'parent_transaction_id' => $transaction->parent_transaction_id,
  255. 'user_id' => $transaction->ext_player_id,
  256. 'operation' => 'refund',
  257. 'amount' => $transaction->amount,
  258. 'currency' => $transaction->currency,
  259. 'balance' => $score+$bet // 示例余额
  260. ]);
  261. }
  262. /**
  263. * Bet Win接口
  264. * 该方法用于处理用户赢取奖金的请求。
  265. *
  266. * 请求参数:
  267. * - `amount` (integer): 金额
  268. * - `currency` (string): 货币
  269. * - `is_cashout` (boolean): 是否兑现
  270. * - `bet_transaction_id` (string): 交易ID
  271. * - `transaction` (array): 交易信息
  272. * - `is_snr_lost` (boolean, 可选): 是否为“无风险免费投注”失利
  273. * - `selections` (array): 投注选择项
  274. * - `odds` (string, 可选): 赔率
  275. * - `bonus_id` (string, 可选): 奖金ID
  276. * - `bonus_type` (string, 可选): 奖金类型
  277. * - `comboboost_multiplier` (string, 可选): 奖金乘数
  278. *
  279. * @param Request $request
  280. * @return JsonResponse
  281. */
  282. public function betWin(Request $request)
  283. {
  284. $data = $this->decode($request);
  285. $transaction = new TransactionItem($data['transaction']);
  286. $this->telLog(compact('data', 'transaction'));
  287. // TODO: 在此处实现具体的赢取奖金逻辑
  288. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $transaction->ext_player_id);
  289. Util::WriteLog('betbytt',[$transaction->ext_player_id,$user]);
  290. $res = SetNXLock::getExclusiveLock('win_betby_'.$transaction->id, 86400);
  291. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  292. $score = $score*NumConfig::NUM_VALUE;
  293. $betKey = 'bet_amount_'.$transaction->parent_transaction_id;
  294. $win = intval($data['amount']);
  295. $bet = Redis::get($betKey)?:0;
  296. if (!$res) {
  297. return response()->json([
  298. 'id' => $transaction->id,
  299. 'ext_transaction_id' => $transaction->id,
  300. 'parent_transaction_id' => $transaction->parent_transaction_id,
  301. 'user_id' => $transaction->ext_player_id,
  302. 'operation' => 'win',
  303. 'amount' => $data['amount'],
  304. 'currency' => $data['currency'],
  305. 'balance' => $score // 示例余额
  306. ]);
  307. }
  308. $userScoreKey = 'userScore-bet-'.$user->UserID;
  309. Redis::set($userScoreKey,$score+$win);
  310. if(true){
  311. PlatformService::platformWin($user->UserID,'betby','',$win,$bet,$score+$win);
  312. OuroGameService::notifyWebHall($user->UserID,"","pay_finish",["Golds"=>$score+$win,"PayNum"=>$win,"event"=>"pay"]);
  313. }
  314. Util::WriteLog('betbytt',['id' => $transaction->id,
  315. 'ext_transaction_id' => $transaction->id,
  316. 'parent_transaction_id' => $transaction->parent_transaction_id,
  317. 'user_id' => $transaction->ext_player_id,
  318. 'operation' => 'win',
  319. 'before_score' => $score,
  320. 'amount' => $data['amount'],
  321. 'currency' => $data['currency'],
  322. 'balance' => $score+$win // 示例余额
  323. ]);
  324. return response()->json([
  325. 'id' => $transaction->id,
  326. 'ext_transaction_id' => $transaction->id,
  327. 'parent_transaction_id' => $transaction->parent_transaction_id,
  328. 'user_id' => $transaction->ext_player_id,
  329. 'operation' => 'win',
  330. 'amount' => $data['amount'],
  331. 'currency' => $data['currency'],
  332. 'balance' => $score+$win // 示例余额
  333. ]);
  334. }
  335. /**
  336. * Bet Lost接口
  337. * 该方法用于处理用户投注失败的请求。
  338. *
  339. * 请求参数:
  340. * - `bet_transaction_id` (string): 交易ID
  341. * - `amount` (integer): 金额
  342. * - `currency` (string): 货币
  343. * - `transaction` (array): 交易信息
  344. * - `selections` (array): 投注选择项
  345. *
  346. * @param Request $request
  347. * @return JsonResponse
  348. */
  349. public function betLost(Request $request)
  350. {
  351. $data = $this->decode($request);
  352. $transaction = new TransactionItem($data['transaction']);
  353. $this->telLog(compact('data', 'transaction'));
  354. // TODO: 在此处实现具体的投注失败逻辑
  355. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $transaction->ext_player_id);
  356. $res = SetNXLock::getExclusiveLock('lost_betby_'.$transaction->id, 86400);
  357. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  358. $score = $score*NumConfig::NUM_VALUE;
  359. if (!$res) {
  360. //Util::WriteLog('24680_win_error_betby',$data);
  361. //return ['success' => false,'code' => 2401,'message' => 'Session not found or expired.'];
  362. return response()->json([
  363. 'id' => $transaction->id,
  364. 'ext_transaction_id' => $transaction->id,
  365. 'parent_transaction_id' => $transaction->parent_transaction_id,
  366. 'user_id' => $transaction->ext_player_id,
  367. 'operation' => 'lost',
  368. 'balance' => $score // 示例余额
  369. ]);
  370. //return $this->getError(2004);
  371. }
  372. $betKey = 'bet_amount_'.$transaction->parent_transaction_id;
  373. $win = 0;
  374. $bet = Redis::get($betKey)?:0;
  375. if(true){
  376. PlatformService::platformWin($user->UserID,'betby','',$win,$bet,$score+$win);
  377. }
  378. return response()->json([
  379. 'id' => $transaction->id,
  380. 'ext_transaction_id' => $transaction->id,
  381. 'parent_transaction_id' => $transaction->parent_transaction_id,
  382. 'user_id' => $transaction->ext_player_id,
  383. 'operation' => 'lost',
  384. 'balance' => $score // 示例余额
  385. ]);
  386. }
  387. /**
  388. * Bet Discard接口
  389. * 该方法用于处理丢弃投注的请求。
  390. *
  391. * 请求参数:
  392. * - `ext_player_id` (string): 玩家外部ID
  393. * - `transaction_id` (string): 交易ID
  394. * - `reason` (string): 丢弃原因
  395. *
  396. * @param Request $request
  397. * @return JsonResponse
  398. */
  399. public function betDiscard(Request $request)
  400. {
  401. $data = $this->decode($request);
  402. $this->telLog(compact('data'));
  403. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $data['ext_player_id']);
  404. $bet=0;
  405. $key='bet_amount_'.$data['transaction_id'];
  406. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  407. $score = $score*NumConfig::NUM_VALUE;
  408. $res = SetNXLock::getExclusiveLock('discard_'.$data['transaction_id'], 86400);
  409. if (!$res) {
  410. return response()->json();
  411. //return ['success' => false,'code' => 2401,'message' => 'Session not found or expired.'];
  412. //return ['success' => true,'balance' => intval($score)];
  413. }
  414. if(Redis::exists($key)){
  415. $bet=Redis::get($key);
  416. if($bet>0){
  417. DB::connection('write')->table('QPTreasureDB.dbo.GameScoreInfo')->where('UserID', $user->UserID)->increment('Score', $bet);
  418. PlatformService::platformBet('betby','',-$bet,$user->UserID);
  419. $userScoreKey = 'userScore-bet-'.$user->UserID;
  420. Redis::set($userScoreKey,$score+$bet);
  421. OuroGameService::notifyWebHall($user->UserID,"","pay_finish",["Golds"=>$score+$bet,"PayNum"=>$bet,"event"=>"pay"]);
  422. Util::WriteLog('betbydiscard', ['before_score'=>$score,'after_score'=>$score+$bet ]);
  423. }
  424. Redis::del($key);
  425. }
  426. Util::WriteLog('betbytt',[
  427. 'user_id' => $data['ext_player_id'],
  428. 'operation' => 'discard',
  429. 'before_score' => $score,
  430. 'amount' => $bet,
  431. 'balance' => $score+$bet // 示例余额
  432. ]);
  433. return response()->json();
  434. }
  435. /**
  436. * Bet Rollback接口
  437. * 该方法用于处理回滚投注的请求。
  438. *
  439. * 请求参数:
  440. * - `bet_transaction_id` (string): 交易ID
  441. * - `parent_transaction_id` (string): 父交易ID
  442. * - `transaction` (array): 交易信息
  443. *
  444. * @param Request $request
  445. * @return JsonResponse
  446. */
  447. public function betRollback(Request $request)
  448. {
  449. $data = $this->decode($request);
  450. $transaction = new TransactionItem($data['transaction']);
  451. $this->telLog(compact('data', 'transaction'));
  452. // TODO: 在此处实现具体的回滚逻辑
  453. $user = GlobalUserInfo::getGameUserInfo('GlobalUID', $transaction->ext_player_id);
  454. $score = GlobalUserInfo::getScoreByUserID($user->UserID);
  455. $score = $score*NumConfig::NUM_VALUE;
  456. $key='bet_amount_'.$data['bet_transaction_id'];
  457. if(!Redis::exists($key)){
  458. Util::WriteLog('betbyeee','######not find bet key######_'.$data['bet_transaction_id']);
  459. return response()->json([
  460. 'id' => $transaction->id,
  461. 'ext_transaction_id' => $transaction->id,
  462. 'parent_transaction_id' => $transaction->parent_transaction_id,
  463. 'user_id' => $transaction->ext_player_id,
  464. 'operation' => 'rollback',
  465. 'amount' => $transaction->amount,
  466. 'currency' => $transaction->currency,
  467. 'balance' => max($score,0) // 示例余额
  468. ]);
  469. //return $this->getError(2004);
  470. }
  471. $res = SetNXLock::getExclusiveLock('rollbackn_'.$transaction->id, 86400);
  472. if (!$res) {
  473. Util::WriteLog('betbyeee','######repeat rollback######_'.$data['bet_transaction_id']);
  474. return response()->json([
  475. 'id' => $transaction->id,
  476. 'ext_transaction_id' => $transaction->id,
  477. 'parent_transaction_id' => $transaction->parent_transaction_id,
  478. 'user_id' => $transaction->ext_player_id,
  479. 'operation' => 'rollback',
  480. 'amount' => $transaction->amount,
  481. 'currency' => $transaction->currency,
  482. 'balance' => max($score,0) // 示例余额
  483. ]);
  484. //return $this->getError(2004);
  485. //return ['success' => false,'code' => 2401,'message' => 'Session not found or expired.'];
  486. //return ['success' => true,'balance' => intval($score)];
  487. }
  488. $amount = $transaction->amount;
  489. if($amount>$score){
  490. $special = ['user_id' => $user->UserID,'current_score' => $score,'rollback_amount' => $amount];
  491. Util::WriteLog('betby_roollback',compact('data',$special));
  492. $amount = $score;
  493. }
  494. //$bet = Redis::get($key)?:0;
  495. if(true){
  496. DB::connection('write')->table('QPTreasureDB.dbo.GameScoreInfo')->where('UserID', $user->UserID)->decrement('Score', $amount);
  497. PlatformService::platformWin($user->UserID,'betby','',-$amount,0,$score-$amount);
  498. }
  499. $userScoreKey = 'userScore-bet-'.$user->UserID;
  500. Redis::set($userScoreKey,max($score-$amount,0));
  501. Util::WriteLog('betbytt',['id' => $transaction->id,
  502. 'ext_transaction_id' => $transaction->id,
  503. 'parent_transaction_id' => $transaction->parent_transaction_id,
  504. 'user_id' => $transaction->ext_player_id,
  505. 'operation' => 'rollback',
  506. 'before_score' => $score,
  507. 'amount' => $transaction->amount,
  508. 'currency' => $transaction->currency,
  509. 'balance' => max($score-$amount,0) // 示例余额
  510. ]);
  511. return response()->json([
  512. 'id' => $transaction->id,
  513. 'ext_transaction_id' => $transaction->id,
  514. 'parent_transaction_id' => $transaction->parent_transaction_id,
  515. 'user_id' => $transaction->ext_player_id,
  516. 'operation' => 'rollback',
  517. 'amount' => $transaction->amount,
  518. 'currency' => $transaction->currency,
  519. 'balance' => max($score-$amount,0) // 示例余额
  520. ]);
  521. }
  522. }