PcIssueService.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. <?php
  2. namespace App\Services;
  3. use App\Jobs\SendTelegramGroupMessageJob;
  4. use App\Models\Config;
  5. use App\Models\PcCao;
  6. use App\Models\PcCaoHistory;
  7. use App\Models\PcIssue;
  8. use App\Models\PcPrediction;
  9. use Illuminate\Support\Facades\Cache;
  10. use Illuminate\Support\Facades\DB;
  11. use Illuminate\Support\Facades\Log;
  12. class PcIssueService extends BaseService
  13. {
  14. public static $MODEL = PcIssue::class;
  15. public static function index(): void
  16. {
  17. //封盘倒数
  18. PcIssueService::syncCountdownIssue();
  19. //对该封盘的进行封盘
  20. PcIssueService:: fengPan();
  21. //将未开奖的进行开奖
  22. PcIssueService:: kaiJiang();
  23. //创建新的期号
  24. PcIssueService::createIssueNo();
  25. }
  26. //封盘倒数
  27. public static function syncCountdownIssue()
  28. {
  29. $info = PcIssue::where('status', PcIssue::STATUS_BETTING)->orderBy('end_time')->first();
  30. if ($info) {
  31. $now_date = date('Y-m-d H:i:s', time() + 60); // 提前60秒
  32. if ($info['end_time'] < $now_date) {
  33. $replyInfo = KeyboardService::findOne(['button' => '封盘倒数']);
  34. if ($replyInfo) {
  35. $text = $replyInfo->reply;
  36. $buttons = json_decode($replyInfo->buttons, true);
  37. $image = $replyInfo->image;
  38. if ($image) {
  39. $image = url($image);
  40. }
  41. if (Cache::has('issue_countdown_' . $info->issue_no)) {
  42. } else {
  43. $pc28Switch = Config::where('field', 'pc28_switch')->first()->val;
  44. if ($pc28Switch == 1) {
  45. self::asyncBettingGroupNotice($text, $buttons, $image);
  46. Cache::put('issue_countdown_' . $info->issue_no, true, 60); // 缓存50秒,防止多次发送
  47. }
  48. }
  49. }
  50. }
  51. }
  52. }
  53. private static function fengPan()
  54. {
  55. $list = PcIssue::where('status', PcIssue::STATUS_BETTING)
  56. ->orderByDesc('id')->get();
  57. $now = time();
  58. $pc28Switch = Config::where('field', 'pc28_switch')->first()->val;
  59. $now += 30;
  60. foreach ($list as $index => $item) {
  61. if (strtotime($item->end_time) < $now) {
  62. $item->status = PcIssue::STATUS_CLOSE;
  63. $item->save();
  64. if ($index == 0 && $pc28Switch == 1) {
  65. $replyInfo = KeyboardService::findOne(['button' => '停止下注']);
  66. if ($replyInfo) {
  67. $text = $replyInfo->reply;
  68. $text .= "PC极速28";
  69. $buttons = json_decode($replyInfo->buttons, true);
  70. $image = $replyInfo->image;
  71. if ($image) $image = url($image);
  72. self::asyncBettingGroupNotice($text, $buttons, $image);
  73. }
  74. // 统计投注情况通知 xxxx期投注统计
  75. BetService::statNotice($item->issue_no);
  76. $replyInfo = KeyboardService::findOne(['button' => '封盘开奖']);
  77. if ($replyInfo) {
  78. $text = $replyInfo->reply;
  79. $text .= "PC极速28";
  80. $buttons = json_decode($replyInfo->buttons, true);
  81. $image = $replyInfo->image;
  82. if ($image) $image = url($image);
  83. self::asyncBettingGroupNotice($text, $buttons, $image);
  84. }
  85. }
  86. }
  87. }
  88. }
  89. private static function kaiJiang(): void
  90. {
  91. $list = PcIssue::where('status', PcIssue::STATUS_CLOSE)
  92. ->orderByDesc('id')->get();
  93. $now = time();
  94. foreach ($list as $index => $item) {
  95. if (strtotime($item->end_time) <= $now) {
  96. DB::beginTransaction();
  97. try {
  98. $keno = static::getKeno();
  99. $item->keno = json_encode($keno);
  100. $winningNumbers = static::getWinningNumbers($keno);
  101. $item->winning_numbers = implode(',', $winningNumbers);
  102. $item->save();
  103. $winArr = array_map('intval', $winningNumbers);
  104. // 组合
  105. $sum = array_sum($winArr);
  106. $combo = [];
  107. $sumSize = IssueService::calculateSumSize($sum); // 总和大小
  108. $combo[] = $sumSize;
  109. $sumOddEven = IssueService::calculateOddEven($sum); // 总和单双
  110. $combo[] = $sumOddEven;
  111. $sumExtremeSize = IssueService::calculateSumExtremeSize($sum); // 总和极值
  112. if ($sumExtremeSize) {
  113. $combo[] = $sumExtremeSize;
  114. }
  115. $sumBaoZi = IssueService::isBaoZi($winArr[0], $winArr[1], $winArr[2]); // 豹子
  116. if ($sumBaoZi) {
  117. $combo[] = $sumBaoZi;
  118. }
  119. $sumPair = IssueService::isPair($winArr[0], $winArr[1], $winArr[2]); // 对子
  120. if ($sumPair) {
  121. $combo[] = $sumPair;
  122. }
  123. $sumStraight = IssueService::isStraight($winArr[0], $winArr[1], $winArr[2]); // 顺子
  124. if ($sumStraight) {
  125. $combo[] = $sumStraight;
  126. }
  127. $tail = IssueService::getLastDigit($sum); // 总和尾数
  128. if ($tail != 0 && $tail != 9) {
  129. $combo[] = '尾' . $tail; // 尾数
  130. }
  131. $key = 'lottery_numbers_' . $item->issue_no;
  132. $combo = implode(' ', $combo);
  133. if (Cache::add($key, $item->winning_numbers, 100)) {
  134. self::lotteryDraw($item->issue_no, $item->winning_numbers, $combo, '', $index);
  135. }
  136. $pc28Switch = Config::where('field', 'pc28_switch')->first()->val;
  137. //更新游戏开关的切换
  138. if ($pc28Switch == 1) Config::setPc28Switch();
  139. $awards = IssueService::award(explode(',', $item->winning_numbers));
  140. //预测结果
  141. PcPrediction::result($item->issue_no, $item->winning_numbers, $awards);
  142. //自开奖以来的结果统计
  143. PcCao::updateData($awards);
  144. //每天开奖结果统计
  145. PcCaoHistory::updateData($awards);
  146. DB::commit();
  147. } catch (\Exception $e) {
  148. Db::rollBack();
  149. }
  150. }
  151. }
  152. }
  153. //开奖
  154. private static function lotteryDraw($issue_no, $winning_numbers, $combo, $recordImage, $index)
  155. {
  156. $info = PcIssue::where('issue_no', $issue_no)->first();
  157. if (!$info) {
  158. return ['code' => self::NOT, 'msg' => '期号不存在'];
  159. }
  160. if ($info->status == PcIssue::STATUS_DRAW) {
  161. return ['code' => self::NOT, 'msg' => '期号状态不正确'];
  162. }
  163. $winArr = array_map('intval', explode(',', $winning_numbers));
  164. // 计算中奖
  165. $awards = IssueService::award(explode(',', $winning_numbers));
  166. DB::beginTransaction();
  167. try {
  168. $info->status = PcIssue::STATUS_DRAW;
  169. $info->combo = $combo;
  170. $info->image = $recordImage;
  171. $info->save();
  172. $pc28Switch = Config::where('field', 'pc28_switch')->first()->val;
  173. $replyInfo = KeyboardService::findOne(['button' => '本期开奖']);
  174. if ($replyInfo) {
  175. $text = $replyInfo->reply;
  176. $text .= "\n";
  177. $text .= $info->issue_no . ": " . implode('+', explode(',', $winning_numbers)) . "=" . array_sum($winArr) . " " . $combo;
  178. $buttons = json_decode($replyInfo->buttons, true);
  179. $image = $replyInfo->image;
  180. if ($image) {
  181. $image = url($image);
  182. }
  183. if (empty($buttons)) {
  184. $serviceAccount = Config::where('field', 'service_account')->first()->val;
  185. $buttons[] = [['text' => lang('✅唯一财务'), 'callback_data' => "", 'url' => "https://t.me/{$serviceAccount}"]];
  186. }
  187. if ($pc28Switch == 1) SendTelegramGroupMessageJob::dispatch($text, $buttons, $image, true);
  188. }
  189. $recordImage = self::lotteryImage($info->issue_no);
  190. if ($recordImage) {
  191. if ($pc28Switch == 1) SendTelegramGroupMessageJob::dispatch('', [], url($recordImage), false);
  192. }
  193. //中奖结算 并发送群通知 xxxx期开奖结果
  194. BetService::betSettled($info->issue_no, $awards);
  195. DB::commit();
  196. return ['code' => self::YES, 'msg' => '开奖成功'];
  197. } catch (\Exception $e) {
  198. DB::rollBack();
  199. Log::error('开奖失败: ' . $e->getMessage() . $winning_numbers);
  200. return ['code' => self::NOT, 'msg' => '开奖失败'];
  201. }
  202. }
  203. // 生成开奖图片
  204. private static function lotteryImage($issue_no)
  205. {
  206. $list = PcIssue::where('issue_no', '<=', $issue_no)->where('status', PcIssue::STATUS_DRAW)->orderBy('issue_no', 'desc')->take(20)->get();
  207. $records = $list->toArray();
  208. foreach ($records as $k => $v) {
  209. $winning_numbers = explode(',', $v['winning_numbers']);
  210. $v['winning_numbers'] = $winning_numbers;
  211. // 组合
  212. $sum = array_sum($winning_numbers);
  213. $v['sum'] = $sum;
  214. $sumOddEven = IssueService::calculateOddEven($sum); // 总和单双
  215. $sumSize = IssueService::calculateSumSize($sum); // 总和大小
  216. $v['combo'] = $sumSize . ' ' . $sumOddEven;
  217. $sumExtremeSize = IssueService::calculateSumExtremeSize($sum); // 总和极值
  218. if (!$sumExtremeSize) {
  219. $sumExtremeSize = '-';
  220. }
  221. $v['extreme'] = $sumExtremeSize;
  222. $tail = IssueService::getLastDigit($sum); // 总和尾数
  223. if ($tail === 0 || $tail === 9) {
  224. $tailStr = '-';
  225. } else {
  226. $tailStr = '尾' . $tail;
  227. }
  228. $v['tail'] = $tailStr;
  229. $records[$k] = $v;
  230. }
  231. $service = new LotteryImageService();
  232. $url = $service->generate($records);
  233. PcIssue::where('issue_no', $issue_no)->update(['image' => $url]);
  234. return $url;
  235. }
  236. //创建新的一期
  237. private static function createIssueNo()
  238. {
  239. $issue = PcIssue::orderByDesc('id')->first();
  240. $now = time();
  241. if (strtotime($issue->end_time) <= $now) {
  242. $issue_no = $issue->issue_no;
  243. $number = preg_replace('/\D/', '', $issue_no);
  244. $number = (int)$number + 1;
  245. $new_str = "P" . $number;
  246. $end_time = strtotime($issue->end_time) + 210;
  247. $status = $end_time - 30 > $now ? PcIssue::STATUS_BETTING : PcIssue::STATUS_CLOSE;
  248. $issue1 = PcIssue::create([
  249. 'issue_no' => $new_str,
  250. 'status' => $status,
  251. 'start_time' => $issue->end_time,
  252. 'end_time' => date('Y-m-d H:i:s', $end_time),
  253. ]);
  254. //预测
  255. PcPrediction::prediction($new_str);
  256. if (strtotime($issue1->end_time) <= $now) {
  257. static::createIssueNo();
  258. } else {
  259. // 群通知,发送玩法规则和开始下注的通知
  260. static::betting($new_str);
  261. }
  262. }
  263. }
  264. //群通知,发送玩法规则和开始下注的通知
  265. private static function betting($issue_no): void
  266. {
  267. $info = PcIssue::where('issue_no', $issue_no)->first();
  268. if (!$info) {
  269. return;
  270. }
  271. if (!in_array($info->status, [PcIssue::STATUS_DRAFT, PcIssue::STATUS_BETTING])) {
  272. return;
  273. }
  274. $pc28Switch = Config::where('field', 'pc28_switch')->first()->val;
  275. $info->status = PcIssue::STATUS_BETTING;
  276. $info->save();
  277. $replyInfo = KeyboardService::findOne(['button' => '玩法规则']);
  278. if ($replyInfo) {
  279. $text = $replyInfo->reply;
  280. $text .= "\n{$issue_no}\nPC极速28";
  281. $buttons = json_decode($replyInfo->buttons, true);
  282. $image = $replyInfo->image;
  283. if ($image) {
  284. $image = url($image);
  285. }
  286. if (empty($buttons)) {
  287. $buttons = self::getOperateButton();
  288. }
  289. if ($pc28Switch == 1) self::asyncBettingGroupNotice($text, $buttons, $image);
  290. }
  291. $replyInfo = KeyboardService::findOne(['button' => '开始下注']);
  292. if ($replyInfo) {
  293. $text = $replyInfo->reply;
  294. $text .= "\n{$issue_no}\nPC极速28";
  295. $buttons = json_decode($replyInfo->buttons, true);
  296. $image = $replyInfo->image;
  297. if ($image) {
  298. $image = url($image);
  299. }
  300. if ($pc28Switch == 1) self::asyncBettingGroupNotice($text, $buttons, $image);
  301. }
  302. }
  303. //获取随机的20个数字
  304. private static function getKeno(): array
  305. {
  306. $numbers = range(1, 80);
  307. shuffle($numbers);
  308. $random_numbers = array_slice($numbers, 0, 20);
  309. sort($random_numbers);
  310. return $random_numbers;
  311. }
  312. //根据20个数字计算开奖号码
  313. private static function getWinningNumbers($keno): array
  314. {
  315. $winningNumbers = [];
  316. $sum = $keno[1] + $keno[4] + $keno[7] + $keno[10] + $keno[13] + $keno[16];
  317. $winningNumbers[0] = $sum % 10;
  318. $sum = $keno[2] + $keno[5] + $keno[8] + $keno[11] + $keno[14] + $keno[17];
  319. $winningNumbers[1] = $sum % 10;
  320. $sum = $keno[3] + $keno[6] + $keno[9] + $keno[12] + $keno[15] + $keno[18];
  321. $winningNumbers[2] = $sum % 10;
  322. return $winningNumbers;
  323. }
  324. //根据指定0-27的数字 得到20个数字
  325. public static function getMatchingNumbers($target): array|null
  326. {
  327. $numbers = range(1, 80);
  328. $bestMatch = null;
  329. for ($i = 0; $i < 100000; $i++) {
  330. shuffle($numbers);
  331. $selectedNumbers = array_slice($numbers, 0, 20);
  332. sort($selectedNumbers);
  333. $firstSum = $selectedNumbers[1] + $selectedNumbers[4] + $selectedNumbers[7] + $selectedNumbers[10] + $selectedNumbers[13] + $selectedNumbers[16];
  334. $firstDigit = $firstSum % 10;
  335. $secondSum = $selectedNumbers[2] + $selectedNumbers[5] + $selectedNumbers[8] + $selectedNumbers[11] + $selectedNumbers[14] + $selectedNumbers[17];
  336. $secondDigit = $secondSum % 10;
  337. $thirdSum = $selectedNumbers[3] + $selectedNumbers[6] + $selectedNumbers[9] + $selectedNumbers[12] + $selectedNumbers[15] + $selectedNumbers[18];
  338. $thirdDigit = $thirdSum % 10;
  339. $result = $firstDigit + $secondDigit + $thirdDigit;
  340. if ($result === $target) {
  341. $bestMatch = $selectedNumbers;
  342. break;
  343. }
  344. }
  345. return $bestMatch;
  346. }
  347. }