Betorder.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567
  1. <?php
  2. namespace App\Api\Controller;
  3. use Biz\Money\Moneycopy;
  4. use Biz\Money\Money;
  5. use Biz\Account\AccountManager;
  6. /*
  7. * Class Register
  8. * @package App\Api\Controller
  9. * User: junghwi
  10. * Date: 2019/3/21
  11. */
  12. class Betorder extends BaseController{
  13. private $simplex_money = 0; //单式投注总额
  14. public function init() {
  15. $this->time = date('Y-m-d H:i:s',time());
  16. $this->MC = new Moneycopy();
  17. $this->M = new Money();
  18. $this->accountManager = new AccountManager();
  19. $this->commonFunction = C()->get('commonFunction');
  20. }
  21. /**
  22. * 单式串式分类
  23. * @param [array] $res 投注数据
  24. * @return [array] $data
  25. *
  26. */
  27. public function Classify(array $res){
  28. if(empty($res)){
  29. Render('', '7003',lang('Errors','Api')->get('error-7003'));
  30. }
  31. $data['single'] = [];//单式
  32. $data['bunch'] = [];//串式
  33. $last[] = $res[count($res)-1];
  34. unset($res[count($res)-1]);
  35. foreach ($last as $k =>$v){
  36. foreach ($v as $kk =>$vv){
  37. $num = $kk+1;
  38. //验证赛事是否存在
  39. $this->verify_match($vv['ballId'],$vv['match_id'],$num);
  40. //验证球类代码是否有效
  41. $this->verify_gameType($vv['ballId'],$num);
  42. //验证赛事下赔率玩法是否有效
  43. $this->verify_odds($vv['ballId'],$vv['odds_only'],$vv['odds'],$vv['condition'],$num);
  44. // if(empty($vv['condition'])){
  45. // $msg = '第'.$num.'条'.lang('Errors','Api')->get('error-5110');
  46. // Render([], '5110',$msg);
  47. // }
  48. }
  49. }
  50. foreach($res as $k => $v){
  51. //根据类型分组
  52. if($v['type'] == '1'){
  53. $data['single'][$k] = array_merge($last[0][$v['index']],$v);
  54. }else{
  55. $data['bunch'][$k] = $v;
  56. }
  57. }
  58. if(!empty($data['bunch'])){
  59. $data['bunch'] = array_merge_recursive($data['bunch'],$last);
  60. }
  61. return $data;
  62. }
  63. /**
  64. * 根据赛事ID分组
  65. * @param [array] $data 球类数据
  66. * @return [array] $data
  67. */
  68. public function Group_ball($data = []){
  69. if(empty($data)){
  70. return $data;
  71. }
  72. foreach($data as $k => $v){
  73. if(!isset($data[$v['match_id']])){
  74. $data[$v['match_id']][] = $v;
  75. unset($data[$k]);
  76. }else{
  77. $data[$v['match_id']][] = $v;
  78. unset($data[$k]);
  79. }
  80. }
  81. return $data;
  82. }
  83. /**
  84. * 单式投注数据处理
  85. * @param [array] $res 单式投注数据
  86. * @return [int] 1 成功 其他失败
  87. * @param [int] $batch_id 批量ID
  88. */
  89. public function Simplex_bet(array $res,array $userInfo,$batch_id){
  90. if(empty($res)||empty($userInfo)||empty($batch_id)){
  91. Render('', '7003',lang('Errors','Api')->get('error-7003'));
  92. }
  93. //验证与判断投注了几种
  94. foreach($res as $k => $v){
  95. $num = $k + 1;
  96. if(empty($v['bet_amount'])){
  97. $msg = '第'.$num.'条'.lang('Errors','Api')->get('error-5105');
  98. Render([], '5105',$msg);
  99. }
  100. //根据类型分组
  101. if(!isset($res[$v['ballId']])){
  102. $res[$v['ballId']][] = $v;
  103. unset($res[$k]);
  104. }else{
  105. $res[$v['ballId']][] = $v;
  106. unset($res[$k]);
  107. }
  108. }
  109. //球类联赛分类并且插入数据库
  110. foreach($res as $k => $v){
  111. $gamedate = $this->Group_ball($res[$k]);
  112. $this->Simplex_bet_insert($gamedate,$userInfo,$k,$batch_id);
  113. }
  114. }
  115. /**
  116. * 单式投注单个球类数据插入
  117. * @param [array] $data 单式投注数据
  118. * @param [array] $userInfo 用户信息
  119. * @param [type] $game_code 球类代码
  120. * @param [int] $batch_id 批量ID
  121. * @return [int] 1 成功 其他失败
  122. */
  123. public function Simplex_bet_insert($data,$userInfo,$game_code,$batch_id){
  124. if(empty($data)||empty($userInfo)||empty($game_code)){
  125. Render('', '7003',lang('Errors','Api')->get('error-7003'));
  126. }
  127. foreach($data as $k => $v){
  128. $OrderID = OrderID();//生成订单ID
  129. $UUID = UUID(); //生成信息ID
  130. $bet_money = 0; //赛事总投注额
  131. $prize_money = 0; //可赢额
  132. foreach($v as $kk => $vv){
  133. $bet_money += $vv['bet_amount'];
  134. $prize_money += $vv['odds']* $vv['bet_amount'];
  135. $matchData =[
  136. 'odds_id' =>$vv['id'],
  137. 'home_team' => $vv['home_team'],
  138. 'guest_team' => $vv['guest_team'],
  139. 'condition' => $vv['condition'],
  140. 'odds' => $vv['odds'],
  141. 'odds_code' => $vv['odds_code'],
  142. 'p_code' => $vv['p_code'],
  143. 'odds_only' => $vv['odds_only'],
  144. 'match_id' => $vv['match_id'],
  145. 'game_code' => $game_code,
  146. 'lg_id' => $vv['lg_id'],
  147. 'batch_id' => $batch_id,
  148. 'bet_type' => 1,
  149. 'bet_money' => $vv['bet_amount'],
  150. 'ctime' => $this->time,
  151. 'utime' => $this->time
  152. ];
  153. //插入数据源
  154. $moneyBuyMatch = lm('MoneyBuyMatch','Api')->insert($matchData);
  155. if(!$moneyBuyMatch){
  156. Render('','3205',lang('errors','Api')->get('error-3205'));
  157. }
  158. }
  159. $order =[
  160. 'info_identity' => $UUID,
  161. 'account_name' => $userInfo['account'],
  162. 'account_identity' => $userInfo['account_identity'],
  163. 'order_id' => $OrderID,
  164. 'game_code' => $game_code,
  165. 'batch_id' => $batch_id,
  166. 'match_id' => $k,
  167. 'game_no' => "",
  168. 'prize_money'=> $prize_money,
  169. 'money' => $bet_money, //投注金额
  170. 'money_time' => $this->time
  171. ];
  172. $insert = lm("MoneyBuySimplex",'Api')->insert($order);
  173. if(!$insert){
  174. Render('','3205',lang('errors','Api')->get('error-3205'));
  175. }
  176. $insertMoney = $this->M->insertMoney($userInfo['account_identity'],$bet_money,$OrderID,1,"",$UUID,$userInfo);
  177. if($insertMoney!=1){
  178. Render('', $insertMoney,lang('Errors','Api')->get('error'.$insertMoney));
  179. }
  180. $this->M->fanshui($userInfo['account_identity'],$userInfo['account'],$bet_money,$OrderID);//反水
  181. }
  182. }
  183. public function Bet(){
  184. $userInfo = $this->getAgent();//获取用户信息
  185. $data = $_REQUEST['data'];
  186. $bet_money = $_REQUEST['bet_money'];
  187. if(empty($data)||empty($bet_money)){
  188. Render('', '51029',lang('Errors','Api')->get('error-51029'));
  189. }
  190. //获取用户金额信息
  191. if(!$this->M->verifyMoney($bet_money,$userInfo['cash'])){
  192. Render('', '4204',lang('Errors','Api')->get('error-4204'));
  193. }
  194. $batch_id = strtotime(date('Y-m-d H:i:s',time())).mt_rand('1','99');//批量ID
  195. //执行数据插入
  196. try {
  197. _beginTransaction();//开启事务
  198. $data = $this->Classify($data);
  199. if(!empty($data['single'])){
  200. $this->Simplex_bet($data['single'],$userInfo,$batch_id);//单式数据处理
  201. }
  202. if(!empty($data['bunch'])){
  203. $data_all = $data['bunch'][count($data['bunch'])-1];//获取最后一个数组
  204. unset($data['bunch'][count($data['bunch'])-1]);//删除最后一个元素
  205. $this->stringBet($data['bunch'],$data_all,$userInfo,$batch_id);
  206. }
  207. _commit();//提交
  208. Render('', '1',lang('Errors','Api')->get('error-1'));
  209. } catch (Exception $e) {
  210. _rollBack();//回滚
  211. print $e->getMessage();
  212. }
  213. }
  214. /**
  215. * token获取用户详情
  216. */
  217. public function getAgent() {
  218. $checkToken = $this->accountManager->checkEffectiveTime();
  219. if ($checkToken['status'] != 1) {
  220. Render($checkToken['data'], $checkToken['status'], lang('commons')->get('user does login'));
  221. };
  222. $result = $this->accountManager->refreshToken();
  223. return $result['data'];
  224. }
  225. /**
  226. * 赛事是否存在或结束
  227. * @param [type] $game_code 球类代码
  228. * @param [type] $match_id 赔率ID
  229. * @param [type] $num 循环键值
  230. * $@param [type]type_str 串式
  231. *
  232. */
  233. public function verify_match($game_code,$match_id,$num = 1){
  234. if(empty($match_id)){
  235. $msg = '第'.$num.'条'.lang('Errors','Api')->get('error-5109');
  236. Render([], '5109',$msg);
  237. }
  238. //根据球类代码 获取相关model
  239. $models = $this->commonFunction->getModels($game_code);
  240. $model_match = $models['model_match'];
  241. $info = lm($model_match,'Sports')
  242. ->select('status')
  243. ->where('status', '<', '2')
  244. // ->where('us_time','>',qgmdate('Y-m-d H:i:s','', -4))
  245. ->where('source','hg3535')
  246. ->where('match_id',$match_id)
  247. ->first();
  248. if(empty($info)){
  249. $msg = '第'.$num.'条'.lang('Errors','Api')->get('error-5103');
  250. Render([], '5103',$msg);
  251. }
  252. $info = $info->toarray();
  253. if($info['status'] == 2){
  254. $msg = '第'.$num.'条'.lang('Errors','Api')->get('error-5108');
  255. Render([], '5108',$msg);
  256. }
  257. return $info;
  258. }
  259. /**
  260. * 验证球类是否存在
  261. * @param [type] $game_code 球类代码
  262. * @param [type] $num 第几条
  263. */
  264. public function verify_gameType($game_code,$num){
  265. if(empty($game_code)){
  266. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-5102');
  267. Render([], '5102', $msg);
  268. }
  269. $info = lm('GameType','Sports')->where('game_code',$game_code)->first();
  270. if(empty($info)){
  271. $msg = '第' . $num . '条数据' . lang('Errors', 'Api')->get('error-5102');
  272. Render([], '5102', $msg);
  273. }
  274. return $info;
  275. }
  276. /**
  277. * 验证赔率是否存在
  278. * @param [type] $game_code 球类代码
  279. * @param [type] $odds_id 赔率IDkk
  280. *
  281. */
  282. public function verify_odds($game_code,$odds_only,$odds,$condition,$num = 1){
  283. if (empty($odds_only)){
  284. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-5111');
  285. Render([], '5111', $msg);
  286. }
  287. if(empty($odds)){
  288. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-7029');
  289. Render([], '7029', $msg);
  290. }
  291. //根据球类代码 获取相关model
  292. $models = $this->commonFunction->getModels($game_code);
  293. $model_odds = $models['model_odds'];
  294. $model_odds_record = $models['model_odds_record'];
  295. //查询赔率记录是否存在该赔率
  296. $is_odds = lm($model_odds_record,'Sports')
  297. ->select('odds')
  298. ->where($model_odds_record.'.odds_only',$odds_only)
  299. ->first();
  300. if(empty($is_odds)){
  301. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-7026');
  302. Render([], '7026', $msg);
  303. }
  304. //匹配赔率是否过期
  305. $odds_info = lm($model_odds,'Sports')
  306. ->join($model_odds_record,$model_odds_record.'.odds_only',$model_odds.'.odds_only')
  307. ->select($model_odds_record.'.odds',$model_odds_record.'.condition')
  308. ->where($model_odds.'.odds_only',$odds_only)
  309. ->first();
  310. if(empty($odds_info)){
  311. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-7027');
  312. Render([], '7027', $msg);
  313. }
  314. $odds_info = $odds_info->toarray();
  315. if($condition != $odds_info['condition']){
  316. $msg = '第'.$num.'条数据'.lang('Errors','Api')->get('error-7025');
  317. Render([], '7025', $msg);
  318. }
  319. return $odds_info;
  320. }
  321. /**
  322. * 处理串式投注
  323. * $data_str 各串式 数据
  324. * $data_all 所有赛事 数据
  325. * $userInfo 用户数据
  326. * $batch_id 批次号
  327. */
  328. public function stringBet($data_str=[],$data_all=[],$userInfo=[],$batch_id=23456){
  329. //验证串式投注是否合法
  330. $this->verify_stringType($data_all);
  331. //获取串式 总赛事数量
  332. $groupNum = count($data_all);//总数量
  333. $orderData = [];
  334. foreach ($data_str as $k=>$v){
  335. //字符串替换
  336. $str = str_ireplace("串","_",$v['type']);
  337. //获取type 3_12
  338. $after = substr($str,strpos($str,"_")+1);
  339. //当前串式 值
  340. $sonNum = substr($str,0,strrpos($str,"_"));
  341. //获取倍数
  342. if((int)$after == 1){
  343. $multiple = $this->getMultiple($groupNum,(int)$sonNum);
  344. }else{
  345. $multiple =(int)$after;
  346. }
  347. //获取当前串式投注金额及可赢金额
  348. if($after == 1){
  349. $money_all = $this->getmakeMoney((int)$sonNum,$v['bet_amount'],$multiple,$data_all,$after);
  350. }else{
  351. $money_all = $this->getStrAllMoney((int)$sonNum,$v['bet_amount'],$multiple,$data_all,$after);
  352. }
  353. //投注金额
  354. $moneyData = $money_all['moneyData'];
  355. //可赢金额
  356. $prize_money = $money_all['prize_money'];
  357. $uuid = UUID();
  358. $orderID = OrderID();
  359. $orderData[$k]['info_identity'] = $uuid;
  360. $orderData[$k]['account_name'] = $userInfo['account'];
  361. $orderData[$k]['account_identity'] = $userInfo['account_identity'];
  362. $orderData[$k]['order_id'] = $orderID;
  363. $orderData[$k]['game_no'] = "";
  364. $orderData[$k]['money'] = $moneyData;//$pay_money//总投注金额
  365. $orderData[$k]['money_time'] = $this->time;//下注时间
  366. $orderData[$k]['prize_money'] = $prize_money;//$expect_money ;//预期总可赢金额
  367. $orderData[$k]['str_type'] = $v['type'];//串式类型
  368. $orderData[$k]['batch_id'] = $batch_id;//批次号
  369. $orderData[$k]['wait_match_num'] = count($data_all);//批次号
  370. //写资金变动表
  371. $insertMoney = $this->M->insertMoney($userInfo['account_identity'],$moneyData,$orderID,1,"",$uuid,$userInfo);
  372. if($insertMoney!=1){
  373. Render('', $insertMoney,lang('Errors','Api')->get('error'.$insertMoney));
  374. }
  375. $this->M->fanshui($userInfo['account_identity'],$userInfo['account'],$moneyData,$orderID);//反水
  376. }
  377. //写注单 表
  378. $moneyBuy = lm('MoneyBuyStr','Api')->insert($orderData);
  379. if(!$moneyBuy){
  380. Render('','3205',lang('errors','Api')->get('error-3205'));
  381. }
  382. foreach ($data_all as $kk=>$vv){
  383. $getModels = $this->commonFunction->getModels($vv['ballId']);
  384. $model_match = $getModels['model_match'];
  385. $matchData[] = [
  386. 'batch_id'=>$batch_id,
  387. 'odds_id'=>$vv['id'],
  388. 'home_team'=>$vv['home_team'],
  389. 'guest_team'=>$vv['guest_team'],
  390. 'condition'=>$vv['condition'],
  391. 'odds'=>$vv['odds'],
  392. 'odds_code'=>$vv['odds_code'],
  393. 'p_code'=>$vv['p_code'],
  394. 'odds_only'=>$vv['odds_only'],
  395. 'status'=>0,
  396. 'match_id'=>$vv['match_id'],
  397. 'game_code'=>$vv['ballId'],
  398. 'lg_id'=> $vv['lg_id'],//lm($model_match,"Sports")->select('lg_id')->where('match_id',$vv['match_id'])->first()->lg_id,
  399. 'ctime'=>$this->time,//写入时间
  400. 'utime' => $this->time,
  401. 'bet_type'=>2
  402. ];
  403. }
  404. //写赔率数据表
  405. $moneyBuyMatch = lm('MoneyBuyMatch','Api')->insert($matchData);
  406. if(!$moneyBuyMatch){
  407. Render('','3205',lang('errors','Api')->get('error-3205'));
  408. }
  409. return true;
  410. }
  411. /**
  412. * 计算串式投注 倍率
  413. * $groupNum 总赛事 数量
  414. * $sonNum 当前串式 数量
  415. */
  416. public function getMultiple($groupNum = 0,$sonNum = 0){
  417. if($groupNum and $sonNum){
  418. $group =1;
  419. for($i = $groupNum ; $i >($groupNum - $sonNum);$i--) {
  420. $group = $group * $i;
  421. }
  422. $son =1;
  423. for($i = $sonNum ; $i >0;$i--) {
  424. $son = $son * $i;
  425. }
  426. }
  427. return $group/$son;
  428. }
  429. /**
  430. * @param int $sonNum 当前串式数量
  431. * @param int $bet_amount 当前串式投注额
  432. * @param int $multiple 当前串式倍数
  433. * @param array $data_all 当前注单 所有投注数据
  434. * @return mixed
  435. * 计算当前串式投注总额、可赢总额
  436. */
  437. public function getmakeMoney($sonNum=0,$bet_amount=0,$multiple=0,$data_all=[],$after = 0){
  438. $allNum = count($data_all);//所有数量
  439. $odds_all = [];//所有投注项数据
  440. foreach ($data_all as $k=>$v){
  441. $odds_all[] = $v['odds']+1;
  442. }
  443. //投注总额
  444. $data['moneyData'] = (double)$bet_amount*(double)$multiple;
  445. if($sonNum == $allNum and (int)$after ==1){//N场赛事 N串一
  446. //获取所有赛事赔率乘积
  447. $odds = array_product($odds_all) - 1;
  448. //获取可赢金额
  449. $data['prize_money'] = sprintf("%.2f",substr(sprintf("%.3f", $bet_amount*$odds), 0, -1));//floor($bet_amount*$odds*100)/100;//保留两位小数
  450. }
  451. if($sonNum < $allNum and (int)$after ==1){//例:N场赛事 <N串一
  452. $group = $this->commonFunction->combination($odds_all, $sonNum);
  453. foreach ($group as $k=>$v){
  454. $odds[] = array_product($v);
  455. }
  456. $odds = (array_sum($odds))-(count($odds));
  457. //获取可赢金额
  458. $data['prize_money'] = sprintf("%.2f",substr(sprintf("%.3f", $bet_amount*$odds), 0, -1));//floor($bet_amount*$odds*100)/100;
  459. }
  460. return $data;
  461. }
  462. /**
  463. * 获取所有可用串式 N场赛事 2串1,3串1...N串1,N<11
  464. * 并计算 其投注可赢金额
  465. */
  466. public function getStrAllMoney($sonNum=0,$bet_amount=0,$multiple=0,$data_all=[],$after = 0){
  467. $d = [];
  468. for ($i=2;$i<($sonNum+1);$i++){
  469. $d[] = $i;
  470. }
  471. foreach ($d as $k=>$v){
  472. $multiple = $this->getMultiple($sonNum,$v);
  473. $money[] = $this->getmakeMoney($v,$bet_amount,$multiple,$data_all,1);
  474. }
  475. $data=array();
  476. foreach($money as $k=>$v){
  477. $data['moneyData']+=$v['moneyData'];
  478. $data['prize_money']+=$v['prize_money'];
  479. }
  480. return $data;
  481. }
  482. /**
  483. * 验证串式投注 是否合法
  484. * $data 串式 所有赛事
  485. */
  486. public function verify_stringType($data=[]){
  487. if($data){
  488. foreach($data as $k=>$v){
  489. if($v['match_id']==$v['match_id']){
  490. $r[$v['match_id']][] = $v;
  491. }
  492. }
  493. }
  494. if($r){
  495. foreach ($r as $k=>$v){
  496. if(count($v) > 1){
  497. $msg = lang('Errors','Api')->get('error-50038');
  498. Render([], '50038',$msg);
  499. }
  500. }
  501. }
  502. }
  503. /**
  504. * 更新投注项相关数据
  505. */
  506. public function getBetData(){
  507. $data = $_REQUEST;
  508. // $data = [['game_code'=>'zq','match_id'=>3095448,'odds_id'=>623078],['game_code'=>'zq','match_id'=>3095448,'odds_id'=>623078]];
  509. foreach ($data as $k=>$v){
  510. $getModels = $this->commonFunction->getModels($v['game_code']);
  511. $model_match = $getModels['model_match'];
  512. $model_odds = $getModels['model_odds'];
  513. $oddsData = lm($model_match,"Sports")
  514. ->join($model_odds,$model_odds.'.match_id',$model_match.'.match_id')
  515. ->select($model_match.'.match_id',$model_match.'.match_date',$model_match.'.home_team',$model_match.'.guest_team',$model_odds.'.id as odds_id',$model_odds.'.p_code',$model_odds.'.odds_code',$model_odds.'.condition',$model_odds.'.odds',$model_odds.'.odds_only',$model_odds.'.status',$model_odds.'.sort')
  516. ->where($model_match.'.source',$this->source)
  517. ->where($model_match.'.match_id',$v['match_id'])
  518. ->where($model_odds.'.id',$v['odds_id'])
  519. //->where($model_odds.'.expire_time','>',date("Y-m-d H:i:s"))
  520. ->first();
  521. $oddsData->game_code = $v['game_code'];
  522. $betData[] = $oddsData;
  523. }
  524. Render($betData, '1', lang('Tips','Sports')->get('success'));
  525. }
  526. }