Passed
Push — master ( f55fd1...466468 )
by Armando
02:34
created

Telegram::handle()   A

Complexity

Conditions 5
Paths 5

Size

Total Lines 22
Code Lines 11

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 30

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 5
eloc 11
c 1
b 0
f 0
nc 5
nop 0
dl 0
loc 22
ccs 0
cts 12
cp 0
crap 30
rs 9.6111
1
<?php
2
/**
3
 * This file is part of the TelegramBot package.
4
 *
5
 * (c) Avtandil Kikabidze aka LONGMAN <[email protected]>
6
 *
7
 * For the full copyright and license information, please view the LICENSE
8
 * file that was distributed with this source code.
9
 */
10
11
namespace Longman\TelegramBot;
12
13 1
defined('TB_BASE_PATH') || define('TB_BASE_PATH', __DIR__);
14 1
defined('TB_BASE_COMMANDS_PATH') || define('TB_BASE_COMMANDS_PATH', TB_BASE_PATH . '/Commands');
15
16
use Exception;
17
use Longman\TelegramBot\Commands\Command;
18
use Longman\TelegramBot\Entities\ServerResponse;
19
use Longman\TelegramBot\Entities\Update;
20
use Longman\TelegramBot\Exception\TelegramException;
21
use PDO;
22
use RecursiveDirectoryIterator;
23
use RecursiveIteratorIterator;
24
use RegexIterator;
25
26
class Telegram
27
{
28
    /**
29
     * Version
30
     *
31
     * @var string
32
     */
33
    protected $version = '0.60.0';
34
35
    /**
36
     * Telegram API key
37
     *
38
     * @var string
39
     */
40
    protected $api_key = '';
41
42
    /**
43
     * Telegram Bot username
44
     *
45
     * @var string
46
     */
47
    protected $bot_username = '';
48
49
    /**
50
     * Telegram Bot id
51
     *
52
     * @var string
53
     */
54
    protected $bot_id = '';
55
56
    /**
57
     * Raw request data (json) for webhook methods
58
     *
59
     * @var string
60
     */
61
    protected $input;
62
63
    /**
64
     * Custom commands paths
65
     *
66
     * @var array
67
     */
68
    protected $commands_paths = [];
69
70
    /**
71
     * Current Update object
72
     *
73
     * @var Update
74
     */
75
    protected $update;
76
77
    /**
78
     * Upload path
79
     *
80
     * @var string
81
     */
82
    protected $upload_path;
83
84
    /**
85
     * Download path
86
     *
87
     * @var string
88
     */
89
    protected $download_path;
90
91
    /**
92
     * MySQL integration
93
     *
94
     * @var boolean
95
     */
96
    protected $mysql_enabled = false;
97
98
    /**
99
     * PDO object
100
     *
101
     * @var PDO
102
     */
103
    protected $pdo;
104
105
    /**
106
     * Commands config
107
     *
108
     * @var array
109
     */
110
    protected $commands_config = [];
111
112
    /**
113
     * Admins list
114
     *
115
     * @var array
116
     */
117
    protected $admins_list = [];
118
119
    /**
120
     * ServerResponse of the last Command execution
121
     *
122
     * @var ServerResponse
123
     */
124
    protected $last_command_response;
125
126
    /**
127
     * Check if runCommands() is running in this session
128
     *
129
     * @var boolean
130
     */
131
    protected $run_commands = false;
132
133
    /**
134
     * Is running getUpdates without DB enabled
135
     *
136
     * @var bool
137
     */
138
    protected $getupdates_without_database = false;
139
140
    /**
141
     * Last update ID
142
     * Only used when running getUpdates without a database
143
     *
144
     * @var integer
145
     */
146
    protected $last_update_id = null;
147
148
    /**
149
     * Telegram constructor.
150
     *
151
     * @param string $api_key
152
     * @param string $bot_username
153
     *
154
     * @throws TelegramException
155
     */
156 30
    public function __construct($api_key, $bot_username = '')
157
    {
158 30
        if (empty($api_key)) {
159 1
            throw new TelegramException('API KEY not defined!');
160
        }
161 30
        preg_match('/(\d+)\:[\w\-]+/', $api_key, $matches);
162 30
        if (!isset($matches[1])) {
163 1
            throw new TelegramException('Invalid API KEY defined!');
164
        }
165 30
        $this->bot_id  = $matches[1];
166 30
        $this->api_key = $api_key;
167
168 30
        if (!empty($bot_username)) {
169 30
            $this->bot_username = $bot_username;
170
        }
171
172
        //Add default system commands path
173 30
        $this->addCommandsPath(TB_BASE_COMMANDS_PATH . '/SystemCommands');
174
175 30
        Request::initialize($this);
176 30
    }
177
178
    /**
179
     * Initialize Database connection
180
     *
181
     * @param array  $credential
182
     * @param string $table_prefix
183
     * @param string $encoding
184
     *
185
     * @return Telegram
186
     * @throws TelegramException
187
     */
188 9
    public function enableMySql(array $credential, $table_prefix = null, $encoding = 'utf8mb4')
189
    {
190 9
        $this->pdo = DB::initialize($credential, $this, $table_prefix, $encoding);
191 9
        ConversationDB::initializeConversation();
192 9
        $this->mysql_enabled = true;
193
194 9
        return $this;
195
    }
196
197
    /**
198
     * Initialize Database external connection
199
     *
200
     * @param PDO    $external_pdo_connection PDO database object
201
     * @param string $table_prefix
202
     *
203
     * @return Telegram
204
     * @throws TelegramException
205
     */
206
    public function enableExternalMySql($external_pdo_connection, $table_prefix = null)
207
    {
208
        $this->pdo = DB::externalInitialize($external_pdo_connection, $this, $table_prefix);
209
        ConversationDB::initializeConversation();
210
        $this->mysql_enabled = true;
211
212
        return $this;
213
    }
214
215
    /**
216
     * Get commands list
217
     *
218
     * @return array $commands
219
     * @throws TelegramException
220
     */
221 1
    public function getCommandsList()
222
    {
223 1
        $commands = [];
224
225 1
        foreach ($this->commands_paths as $path) {
226
            try {
227
                //Get all "*Command.php" files
228 1
                $files = new RegexIterator(
229 1
                    new RecursiveIteratorIterator(
230 1
                        new RecursiveDirectoryIterator($path)
231
                    ),
232 1
                    '/^.+Command.php$/'
233
                );
234
235 1
                foreach ($files as $file) {
236
                    //Remove "Command.php" from filename
237 1
                    $command      = $this->sanitizeCommand(substr($file->getFilename(), 0, -11));
238 1
                    $command_name = strtolower($command);
239
240 1
                    if (array_key_exists($command_name, $commands)) {
241
                        continue;
242
                    }
243
244 1
                    require_once $file->getPathname();
245
246 1
                    $command_obj = $this->getCommandObject($command);
247 1
                    if ($command_obj instanceof Command) {
248 1
                        $commands[$command_name] = $command_obj;
249
                    }
250
                }
251
            } catch (Exception $e) {
252
                throw new TelegramException('Error getting commands from path: ' . $path);
253
            }
254
        }
255
256 1
        return $commands;
257
    }
258
259
    /**
260
     * Get an object instance of the passed command
261
     *
262
     * @param string $command
263
     *
264
     * @return Command|null
265
     */
266 1
    public function getCommandObject($command)
267
    {
268 1
        $which = ['System'];
269 1
        $this->isAdmin() && $which[] = 'Admin';
270 1
        $which[] = 'User';
271
272 1
        foreach ($which as $auth) {
273 1
            $command_namespace = __NAMESPACE__ . '\\Commands\\' . $auth . 'Commands\\' . $this->ucfirstUnicode($command) . 'Command';
274 1
            if (class_exists($command_namespace)) {
275 1
                return new $command_namespace($this, $this->update);
276
            }
277
        }
278
279
        return null;
280
    }
281
282
    /**
283
     * Set custom input string for debug purposes
284
     *
285
     * @param string $input (json format)
286
     *
287
     * @return Telegram
288
     */
289
    public function setCustomInput($input)
290
    {
291
        $this->input = $input;
292
293
        return $this;
294
    }
295
296
    /**
297
     * Get custom input string for debug purposes
298
     *
299
     * @return string
300
     */
301
    public function getCustomInput()
302
    {
303
        return $this->input;
304
    }
305
306
    /**
307
     * Get the ServerResponse of the last Command execution
308
     *
309
     * @return ServerResponse
310
     */
311
    public function getLastCommandResponse()
312
    {
313
        return $this->last_command_response;
314
    }
315
316
    /**
317
     * Handle getUpdates method
318
     *
319
     * @param int|null $limit
320
     * @param int|null $timeout
321
     *
322
     * @return ServerResponse
323
     * @throws TelegramException
324
     */
325
    public function handleGetUpdates($limit = null, $timeout = null)
326
    {
327
        if (empty($this->bot_username)) {
328
            throw new TelegramException('Bot Username is not defined!');
329
        }
330
331
        if (!DB::isDbConnected() && !$this->getupdates_without_database) {
332
            return new ServerResponse(
333
                [
334
                    'ok'          => false,
335
                    'description' => 'getUpdates needs MySQL connection! (This can be overridden - see documentation)',
336
                ],
337
                $this->bot_username
338
            );
339
        }
340
341
        $offset = 0;
342
343
        //Take custom input into account.
344
        if ($custom_input = $this->getCustomInput()) {
345
            $response = new ServerResponse(json_decode($custom_input, true), $this->bot_username);
346
        } else {
347
            if (DB::isDbConnected() && $last_update = DB::selectTelegramUpdate(1)) {
348
                //Get last update id from the database
349
                $last_update = reset($last_update);
350
351
                $this->last_update_id = isset($last_update['id']) ? $last_update['id'] : null;
352
            }
353
354
            if ($this->last_update_id !== null) {
355
                $offset = $this->last_update_id + 1;    //As explained in the telegram bot API documentation
356
            }
357
358
            $response = Request::getUpdates(
359
                [
360
                    'offset'  => $offset,
361
                    'limit'   => $limit,
362
                    'timeout' => $timeout,
363
                ]
364
            );
365
        }
366
367
        if ($response->isOk()) {
368
            $results = $response->getResult();
369
370
            //Process all updates
371
            /** @var Update $result */
372
            foreach ($results as $result) {
373
                $this->processUpdate($result);
374
            }
375
376
            if (!DB::isDbConnected() && !$custom_input && $this->last_update_id !== null && $offset === 0) {
377
                //Mark update(s) as read after handling
378
                Request::getUpdates(
379
                    [
380
                        'offset'  => $this->last_update_id + 1,
381
                        'limit'   => 1,
382
                        'timeout' => $timeout,
383
                    ]
384
                );
385
            }
386
        }
387
388
        return $response;
389
    }
390
391
    /**
392
     * Handle bot request from webhook
393
     *
394
     * @return bool
395
     *
396
     * @throws TelegramException
397
     */
398
    public function handle()
399
    {
400
        if (empty($this->bot_username)) {
401
            throw new TelegramException('Bot Username is not defined!');
402
        }
403
404
        $this->input = Request::getInput();
405
406
        if (empty($this->input)) {
407
            throw new TelegramException('Input is empty!');
408
        }
409
410
        $post = json_decode($this->input, true);
411
        if (empty($post)) {
412
            throw new TelegramException('Invalid JSON!');
413
        }
414
415
        if ($response = $this->processUpdate(new Update($post, $this->bot_username))) {
416
            return $response->isOk();
417
        }
418
419
        return false;
420
    }
421
422
    /**
423
     * Get the command name from the command type
424
     *
425
     * @param string $type
426
     *
427
     * @return string
428
     */
429
    protected function getCommandFromType($type)
430
    {
431
        return $this->ucfirstUnicode(str_replace('_', '', $type));
432
    }
433
434
    /**
435
     * Process bot Update request
436
     *
437
     * @param Update $update
438
     *
439
     * @return ServerResponse
440
     * @throws TelegramException
441
     */
442
    public function processUpdate(Update $update)
443
    {
444
        $this->update         = $update;
445
        $this->last_update_id = $update->getUpdateId();
446
447
        //Load admin commands
448
        if ($this->isAdmin()) {
449
            $this->addCommandsPath(TB_BASE_COMMANDS_PATH . '/AdminCommands', false);
450
        }
451
452
        //Make sure we have an up-to-date command list
453
        //This is necessary to "require" all the necessary command files!
454
        $this->getCommandsList();
455
456
        //If all else fails, it's a generic message.
457
        $command = 'genericmessage';
458
459
        $update_type = $this->update->getUpdateType();
460
        if ($update_type === 'message') {
461
            $message = $this->update->getMessage();
462
            $type    = $message->getType();
463
464
            // Let's check if the message object has the type field we're looking for...
465
            $command_tmp = $type === 'command' ? $message->getCommand() : $this->getCommandFromType($type);
466
            // ...and if a fitting command class is available.
467
            $command_obj = $this->getCommandObject($command_tmp);
468
469
            // Empty usage string denotes a non-executable command.
470
            // @see https://github.com/php-telegram-bot/core/issues/772#issuecomment-388616072
471
            if (($command_obj === null && $type === 'command')
472
                || ($command_obj !== null && $command_obj->getUsage() !== '')
473
            ) {
474
                $command = $command_tmp;
475
            }
476
        } else {
477
            $command = $this->getCommandFromType($update_type);
478
        }
479
480
        //Make sure we don't try to process update that was already processed
481
        $last_id = DB::selectTelegramUpdate(1, $this->update->getUpdateId());
482
        if ($last_id && count($last_id) === 1) {
483
            TelegramLog::debug('Duplicate update received, processing aborted!');
484
            return Request::emptyResponse();
485
        }
486
487
        DB::insertRequest($this->update);
488
489
        return $this->executeCommand($command);
490
    }
491
492
    /**
493
     * Execute /command
494
     *
495
     * @param string $command
496
     *
497
     * @return mixed
498
     * @throws TelegramException
499
     */
500
    public function executeCommand($command)
501
    {
502
        $command     = strtolower($command);
503
        $command_obj = $this->getCommandObject($command);
504
505
        if (!$command_obj || !$command_obj->isEnabled()) {
506
            //Failsafe in case the Generic command can't be found
507
            if ($command === 'generic') {
508
                throw new TelegramException('Generic command missing!');
509
            }
510
511
            //Handle a generic command or non existing one
512
            $this->last_command_response = $this->executeCommand('generic');
513
        } else {
514
            //execute() method is executed after preExecute()
515
            //This is to prevent executing a DB query without a valid connection
516
            $this->last_command_response = $command_obj->preExecute();
517
        }
518
519
        return $this->last_command_response;
520
    }
521
522
    /**
523
     * Sanitize Command
524
     *
525
     * @param string $command
526
     *
527
     * @return string
528
     */
529 1
    protected function sanitizeCommand($command)
530
    {
531 1
        return str_replace(' ', '', $this->ucwordsUnicode(str_replace('_', ' ', $command)));
532
    }
533
534
    /**
535
     * Enable a single Admin account
536
     *
537
     * @param integer $admin_id Single admin id
538
     *
539
     * @return Telegram
540
     */
541 1
    public function enableAdmin($admin_id)
542
    {
543 1
        if (!is_int($admin_id) || $admin_id <= 0) {
0 ignored issues
show
introduced by
The condition is_int($admin_id) is always true.
Loading history...
544 1
            TelegramLog::error('Invalid value "' . $admin_id . '" for admin.');
545 1
        } elseif (!in_array($admin_id, $this->admins_list, true)) {
546 1
            $this->admins_list[] = $admin_id;
547
        }
548
549 1
        return $this;
550
    }
551
552
    /**
553
     * Enable a list of Admin Accounts
554
     *
555
     * @param array $admin_ids List of admin ids
556
     *
557
     * @return Telegram
558
     */
559 1
    public function enableAdmins(array $admin_ids)
560
    {
561 1
        foreach ($admin_ids as $admin_id) {
562 1
            $this->enableAdmin($admin_id);
563
        }
564
565 1
        return $this;
566
    }
567
568
    /**
569
     * Get list of admins
570
     *
571
     * @return array
572
     */
573 1
    public function getAdminList()
574
    {
575 1
        return $this->admins_list;
576
    }
577
578
    /**
579
     * Check if the passed user is an admin
580
     *
581
     * If no user id is passed, the current update is checked for a valid message sender.
582
     *
583
     * @param int|null $user_id
584
     *
585
     * @return bool
586
     */
587 1
    public function isAdmin($user_id = null)
588
    {
589 1
        if ($user_id === null && $this->update !== null) {
590
            //Try to figure out if the user is an admin
591
            $update_methods = [
592
                'getMessage',
593
                'getEditedMessage',
594
                'getChannelPost',
595
                'getEditedChannelPost',
596
                'getInlineQuery',
597
                'getChosenInlineResult',
598
                'getCallbackQuery',
599
            ];
600
            foreach ($update_methods as $update_method) {
601
                $object = call_user_func([$this->update, $update_method]);
602
                if ($object !== null && $from = $object->getFrom()) {
603
                    $user_id = $from->getId();
604
                    break;
605
                }
606
            }
607
        }
608
609 1
        return ($user_id === null) ? false : in_array($user_id, $this->admins_list, true);
610
    }
611
612
    /**
613
     * Check if user required the db connection
614
     *
615
     * @return bool
616
     */
617
    public function isDbEnabled()
618
    {
619
        if ($this->mysql_enabled) {
620
            return true;
621
        } else {
622
            return false;
623
        }
624
    }
625
626
    /**
627
     * Add a single custom commands path
628
     *
629
     * @param string $path   Custom commands path to add
630
     * @param bool   $before If the path should be prepended or appended to the list
631
     *
632
     * @return Telegram
633
     */
634 30
    public function addCommandsPath($path, $before = true)
635
    {
636 30
        if (!is_dir($path)) {
637 1
            TelegramLog::error('Commands path "' . $path . '" does not exist.');
638 30
        } elseif (!in_array($path, $this->commands_paths, true)) {
639 30
            if ($before) {
640 30
                array_unshift($this->commands_paths, $path);
641
            } else {
642
                $this->commands_paths[] = $path;
643
            }
644
        }
645
646 30
        return $this;
647
    }
648
649
    /**
650
     * Add multiple custom commands paths
651
     *
652
     * @param array $paths  Custom commands paths to add
653
     * @param bool  $before If the paths should be prepended or appended to the list
654
     *
655
     * @return Telegram
656
     */
657 1
    public function addCommandsPaths(array $paths, $before = true)
658
    {
659 1
        foreach ($paths as $path) {
660 1
            $this->addCommandsPath($path, $before);
661
        }
662
663 1
        return $this;
664
    }
665
666
    /**
667
     * Return the list of commands paths
668
     *
669
     * @return array
670
     */
671 1
    public function getCommandsPaths()
672
    {
673 1
        return $this->commands_paths;
674
    }
675
676
    /**
677
     * Set custom upload path
678
     *
679
     * @param string $path Custom upload path
680
     *
681
     * @return Telegram
682
     */
683
    public function setUploadPath($path)
684
    {
685
        $this->upload_path = $path;
686
687
        return $this;
688
    }
689
690
    /**
691
     * Get custom upload path
692
     *
693
     * @return string
694
     */
695
    public function getUploadPath()
696
    {
697
        return $this->upload_path;
698
    }
699
700
    /**
701
     * Set custom download path
702
     *
703
     * @param string $path Custom download path
704
     *
705
     * @return Telegram
706
     */
707
    public function setDownloadPath($path)
708
    {
709
        $this->download_path = $path;
710
711
        return $this;
712
    }
713
714
    /**
715
     * Get custom download path
716
     *
717
     * @return string
718
     */
719
    public function getDownloadPath()
720
    {
721
        return $this->download_path;
722
    }
723
724
    /**
725
     * Set command config
726
     *
727
     * Provide further variables to a particular commands.
728
     * For example you can add the channel name at the command /sendtochannel
729
     * Or you can add the api key for external service.
730
     *
731
     * @param string $command
732
     * @param array  $config
733
     *
734
     * @return Telegram
735
     */
736 14
    public function setCommandConfig($command, array $config)
737
    {
738 14
        $this->commands_config[$command] = $config;
739
740 14
        return $this;
741
    }
742
743
    /**
744
     * Get command config
745
     *
746
     * @param string $command
747
     *
748
     * @return array
749
     */
750 15
    public function getCommandConfig($command)
751
    {
752 15
        return isset($this->commands_config[$command]) ? $this->commands_config[$command] : [];
753
    }
754
755
    /**
756
     * Get API key
757
     *
758
     * @return string
759
     */
760 1
    public function getApiKey()
761
    {
762 1
        return $this->api_key;
763
    }
764
765
    /**
766
     * Get Bot name
767
     *
768
     * @return string
769
     */
770 1
    public function getBotUsername()
771
    {
772 1
        return $this->bot_username;
773
    }
774
775
    /**
776
     * Get Bot Id
777
     *
778
     * @return string
779
     */
780
    public function getBotId()
781
    {
782
        return $this->bot_id;
783
    }
784
785
    /**
786
     * Get Version
787
     *
788
     * @return string
789
     */
790
    public function getVersion()
791
    {
792
        return $this->version;
793
    }
794
795
    /**
796
     * Set Webhook for bot
797
     *
798
     * @param string $url
799
     * @param array  $data Optional parameters.
800
     *
801
     * @return ServerResponse
802
     * @throws TelegramException
803
     */
804
    public function setWebhook($url, array $data = [])
805
    {
806
        if (empty($url)) {
807
            throw new TelegramException('Hook url is empty!');
808
        }
809
810
        $data        = array_intersect_key($data, array_flip([
811
            'certificate',
812
            'max_connections',
813
            'allowed_updates',
814
        ]));
815
        $data['url'] = $url;
816
817
        // If the certificate is passed as a path, encode and add the file to the data array.
818
        if (!empty($data['certificate']) && is_string($data['certificate'])) {
819
            $data['certificate'] = Request::encodeFile($data['certificate']);
820
        }
821
822
        $result = Request::setWebhook($data);
823
824
        if (!$result->isOk()) {
825
            throw new TelegramException(
826
                'Webhook was not set! Error: ' . $result->getErrorCode() . ' ' . $result->getDescription()
827
            );
828
        }
829
830
        return $result;
831
    }
832
833
    /**
834
     * Delete any assigned webhook
835
     *
836
     * @return mixed
837
     * @throws TelegramException
838
     */
839
    public function deleteWebhook()
840
    {
841
        $result = Request::deleteWebhook();
842
843
        if (!$result->isOk()) {
844
            throw new TelegramException(
845
                'Webhook was not deleted! Error: ' . $result->getErrorCode() . ' ' . $result->getDescription()
846
            );
847
        }
848
849
        return $result;
850
    }
851
852
    /**
853
     * Replace function `ucwords` for UTF-8 characters in the class definition and commands
854
     *
855
     * @param string $str
856
     * @param string $encoding (default = 'UTF-8')
857
     *
858
     * @return string
859
     */
860 1
    protected function ucwordsUnicode($str, $encoding = 'UTF-8')
861
    {
862 1
        return mb_convert_case($str, MB_CASE_TITLE, $encoding);
863
    }
864
865
    /**
866
     * Replace function `ucfirst` for UTF-8 characters in the class definition and commands
867
     *
868
     * @param string $str
869
     * @param string $encoding (default = 'UTF-8')
870
     *
871
     * @return string
872
     */
873 1
    protected function ucfirstUnicode($str, $encoding = 'UTF-8')
874
    {
875 1
        return mb_strtoupper(mb_substr($str, 0, 1, $encoding), $encoding)
876 1
               . mb_strtolower(mb_substr($str, 1, mb_strlen($str), $encoding), $encoding);
877
    }
878
879
    /**
880
     * Enable requests limiter
881
     *
882
     * @param array $options
883
     *
884
     * @return Telegram
885
     * @throws TelegramException
886
     */
887
    public function enableLimiter(array $options = [])
888
    {
889
        Request::setLimiter(true, $options);
890
891
        return $this;
892
    }
893
894
    /**
895
     * Run provided commands
896
     *
897
     * @param array $commands
898
     *
899
     * @throws TelegramException
900
     */
901
    public function runCommands($commands)
902
    {
903
        if (!is_array($commands) || empty($commands)) {
0 ignored issues
show
introduced by
The condition is_array($commands) is always true.
Loading history...
904
            throw new TelegramException('No command(s) provided!');
905
        }
906
907
        $this->run_commands = true;
908
909
        $result = Request::getMe();
910
911
        if ($result->isOk()) {
912
            $result = $result->getResult();
913
914
            $bot_id       = $result->getId();
915
            $bot_name     = $result->getFirstName();
916
            $bot_username = $result->getUsername();
917
        } else {
918
            $bot_id       = $this->getBotId();
919
            $bot_name     = $this->getBotUsername();
920
            $bot_username = $this->getBotUsername();
921
        }
922
923
924
        $this->enableAdmin($bot_id);    // Give bot access to admin commands
925
        $this->getCommandsList();       // Load full commands list
926
927
        foreach ($commands as $command) {
928
            $this->update = new Update(
929
                [
930
                    'update_id' => 0,
931
                    'message'   => [
932
                        'message_id' => 0,
933
                        'from'       => [
934
                            'id'         => $bot_id,
935
                            'first_name' => $bot_name,
936
                            'username'   => $bot_username,
937
                        ],
938
                        'date'       => time(),
939
                        'chat'       => [
940
                            'id'   => $bot_id,
941
                            'type' => 'private',
942
                        ],
943
                        'text'       => $command,
944
                    ],
945
                ]
946
            );
947
948
            $this->executeCommand($this->update->getMessage()->getCommand());
949
        }
950
    }
951
952
    /**
953
     * Is this session initiated by runCommands()
954
     *
955
     * @return bool
956
     */
957
    public function isRunCommands()
958
    {
959
        return $this->run_commands;
960
    }
961
962
    /**
963
     * Switch to enable running getUpdates without a database
964
     *
965
     * @param bool $enable
966
     */
967
    public function useGetUpdatesWithoutDatabase($enable = true)
968
    {
969
        $this->getupdates_without_database = $enable;
970
    }
971
972
    /**
973
     * Return last update id
974
     *
975
     * @return int
976
     */
977
    public function getLastUpdateId()
978
    {
979
        return $this->last_update_id;
980
    }
981
}
982