PcIssueService.php 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  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. class PcIssueService extends BaseService
  11. {
  12. public static $MODEL = PcIssue::class;
  13. public static function index(): void
  14. {
  15. //创建新的期号
  16. PcIssueService::createIssueNo();
  17. //对该封盘的进行封盘
  18. PcIssueService:: fengPan();
  19. //将未开奖的进行开奖
  20. PcIssueService:: kaiJiang();
  21. }
  22. private static function fengPan()
  23. {
  24. $list = PcIssue::where('status', PcIssue::STATUS_BETTING)->get();
  25. $now = time();
  26. foreach ($list as $item) {
  27. if (strtotime($item->end_time) - 30 <= $now) {
  28. $item->status = PcIssue::STATUS_CLOSE;
  29. $item->save();
  30. }
  31. }
  32. }
  33. private static function kaiJiang(): void
  34. {
  35. $list = PcIssue::where('status', PcIssue::STATUS_CLOSE)->get();
  36. $now = time();
  37. foreach ($list as $item) {
  38. if (strtotime($item->end_time) <= $now) {
  39. $keno = static::getKeno();
  40. $item->keno = json_encode($keno);
  41. $winningNumbers = static::getWinningNumbers($keno);
  42. $item->winning_numbers = implode(',', $winningNumbers);
  43. $item->status = PcIssue::STATUS_DRAW;
  44. $item->save();
  45. $winArr = array_map('intval', $winningNumbers);
  46. // 组合
  47. $sum = array_sum($winArr);
  48. $combo = [];
  49. $sumOddEven = IssueService::calculateOddEven($sum); // 总和单双
  50. $combo[] = $sumOddEven;
  51. $sumSize = IssueService::calculateSumSize($sum); // 总和大小
  52. $combo[] = $sumSize;
  53. $sumExtremeSize = IssueService::calculateSumExtremeSize($sum); // 总和极值
  54. if ($sumExtremeSize) {
  55. $combo[] = $sumExtremeSize;
  56. }
  57. $sumBaoZi = IssueService::isBaoZi($winArr[0], $winArr[1], $winArr[2]); // 豹子
  58. if ($sumBaoZi) {
  59. $combo[] = $sumBaoZi;
  60. }
  61. $sumPair = IssueService::isPair($winArr[0], $winArr[1], $winArr[2]); // 对子
  62. if ($sumPair) {
  63. $combo[] = $sumPair;
  64. }
  65. $sumStraight = IssueService::isStraight($winArr[0], $winArr[1], $winArr[2]); // 顺子
  66. if ($sumStraight) {
  67. $combo[] = $sumStraight;
  68. }
  69. $tail = IssueService::getLastDigit($sum); // 总和尾数
  70. if ($tail != 0 && $tail != 9) {
  71. $combo[] = '尾' . $tail; // 尾数
  72. }
  73. $key = 'lottery_numbers_' . $item->issue_no;
  74. $combo = implode(' ', $combo);
  75. if (Cache::add($key, $item->winning_numbers, 100)) {
  76. self::lotteryDraw($item->issue_no, $item->winning_numbers, $combo, '');
  77. }
  78. $awards = IssueService::award(explode(',', $item->winning_numbers));
  79. //预测结果
  80. PcPrediction::result($item->issue_no, $item->winning_numbers, $awards);
  81. //自开奖以来的结果统计
  82. PcCao::updateData($awards);
  83. //每天开奖结果统计
  84. PcCaoHistory::updateData($awards);
  85. }
  86. }
  87. }
  88. private static function lotteryDraw($issue_no, $winning_numbers, $combo, $recordImage)
  89. {
  90. $info = PcIssue::where('issue_no', $issue_no)->first();
  91. if (!$info) {
  92. return ['code' => self::NOT, 'msg' => '期号不存在'];
  93. }
  94. if ($info->status == PcIssue::STATUS_DRAW) {
  95. return ['code' => self::NOT, 'msg' => '期号状态不正确'];
  96. }
  97. $winArr = array_map('intval', explode(',', $winning_numbers));
  98. // 计算中奖
  99. $awards = IssueService::award(explode(',', $winning_numbers));
  100. $info->status = self::model()::STATUS_DRAW;
  101. $info->winning_numbers = $winning_numbers;
  102. $info->combo = $combo;
  103. $info->image = $recordImage;
  104. $info->save();
  105. $replyInfo = KeyboardService::findOne(['button' => '本期开奖']);
  106. if ($replyInfo) {
  107. $text = $replyInfo->reply;
  108. $text .= "\n";
  109. $text .= $info->issue_no . "期 " . implode('+', explode(',', $winning_numbers)) . "=" . array_sum($winArr) . " " . $combo;
  110. $buttons = json_decode($replyInfo->buttons, true);
  111. $image = $replyInfo->image;
  112. if ($image) {
  113. $image = url($image);
  114. }
  115. if (empty($buttons)) {
  116. $serviceAccount = Config::where('field', 'service_account')->first()->val;
  117. $buttons[] = [['text' => '✅唯一财务', 'callback_data' => "", 'url' => "https://t.me/{$serviceAccount}"]];
  118. }
  119. // self::bettingGroupNotice($text, $buttons, $image, true);
  120. SendTelegramGroupMessageJob::dispatch($text, $buttons, $image, true);
  121. }
  122. if (config('app.url') != 'https://bot28dev.testx2.cc') {
  123. $recordImage = self::lotteryImage($info->issue_no);
  124. }
  125. if ($recordImage) {
  126. // self::bettingGroupNotice('', [], url($recordImage));
  127. SendTelegramGroupMessageJob::dispatch('', [], url($recordImage), false);
  128. }
  129. BetService::betSettled($info->issue_no, $awards);
  130. return ['code' => self::YES, 'msg' => '开奖成功'];
  131. }
  132. // 生成开奖图片
  133. private static function lotteryImage($issue_no)
  134. {
  135. $list = PcIssue::where('issue_no', '<=', $issue_no)->where('status' , PcIssue::STATUS_DRAW)->orderBy('issue_no', 'desc')->take(20)->get();
  136. $records = $list->toArray();
  137. foreach ($records as $k => $v) {
  138. $winning_numbers = explode(',', $v['winning_numbers']);
  139. $v['winning_numbers'] = $winning_numbers;
  140. // 组合
  141. $sum = array_sum($winning_numbers);
  142. $v['sum'] = $sum;
  143. $sumOddEven = IssueService::calculateOddEven($sum); // 总和单双
  144. $sumSize = IssueService::calculateSumSize($sum); // 总和大小
  145. $v['combo'] = $sumSize . ' ' . $sumOddEven;
  146. $sumExtremeSize = IssueService::calculateSumExtremeSize($sum); // 总和极值
  147. if (!$sumExtremeSize) {
  148. $sumExtremeSize = '-';
  149. }
  150. $v['extreme'] = $sumExtremeSize;
  151. $tail = IssueService::getLastDigit($sum); // 总和尾数
  152. if ($tail === 0 || $tail === 9) {
  153. $tailStr = '-';
  154. } else {
  155. $tailStr = '尾' . $tail;
  156. }
  157. $v['tail'] = $tailStr;
  158. $records[$k] = $v;
  159. }
  160. $service = new LotteryImageService();
  161. $url = $service->generate($records);
  162. PcIssue::where('issue_no', $issue_no)->update(['image' => $url]);
  163. return $url;
  164. }
  165. private static function createIssueNo()
  166. {
  167. $issue = PcIssue::orderByDesc('id')->first();
  168. $now = time();
  169. if (strtotime($issue->end_time) <= $now) {
  170. $issue_no = $issue->issue_no;
  171. $number = preg_replace('/\D/', '', $issue_no);
  172. $number = (int)$number + 1;
  173. $new_str = "P" . $number;
  174. $end_time = strtotime($issue->end_time) + 210;
  175. $status = $end_time - 30 > $now ? PcIssue::STATUS_BETTING : PcIssue::STATUS_CLOSE;
  176. $issue = PcIssue::create([
  177. 'issue_no' => $new_str,
  178. 'status' => $status,
  179. 'start_time' => $issue->end_time,
  180. 'end_time' => date('Y-m-d H:i:s', $end_time),
  181. ]);
  182. //预测
  183. PcPrediction::prediction($new_str);
  184. if (strtotime($issue->end_time) <= $now) {
  185. static::createIssueNo();
  186. }
  187. }
  188. }
  189. //获取随机的20个数字
  190. private static function getKeno(): array
  191. {
  192. $numbers = range(1, 80);
  193. shuffle($numbers);
  194. $random_numbers = array_slice($numbers, 0, 20);
  195. sort($random_numbers);
  196. return $random_numbers;
  197. }
  198. //根据20个数字计算开奖号码
  199. private static function getWinningNumbers($keno): array
  200. {
  201. $winningNumbers = [];
  202. $sum = $keno[1] + $keno[4] + $keno[7] + $keno[10] + $keno[13] + $keno[16];
  203. $winningNumbers[0] = $sum % 10;
  204. $sum = $keno[2] + $keno[5] + $keno[8] + $keno[11] + $keno[14] + $keno[17];
  205. $winningNumbers[1] = $sum % 10;
  206. $sum = $keno[3] + $keno[6] + $keno[9] + $keno[12] + $keno[15] + $keno[18];
  207. $winningNumbers[2] = $sum % 10;
  208. return $winningNumbers;
  209. }
  210. //根据指定0-27的数字 得到20个数字
  211. public static function getMatchingNumbers($target): array|null
  212. {
  213. $numbers = range(1, 80);
  214. $bestMatch = null;
  215. for ($i = 0; $i < 100000; $i++) {
  216. shuffle($numbers);
  217. $selectedNumbers = array_slice($numbers, 0, 20);
  218. sort($selectedNumbers);
  219. $firstSum = $selectedNumbers[1] + $selectedNumbers[4] + $selectedNumbers[7] + $selectedNumbers[10] + $selectedNumbers[13] + $selectedNumbers[16];
  220. $firstDigit = $firstSum % 10;
  221. $secondSum = $selectedNumbers[2] + $selectedNumbers[5] + $selectedNumbers[8] + $selectedNumbers[11] + $selectedNumbers[14] + $selectedNumbers[17];
  222. $secondDigit = $secondSum % 10;
  223. $thirdSum = $selectedNumbers[3] + $selectedNumbers[6] + $selectedNumbers[9] + $selectedNumbers[12] + $selectedNumbers[15] + $selectedNumbers[18];
  224. $thirdDigit = $thirdSum % 10;
  225. $result = $firstDigit + $secondDigit + $thirdDigit;
  226. if ($result === $target) {
  227. $bestMatch = $selectedNumbers;
  228. break;
  229. }
  230. }
  231. return $bestMatch;
  232. }
  233. }