Telegram services refactoring.

This commit is contained in:
Alexey Skobkin 2017-01-06 20:18:39 +03:00
parent a84c2dcd3b
commit e6338b0cbd
4 changed files with 166 additions and 151 deletions

View file

@ -106,9 +106,9 @@ services:
class: unreal4u\TelegramAPI\TgLog class: unreal4u\TelegramAPI\TgLog
arguments: [%telegram_token%, @logger, @point_tools.http.telegram_client] arguments: [%telegram_token%, @logger, @point_tools.http.telegram_client]
# Simple message sender # Message sender
point_tools.telegram.simple_sender: point_tools.telegram.message_sender:
class: Skobkin\Bundle\PointToolsBundle\Service\Telegram\SimpleSender class: Skobkin\Bundle\PointToolsBundle\Service\Telegram\MessageSender
arguments: [@point_tools.telegram.api_client] arguments: [@point_tools.telegram.api_client]
# Common incoming message processor # Common incoming message processor
@ -129,7 +129,7 @@ services:
class: Skobkin\Bundle\PointToolsBundle\Service\Telegram\PrivateMessageProcessor class: Skobkin\Bundle\PointToolsBundle\Service\Telegram\PrivateMessageProcessor
lazy: true lazy: true
arguments: arguments:
- @point_tools.telegram.api_client - @point_tools.telegram.message_sender
- @skobkin_point_tools.api_user - @skobkin_point_tools.api_user
- @point_tools.factory.telegram_account - @point_tools.factory.telegram_account
- @doctrine.orm.entity_manager - @doctrine.orm.entity_manager

View file

@ -0,0 +1,66 @@
<?php
namespace Skobkin\Bundle\PointToolsBundle\Service\Telegram;
use GuzzleHttp\Exception\ClientException;
use Skobkin\Bundle\PointToolsBundle\Entity\Telegram\Account;
use unreal4u\TelegramAPI\Telegram\Methods\SendMessage;
use unreal4u\TelegramAPI\Telegram\Types\ReplyKeyboardMarkup;
use unreal4u\TelegramAPI\TgLog;
/**
* Service which sends simple messages to Telegram users
*/
class MessageSender
{
const PARSE_MODE_NOPARSE = '';
const PARSE_MODE_MARKDOWN = 'Markdown';
const PARSE_MODE_HTML5 = 'HTML';
/**
* @var TgLog
*/
private $client;
/**
* @param TgLog $client
*/
public function __construct(TgLog $client)
{
$this->client = $client;
}
public function sendMessageToUser(
Account $account,
string $text,
string $parseMode = self::PARSE_MODE_NOPARSE,
ReplyKeyboardMarkup $keyboardMarkup = null,
bool $disableWebPreview = false,
$disableNotifications = false
): bool
{
return $this->sendMessageToChat($account->getChatId(), $text, $parseMode, $keyboardMarkup, $disableWebPreview, $disableNotifications);
}
public function sendMessageToChat(
int $chatId,
string $text,
string $parseMode = self::PARSE_MODE_NOPARSE,
ReplyKeyboardMarkup $keyboardMarkup = null,
bool $disableWebPreview = false,
$disableNotifications = false
): bool
{
$sendMessage = new SendMessage();
$sendMessage->chat_id = (string) $chatId;
$sendMessage->text = $text;
try {
$this->client->performApiRequest($sendMessage);
return true;
} catch (ClientException $e) {
return false;
}
}
}

View file

