Commit 34ad296b authored by Jack'lul's avatar Jack'lul

Botan integration improvements

parent 1c54e2c0
<?php
/**
* This file is part of the TelegramBot package.
*
* (c) Avtandil Kikabidze aka LONGMAN <akalongman@gmail.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Longman\TelegramBot\Commands\UserCommands;
use Longman\TelegramBot\Botan;
use Longman\TelegramBot\Commands\UserCommand;
use Longman\TelegramBot\Request;
/**
* User "/shortener" command
*/
class ShortenerCommand extends UserCommand
{
/**#@+
* {@inheritdoc}
*/
protected $name = 'shortener';
protected $description = 'Botan Shortener example';
protected $usage = '/shortener';
protected $version = '1.0.0';
/**#@-*/
/**
* {@inheritdoc}
*/
public function execute()
{
$message = $this->getMessage();
$chat_id = $message->getChat()->getId();
$user_id = $message->getFrom()->getId();
$data = [];
$data['chat_id'] = $chat_id;
$text = Botan::shortenUrl("https://github.com/akalongman/php-telegram-bot", $user_id);
$data['text'] = $text;
return Request::sendMessage($data);
}
}
...@@ -62,7 +62,9 @@ try { ...@@ -62,7 +62,9 @@ try {
//$telegram->setUploadPath('../Upload'); //$telegram->setUploadPath('../Upload');
// Botan.io integration // Botan.io integration
// Second argument is optional maximum timeout
//$telegram->enableBotan('your_token'); //$telegram->enableBotan('your_token');
//$telegram->enableBotan('your_token', 3);
// Handle telegram getUpdates request // Handle telegram getUpdates request
$serverResponse = $telegram->handleGetUpdates(); $serverResponse = $telegram->handleGetUpdates();
......
...@@ -61,7 +61,9 @@ try { ...@@ -61,7 +61,9 @@ try {
//$telegram->setUploadPath('../Upload'); //$telegram->setUploadPath('../Upload');
// Botan.io integration // Botan.io integration
// Second argument is optional maximum timeout
//$telegram->enableBotan('your_token'); //$telegram->enableBotan('your_token');
//$telegram->enableBotan('your_token', 3);
// Handle telegram webhook request // Handle telegram webhook request
$telegram->handle(); $telegram->handle();
......
...@@ -10,6 +10,9 @@ ...@@ -10,6 +10,9 @@
namespace Longman\TelegramBot; namespace Longman\TelegramBot;
use GuzzleHttp\Client;
use GuzzleHttp\Exception\RequestException;
use Longman\TelegramBot\Entities\Update;
use Longman\TelegramBot\Exception\TelegramException; use Longman\TelegramBot\Exception\TelegramException;
/** /**
...@@ -20,82 +23,107 @@ use Longman\TelegramBot\Exception\TelegramException; ...@@ -20,82 +23,107 @@ use Longman\TelegramBot\Exception\TelegramException;
class Botan class Botan
{ {
/** /**
* @var string Tracker request url * Botan.io API URL
*
* @var string
*/ */
protected static $track_url = 'https://api.botan.io/track?token=#TOKEN&uid=#UID&name=#NAME'; protected static $api_base_uri = 'https://api.botan.io';
/** /**
* @var string Url Shortener request url * Yandex AppMetrica application key
*
* @var string
*/ */
protected static $shortener_url = 'https://api.botan.io/s/?token=#TOKEN&user_ids=#UID&url=#URL'; protected static $token = '';
/** /**
* @var string Yandex AppMetrica application key * Guzzle Client object
*
* @var \GuzzleHttp\Client
*/ */
protected static $token = ''; private static $client;
/** /**
* @var string The actual command that is going to be reported * The actual command that is going to be reported
*
* Set as public to let the developers either:
* - block tracking from inside commands by setting the value to non-existent command
* - override which command is tracked when commands call other commands with executedCommand()
*
* @var string
*/ */
public static $command = ''; public static $command = '';
/** /**
* Initialize Botan * Initialize Botan
* *
* @param $token * @param string $token
* @param integer $timeout
*
* @throws \Longman\TelegramBot\Exception\TelegramException * @throws \Longman\TelegramBot\Exception\TelegramException
*/ */
public static function initializeBotan($token) public static function initializeBotan($token, $timeout = 3)
{ {
if (empty($token) || !is_string($token)) { if (empty($token)) {
throw new TelegramException('Botan token should be a string!'); throw new TelegramException('Botan token is empty!');
} }
if (!is_numeric($timeout)) {
throw new TelegramException('Timeout must be a number!');
}
self::$token = $token; self::$token = $token;
self::$client = new Client(['base_uri' => self::$api_base_uri, 'timeout' => $timeout]);
BotanDB::initializeBotanDb(); BotanDB::initializeBotanDb();
} }
/** /**
* Lock function to make sure only the first command is reported * Lock function to make sure only the first command is reported (the one user requested)
* ( in case commands are calling other commands $telegram->executedCommand() ) *
* This is in case commands are calling other commands with executedCommand()
* *
* @param string $command * @param string $command
*/ */
public static function lock($command = '') public static function lock($command = '')
{ {
if (empty(self::$command)) { if (empty(self::$command)) {
self::$command = $command; self::$command = strtolower($command);
} }
} }
/** /**
* Track function * Track function
* *
* @param string $input * @param \Longman\TelegramBot\Entities\Update $update
* @param string $command * @param string $command
* *
* @return bool|string * @return bool|string
* @throws \Longman\TelegramBot\Exception\TelegramException * @throws \Longman\TelegramBot\Exception\TelegramException
*/ */
public static function track($input, $command = '') public static function track(Update $update, $command = '')
{ {
if (empty(self::$token) || $command !== self::$command) { if (empty(self::$token) || strtolower($command) !== self::$command) {
return false; return false;
} }
if (empty($input)) { if (empty($update)) {
throw new TelegramException('Input is empty!'); throw new TelegramException('Update object is empty!');
} }
// Release the lock in case someone runs getUpdates in foreach loop
self::$command = ''; self::$command = '';
$obj = json_decode($input, true); // For now, this is the only way
$update_data = (array) $update;
$data = []; $data = [];
if (isset($obj['message'])) { if ($update->getMessage()) {
$data = $obj['message']; $data = $update_data['message'];
$event_name = 'Message'; $event_name = 'Generic Message';
if (isset($obj['message']['entities']) && is_array($obj['message']['entities'])) { if (!empty($data['entities']) && is_array($data['entities'])) {
foreach ($obj['message']['entities'] as $entity) { foreach ($data['entities'] as $entity) {
if ($entity['type'] === 'bot_command' && $entity['offset'] === 0) { if ($entity['type'] === 'bot_command' && $entity['offset'] === 0) {
if (strtolower($command) === 'generic') { if (strtolower($command) === 'generic') {
$command = 'Generic'; $command = 'Generic';
...@@ -110,23 +138,23 @@ class Botan ...@@ -110,23 +138,23 @@ class Botan
} }
} }
} }
} elseif (isset($obj['edited_message'])) { } elseif ($update->getEditedMessage()) {
$data = $obj['edited_message']; $data = $update_data['edited_message'];
$event_name = 'Edited Message'; $event_name = 'Edited Message';
} elseif (isset($obj['channel_post'])) { } elseif ($update->getChannelPost()) {
$data = $obj['channel_post']; $data = $update_data['channel_post'];
$event_name = 'Channel Message'; $event_name = 'Channel Post';
} elseif (isset($obj['edited_channel_post'])) { } elseif ($update->getEditedChannelPost()) {
$data = $obj['edited_channel_post']; $data = $update_data['edited_channel_post'];
$event_name = 'Edited Channel Message'; $event_name = 'Edited Channel Post';
} elseif (isset($obj['inline_query'])) { } elseif ($update->getInlineQuery()) {
$data = $obj['inline_query']; $data = $update_data['inline_query'];
$event_name = 'Inline Query'; $event_name = 'Inline Query';
} elseif (isset($obj['chosen_inline_result'])) { } elseif ($update->getChosenInlineResult()) {
$data = $obj['chosen_inline_result']; $data = $update_data['chosen_inline_result'];
$event_name = 'Chosen Inline Result'; $event_name = 'Chosen Inline Result';
} elseif (isset($obj['callback_query'])) { } elseif ($update->getCallbackQuery()) {
$data = $obj['callback_query']; $data = $update_data['callback_query'];
$event_name = 'Callback Query'; $event_name = 'Callback Query';
} }
...@@ -134,38 +162,51 @@ class Botan ...@@ -134,38 +162,51 @@ class Botan
return false; return false;
} }
// In case there is no from field (channel posts) assign chat id
if (isset($data['from']['id'])) {
$uid = $data['from']['id']; $uid = $data['from']['id'];
$request = str_replace( } elseif (isset($data['chat']['id'])) {
$uid = $data['chat']['id'];
} else {
$uid = 0; // if that fails too assign id = 0
}
try {
$response = self::$client->request(
'POST',
str_replace(
['#TOKEN', '#UID', '#NAME'], ['#TOKEN', '#UID', '#NAME'],
[self::$token, $uid, urlencode($event_name)], [self::$token, $uid, urlencode($event_name)],
self::$track_url '/track?token=#TOKEN&uid=#UID&name=#NAME'
); ),
[
$options = [ 'headers' => [
'http' => [ 'Content-Type' => 'application/json'
'header' => 'Content-Type: application/json',
'method' => 'POST',
'content' => json_encode($data),
'ignore_errors' => true,
], ],
]; 'json' => $data
]
);
$context = stream_context_create($options); $response = (string) $response->getBody();
$response = @file_get_contents($request, false, $context); } catch (RequestException $e) {
$response = ($e->getResponse()) ? (string) $e->getResponse()->getBody() : '';
} finally {
$responseData = json_decode($response, true); $responseData = json_decode($response, true);
if ($responseData['status'] !== 'accepted') { if ($responseData['status'] !== 'accepted') {
TelegramLog::debug('Botan.io API replied with error: ' . $response); TelegramLog::debug("Botan.io API replied with error:\n$response\n\n");
return false;
} }
return $responseData; return $responseData;
} }
}
/** /**
* Url Shortener function * Url Shortener function
* *
* @param $url * @param string $url
* @param $user_id * @param integer $user_id
* *
* @return string * @return string
* @throws \Longman\TelegramBot\Exception\TelegramException * @throws \Longman\TelegramBot\Exception\TelegramException
...@@ -181,33 +222,32 @@ class Botan ...@@ -181,33 +222,32 @@ class Botan
} }
$cached = BotanDB::selectShortUrl($user_id, $url); $cached = BotanDB::selectShortUrl($user_id, $url);
if (!empty($cached[0]['short_url'])) {
return $cached[0]['short_url']; if (!empty($cached)) {
return $cached;
} }
$request = str_replace( try {
$response = self::$client->request(
'POST',
str_replace(
['#TOKEN', '#UID', '#URL'], ['#TOKEN', '#UID', '#URL'],
[self::$token, $user_id, urlencode($url)], [self::$token, $user_id, urlencode($url)],
self::$shortener_url '/s/?token=#TOKEN&user_ids=#UID&url=#URL'
)
); );
$options = [ $response = (string) $response->getBody();
'http' => [ } catch (RequestException $e) {
'ignore_errors' => true, $response = ($e->getResponse()) ? (string) $e->getResponse()->getBody() : '';
'timeout' => 3, } finally {
],
];
$context = stream_context_create($options);
$response = @file_get_contents($request, false, $context);
if (!filter_var($response, FILTER_VALIDATE_URL) === false) { if (!filter_var($response, FILTER_VALIDATE_URL) === false) {
BotanDB::insertShortUrl($user_id, $url, $response); BotanDB::insertShortUrl($user_id, $url, $response);
return $response;
} else { } else {
TelegramLog::debug('Botan.io API replied with error: ' . $response); TelegramLog::error("Botan.io URL shortening failed - API replied with error:\n$response\n\n");
return $url; return $url;
} }
}
return $response;
} }
} }
...@@ -47,13 +47,21 @@ class BotanDB extends DB ...@@ -47,13 +47,21 @@ class BotanDB extends DB
try { try {
$sth = self::$pdo->prepare('SELECT * FROM `' . TB_BOTAN_SHORTENER . '` $sth = self::$pdo->prepare('SELECT * FROM `' . TB_BOTAN_SHORTENER . '`
WHERE `user_id` = :user_id AND `url` = :url WHERE `user_id` = :user_id AND `url` = :url
ORDER BY `created_at` DESC
LIMIT 1
'); ');
$sth->bindParam(':user_id', $user_id, PDO::PARAM_INT); $sth->bindParam(':user_id', $user_id, PDO::PARAM_INT);
$sth->bindParam(':url', $url, PDO::PARAM_INT); $sth->bindParam(':url', $url, PDO::PARAM_STR);
$sth->execute(); $sth->execute();
return $sth->fetchAll(PDO::FETCH_ASSOC); $result = $sth->fetchAll(PDO::FETCH_ASSOC);
if (isset($result[0]['short_url'])) {
return $result[0]['short_url'];
}
return $result;
} catch (Exception $e) { } catch (Exception $e) {
throw new TelegramException($e->getMessage()); throw new TelegramException($e->getMessage());
} }
......
...@@ -455,7 +455,7 @@ class Telegram ...@@ -455,7 +455,7 @@ class Telegram
//Handle a generic command or non existing one //Handle a generic command or non existing one
$this->last_command_response = $this->executeCommand('Generic'); $this->last_command_response = $this->executeCommand('Generic');
} else { } else {
//Botan.io integration, make sure only the command user executed is reported //Botan.io integration, make sure only the actual command user executed is reported
if ($this->botan_enabled) { if ($this->botan_enabled) {
Botan::lock($command); Botan::lock($command);
} }
...@@ -815,14 +815,15 @@ class Telegram ...@@ -815,14 +815,15 @@ class Telegram
/** /**
* Enable Botan.io integration * Enable Botan.io integration
* *
* @param $token * @param string $token
* @param integer $timeout
* *
* @return \Longman\TelegramBot\Telegram * @return \Longman\TelegramBot\Telegram
* @throws \Longman\TelegramBot\Exception\TelegramException * @throws \Longman\TelegramBot\Exception\TelegramException
*/ */
public function enableBotan($token) public function enableBotan($token, $timeout = 3)
{ {
Botan::initializeBotan($token); Botan::initializeBotan($token, $timeout);
$this->botan_enabled = true; $this->botan_enabled = true;
return $this; return $this;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment