TelegramWebHook.php 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632
  1. <?php
  2. namespace App\Http\Controllers\api;
  3. use App\Constants\StepStatus;
  4. use App\Constants\Util;
  5. use App\Exceptions\MessageException;
  6. use App\Http\Controllers\Controller;
  7. use App\Models\Message;
  8. use App\Models\User;
  9. use App\Services\OnLineService;
  10. use App\Services\QianBaoWithdrawService;
  11. use App\Services\SanJinRechargeService;
  12. use App\Services\RecordService;
  13. use App\Services\RoomService;
  14. use App\Services\SecretService;
  15. use App\Services\SettlementService;
  16. use App\Services\TopUpService;
  17. use App\Services\TutorialService;
  18. use App\Services\UserService;
  19. use App\Services\WalletService;
  20. use App\Services\WithdrawService;
  21. use App\Services\BalanceLogService;
  22. use Carbon\Carbon;
  23. use Illuminate\Http\Request;
  24. use Illuminate\Support\Facades\App;
  25. use Illuminate\Support\Facades\Cache;
  26. use Illuminate\Support\Facades\DB;
  27. use Telegram\Bot\Api;
  28. use Telegram\Bot\Exceptions\TelegramSDKException;
  29. use Illuminate\Support\Facades\Log;
  30. use App\Models\Config;
  31. use App\Services\GameplayRuleService;
  32. use App\Services\BetService;
  33. use App\Services\IssueService;
  34. use App\Services\KeyboardService;
  35. use Google\Service\Dfareporting\Language;
  36. use Telegram\Bot\FileUpload\InputFile;
  37. class TelegramWebHook extends BaseController
  38. {
  39. protected Api $telegram;
  40. public function __construct(Api $telegram)
  41. {
  42. $this->telegram = $telegram;
  43. parent::__construct();
  44. }
  45. public function handle(Request $request)
  46. {
  47. Log::error('Telegram 日志写入测试: ' . json_encode([$request->ip()], JSON_UNESCAPED_UNICODE));
  48. try {
  49. $telegram = new Api(config('services.telegram.token'));
  50. } catch (TelegramSDKException $e) {
  51. return response()->json(['status' => 'ok']);
  52. }
  53. try {
  54. $update = $telegram->getWebhookUpdate(); // 获取更新数据
  55. $update->callbackQuery;
  56. if ($update->has('callback_query')) {
  57. $callbackQuery = $update->callbackQuery;
  58. $message = $callbackQuery->message;
  59. $from = $callbackQuery->from;
  60. $data = $callbackQuery->data; // 获取 callback_data
  61. $callbackId = $callbackQuery->id; // 获取 callback_query 的 ID
  62. Util::delCache($message->chat->id);
  63. // Log::error('Telegram 回调数据(JSON): ' . json_encode($update, JSON_UNESCAPED_UNICODE));
  64. DB::beginTransaction();
  65. try {
  66. $chatId = $message->chat->id;
  67. $firstName = $message->chat->firstName;
  68. $username = '';
  69. $messageId = $message->messageId;
  70. if (!$from->isBot) {
  71. $chatId = $from->id;
  72. $firstName = $from->firstName;
  73. $username = $from->username;
  74. }
  75. $user = User::where('member_id', $chatId)->first();
  76. if (!$user) {
  77. $user = new User();
  78. $user->member_id = $chatId;
  79. }
  80. if ($username) $user->username = $username;
  81. $user->first_name = $firstName;
  82. $user->save();
  83. App::setLocale($user->language);
  84. //给每个用户生成一个专属的USDT钱包
  85. WalletService::getUserWallet($chatId);
  86. QianBaoWithdrawService::init($telegram, $data, $chatId, $firstName, $messageId);
  87. SanJinRechargeService::init($telegram, $data, $chatId, $firstName, $messageId);
  88. SecretService::init($telegram, $data, $chatId, $firstName, $messageId);
  89. UserService::init($telegram, $data, $chatId, $firstName, $messageId);
  90. // 查看余额弹窗
  91. if ($data === 'balanceAlert') {
  92. $alertText = WalletService::getBalance($chatId)['text'];
  93. WalletService::alertNotice($callbackId, $alertText);
  94. }
  95. // 今日流水弹窗
  96. if ($data === 'todayFlowAlert') {
  97. $alertText = BalanceLogService::getTodayFlowing($chatId)['text'];
  98. BalanceLogService::alertNotice($callbackId, $alertText);
  99. }
  100. // 近期注单弹窗
  101. if ($data === 'betsAlert') {
  102. Cache::put('message_id_bet_record_' . $chatId, 0, 600);
  103. $alertText = BetService::recentlyRecord($chatId);
  104. BetService::alertNotice($callbackId, $alertText);
  105. }
  106. //选择充值地址
  107. if ($data === "topUp@@TRC20" || $data === "topUp@@ERC20") {
  108. $type = preg_replace('/^topUp@@/', '', $data);
  109. $topService = new TopUpService();
  110. $res = $topService->scan($chatId, $messageId, $type);
  111. $telegram->deleteMessage([
  112. 'chat_id' => $chatId,
  113. 'message_id' => $messageId,
  114. ]);
  115. $telegram->sendPhoto($res);
  116. }
  117. //点击充值按钮
  118. if ($data === 'topup@@topup') {
  119. $res = TopUpService::chooseAddress($chatId, $messageId);
  120. $telegram->editMessageText($res);
  121. }
  122. //点击充值的账单按钮
  123. if ($data === 'topup@@bill') {
  124. $res = (new TopUpService())->bill($chatId, $firstName, $messageId);
  125. $telegram->editMessageText($res);
  126. }
  127. //点击我已付款按钮
  128. //手动充值(后台审核后到账)
  129. if ($data === 'topUp@@pay2') {
  130. $telegram->deleteMessage([
  131. 'chat_id' => $chatId,
  132. 'message_id' => $messageId
  133. ]);
  134. $res = TopUpService::pay2($chatId);
  135. $telegram->sendMessage($res);
  136. } //
  137. //自动充值
  138. elseif ($data === 'topUp@@pay') {
  139. $telegram->deleteMessage([
  140. 'chat_id' => $chatId,
  141. 'message_id' => $messageId
  142. ]);
  143. $topService = new TopUpService();
  144. $res = $topService->done($chatId);
  145. $telegram->sendMessage($res);
  146. }
  147. //充值首页
  148. if ($data === "topUp@@home" || $data === "topUp@@home1") {
  149. $returnMsg = WalletService::getBalance($chatId);
  150. if ($returnMsg) {
  151. if ($data === "topUp@@home1") {
  152. $telegram->deleteMessage([
  153. 'chat_id' => $chatId,
  154. 'message_id' => $messageId
  155. ]);
  156. $this->telegram->sendMessage($returnMsg);
  157. } else {
  158. $returnMsg['message_id'] = $messageId;
  159. $telegram->editMessageText($returnMsg);
  160. }
  161. }
  162. // $res = (new TopUpService())->index($chatId, $firstName, $messageId);
  163. // if ($data === "topUp@@home1") {
  164. // $telegram->deleteMessage([
  165. // 'chat_id' => $chatId,
  166. // 'message_id' => $messageId
  167. // ]);
  168. // $telegram->sendMessage($res);
  169. // } else {
  170. // $telegram->editMessageText($res);
  171. // }
  172. }
  173. //点击提现按钮
  174. if ($data === "withdraw@@apply") {
  175. $res = (new WithdrawService())->apply($chatId, $messageId);
  176. $telegram->editMessageText($res);
  177. }
  178. //地址管理
  179. if ($data === 'withdraw@@address') {
  180. $res = WithdrawService::getAddress($chatId, $messageId);
  181. $telegram->editMessageText($res);
  182. }
  183. //关闭本条消息
  184. if ($data === 'message@@close') {
  185. $telegram->deleteMessage([
  186. 'chat_id' => $chatId,
  187. 'message_id' => $messageId
  188. ]);
  189. }
  190. if ($data === 'withdrawAddress@@add') {
  191. $res = WithdrawService::addAddress($chatId, $messageId);
  192. $telegram->editMessageText($res);
  193. }
  194. //提现管理
  195. if ($data === "withdraw@@home") {
  196. // $res = WithdrawService::index($chatId, $firstName, $messageId);
  197. // $telegram->editMessageText($res);
  198. $telegram->deleteMessage([
  199. 'chat_id' => $chatId,
  200. 'message_id' => $messageId
  201. ]);
  202. $returnMsg = WalletService::getBalance($chatId);
  203. if ($returnMsg) {
  204. $this->telegram->sendMessage($returnMsg);
  205. }
  206. }
  207. //点击提现的账单按钮
  208. if ($data === "withdraw@@bill") {
  209. $res = (new WithdrawService())->bill($chatId, $firstName, $messageId);
  210. $telegram->editMessageText($res);
  211. // $telegram->sendMessage($res);
  212. // $text = "📅 请输入查询日期\n";
  213. // $date = date('Y-m-d');
  214. // $text .= "例如您要查询的日期 {$date}\n";
  215. // $text .= "那么请发送:【提现账单】{$date}\n";
  216. // $telegram->sendMessage([
  217. // 'chat_id' => $chatId,
  218. // 'text' => $text
  219. // ]);
  220. }
  221. if ($data === 'withdrawAddress@@done') {
  222. $res = WithdrawService::done($chatId, $messageId, $firstName);
  223. $telegram->editMessageText($res);
  224. }
  225. // 今日汇率
  226. if ($data === 'todayExchangeRate@@rate') {
  227. BetService::todayExchangeRate($chatId);
  228. // $telegram->sendMessage($res);
  229. }
  230. //查看开奖历史图片
  231. $pattern = "/^showLotteryHistory@@\d+$/";
  232. if (preg_match($pattern, $data)) {
  233. $id = preg_replace('/^showLotteryHistory@@/', '', $data);
  234. IssueService::sendLotteryImage($chatId, $id);
  235. }
  236. //选择投注记录
  237. $pattern = "/^betRecordType@@\d+$/";
  238. if (preg_match($pattern, $data)) {
  239. $type = preg_replace('/^betRecordType@@/', '', $data);
  240. Cache::put('message_id_bet_record_' . $chatId, intval($type), 600);
  241. $telegram->deleteMessage([
  242. 'chat_id' => $chatId,
  243. 'message_id' => $messageId,
  244. ]);
  245. $returnMsg = BetService::record($chatId);
  246. $telegram->sendMessage($returnMsg);
  247. }
  248. //选择提现地址
  249. $pattern = "/^withdrawAddress@@choose\d+$/";
  250. if (preg_match($pattern, $data)) {
  251. $id = preg_replace('/^withdrawAddress@@choose/', '', $data);
  252. $res = WithdrawService::chooseAddress($chatId, $firstName, $messageId, $id);
  253. $telegram->editMessageText($res);
  254. }
  255. //删除地址
  256. $pattern = "/^withdrawAddress@@del\d+$/";
  257. if (preg_match($pattern, $data)) {
  258. $id = preg_replace('/^withdrawAddress@@del/', '', $data);
  259. $res = WithdrawService::delAddress($chatId, $id, $messageId);
  260. $telegram->editMessageText($res);
  261. }
  262. //地址详情
  263. $pattern = "/^withdrawAddress@@detail\d+$/";
  264. if (preg_match($pattern, $data)) {
  265. $id = preg_replace('/^withdrawAddress@@detail/', '', $data);
  266. $res = WithdrawService::addressDetails($chatId, $messageId, $id);
  267. $telegram->editMessageText($res);
  268. }
  269. //充值账单,下一页
  270. $pattern = "/^topUpBillNextPage@@\d+$/";
  271. if (preg_match($pattern, $data)) {
  272. $page = preg_replace('/^topUpBillNextPage@@/', '', $data);
  273. $page = intval($page);
  274. $res = (new TopUpService())->bill($chatId, $firstName, $messageId, $page);
  275. $telegram->editMessageText($res);
  276. }
  277. //流水列表,下一页
  278. $pattern = "/^FlowingHistoryPage@@\d+$/";
  279. if (preg_match($pattern, $data)) {
  280. $page = preg_replace('/^FlowingHistoryPage@@/', '', $data);
  281. $page = intval($page);
  282. $returnMsg = BalanceLogService::getFlowingHistory($chatId, $messageId, $page);
  283. $telegram->editMessageText($returnMsg);
  284. }
  285. //提现账单,下一页
  286. $pattern = "/^withdrawBillNextPage@@\d+$/";
  287. if (preg_match($pattern, $data)) {
  288. $page = preg_replace('/^withdrawBillNextPage@@/', '', $data);
  289. $page = intval($page);
  290. $res = (new WithdrawService())->bill($chatId, $firstName, $messageId, $page);
  291. $telegram->editMessageText($res);
  292. }
  293. //近期注单,下一页
  294. $pattern = "/^betRecordNextPage@@\d+$/";
  295. if (preg_match($pattern, $data)) {
  296. $page = preg_replace('/^betRecordNextPage@@/', '', $data);
  297. $page = intval($page);
  298. $res = BetService::record($chatId, $messageId, $page);
  299. $telegram->editMessageText($res);
  300. }
  301. DB::commit();
  302. } //
  303. catch (MessageException $e) {
  304. DB::rollBack();
  305. $msg = $e->getMessage();
  306. $msg = json_decode($msg, true);
  307. $telegram->sendMessage($msg);
  308. } //
  309. catch (TelegramSDKException $e) {
  310. DB::rollBack();
  311. $m = new Message();
  312. $m->json = $e->getMessage();
  313. $m->save();
  314. $telegram->sendMessage([
  315. 'chat_id' => $chatId,
  316. 'text' => '‼️‼️系统发生了错误,请联系客服'
  317. ]);
  318. }//
  319. catch (\Exception $e) {
  320. DB::rollBack();
  321. $m = new Message();
  322. $m->json = json_encode([
  323. 'line' => $e->getLine(),
  324. 'message' => $e->getMessage()
  325. ]);
  326. $m->save();
  327. $telegram->sendMessage([
  328. 'chat_id' => $chatId,
  329. 'text' => '‼️‼️系统发生了错误,请联系客服'
  330. ]);
  331. }
  332. } //
  333. else {
  334. $update = $request->all();
  335. Log::error('Telegram 文字消息回复: ' . json_encode($update, JSON_UNESCAPED_UNICODE));
  336. if (isset($update['message'])) {
  337. $message = $update['message'];
  338. $chatId = $message['chat']['id'];
  339. $messageId = $message['message_id'];
  340. DB::beginTransaction();
  341. try {
  342. $m = new Message();
  343. $m->json = json_encode($update);
  344. $m->save();
  345. $returnMsg = $this->processChatMessage($chatId, $messageId, $message, $message['from']);
  346. if ($returnMsg) {
  347. if (isset($returnMsg['image']) && $returnMsg['image'] != '') {
  348. KeyboardService::sendMessage($returnMsg['chat_id'], $returnMsg['text'] ?? '', $returnMsg['keyboard'] ?? [], $returnMsg['image'] ?? '');
  349. } else if (isset($returnMsg['photo']) && $returnMsg['photo'] != '') {
  350. $this->telegram->sendPhoto($returnMsg);
  351. } else {
  352. $this->telegram->sendMessage($returnMsg);
  353. }
  354. }
  355. DB::commit();
  356. } catch (MessageException $e) {
  357. DB::rollBack();
  358. $msg = $e->getMessage();
  359. $msg = json_decode($msg, true);
  360. $telegram->sendMessage($msg);
  361. } //
  362. catch (TelegramSDKException $e) {
  363. DB::rollBack();
  364. $m = new Message();
  365. $m->json = $e->getMessage();
  366. $m->save();
  367. $telegram->sendMessage([
  368. 'chat_id' => $chatId,
  369. 'text' => '‼️‼️系统发生了错误,请联系客服'
  370. ]);
  371. }//
  372. catch (\Exception $e) {
  373. DB::rollBack();
  374. $m = new Message();
  375. $m->json = json_encode([
  376. 'line' => $e->getLine(),
  377. 'message' => $e->getMessage()
  378. ]);
  379. $m->save();
  380. Log::error('Telegram 处理消息异常: ' . $e->getMessage());
  381. $telegram->sendMessage([
  382. 'chat_id' => $chatId,
  383. 'text' => '‼️‼️系统发生了错误,请联系客服'
  384. ]);
  385. }
  386. }
  387. }
  388. } //
  389. catch (\Exception $e) {
  390. $m = new Message();
  391. $m->json = $e->getMessage();
  392. $m->save();
  393. if (!empty($chatId)) {
  394. $telegram->sendMessage([
  395. 'chat_id' => $chatId,
  396. 'text' => '‼️‼️系统发生了错误,请联系客服'
  397. ]);
  398. }
  399. }
  400. return response()->json(['status' => 'ok']);
  401. }
  402. /**
  403. * @description: 处理聊天消息
  404. * @param {*} $chatId
  405. * @param {*} $messageId
  406. * @param {*} $message
  407. * @param {*} $from
  408. * @throws TelegramSDKException
  409. */
  410. public function processChatMessage($chatId, $messageId, $message, $from)
  411. {
  412. $returnMsg = [];
  413. //用户发送图片,结算截图
  414. if (isset($message['photo'])) {
  415. $stepStatus = Cache::get(get_step_key($chatId), -1);
  416. $stepStatus = intval($stepStatus);
  417. // //结算截图
  418. if ($stepStatus === StepStatus::INPUT_IMAGE) {
  419. $photo = $message['photo'][count($message['photo']) - 1];
  420. return (new SettlementService())->photo($photo, $chatId);
  421. }//
  422. //充值截图
  423. else if ($stepStatus === StepStatus::INPUT_TOP_UP_IMAGE) {
  424. $photo = $message['photo'][count($message['photo']) - 1];
  425. return TopUpService::photo($chatId, $photo);
  426. } else {
  427. return [];
  428. }
  429. } //用户发送了消息
  430. else if (isset($message['text'])) {
  431. $text = $message['text'];
  432. $user = User::where('member_id', $chatId)->first();
  433. if (!$user) {
  434. $user = new User();
  435. $user->member_id = $chatId;
  436. }
  437. $user->first_name = $message['chat']['first_name'];
  438. if (isset($message['chat']['username'])) {
  439. $user->username = $message['chat']['username'];
  440. }
  441. $user->save();
  442. App::setLocale($user->language);
  443. if ($message['chat']['type'] === 'private') {
  444. // 校验开始菜单事件
  445. $returnMsg = KeyboardService::checkStart($chatId, $text);
  446. if ($returnMsg) {
  447. return $returnMsg;
  448. }
  449. switch ($text) {
  450. case "/start":
  451. Util::delCache($chatId);
  452. //给每个用户生成一个专属的USDT钱包
  453. WalletService::getUserWallet($chatId);
  454. self::setReplyKeyboard($chatId, $user->language);
  455. break;
  456. default:
  457. // 关键字回复
  458. $keyboardText = KeyboardService::findOne(['button' => $text]);
  459. if ($keyboardText) {
  460. $keyboard = [];
  461. if ($keyboardText['buttons']) {
  462. $keyboard = json_decode($keyboardText['buttons'], true);
  463. }
  464. Util::delCache($chatId);
  465. $res = [
  466. 'chat_id' => $chatId,
  467. 'text' => $keyboardText['reply'],
  468. // 'reply_to_message_id' => $messageId
  469. ];
  470. if ($keyboard) {
  471. $res['reply_markup'] = json_encode(['inline_keyboard' => $keyboard]);
  472. }
  473. if ($keyboardText['image']) {
  474. $res['photo'] = InputFile::create(url($keyboardText['image']));
  475. unset($res['text']);
  476. $res['caption'] = $text;
  477. $res['protect_content'] = true; // 防止转发
  478. }
  479. return $res;
  480. }
  481. $stepStatus = Cache::get(get_step_key($chatId), -1);
  482. $stepStatus = intval($stepStatus);
  483. $res = QianBaoWithdrawService::onMessage($chatId, $text, $messageId, $stepStatus);
  484. if (empty($res)) $res = SanJinRechargeService::onMessage($chatId, $text, $messageId, $stepStatus);
  485. if (empty($res)) $res = SecretService::onMessage($chatId, $text, $messageId, $stepStatus);
  486. if (!empty($res)) return $res;
  487. switch ($stepStatus) {
  488. case StepStatus::INPUT_TOP_UP_MONEY:
  489. return TopUpService::inputAmount($chatId, $text, $messageId);
  490. case StepStatus::INPUT_WITHDRAW_MONEY:
  491. $res = (new WithdrawService())->inputAmount($chatId, $text, $messageId);
  492. return $res[0];
  493. case StepStatus::INPUT_ADDRESS_TRC20:
  494. return WithdrawService::inputAddress($chatId, $text, $messageId);
  495. case StepStatus::INPUT_ADDRESS_ALIAS:
  496. return WithdrawService::inputAlias($chatId, $text, $messageId);
  497. }
  498. $returnMsg = BetService::bet($chatId, $text, $messageId);
  499. }
  500. return $returnMsg;
  501. }
  502. }
  503. return [];
  504. }
  505. /**
  506. * @description: 设置 start 回复菜单
  507. * @param {*} $chatId
  508. * @throws TelegramSDKException
  509. */
  510. public static function setReplyKeyboard($chatId, $language = 'en'): void
  511. {
  512. $replyInfo = KeyboardService::findOne(['button' => '开始使用', 'language' => $language]);
  513. if (empty($replyInfo)) {
  514. $replyInfo = KeyboardService::findOne(['button' => '开始使用', 'language' => 'en']);
  515. }
  516. $telegram = new Api(config('services.telegram.token'));
  517. $keyboard = [
  518. [lang('近期注单'), lang('今日流水'), lang('联系客服')], // 第一排按钮
  519. [lang('开奖历史'), lang('当期下注'), lang('查看余额')], // 第二排按钮
  520. [lang('投注大群')]
  521. ];
  522. if ($replyInfo && $replyInfo->buttons) {
  523. $keyboard = [];
  524. $buttons = json_decode($replyInfo->buttons, true);
  525. foreach ($buttons as $rowIndex => $row) {
  526. if (!empty($row)) {
  527. foreach ($row as $buttonIndex => $button) {
  528. // $keyboard[$rowIndex][$buttonIndex] = lang($button['text']);
  529. $keyboard[$rowIndex][$buttonIndex] = $button['text'];
  530. }
  531. }
  532. }
  533. $keyboard = array_values($keyboard); // 重新索引数组
  534. Log::error('自定义开始使用按钮: ' . json_encode($keyboard));
  535. }
  536. $botMsg = [];
  537. $botMsg['chat_id'] = $chatId;
  538. $replyMarkup = [
  539. 'keyboard' => $keyboard,
  540. 'resize_keyboard' => true, // 自适应大小
  541. 'one_time_keyboard' => false, // 保持显示,不会点击后收起
  542. ];
  543. $botMsg['reply_markup'] = json_encode($replyMarkup);
  544. if ($replyInfo) {
  545. $image = '';
  546. if ($replyInfo->image) {
  547. $image = url($replyInfo->image);
  548. }
  549. if ($image != '') {
  550. $botMsg['photo'] = InputFile::create($image);
  551. $botMsg['caption'] = lang($replyInfo->reply);
  552. $botMsg['protect_content'] = true; // 防止转发
  553. KeyboardService::telegram()->sendPhoto($botMsg);
  554. } else {
  555. $botMsg['text'] = lang($replyInfo->reply);
  556. KeyboardService::telegram()->sendMessage($botMsg);
  557. }
  558. } else {
  559. $telegram->sendMessage([
  560. 'chat_id' => $chatId,
  561. 'text' => lang('你好,请选择功能菜单'),
  562. 'reply_markup' => json_encode($replyMarkup),
  563. ]);
  564. }
  565. }
  566. }