@ -11,29 +11,18 @@ use Skobkin\Bundle\PointToolsBundle\Repository\SubscriptionRepository;
use Skobkin\Bundle\PointToolsBundle\Repository\UserRepository; use Skobkin\Bundle\PointToolsBundle\Repository\UserRepository;
use Skobkin\Bundle\PointToolsBundle\Service\Factory\Telegram\AccountFactory; use Skobkin\Bundle\PointToolsBundle\Service\Factory\Telegram\AccountFactory;
use Skobkin\Bundle\PointToolsBundle\Service\UserApi; use Skobkin\Bundle\PointToolsBundle\Service\UserApi;
use unreal4u\TelegramAPI\Telegram\Methods\SendMessage;
use unreal4u\TelegramAPI\Telegram\Types\Message; use unreal4u\TelegramAPI\Telegram\Types\Message;
use unreal4u\TelegramAPI\TgLog; use unreal4u\TelegramAPI\Telegram\Types\ReplyKeyboardMarkup;
/** /**
* Processes all private messages * Processes all private messages
*/ */
class PrivateMessageProcessor class PrivateMessageProcessor
{ {
const TEMPLATE_ERROR = '@SkobkinPointTools/Telegram/error.md.twig';
const TEMPLATE_STATS = '@SkobkinPointTools/Telegram/stats.md.twig';
const TEMPLATE_HELP = '@SkobkinPointTools/Telegram/help.md.twig';
const TEMPLATE_LAST_EVENTS = '@SkobkinPointTools/Telegram/last_global_subscriptions.md.twig';
const TEMPLATE_LAST_USER_SUB_EVENTS = '@SkobkinPointTools/Telegram/last_user_subscriptions.md.twig';
const TEMPLATE_USER_SUBSCRIBERS = '@SkobkinPointTools/Telegram/user_subscribers.md.twig';
const PARSE_MODE_MARKDOWN = 'Markdown';
const PARSE_MODE_HTML5 = 'HTML';
/** /**
* @var TgLog * @var MessageSender
*/ */
private $client; private $messenger;
/** /**
* @var UserApi * @var UserApi
@ -77,7 +66,7 @@ class PrivateMessageProcessor
public function __construct( public function __construct(
TgLog $client, MessageSender $messageSender,
UserApi $userApi, UserApi $userApi,
AccountFactory $accountFactory, AccountFactory $accountFactory,
EntityManagerInterface $em, EntityManagerInterface $em,
@ -85,7 +74,7 @@ class PrivateMessageProcessor
int $pointUserId int $pointUserId
) )
{ {
$this->client = $client; $this->messenger = $messageSender;
$this->userApi = $userApi; $this->userApi = $userApi;
$this->accountFactory = $accountFactory; $this->accountFactory = $accountFactory;
$this->em = $em; $this->em = $em;
@ -100,24 +89,26 @@ class PrivateMessageProcessor
public function process(Message $message) public function process(Message $message)
{ {
if (!IncomingUpdateDispatcher::CHAT_TYPE_PRIVATE === $message->chat->type) { if (!IncomingUpdateDispatcher::CHAT_TYPE_PRIVATE === $message->chat->type) {
throw new \InvalidArgumentException('This service can process only private chat messages'); throw new \LogicException('This service can process only private chat messages');
} }
try {
// Registering Telegram user // Registering Telegram user
/** @var Account $account */ /** @var Account $account */
$account = $this->accountFactory->findOrCreateFromMessage($message); $account = $this->accountFactory->findOrCreateFromMessage($message);
$this->em->flush(); $this->em->flush();
} catch (\Exception $e) {
// Low-level message in case of incorrect $account
$this->messenger->sendMessageToChat($message->chat->id, 'There was an error during your Telegram account registration. Try again or report the bug.');
}
// Creating blank response for later use try {
$sendMessage = $this->createResponseMessage($message, self::PARSE_MODE_MARKDOWN, true); $words = explode(' ', $message->text, 10);
$words = explode(' ', $message->text, 4);
if (0 === count($words)) { if (0 === count($words)) {
return; return;
} }
try {
switch ($words[0]) { switch ($words[0]) {
case '/link': case '/link':
case 'link': case 'link':
@ -125,35 +116,34 @@ class PrivateMessageProcessor
if ($this->linkAccount($account, $words[1], $words[2])) { if ($this->linkAccount($account, $words[1], $words[2])) {
// Saving linking status // Saving linking status
$this->em->flush(); $this->em->flush();
$this->sendAccountLinked($sendMessage); $this->sendAccountLinked($account);
} else { } else {
$this->sendError($sendMessage, 'Account linking error', 'Check login and password or try again later.'); $this->sendError($account, 'Account linking error', 'Check login and password or try again later.');
} }
} else { } else {
$this->sendError($sendMessage, 'Login/Password error', 'You need to specify login and password separated by space after /link (example: `/link mylogin MypASSw0rd`)'); $this->sendError($account, 'Login/Password error', 'You need to specify login and password separated by space after /link (example: `/link mylogin MypASSw0rd`)');
} }
break; break;
case '/me': case '/me':
case 'me': case 'me':
if ($user = $account->getUser()) { if ($user = $account->getUser()) {
$this->sendUserEvents($sendMessage, $user); $this->sendUserEvents($account, $user);
} else { } else {
$this->sendError($sendMessage, 'Account not linked', 'You must /link your account first to be able to use this command.'); $this->sendError($account, 'Account not linked', 'You must /link your account first to be able to use this command.');
} }
break; break;
case '/last': case '/last':
case 'l':
case 'last': case 'last':
if (array_key_exists(1, $words)) { if (array_key_exists(1, $words)) {
if (null !== $user = $this->userRepo->findUserByLogin($words[1])) { if (null !== $user = $this->userRepo->findUserByLogin($words[1])) {
$this->sendUserEvents($sendMessage, $user); $this->sendUserEvents($account, $user);
} else { } else {
$this->sendError($sendMessage, 'User not found'); $this->sendError($account, 'User not found');
} }
} else { } else {
$this->sendGlobalEvents($sendMessage); $this->sendGlobalEvents($account);
} }
break; break;
@ -162,15 +152,15 @@ class PrivateMessageProcessor
case 'sub': case 'sub':
if (array_key_exists(1, $words)) { if (array_key_exists(1, $words)) {
if (null !== $user = $this->userRepo->findUserByLogin($words[1])) { if (null !== $user = $this->userRepo->findUserByLogin($words[1])) {
$this->sendUserSubscribers($sendMessage, $user); $this->sendUserSubscribers($account, $user);
} else { } else {
$this->sendError($sendMessage, 'User not found'); $this->sendError($account, 'User not found');
} }
} else { } else {
if ($user = $account->getUser()) { if ($user = $account->getUser()) {
$this->sendUserSubscribers($sendMessage, $user); $this->sendUserSubscribers($account, $user);
} else { } else {
$this->sendError($sendMessage, 'Account not linked', 'You must /link your account first to be able to use this command.'); $this->sendError($account, 'Account not linked', 'You must /link your account first to be able to use this command.');
} }
} }
@ -178,23 +168,23 @@ class PrivateMessageProcessor
case '/stats': case '/stats':
case 'stats': case 'stats':
$this->sendStats($sendMessage); $this->sendStats($account);
break; break;
case '/help': case '/help':
default: default:
$this->sendHelp($sendMessage); $this->sendHelp($account);
break; break;
} }
} catch (CommandProcessingException $e) { } catch (CommandProcessingException $e) {
$this->sendError($sendMessage, 'Processing error', $e->getMessage()); $this->sendError($account, 'Processing error', $e->getMessage());
if ($e->getPrevious()) { if ($e->getPrevious()) {
throw $e->getPrevious(); throw $e->getPrevious();
} }
} catch (\Exception $e) { } catch (\Exception $e) {
$this->sendError($sendMessage, 'Unknown error'); $this->sendError($account, 'Unknown error');
throw $e; throw $e;
} }
@ -216,87 +206,95 @@ class PrivateMessageProcessor
return false; return false;
} }
private function sendAccountLinked(SendMessage $sendMessage) private function sendAccountLinked(Account $account)
{ {
$sendMessage->text = 'Account linked. Try using /me now.'; $this->messenger->sendMessageToUser($account, 'Account linked. Try using /me now.');
$this->client->performApiRequest($sendMessage);
} }
private function sendUserSubscribers(SendMessage $sendMessage, User $user) private function sendUserSubscribers(Account $account, User $user)
{ {
$subscribers = []; $subscribers = [];
foreach ($user->getSubscribers() as $subscription) { foreach ($user->getSubscribers() as $subscription) {
$subscribers[] = '@'.$subscription->getSubscriber()->getLogin(); $subscribers[] = '@'.$subscription->getSubscriber()->getLogin();
} }
$sendMessage->text = $this->twig->render(self::TEMPLATE_USER_SUBSCRIBERS, [ $this->sendTemplatedMessage(
$account,
'@SkobkinPointTools/Telegram/user_subscribers.md.twig',
[
'user' => $user, 'user' => $user,
'subscribers' => $subscribers, 'subscribers' => $subscribers,
]); ]
);
$this->client->performApiRequest($sendMessage);
} }
/** private function sendUserEvents(Account $account, User $user)
* @param SendMessage $sendMessage
* @param User $user
*/
private function sendUserEvents(SendMessage $sendMessage, User $user)
{ {
$events = $this->subscriptionEventRepo->getUserLastSubscribersEvents($user, 10); $events = $this->subscriptionEventRepo->getUserLastSubscribersEvents($user, 10);
$sendMessage->text = $this->twig->render(self::TEMPLATE_LAST_USER_SUB_EVENTS, [
$this->sendTemplatedMessage(
$account,
'@SkobkinPointTools/Telegram/last_user_subscriptions.md.twig',
[
'user' => $user, 'user' => $user,
'events' => $events, 'events' => $events,
]); ]
);
$this->client->performApiRequest($sendMessage);
} }
private function sendGlobalEvents(SendMessage $sendMessage) private function sendGlobalEvents(Account $account)
{ {
$events = $this->subscriptionEventRepo->getLastSubscriptionEvents(10); $events = $this->subscriptionEventRepo->getLastSubscriptionEvents(10);
$sendMessage->text = $this->twig->render(self::TEMPLATE_LAST_EVENTS, ['events' => $events]);
$this->client->performApiRequest($sendMessage); $this->sendTemplatedMessage($account, '@SkobkinPointTools/Telegram/last_global_subscriptions.md.twig', ['events' => $events]);
} }
private function sendStats(SendMessage $sendMessage) private function sendStats(Account $account)
{ {
$sendMessage->text = $this->twig->render(self::TEMPLATE_STATS, [ $this->sendTemplatedMessage(
$account,
'@SkobkinPointTools/Telegram/stats.md.twig',
[
'total_users' => $this->userRepo->getUsersCount(), 'total_users' => $this->userRepo->getUsersCount(),
'active_users' => $this->subscriptionRepo->getUserSubscribersCountById($this->pointUserId), 'active_users' => $this->subscriptionRepo->getUserSubscribersCountById($this->pointUserId),
'today_events' => $this->subscriptionEventRepo->getLastDayEventsCount(), 'today_events' => $this->subscriptionEventRepo->getLastDayEventsCount(),
]); ]
);
$this->client->performApiRequest($sendMessage);
} }
private function sendHelp(SendMessage $sendMessage) private function sendHelp(Account $account)
{ {
$sendMessage->text = $this->twig->render(self::TEMPLATE_HELP); $this->sendTemplatedMessage($account, '@SkobkinPointTools/Telegram/help.md.twig');
$this->client->performApiRequest($sendMessage);
} }
private function sendError(SendMessage $sendMessage, string $title, string $text = '') private function sendError(Account $account, string $title, string $text = '')
{ {
$sendMessage->text = $this->twig->render(self::TEMPLATE_ERROR, [ $this->sendTemplatedMessage(
$account,
'@SkobkinPointTools/Telegram/error.md.twig',
[
'title' => $title, 'title' => $title,
'text' => $text, 'text' => $text,
]); ]
);
$this->client->performApiRequest($sendMessage);
} }
private function createResponseMessage(Message $message, string $parseMode = self::PARSE_MODE_MARKDOWN, bool $disableWebPreview = false): SendMessage private function sendTemplatedMessage(
Account $account,
string $template,
array $templateData = [],
ReplyKeyboardMarkup $keyboardMarkup = null,
bool $disableWebPreview = true,
string $format = MessageSender::PARSE_MODE_MARKDOWN
): bool
{ {
$sendMessage = new SendMessage(); $text = $this->twig->render($template, $templateData);
$sendMessage->chat_id = (string) $message->chat->id;
$sendMessage->parse_mode = $parseMode;
$sendMessage->disable_web_page_preview = $disableWebPreview;
return $sendMessage; return $this->messenger->sendMessageToUser($account, $text, $format, $keyboardMarkup, $disableWebPreview, false);
}
private function sendPlainTextMessage(Account $account, string $text, ReplyKeyboardMarkup $keyboardMarkup = null, bool $disableWebPreview = true): bool
{
return $this->messenger->sendMessageToUser($account, $text, MessageSender::PARSE_MODE_NOPARSE, $keyboardMarkup, $disableWebPreview);
} }
} }

View file

@ -1,49 +0,0 @@
<?php
namespace Skobkin\Bundle\PointToolsBundle\Service\Telegram;
use GuzzleHttp\Exception\ClientException;
use unreal4u\TelegramAPI\Telegram\Methods\SendMessage;
use unreal4u\TelegramAPI\TgLog;
/**
* Service which sends simple messages to Telegram users
*/
class SimpleSender
{
/**
* @var TgLog
*/
private $client;
/**
* @param TgLog $client
*/
public function __construct(TgLog $client)
{
$this->client = $client;
}
/**
* Send simple message
*
* @param int $chatId
* @param string $text
*
* @return bool
*/
public function sendMessage(int $chatId, string $text): bool
{
$sendMessage = new SendMessage();
$sendMessage->chat_id = (string) $chatId;
$sendMessage->text = $text;
try {
$this->client->performApiRequest($sendMessage);
return true;
} catch (ClientException $e) {
return false;
}
}
}