SportOdds.php 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. <?php
  2. namespace App\Console\Commands;
  3. use Illuminate\Console\Command;
  4. use App\Models\Sport as SportModel;
  5. use App\Models\SportOdds as SportOddsModel;
  6. use App\Services\SportClientService;
  7. use Illuminate\Support\Facades\Log;
  8. use App\Models\Config;
  9. use Illuminate\Support\Facades\Cache;
  10. use Throwable; // 使用 Throwable 可以捕获所有 PHP 7+ 的异常和错误
  11. class SportOdds extends Command
  12. {
  13. /**
  14. * 命令名称和签名
  15. *
  16. * @var string
  17. */
  18. protected $signature = 'sport:odds {is_live=0}';
  19. protected $is_live = 0;
  20. protected $long_status = [
  21. 'Time To Be Defined' => 0,
  22. 'Not Started' => 0,
  23. 'First Half' => 1,
  24. 'First Half, Kick Off' => 1,
  25. 'Halftime' => 1,
  26. 'Second Half' => 1,
  27. 'Second Half, 2nd Half Started' => 1,
  28. 'Extra Time' => 1,
  29. 'Break Time' => 1,
  30. 'Penalty In Progress' => 1,
  31. 'Match Suspended' => 1,
  32. 'Match Interrupted' => 4,
  33. 'Match Finished' => 2,
  34. 'Match Postponed' => 3,
  35. 'Match Cancelled' => 4,
  36. 'Match Abandoned' => 4,
  37. 'Technical Loss' => 4,
  38. 'WalkOver' => 4,
  39. 'In Progress' => 1,
  40. ];
  41. protected $fixture_status = [
  42. 'First Half' => 1,
  43. 'First Half, Kick Off' => 1,
  44. 'Halftime' => 1,
  45. 'Second Half' => 1,
  46. 'Second Half, 2nd Half Started' => 1,
  47. 'Extra Time' => 1,
  48. 'Break Time' => 1,
  49. 'Penalty In Progress' => 1,
  50. ];
  51. /**
  52. * 命令描述
  53. *
  54. * @var string
  55. */
  56. protected $description = '赔率(直播赔率5秒更新一次,塞前赔率3小时更新一次)';
  57. /**
  58. * 执行命令
  59. *
  60. * @return int
  61. */
  62. public function handle()
  63. {
  64. $this->is_live = $this->argument('is_live');
  65. $this->sportOddsData($this->is_live);
  66. }
  67. public function sportOddsData($is_live)
  68. {
  69. if ($is_live == 1) {
  70. // 直播赔率通常是一次获取全量,但也需要对整体更新逻辑做保护
  71. $data = SportClientService::oddsLive([]);
  72. if ($data) {
  73. // file_put_contents("oddsLive.json",json_encode($data));
  74. $this->updateOddsLive($data);
  75. }
  76. } else {
  77. $limit = 3000;
  78. //普通球赛是开赛前购买,更新数据
  79. $where['state'] = 0; //比赛状态:0未开始1进行中2已完场3延期4取消
  80. $list = SportModel::where($where)->where('odds',null)->limit($limit)->get()->toArray();
  81. foreach ($list as $item) {
  82. // --- 关键优化:在循环内部嵌套 try-catch ---
  83. try {
  84. $data = SportClientService::odds([
  85. 'fixture' => $item['data_id'],
  86. ]);
  87. $this->updateOdds($item['data_id'], $data);
  88. } catch (Throwable $e) {
  89. // 记录具体哪条赛事出错了,但不抛出,让循环继续
  90. Log::error($item['data_id'].",赛前赔率更新失败: " . $e->getMessage());
  91. continue;
  92. }
  93. }
  94. }
  95. return true;
  96. }
  97. public function updateOddsLive($data)
  98. {
  99. if (empty($data['response'])) return;
  100. //体育赛事结束前几(分钟)锁盘,90分钟结束
  101. $sport_locked = Config::where('field', 'sport_locked')->first()->val ?? 1;
  102. $responses = $data['response'];
  103. foreach ($responses as $item) {
  104. // --- 关键优化:直播赔率逐条更新也要包裹,防止单条数据格式问题挂掉全盘 ---
  105. try {
  106. $data_id = $item['fixture']['id'];
  107. $sport_info = SportModel::where('data_id',$data_id)->select('odds','odd_ids_locked')->first();
  108. if (!$sport_info) continue;
  109. $old_odd_ids_locked = $sport_info['odd_ids_locked'];
  110. $odds_data = $this->doOdds($item['odds'], $sport_info['odds']);
  111. $odds = $odds_data['odds'];
  112. $odds = !empty($odds) ? json_encode($odds) : null;
  113. $odd_ids_locked = array_merge($old_odd_ids_locked, $odds_data['odd_ids_locked']);
  114. $update_data = [
  115. 'is_send' => 0,
  116. 'is_roll' => 1,
  117. 'is_locked' => 0,
  118. 'odd_ids_locked' => array_unique($odd_ids_locked),
  119. 'odds' => $odds,
  120. 'error' => 0, //异常类型:0正常;1直播赔率的赛事时异常
  121. ];
  122. $odd_fixture_status = null;//直播赔率中的赛事时间和状态(与直播赛事信息中的赛事时间和状态存在较大差异)
  123. if (isset($item['fixture']['status']['long']) ) {
  124. // $long = $item['fixture']['status']['long'];
  125. // if (isset($this->fixture_status[$long])) {
  126. $sport_info = SportModel::where('data_id',$data_id)->first();
  127. $fixture_status = $sport_info['fixture_status'] ? json_decode($sport_info['fixture_status'],true) : [];
  128. $odd_fixture_status = $item['fixture']['status'];
  129. $update_data['odd_fixture_status'] = json_encode($odd_fixture_status);
  130. //如果时间差距超过3分钟,则锁盘(赛事时间90分钟内)
  131. if (isset($fixture_status['elapsed']) && $fixture_status['elapsed'] < 90 && isset($odd_fixture_status['elapsed'])) {
  132. if (abs($fixture_status['elapsed'] - $odd_fixture_status['elapsed']) >= 3) {
  133. $update_data['error'] = 1; //异常
  134. $update_data['is_locked'] = 1; //锁盘
  135. unset($update_data['odds']);//不更新赔率
  136. } else {
  137. $update_data['error'] = 0;
  138. }
  139. }
  140. // }
  141. }
  142. //锁盘
  143. if (isset($item['fixture']['status']['blocked']) && $item['fixture']['status']['blocked']) {
  144. $update_data['is_locked'] = 1;
  145. }
  146. //提前锁盘(比赛进行时长,分钟)
  147. if (isset($item['fixture']['status']['elapsed'])) {
  148. $elapsed = $item['fixture']['status']['elapsed'];
  149. if ((int)$elapsed >= 90 - $sport_locked ) {
  150. $update_data['is_locked'] = 1;
  151. }
  152. }
  153. //已结束
  154. if (isset($item['fixture']['status']['finished']) && $item['fixture']['status']['finished']) {
  155. $update_data['state'] = 2;
  156. }
  157. //如果赛事取消、延期等,标记需要退款
  158. if (isset($update_data['state']) && $update_data['state'] > 2) {
  159. $update_data['refund_status'] = 1;
  160. }
  161. SportModel::where('data_id', $data_id)->update($update_data);
  162. } catch (Throwable $e) {
  163. Log::error("直播赔率单条更新失败 [ID: {$data_id}]: " . $e->getMessage());
  164. continue;
  165. }
  166. }
  167. }
  168. public function updateOdds($data_id, $data)
  169. {
  170. if (!empty($data['response'][0]['bookmakers'][0]['bets'])) {
  171. $odds = $data['response'][0]['bookmakers'][0]['bets'];
  172. SportModel::where('data_id',$data_id)->update(['odds' => json_encode($odds), 'is_send' => 0]);
  173. }
  174. }
  175. //去掉无效的赔率
  176. private function doOdds($odds, $old_odds) {
  177. // 1. 获取赔率,缓存数据
  178. $sport_odds = cache('sport_odds');
  179. if (!$sport_odds) {
  180. $sport_odds = SportOddsModel::where('function_name', '<>', null)->get()->toArray();
  181. Cache::set('sport_odds', $sport_odds, 300); //有效期5分钟
  182. }
  183. $old_odds = $old_odds ? json_decode($old_odds, true) : [];
  184. $odds_ids = []; //新的玩法id集合
  185. $sport_odds = array_column($sport_odds, null,'odd_name_en');
  186. $new_odds = [];
  187. foreach($odds as $item) {
  188. if (!isset($sport_odds[$item['name']])) {
  189. continue;
  190. }
  191. $new_odds[] = $item;
  192. $odds_ids[] = $item['id'];
  193. }
  194. $odd_ids_locked = [];
  195. foreach($old_odds as $item) {
  196. if (!in_array($item['id'], $odds_ids)) {
  197. $new_odds[] = $item;
  198. //历史玩法如果没有了,数据保存,加锁
  199. $odd_ids_locked[] = $item['id'];
  200. }
  201. }
  202. return [
  203. 'odds' => $new_odds,
  204. 'odd_ids_locked' => $odd_ids_locked,
  205. ];
  206. }
  207. }