diff --git a/README.md b/README.md index 2d323c8..108042f 100644 --- a/README.md +++ b/README.md @@ -16,7 +16,6 @@ - PHP ^8.0 - Composer - Telegram Bot -- SSL Certificate ## Installation @@ -70,12 +69,14 @@ APP_URL=https://123456789.ngrok.io ### Set the webhook -#### Set the webhook from the source code +We have two ways to set the webhook: + +#### 1. Set the webhook from this project After setting up your domain and SSL certificate, you need to set up the webhook for your bot. Go to: ```text -/setWebhook.php +/webhook/set.php ``` > **Note:** Replace `` with your app URL in .env file. @@ -86,7 +87,7 @@ If you see the following message, it means that the webhook has been sent succes {"ok":true,"result":true,"description":"Webhook was set"} ``` -#### Set the webhook manually +#### 2. Set the webhook manually If you want to set the webhook manually, you can use the following URL: @@ -140,4 +141,4 @@ Now you can send a message to your bot, and you will receive a notification. Here is the first notification you will receive: ♻️ **Connection Successful** -> **You can add multiple webhooks to your repository.** +> **Note: You can add multiple webhooks to your repository. Please similarly set up the webhook for each repository.** diff --git a/resources/tools/custom_event_actions.php b/resources/tools/custom_event_actions.php new file mode 100644 index 0000000..848cdbc --- /dev/null +++ b/resources/tools/custom_event_actions.php @@ -0,0 +1,8 @@ + +Setting actions for the event: +Please select an action of this event to enable or disable notifications: diff --git a/resources/tools/custom_events.php b/resources/tools/custom_events.php new file mode 100644 index 0000000..d71fdf6 --- /dev/null +++ b/resources/tools/custom_events.php @@ -0,0 +1,4 @@ +Go to check the GitHub documentation for more information about events. +--- +Click and configure child events if the option has the ⚙ icon. +And select an event to enable or disable notifications: diff --git a/resources/tools/event.php b/resources/tools/event.php deleted file mode 100644 index a753ea8..0000000 --- a/resources/tools/event.php +++ /dev/null @@ -1,4 +0,0 @@ -Please select an event to enable or disable notifications. -Click and configure child events if the option has the ⚙ icon. ---- -Please check the GitHub documentation for more information about events. diff --git a/resources/tools/menu.php b/resources/tools/menu.php index 37db9f4..589e98e 100644 --- a/resources/tools/menu.php +++ b/resources/tools/menu.php @@ -4,7 +4,7 @@ /id - To get your Chat ID. /token - To get this bot token. /usage - How to use me. -/settings - Settings github notify. +/settings - Settings GitHub notify. /menu - To get this menu. Select a button: diff --git a/resources/tools/settings.php b/resources/tools/settings.php index fa2ed29..cd7de79 100644 --- a/resources/tools/settings.php +++ b/resources/tools/settings.php @@ -1 +1 @@ -Settings for github notify. +Settings for GitHub notify. diff --git a/resources/tools/start.php b/resources/tools/start.php index 4ec152b..30d42ec 100644 --- a/resources/tools/start.php +++ b/resources/tools/start.php @@ -9,4 +9,4 @@ Hey , I can send you notifications from your GitHub Repository instantly to your Telegram. -Use /menu for more information about me. +Use /menu for more options. diff --git a/src/Models/Event.php b/src/Models/Event.php index 25f735e..be2c605 100644 --- a/src/Models/Event.php +++ b/src/Models/Event.php @@ -6,7 +6,7 @@ class Event { public const EVENT_FILE = __DIR__ . '/../../storage/tg-event.json'; - public const EVENT_PREFIX = Setting::SETTING_PREFIX . '.event.'; + public const EVENT_PREFIX = Setting::SETTING_CUSTOM_EVENTS . '.evt.'; public array $eventConfig = []; @@ -37,4 +37,35 @@ public function getEventConfig(): array { return $this->eventConfig; } + + /** + * Update event config by event and action + * + * @param string $event + * @param string|null $action + * @return void + */ + public function updateEvent(string $event, string|null $action): void + { + if (!empty($action)) { + $this->eventConfig[$event][$action] = !$this->eventConfig[$event][$action]; + } else { + $this->eventConfig[$event] = !$this->eventConfig[$event]; + } + + $this->saveEventConfig(); + } + + /** + * Save event config + * + * @return void + */ + private function saveEventConfig(): void + { + if (file_exists(self::EVENT_FILE)) { + $json = json_encode($this->eventConfig, JSON_PRETTY_PRINT); + file_put_contents(self::EVENT_FILE, $json, LOCK_EX); + } + } } diff --git a/src/Models/Setting.php b/src/Models/Setting.php index c1b03f6..e9e6aef 100644 --- a/src/Models/Setting.php +++ b/src/Models/Setting.php @@ -5,11 +5,12 @@ class Setting { public const SETTING_FILE = __DIR__ . '/../../storage/tg-setting.json'; - public const SETTING_PREFIX = 'setting.'; + public const SETTING_PREFIX = 'stg.'; public const SETTING_IS_NOTIFIED = self::SETTING_PREFIX . 'is_notified'; public const SETTING_ALL_EVENTS_NOTIFY = self::SETTING_PREFIX . 'all_events_notify'; - public const SETTING_CUSTOM_EVENTS = self::SETTING_PREFIX . 'custom_events'; + public const SETTING_CUSTOM_EVENTS = self::SETTING_PREFIX . 'cus'; + public const SETTING_BACK = self::SETTING_PREFIX . 'back.'; public array $settings = []; @@ -64,4 +65,51 @@ public function isNotified(): bool return false; } + + /** + * Update setting item value and save to file + * + * @param string $settingName + * @param $settingValue + * @return bool + */ + public function updateSettingItem(string $settingName, $settingValue = null): bool + { + $keys = explode('.', $settingName); + $lastKey = array_pop($keys); + $nestedSettings = &$this->settings; + + foreach ($keys as $key) { + if (!isset($nestedSettings[$key]) || !is_array($nestedSettings[$key])) { + return false; + } + $nestedSettings = &$nestedSettings[$key]; + } + + if (isset($nestedSettings[$lastKey])) { + $nestedSettings[$lastKey] = $settingValue ?? !$nestedSettings[$lastKey]; + if ($this->saveSettingsToFile()) { + return true; + } + } + + return false; + } + + /** + * Save settings to json file + * + * @return bool + */ + private function saveSettingsToFile(): bool + { + if (file_exists(self::SETTING_FILE)) { + $json = json_encode($this->settings, JSON_PRETTY_PRINT); + file_put_contents(self::SETTING_FILE, $json, LOCK_EX); + + return true; + } + + return false; + } } diff --git a/src/Services/AppService.php b/src/Services/AppService.php index 24a173f..71e317e 100644 --- a/src/Services/AppService.php +++ b/src/Services/AppService.php @@ -33,22 +33,16 @@ public function sendMessage(string $message = '', array $options = [], string $s 'parse_mode' => 'HTML' ); - try { - if ($sendType === 'Message') { - $content['text'] = $message; - } elseif ($sendType === 'Photo' && !empty($options)) { - $content['photo'] = $options['photo']; - $content['caption'] = $message; - } - - if (!empty($options) && isset($options['reply_markup'])) { - $content['reply_markup'] = $this->telegram->buildInlineKeyBoard($options['reply_markup']); - } - - $this->telegram->{'send' . $sendType}($content); - } catch (Exception $e) { - error_log($e->getMessage()); + if ($sendType === 'Message') { + $content['text'] = $message; + } elseif ($sendType === 'Photo') { + $content['photo'] = $options['photo'] ?? null; + $content['caption'] = $message; } + + $content['reply_markup'] = $options['reply_markup'] ? $this->telegram->buildInlineKeyBoard($options['reply_markup']) : null; + + $this->telegram->{'send' . $sendType}($content); } /** @@ -81,10 +75,9 @@ public function answerCallbackQuery(string $text = null): void public function editMessageText(?string $text = null, array $options = []): void { try { - $content = [ + $content = array_merge([ 'text' => $text ?? $this->Callback_Message_Text() - ]; - $content = array_merge($content, $this->setContentEditMessage($options)); + ], $this->setCallbackContentMessage($options)); $this->telegram->editMessageText($content); } catch (Exception $e) { @@ -102,7 +95,7 @@ public function editMessageText(?string $text = null, array $options = []): void public function editMessageReplyMarkup(array $options = []): void { try { - $this->telegram->editMessageReplyMarkup($this->setContentEditMessage($options)); + $this->telegram->editMessageReplyMarkup($this->setCallbackContentMessage($options)); } catch (Exception $e) { error_log($e->getMessage()); } @@ -119,10 +112,12 @@ public function Callback_Message_Text(): string } /** + * Create content for a callback message + * * @param array $options * @return array */ - public function setContentEditMessage(array $options = []): array + public function setCallbackContentMessage(array $options = []): array { $content = array( 'chat_id' => $this->telegram->Callback_ChatID(), @@ -131,10 +126,25 @@ public function setContentEditMessage(array $options = []): array 'parse_mode' => 'HTML', ); - if (!empty($options) && isset($options['reply_markup'])) { - $content['reply_markup'] = $this->telegram->buildInlineKeyBoard($options['reply_markup']); - } + $content['reply_markup'] = $options['reply_markup'] ? $this->telegram->buildInlineKeyBoard($options['reply_markup']) : null; return $content; } + + /** + * Generate menu markup + * + * @return array[] + */ + public function menuMarkup(): array + { + return [ + [ + $this->telegram->buildInlineKeyBoardButton("📰 About", "", "about", ""), + $this->telegram->buildInlineKeyBoardButton("📞 Contact", config('author.contact')) + ], [ + $this->telegram->buildInlineKeyBoardButton("💠 Source Code", config('author.source_code')) + ] + ]; + } } diff --git a/src/Services/EventService.php b/src/Services/EventService.php index 19e3c4c..572bceb 100644 --- a/src/Services/EventService.php +++ b/src/Services/EventService.php @@ -10,6 +10,10 @@ class EventService extends AppService { public const LINE_ITEM_COUNT = 2; + public const EVENT_HAS_ACTION_SEPARATOR = 'atc.'; + + public const EVENT_UPDATE_SEPARATOR = '.upd'; + protected Setting $setting; protected Event $event; @@ -67,22 +71,14 @@ public function eventMarkup(?string $event = null): array $events = $event === null ? $this->event->eventConfig : $this->event->eventConfig[$event]; - foreach ($events as $event => $value) { + foreach ($events as $key => $value) { if (count($replyMarkupItem) === self::LINE_ITEM_COUNT) { $replyMarkup[] = $replyMarkupItem; $replyMarkupItem = []; } - $callbackData = $this->event::EVENT_PREFIX . $event; - - if (is_array($value)) { - $eventName = '⚙ ' . $event; - $callbackData .= '.child'; - } elseif ($value) { - $eventName = '✅ ' . $event; - } else { - $eventName = '❌ ' . $event; - } + $callbackData = $this->getCallbackData($key, $value, $event); + $eventName = $this->getEventName($key, $value); $replyMarkupItem[] = $this->telegram->buildInlineKeyBoardButton($eventName, '', $callbackData); } @@ -92,20 +88,116 @@ public function eventMarkup(?string $event = null): array $replyMarkup[] = $replyMarkupItem; } - $replyMarkup[] = [$this->telegram->buildInlineKeyBoardButton('🔙 Back', '', 'back')]; - $replyMarkup[] = [$this->telegram->buildInlineKeyBoardButton('📚 Menu', '', $this->setting::SETTING_PREFIX . '.back.menu')]; + $replyMarkup[] = $this->getEndKeyboard($event); return $replyMarkup; } /** + * Get event name for markup + * + * @param string $event + * @param $value + * @return string + */ + private function getEventName(string $event, $value): string + { + if (is_array($value)) { + return '⚙ ' . $event; + } elseif ($value) { + return '✅ ' . $event; + } + + return '❌ ' . $event; + } + + /** + * Get callback data for markup + * + * @param string $event + * @param array|string $value + * @param string|null $parentEvent + * + * @return string + */ + private function getCallbackData(string $event, array|string $value, ?string $parentEvent = null): string + { + if (is_array($value)) { + return $this->event::EVENT_PREFIX . self::EVENT_HAS_ACTION_SEPARATOR . $event; + } elseif ($parentEvent) { + return $this->event::EVENT_PREFIX . $parentEvent . '.' . $event . self::EVENT_UPDATE_SEPARATOR; + } + + return $this->event::EVENT_PREFIX . $event . self::EVENT_UPDATE_SEPARATOR; + } + + /** + * Get end keyboard buttons + * + * @param string|null $event + * @return array + */ + public function getEndKeyboard(?string $event = null): array + { + $back = $this->setting::SETTING_BACK . 'settings'; + + if ($event) { + $back = $this->setting::SETTING_BACK . 'settings.custom_events'; + } + + return [ + $this->telegram->buildInlineKeyBoardButton('🔙 Back', '', $back), + $this->telegram->buildInlineKeyBoardButton('📚 Menu', '', $this->setting::SETTING_BACK . 'menu') + ]; + } + + /** + * Handle event callback settings + * + * @param string|null $callback * @return void */ - public function eventHandle(): void + public function eventHandle(?string $callback = null): void { - $this->editMessageText( - view('tools.event'), - ['reply_markup' => $this->eventMarkup()] - ); + // first event settings + if ($this->setting::SETTING_CUSTOM_EVENTS === $callback || empty($callback)) { + $this->editMessageText( + view('tools.custom_events'), + ['reply_markup' => $this->eventMarkup()] + ); + return; + } + + $event = str_replace($this->event::EVENT_PREFIX, '', $callback); + + // if event has actions + if (str_contains($callback, self::EVENT_HAS_ACTION_SEPARATOR)) { + $event = str_replace(self::EVENT_HAS_ACTION_SEPARATOR, '', $event); + $this->editMessageText( + view('tools.custom_event_actions', compact('event')), + ['reply_markup' => $this->eventMarkup($event)] + ); + } + + if (str_contains($event, self::EVENT_UPDATE_SEPARATOR)) { + $event = str_replace(self::EVENT_UPDATE_SEPARATOR, '', $event); + $this->eventUpdateHandle($event); + } + } + + /** + * Handle event update + * + * @param string $event + * @return void + */ + public function eventUpdateHandle(string $event): void + { + $event = explode('.', $event); + $action = $event[1] ?? null; + $event = $event[0]; + + $this->event->updateEvent($event, $action); + $this->eventHandle($action ? self::EVENT_HAS_ACTION_SEPARATOR . $event : null); } } diff --git a/src/Services/NotificationService.php b/src/Services/NotificationService.php index fa7aa8f..996619e 100644 --- a/src/Services/NotificationService.php +++ b/src/Services/NotificationService.php @@ -69,7 +69,7 @@ private function setMessage(string $typeEvent): void } /** - * Send notify to telegram + * Send notification to telegram * * @param string $chatId * @param string|null $message diff --git a/src/Services/SettingService.php b/src/Services/SettingService.php index cee797e..ccba48c 100644 --- a/src/Services/SettingService.php +++ b/src/Services/SettingService.php @@ -8,16 +8,15 @@ class SettingService extends AppService { protected Setting $setting; - protected array $settingConfig = []; - public function __construct() { parent::__construct(); $this->setting = new Setting(); - $this->settingConfig = $this->setting->getSettingConfig(); } /** + * Send a setting message + * * @return void */ public function settingHandle(): void @@ -29,57 +28,70 @@ public function settingHandle(): void } /** + * Generate setting markup + * * @return array[] */ public function settingMarkup(): array { + $allEventKeyboard = [ + $this->telegram->buildInlineKeyBoardButton( + $this->setting->settings['all_events_notify'] + ? '✅ Enable All Events Notify' : 'Enable All Events Notify', + '', + $this->setting::SETTING_ALL_EVENTS_NOTIFY + ), + ]; + + if (!$this->setting->settings['all_events_notify']) { + $allEventKeyboard[] = $this->telegram->buildInlineKeyBoardButton( + '⚙ Custom individual events', + '', + $this->setting::SETTING_CUSTOM_EVENTS + ); + } + return [ [ $this->telegram->buildInlineKeyBoardButton( - $this->settingConfig['is_notified'] + $this->setting->settings['is_notified'] ? '✅ Github notification' : 'Github notification', '', $this->setting::SETTING_IS_NOTIFIED ), ], - [ - $this->telegram->buildInlineKeyBoardButton( - $this->settingConfig['all_events_notify'] - ? '✅ Enable All Events Notify' - : 'Enable All Events Notify', - '', - $this->setting::SETTING_ALL_EVENTS_NOTIFY - ), - $this->telegram->buildInlineKeyBoardButton( - '⚙ Custom individual events', - '', - $this->setting::SETTING_CUSTOM_EVENTS - ), - ], + $allEventKeyboard, [ $this->telegram->buildInlineKeyBoardButton( '🔙 Back to menu', '', - $this->setting::SETTING_PREFIX . '.back.menu' + $this->setting::SETTING_BACK . 'menu' ), ] ]; } /** + * Setting callback handler + * * @param string $callback * @return void */ public function settingCallbackHandler(string $callback): void { - if ($callback === $this->setting::SETTING_CUSTOM_EVENTS) { - (new EventService())->eventHandle(); + if (str_contains($callback, $this->setting::SETTING_CUSTOM_EVENTS)) { + (new EventService())->eventHandle($callback); + return; + } + + if (str_contains($callback, $this->setting::SETTING_BACK)) { + $this->answerBackButton($callback); return; } $callback = str_replace($this->setting::SETTING_PREFIX, '', $callback); - if ($this->updateSettingItem($callback, !$this->settingConfig[$callback])) { + if ($this->setting->updateSettingItem($callback, !$this->setting->settings[$callback])) { $this->editMessageReplyMarkup([ 'reply_markup' => $this->settingMarkup(), ]); @@ -89,45 +101,32 @@ public function settingCallbackHandler(string $callback): void } /** - * @param string $settingName - * @param $settingValue - * @return bool - */ - public function updateSettingItem(string $settingName, $settingValue = null): bool - { - $keys = explode('.', $settingName); - $lastKey = array_pop($keys); - $nestedSettings = &$this->settingConfig; - - foreach ($keys as $key) { - if (!isset($nestedSettings[$key]) || !is_array($nestedSettings[$key])) { - return false; - } - $nestedSettings = &$nestedSettings[$key]; - } - - if (isset($nestedSettings[$lastKey])) { - $nestedSettings[$lastKey] = $settingValue ?? !$nestedSettings[$lastKey]; - if ($this->saveSettingsToFile()) { - return true; - } - } - - return false; - } - - /** - * @return bool + * Answer the back button + * + * @param string $callback + * @return void */ - private function saveSettingsToFile(): bool + public function answerBackButton(string $callback): void { - if (file_exists($this->setting::SETTING_FILE)) { - $json = json_encode($this->settingConfig, JSON_PRETTY_PRINT); - file_put_contents($this->setting::SETTING_FILE, $json, LOCK_EX); - - return true; + $callback = str_replace($this->setting::SETTING_BACK, '', $callback); + + switch ($callback) { + case 'settings': + $view = view('tools.settings'); + $markup = $this->settingMarkup(); + break; + case 'settings.custom_events': + $view = view('tools.custom_events'); + $markup = (new EventService())->eventMarkup(); + break; + default: + $view = view('tools.menu'); + $markup = $this->menuMarkup(); + break; } - return false; + $this->editMessageText($view, [ + 'reply_markup' => $markup, + ]); } } diff --git a/src/Services/TelegramService.php b/src/Services/TelegramService.php index 708f3ab..3ab1277 100644 --- a/src/Services/TelegramService.php +++ b/src/Services/TelegramService.php @@ -80,19 +80,4 @@ public function checkCallback(): bool } return false; } - - /** - * @return array[] - */ - private function menuMarkup(): array - { - return [ - [ - $this->telegram->buildInlineKeyBoardButton("📰 About", "", "about", ""), - $this->telegram->buildInlineKeyBoardButton("📞 Contact", config('author.contact')) - ], [ - $this->telegram->buildInlineKeyBoardButton("💠 Source Code", config('author.source_code')) - ] - ]; - } }