2022-12-30 21:54:44 +01:00
< ? php
declare ( strict_types = 1 );
Documentation rework (#349)
* Documentation rework
* Apply fixes from StyleCI
* Documentation fixes
* Login as bot through web/cli API, allow using invite links in joinChannel, full invite links in importChatInvite and checkChatInvite, non-invite links in importChatInvite
* Apply fixes from StyleCI
* Logging fixes
* Build docs
* Add methods to modify start template, bugfix to logging and keyboard conversion
* Add TL documentator
* Document MTProto methods
* Documenting methods...
* 7% documented
* Bugfixes
* Update docs
* Update docs
* Simplify file management
* Implement automatic object conversion for media, and more awesome stuff
* Implement automatic object conversion for media, and more awesome stuff
* Implement event update handler and file upload/download callback
* Auto-detect mime type, duration, width and height of media
* Update docs
* Document new file functions
* Fix links
* Fix links
* Update bot.php to use event loop
* Implement webhook update handler and forking in main loop
* Build docs
* Better docs
* Fixes to secret chats
* Almost finished updating docs
* Bugfixes, implemented infinite loop for loop() method, almost finished docs
* Finish writing docs
* Add automatic documentation builder script
* Finished writing docs
2018-03-20 12:48:05 +01:00
Merge alpha into master (async, huge bugfixes and more) (#546)
* Implement async and lots of bugfixes
* Implement more async
* Implement async, implement bugfixes for the connection module, for the datacenter module, huge bugfixes, huge perfomance improvements, media DCs for https, advanced selecting, custom var_dump, totally rewritten IOLoop and response mechanism, promises, improvements to the TL parser, custom mb_substr
* Apply fixes from StyleCI
* Bugfixes
* Apply fixes from StyleCI
* Bugfixes, implement combined promises
* Apply fixes from StyleCI
* Support passing method arguments as callable
* Starting to write async upload logic
* Apply fixes from StyleCI
* Start implementing async file upload
* Apply fixes from StyleCI
* bugfix
* Apply fixes from StyleCI
* Start rewriting connection module
* Add PHP file docblocks for all classes
* Start working on new async stream API
* Finish writing stream API
* More stream API fixes
* Apply fixes from StyleCI
* Rewrite DataCenter and Connection modules
* Clean up stream API documentation
* Fixes
* Apply fixes from StyleCI
* Add referenced parameter to get length of buffer to read in getReadBuffer API
* Moved all MessageHandler code in the Connection module, added a PHP version warning in the phar
* Start fixing reads
* Fix all protocol stream wrappers
* Apply fixes from StyleCI
* Implement disconnection, and remove end function
* Working async RPC
* Implement async file upload
* Bugfix
* Method recall bugfixes
* Bugfixes
* Trait bugfixes
* Fix FIFO buffer
* Bugfixes and speedtests
* Async logging
* Implement websocket streams
* Implement loop API, signal API, clean closing and start changing layer
* Small magna, websocket and HTTP fixes
* Clean up loop API
* Improved stack traces, 2FA and async
* Login fixes
* Added instructions for manual verification
* Small fixes
* More app info improvements
* More app info improvements
* TL and 2FA fixes
* Update to layer 89
* More bugfixes
* Implement broken media reporting
* Remove debug comments
* PHP 7.2 backwards compatibility
* Bugfixes
* Async key generation
* Some simplifications
* Transport fixes
* Cleanup
* async API
* Performance fixes
* Fixes to async API
* Bugfixes
* Implement one-time async loop
* Authorization and logging fixes
* Update to layer 91
* 7to5 fix
* Null coalesce conversion
* Implement socks5 proxy
* Implement HTTP proxy
* Fixes to HTTP proxy
* MTProxy and socks5 fixes
* Disable PHP 5 conversion
* Proxies have higher priority
* Avoid error handling in vendor
* Override composer dependencies
* Fix travis build
* Final composer fixes
* Proxy logic fixes
* Fix get_updates update handling
* Do not use parallel file driver if not supported
* Refactor loader and implement HTTP fixes
* Suppress errors in loader
* HTTP and authorization fixes
* HTTP fixes
* Improved peer management
* Use HTTP protocol on altervista
* Small bugfixes
* Minor fixes
* Docufix
* Docufix
* Legacy fixes
* Fix message queue
* Avoid updating if using MTProxy
* Improve logs and examples
* Trim final newlines while converting parse mode
* Reimplement noResponse flag
* Async combined event handler and APIFactory fixes
* Actually return config
* Case-insensitive methods
* Bugfix
* Apply fixes from StyleCI (#545)
* MTProxy fixes
* PHP 5 warning
* Improved PHP 5 warning
* Use <br> along with newlines in web logs
* Update docs
2018-12-26 20:51:14 +01:00
/**
* EventHandler module .
*
* This file is part of MadelineProto .
* MadelineProto is free software : you can redistribute it and / or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation , either version 3 of the License , or ( at your option ) any later version .
* MadelineProto is distributed in the hope that it will be useful , but WITHOUT ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE .
* See the GNU Affero General Public License for more details .
* You should have received a copy of the GNU General Public License along with MadelineProto .
* If not , see < http :// www . gnu . org / licenses />.
*
* @ author Daniil Gentili < daniil @ daniil . it >
2023-01-04 12:43:01 +01:00
* @ copyright 2016 - 2023 Daniil Gentili < daniil @ daniil . it >
Merge alpha into master (async, huge bugfixes and more) (#546)
* Implement async and lots of bugfixes
* Implement more async
* Implement async, implement bugfixes for the connection module, for the datacenter module, huge bugfixes, huge perfomance improvements, media DCs for https, advanced selecting, custom var_dump, totally rewritten IOLoop and response mechanism, promises, improvements to the TL parser, custom mb_substr
* Apply fixes from StyleCI
* Bugfixes
* Apply fixes from StyleCI
* Bugfixes, implement combined promises
* Apply fixes from StyleCI
* Support passing method arguments as callable
* Starting to write async upload logic
* Apply fixes from StyleCI
* Start implementing async file upload
* Apply fixes from StyleCI
* bugfix
* Apply fixes from StyleCI
* Start rewriting connection module
* Add PHP file docblocks for all classes
* Start working on new async stream API
* Finish writing stream API
* More stream API fixes
* Apply fixes from StyleCI
* Rewrite DataCenter and Connection modules
* Clean up stream API documentation
* Fixes
* Apply fixes from StyleCI
* Add referenced parameter to get length of buffer to read in getReadBuffer API
* Moved all MessageHandler code in the Connection module, added a PHP version warning in the phar
* Start fixing reads
* Fix all protocol stream wrappers
* Apply fixes from StyleCI
* Implement disconnection, and remove end function
* Working async RPC
* Implement async file upload
* Bugfix
* Method recall bugfixes
* Bugfixes
* Trait bugfixes
* Fix FIFO buffer
* Bugfixes and speedtests
* Async logging
* Implement websocket streams
* Implement loop API, signal API, clean closing and start changing layer
* Small magna, websocket and HTTP fixes
* Clean up loop API
* Improved stack traces, 2FA and async
* Login fixes
* Added instructions for manual verification
* Small fixes
* More app info improvements
* More app info improvements
* TL and 2FA fixes
* Update to layer 89
* More bugfixes
* Implement broken media reporting
* Remove debug comments
* PHP 7.2 backwards compatibility
* Bugfixes
* Async key generation
* Some simplifications
* Transport fixes
* Cleanup
* async API
* Performance fixes
* Fixes to async API
* Bugfixes
* Implement one-time async loop
* Authorization and logging fixes
* Update to layer 91
* 7to5 fix
* Null coalesce conversion
* Implement socks5 proxy
* Implement HTTP proxy
* Fixes to HTTP proxy
* MTProxy and socks5 fixes
* Disable PHP 5 conversion
* Proxies have higher priority
* Avoid error handling in vendor
* Override composer dependencies
* Fix travis build
* Final composer fixes
* Proxy logic fixes
* Fix get_updates update handling
* Do not use parallel file driver if not supported
* Refactor loader and implement HTTP fixes
* Suppress errors in loader
* HTTP and authorization fixes
* HTTP fixes
* Improved peer management
* Use HTTP protocol on altervista
* Small bugfixes
* Minor fixes
* Docufix
* Docufix
* Legacy fixes
* Fix message queue
* Avoid updating if using MTProxy
* Improve logs and examples
* Trim final newlines while converting parse mode
* Reimplement noResponse flag
* Async combined event handler and APIFactory fixes
* Actually return config
* Case-insensitive methods
* Bugfix
* Apply fixes from StyleCI (#545)
* MTProxy fixes
* PHP 5 warning
* Improved PHP 5 warning
* Use <br> along with newlines in web logs
* Update docs
2018-12-26 20:51:14 +01:00
* @ license https :// opensource . org / licenses / AGPL - 3.0 AGPLv3
2019-10-31 15:07:35 +01:00
* @ link https :// docs . madelineproto . xyz MadelineProto documentation
Merge alpha into master (async, huge bugfixes and more) (#546)
* Implement async and lots of bugfixes
* Implement more async
* Implement async, implement bugfixes for the connection module, for the datacenter module, huge bugfixes, huge perfomance improvements, media DCs for https, advanced selecting, custom var_dump, totally rewritten IOLoop and response mechanism, promises, improvements to the TL parser, custom mb_substr
* Apply fixes from StyleCI
* Bugfixes
* Apply fixes from StyleCI
* Bugfixes, implement combined promises
* Apply fixes from StyleCI
* Support passing method arguments as callable
* Starting to write async upload logic
* Apply fixes from StyleCI
* Start implementing async file upload
* Apply fixes from StyleCI
* bugfix
* Apply fixes from StyleCI
* Start rewriting connection module
* Add PHP file docblocks for all classes
* Start working on new async stream API
* Finish writing stream API
* More stream API fixes
* Apply fixes from StyleCI
* Rewrite DataCenter and Connection modules
* Clean up stream API documentation
* Fixes
* Apply fixes from StyleCI
* Add referenced parameter to get length of buffer to read in getReadBuffer API
* Moved all MessageHandler code in the Connection module, added a PHP version warning in the phar
* Start fixing reads
* Fix all protocol stream wrappers
* Apply fixes from StyleCI
* Implement disconnection, and remove end function
* Working async RPC
* Implement async file upload
* Bugfix
* Method recall bugfixes
* Bugfixes
* Trait bugfixes
* Fix FIFO buffer
* Bugfixes and speedtests
* Async logging
* Implement websocket streams
* Implement loop API, signal API, clean closing and start changing layer
* Small magna, websocket and HTTP fixes
* Clean up loop API
* Improved stack traces, 2FA and async
* Login fixes
* Added instructions for manual verification
* Small fixes
* More app info improvements
* More app info improvements
* TL and 2FA fixes
* Update to layer 89
* More bugfixes
* Implement broken media reporting
* Remove debug comments
* PHP 7.2 backwards compatibility
* Bugfixes
* Async key generation
* Some simplifications
* Transport fixes
* Cleanup
* async API
* Performance fixes
* Fixes to async API
* Bugfixes
* Implement one-time async loop
* Authorization and logging fixes
* Update to layer 91
* 7to5 fix
* Null coalesce conversion
* Implement socks5 proxy
* Implement HTTP proxy
* Fixes to HTTP proxy
* MTProxy and socks5 fixes
* Disable PHP 5 conversion
* Proxies have higher priority
* Avoid error handling in vendor
* Override composer dependencies
* Fix travis build
* Final composer fixes
* Proxy logic fixes
* Fix get_updates update handling
* Do not use parallel file driver if not supported
* Refactor loader and implement HTTP fixes
* Suppress errors in loader
* HTTP and authorization fixes
* HTTP fixes
* Improved peer management
* Use HTTP protocol on altervista
* Small bugfixes
* Minor fixes
* Docufix
* Docufix
* Legacy fixes
* Fix message queue
* Avoid updating if using MTProxy
* Improve logs and examples
* Trim final newlines while converting parse mode
* Reimplement noResponse flag
* Async combined event handler and APIFactory fixes
* Actually return config
* Case-insensitive methods
* Bugfix
* Apply fixes from StyleCI (#545)
* MTProxy fixes
* PHP 5 warning
* Improved PHP 5 warning
* Use <br> along with newlines in web logs
* Update docs
2018-12-26 20:51:14 +01:00
*/
Documentation rework (#349)
* Documentation rework
* Apply fixes from StyleCI
* Documentation fixes
* Login as bot through web/cli API, allow using invite links in joinChannel, full invite links in importChatInvite and checkChatInvite, non-invite links in importChatInvite
* Apply fixes from StyleCI
* Logging fixes
* Build docs
* Add methods to modify start template, bugfix to logging and keyboard conversion
* Add TL documentator
* Document MTProto methods
* Documenting methods...
* 7% documented
* Bugfixes
* Update docs
* Update docs
* Simplify file management
* Implement automatic object conversion for media, and more awesome stuff
* Implement automatic object conversion for media, and more awesome stuff
* Implement event update handler and file upload/download callback
* Auto-detect mime type, duration, width and height of media
* Update docs
* Document new file functions
* Fix links
* Fix links
* Update bot.php to use event loop
* Implement webhook update handler and forking in main loop
* Build docs
* Better docs
* Fixes to secret chats
* Almost finished updating docs
* Bugfixes, implemented infinite loop for loop() method, almost finished docs
* Finish writing docs
* Add automatic documentation builder script
* Finished writing docs
2018-03-20 12:48:05 +01:00
namespace danog\MadelineProto ;
2023-01-22 20:03:51 +01:00
use Amp\DeferredFuture ;
2023-01-21 21:21:35 +01:00
use Amp\Future ;
2022-08-24 22:16:57 +02:00
use Amp\Sync\LocalMutex ;
2023-07-09 13:54:41 +02:00
use AssertionError ;
2023-07-08 21:42:18 +02:00
use Closure ;
2023-07-09 17:42:51 +02:00
use danog\Loop\PeriodicLoop ;
2020-10-07 16:48:34 +02:00
use danog\MadelineProto\Db\DbPropertiesTrait ;
2023-07-09 18:13:37 +02:00
use danog\MadelineProto\EventHandler\Attributes\Cron ;
2023-07-10 10:12:46 +02:00
use danog\MadelineProto\EventHandler\Attributes\Handler ;
2023-07-09 17:42:51 +02:00
use danog\MadelineProto\EventHandler\Attributes\Periodic ;
2023-07-07 22:19:34 +02:00
use danog\MadelineProto\EventHandler\Filter\Combinator\FiltersAnd ;
2023-07-06 22:09:09 +02:00
use danog\MadelineProto\EventHandler\Filter\Filter ;
2023-07-10 10:12:46 +02:00
use danog\MadelineProto\EventHandler\Filter\FilterAllowAll ;
2023-07-08 15:45:14 +02:00
use danog\MadelineProto\EventHandler\Update ;
2023-01-21 21:21:35 +01:00
use Generator ;
2023-07-09 16:55:08 +02:00
use mysqli ;
use PDO ;
use PhpParser\Node\Expr\FuncCall ;
2023-07-10 18:42:58 +02:00
use PhpParser\Node\Expr\Include_ ;
2023-07-09 16:55:08 +02:00
use PhpParser\Node\Expr\New_ ;
use PhpParser\Node\Name ;
2023-07-09 17:23:54 +02:00
use PhpParser\Node\Scalar\LNumber ;
use PhpParser\Node\Stmt\DeclareDeclare ;
2023-07-09 16:55:08 +02:00
use PhpParser\NodeFinder ;
use PhpParser\NodeVisitor\NameResolver ;
use PhpParser\ParserFactory ;
use PHPStan\PhpDocParser\Ast\NodeTraverser ;
2023-07-06 22:09:09 +02:00
use ReflectionAttribute ;
2023-06-25 19:52:00 +02:00
use ReflectionClass ;
use ReflectionMethod ;
use Revolt\EventLoop ;
use Webmozart\Assert\Assert ;
2020-10-07 16:48:34 +02:00
2023-07-08 22:14:49 +02:00
use function Amp\File\isDirectory ;
use function Amp\File\isFile ;
use function Amp\File\listFiles ;
2023-07-09 16:55:08 +02:00
use function Amp\File\read ;
2023-07-08 22:14:49 +02:00
2019-10-31 15:06:25 +01:00
/**
2019-10-31 20:48:06 +01:00
* Event handler .
2019-10-31 15:06:25 +01:00
*/
2023-01-27 14:20:47 +01:00
abstract class EventHandler extends AbstractAPI
Documentation rework (#349)
* Documentation rework
* Apply fixes from StyleCI
* Documentation fixes
* Login as bot through web/cli API, allow using invite links in joinChannel, full invite links in importChatInvite and checkChatInvite, non-invite links in importChatInvite
* Apply fixes from StyleCI
* Logging fixes
* Build docs
* Add methods to modify start template, bugfix to logging and keyboard conversion
* Add TL documentator
* Document MTProto methods
* Documenting methods...
* 7% documented
* Bugfixes
* Update docs
* Update docs
* Simplify file management
* Implement automatic object conversion for media, and more awesome stuff
* Implement automatic object conversion for media, and more awesome stuff
* Implement event update handler and file upload/download callback
* Auto-detect mime type, duration, width and height of media
* Update docs
* Document new file functions
* Fix links
* Fix links
* Update bot.php to use event loop
* Implement webhook update handler and forking in main loop
* Build docs
* Better docs
* Fixes to secret chats
* Almost finished updating docs
* Bugfixes, implemented infinite loop for loop() method, almost finished docs
* Finish writing docs
* Add automatic documentation builder script
* Finished writing docs
2018-03-20 12:48:05 +01:00
{
2020-10-07 16:48:34 +02:00
use DbPropertiesTrait {
DbPropertiesTrait :: initDb as private internalInitDb ;
}
2023-07-08 22:14:49 +02:00
private static bool $includingPlugins = false ;
2021-12-09 13:25:14 +01:00
/**
2023-01-26 19:52:35 +01:00
* Start MadelineProto and the event handler .
2021-12-09 13:25:14 +01:00
*
* Also initializes error reporting , catching and reporting all errors surfacing from the event loop .
*
* @ param string $session Session name
2023-07-09 14:38:39 +02:00
* @ param ? SettingsAbstract $settings Settings
2021-12-09 13:25:14 +01:00
*/
2023-07-09 14:38:39 +02:00
final public static function startAndLoop ( string $session , ? SettingsAbstract $settings = null ) : void
2021-12-09 13:25:14 +01:00
{
2023-07-09 13:54:41 +02:00
if ( self :: $includingPlugins ) {
2023-07-09 16:39:43 +02:00
throw new PluginRegistration ( static :: class );
2023-07-09 13:54:41 +02:00
}
2023-07-09 14:38:39 +02:00
$settings ? ? = new SettingsEmpty ;
2021-12-09 13:25:14 +01:00
$API = new API ( $session , $settings );
2023-01-26 19:57:50 +01:00
$API -> startAndLoopInternal ( static :: class );
2021-12-09 13:25:14 +01:00
}
2022-08-29 16:46:50 +02:00
/**
2023-01-26 19:52:35 +01:00
* Start MadelineProto as a bot and the event handler .
2022-08-29 16:46:50 +02:00
*
* Also initializes error reporting , catching and reporting all errors surfacing from the event loop .
*
* @ param string $session Session name
* @ param string $token Bot token
2023-07-09 14:38:39 +02:00
* @ param ? SettingsAbstract $settings Settings
2022-08-29 16:46:50 +02:00
*/
2023-07-09 14:38:39 +02:00
final public static function startAndLoopBot ( string $session , string $token , ? SettingsAbstract $settings = null ) : void
2022-08-29 16:46:50 +02:00
{
2023-07-09 13:54:41 +02:00
if ( self :: $includingPlugins ) {
2023-07-09 16:39:43 +02:00
throw new PluginRegistration ( static :: class );
2023-07-09 13:54:41 +02:00
}
2023-07-09 14:38:39 +02:00
$settings ? ? = new SettingsEmpty ;
2022-08-29 16:46:50 +02:00
$API = new API ( $session , $settings );
$API -> botLogin ( $token );
2023-01-26 19:57:50 +01:00
$API -> startAndLoopInternal ( static :: class );
2022-08-29 16:46:50 +02:00
}
2023-06-25 19:52:00 +02:00
/** @internal */
final protected function reconnectFull () : bool
2023-01-27 14:20:47 +01:00
{
return true ;
}
2023-01-28 20:08:56 +01:00
/**
* Whether the event handler was started .
*/
private bool $startedInternal = false ;
private ? LocalMutex $startMutex = null ;
private ? DeferredFuture $startDeferred = null ;
2023-07-09 17:42:51 +02:00
/**
* @ var array < PeriodicLoop >
*/
private array $periodicLoops = [];
2020-10-06 17:33:18 +02:00
/**
* Start method handler .
*
* @ internal
*/
2023-06-25 19:52:00 +02:00
final public function internalStart ( APIWrapper $MadelineProto , array $pluginsPrev , array & $pluginsNew , bool $main = true ) : ? array
2020-10-06 17:33:18 +02:00
{
2023-06-25 20:00:41 +02:00
if ( $this -> startedInternal ) {
return null ;
}
2022-08-24 22:16:57 +02:00
$this -> startMutex ? ? = new LocalMutex ;
2023-01-28 20:08:56 +01:00
$this -> startDeferred ? ? = new DeferredFuture ;
$startDeferred = $this -> startDeferred ;
2022-12-30 22:31:20 +01:00
$lock = $this -> startMutex -> acquire ();
2022-08-24 22:16:57 +02:00
try {
2023-06-25 19:52:00 +02:00
$this -> wrapper = $MadelineProto ;
$this -> exportNamespaces ();
2022-08-24 22:16:57 +02:00
if ( isset ( static :: $dbProperties )) {
2023-01-26 14:33:30 +01:00
$this -> internalInitDb ( $this -> wrapper -> getAPI ());
2022-08-24 22:16:57 +02:00
}
2023-06-25 19:52:00 +02:00
if ( $main ) {
$this -> setReportPeers ( Tools :: call ( $this -> getReportPeers ()) -> await ());
}
2022-08-24 22:16:57 +02:00
if ( \method_exists ( $this , 'onStart' )) {
2023-01-21 21:21:35 +01:00
$r = $this -> onStart ();
if ( $r instanceof Generator ) {
$r = Tools :: consumeGenerator ( $r );
}
if ( $r instanceof Future ) {
$r = $r -> await ();
}
2022-08-24 22:16:57 +02:00
}
2023-06-25 19:52:00 +02:00
if ( $main ) {
$this -> setReportPeers ( Tools :: call ( $this -> getReportPeers ()) -> await ());
}
$constructors = $this -> getTL () -> getConstructors ();
$methods = [];
2023-07-08 15:45:14 +02:00
$handlers = [];
2023-06-25 19:52:00 +02:00
$has_any = false ;
2023-07-08 21:42:18 +02:00
$basic_handler = static function ( array $update , Closure $closure ) : void {
$r = $closure ( $update );
if ( $r instanceof Generator ) {
Tools :: consumeGenerator ( $r );
}
};
2023-06-25 19:52:00 +02:00
foreach (( new ReflectionClass ( $this )) -> getMethods ( ReflectionMethod :: IS_PUBLIC ) as $methodRefl ) {
$method = $methodRefl -> getName ();
if ( $method === 'onAny' ) {
$has_any = true ;
continue ;
}
2023-07-08 21:42:18 +02:00
$closure = $this -> $method ( ... );
2023-06-25 19:52:00 +02:00
$method_name = \lcfirst ( \substr ( $method , 2 ));
if (( $constructor = $constructors -> findByPredicate ( $method_name )) && $constructor [ 'type' ] === 'Update' ) {
2023-07-08 21:42:18 +02:00
$methods [ $method_name ] = [
function ( array $update ) use ( $basic_handler , $closure ) : void {
2023-07-08 22:14:49 +02:00
EventLoop :: queue ( $basic_handler , $update , $closure );
2023-07-08 21:42:18 +02:00
}
];
2023-07-06 22:09:09 +02:00
continue ;
2023-07-07 22:19:34 +02:00
}
2023-07-10 21:15:35 +02:00
if ( ! $this instanceof SimpleEventHandler ) {
continue ;
}
2023-07-09 18:13:37 +02:00
if ( $periodic = $methodRefl -> getAttributes ( Cron :: class )) {
2023-07-09 17:42:51 +02:00
$periodic = $periodic [ 0 ] -> newInstance ();
$this -> periodicLoops [ $method ] = new PeriodicLoop (
2023-07-09 18:10:58 +02:00
function ( PeriodicLoop $loop ) use ( $closure ) : bool {
return $closure ( $loop ) ? ? false ;
},
2023-07-09 17:42:51 +02:00
$method ,
$periodic -> period
);
2023-07-09 18:07:23 +02:00
$this -> periodicLoops [ $method ] -> start ();
2023-07-09 17:42:51 +02:00
continue ;
}
2023-07-06 22:09:09 +02:00
$filter = $methodRefl -> getAttributes (
Filter :: class ,
ReflectionAttribute :: IS_INSTANCEOF
)[ 0 ] ? ? null ;
if ( ! $filter ) {
2023-07-10 10:12:46 +02:00
if ( ! ( $handler = $methodRefl -> getAttributes ( Handler :: class ))) {
continue ;
}
$filter = new FilterAllowAll ;
} else {
$filter = $filter -> newInstance ();
2023-06-25 19:52:00 +02:00
}
2023-07-07 22:19:34 +02:00
$filter = new FiltersAnd (
2023-07-10 10:12:46 +02:00
$filter ,
2023-07-07 22:19:34 +02:00
Filter :: fromReflectionType ( $methodRefl -> getParameters ()[ 0 ] -> getType ())
);
$filter = $filter -> initialize ( $this ) ? ? $filter ;
2023-07-08 21:42:18 +02:00
$handlers [] = function ( Update $update ) use ( $closure , $filter ) : void {
if ( $filter -> apply ( $update )) {
EventLoop :: queue ( $closure , $update );
2023-07-08 15:45:14 +02:00
}
2023-07-08 21:42:18 +02:00
};
2023-07-10 20:05:46 +02:00
}
2023-07-10 21:15:35 +02:00
if ( $this instanceof SimpleEventHandler ) {
2023-07-10 20:05:46 +02:00
self :: validateEventHandler ( static :: class );
2023-06-25 19:52:00 +02:00
}
if ( $has_any ) {
2023-07-06 21:59:36 +02:00
$onAny = $this -> onAny ( ... );
2023-06-25 19:52:00 +02:00
foreach ( $constructors -> by_id as $constructor ) {
if ( $constructor [ 'type' ] === 'Update' && ! isset ( $methods [ $constructor [ 'predicate' ]])) {
2023-07-08 21:42:18 +02:00
$methods [ $constructor [ 'predicate' ]] = [ $onAny ];
2023-06-25 19:52:00 +02:00
}
}
}
2023-07-08 22:14:49 +02:00
$plugins = $this -> internalGetPlugins ();
foreach ( $plugins as $class ) {
2023-06-25 19:52:00 +02:00
$plugin = $pluginsPrev [ $class ] ? ? $pluginsNew [ $class ] ? ? new $class ;
$pluginsNew [ $class ] = $plugin ;
2023-07-08 21:42:18 +02:00
[ $newMethods , $newHandlers ] = $plugin -> internalStart ( $MadelineProto , $pluginsPrev , $pluginsNew , false ) ? ? [];
2023-07-10 21:15:35 +02:00
if ( ! $plugin -> isPluginEnabled ()) {
continue ;
}
2023-07-08 21:42:18 +02:00
foreach ( $newMethods as $update => $method ) {
$methods [ $update ] ? ? = [];
$methods [ $update ][] = $method ;
2023-06-25 19:52:00 +02:00
}
2023-07-08 21:42:18 +02:00
$handlers = \array_merge ( $handler , $newHandlers );
2023-06-25 19:52:00 +02:00
}
2022-08-24 22:16:57 +02:00
$this -> startedInternal = true ;
2023-07-08 21:42:18 +02:00
return [ $methods , $handlers ];
2022-08-24 22:16:57 +02:00
} finally {
2023-01-28 20:08:56 +01:00
$this -> startDeferred = null ;
2023-01-22 20:03:51 +01:00
$startDeferred -> complete ();
2022-08-24 22:16:57 +02:00
$lock -> release ();
2020-10-06 17:33:18 +02:00
}
}
2023-07-09 17:42:51 +02:00
/**
* Obtain a PeriodicLoop instance created by the Periodic attribute .
*
* @ param string $name Method name
*/
final public function getPeriodicLoop ( string $name ) : PeriodicLoop
{
return $this -> periodicLoops [ $name ];
}
2023-01-22 20:03:51 +01:00
/**
* @ internal
*/
2023-06-25 19:52:00 +02:00
final public function waitForInternalStart () : ? Future
2023-01-22 20:06:34 +01:00
{
2023-01-28 20:08:56 +01:00
if ( ! $this -> startedInternal && ! $this -> startDeferred ) {
$this -> startDeferred = new DeferredFuture ;
}
return $this -> startDeferred ? -> getFuture ();
2023-01-22 20:03:51 +01:00
}
2020-02-23 19:28:42 +01:00
/**
* Get peers where to send error reports .
*
2023-01-04 15:26:06 +01:00
* @ return string | int | array < string | int >
2020-02-23 19:28:42 +01:00
*/
2023-01-04 15:26:06 +01:00
public function getReportPeers ()
2020-02-23 19:28:42 +01:00
{
return [];
}
2023-06-25 19:52:00 +02:00
/**
2023-07-08 22:14:49 +02:00
* Obtain a path or a list of paths that will be recursively searched for plugins .
2023-07-08 22:24:31 +02:00
*
2023-07-09 16:36:30 +02:00
* Plugin filenames end with Plugin . php , and will be included automatically .
2023-07-08 22:14:49 +02:00
*
* @ return non - empty - string | non - empty - list < non - empty - string >| null
*/
2023-07-08 22:24:31 +02:00
public function getPluginPaths () : string | array | null
{
2023-07-08 22:14:49 +02:00
return null ;
}
/**
* Obtain a list of plugin event handlers to use , in addition with those found by getPluginPath .
2023-06-25 19:52:00 +02:00
*
* @ return array < class - string < EventHandler >>
*/
public function getPlugins () : array
{
return [];
}
2023-07-08 22:14:49 +02:00
2023-07-09 16:36:30 +02:00
private static array $includedPaths = [];
2023-07-08 22:14:49 +02:00
/**
* Obtain a list of plugin event handlers .
2023-07-10 21:15:35 +02:00
*
* @ return list < class - string < PluginEventHandler >>
2023-07-08 22:14:49 +02:00
*/
2023-07-08 22:24:31 +02:00
private function internalGetPlugins () : array
{
2023-07-10 21:15:35 +02:00
$plugins = $this -> getPlugins ();
$this -> internalGetDirectoryPlugins ( $plugins );
$plugins = \array_values ( \array_unique ( $plugins , SORT_REGULAR ));
foreach ( $plugins as $plugin ) {
Assert :: classExists ( $plugin );
Assert :: true ( \is_subclass_of ( $plugin , PluginEventHandler :: class ), " $plugin must extend " . PluginEventHandler :: class );
Assert :: notEq ( $plugin , PluginEventHandler :: class );
Assert :: true ( \str_contains ( \ltrim ( $plugin , '\\' ), '\\' ), " $plugin must be in a namespace! " );
self :: validateEventHandler ( $plugin );
}
return $plugins ;
}
private function internalGetDirectoryPlugins ( array & $plugins ) : void
{
if ( $this instanceof PluginEventHandler ) {
return ;
}
2023-07-08 22:14:49 +02:00
$paths = $this -> getPluginPaths ();
2023-07-08 22:24:31 +02:00
if ( \is_string ( $paths )) {
2023-07-08 22:14:49 +02:00
$paths = [ $paths ];
} elseif ( $paths === null ) {
$paths = [];
}
2023-07-10 21:15:35 +02:00
if ( ! $paths ) {
return ;
}
$paths = \array_map ( realpath ( ... ), $paths );
2023-07-08 22:14:49 +02:00
2023-07-10 21:15:35 +02:00
$recurse = static function ( string $path , string $namespace = 'MadelinePlugin' ) use ( & $recurse , & $plugins ) : void {
2023-07-08 22:14:49 +02:00
foreach ( listFiles ( $path ) as $file ) {
if ( isDirectory ( $file )) {
2023-07-10 21:15:35 +02:00
$recurse ( $file , $namespace . '\\' . \basename ( $file ));
2023-07-09 16:36:30 +02:00
} elseif ( isFile ( $file ) && \str_ends_with ( $file , " Plugin.php " )) {
2023-07-09 16:39:43 +02:00
$file = \realpath ( $file );
2023-07-09 16:36:30 +02:00
if ( isset ( self :: $includedPaths [ $file ])) {
continue ;
}
self :: $includedPaths [ $file ] = true ;
try {
2023-07-09 16:39:43 +02:00
require $file ;
2023-07-09 16:36:30 +02:00
} catch ( PluginRegistration $e ) {
2023-07-10 21:15:35 +02:00
Assert :: eq ( $e -> plugin , $namespace . '\\' . \basename ( $file , '.php' ));
2023-07-09 16:36:30 +02:00
$plugins [] = $e -> plugin ;
continue ;
}
throw new AssertionError ( " No plugin was registered after including $file ! " );
2023-07-08 22:14:49 +02:00
}
}
};
try {
2023-07-08 22:24:31 +02:00
self :: $includingPlugins = true ;
\array_map ( $recurse , $paths );
2023-07-08 22:14:49 +02:00
} finally {
self :: $includingPlugins = false ;
}
2023-07-10 21:15:35 +02:00
\spl_autoload_register ( function ( string $class ) use ( $paths ) : void {
if ( ! \str_starts_with ( $class , 'MadelinePlugin\\' )) {
return ;
}
// Has leading /
$file = \str_replace ( '\\' , DIRECTORY_SEPARATOR , \substr ( $class , 14 )) . '.php' ;
foreach ( $paths as $path ) {
if ( \file_exists ( $path . $file )) {
require $path . $file ;
Assert :: classExists ( $class );
}
}
});
2023-07-08 22:14:49 +02:00
}
2023-07-09 16:55:08 +02:00
private const BANNED_FUNCTIONS = [
'file_get_contents' ,
'file_put_contents' ,
2023-07-10 18:42:58 +02:00
'unlink' ,
2023-07-09 16:55:08 +02:00
'curl_exec' ,
2023-07-10 18:42:58 +02:00
'mysqli_query' ,
2023-07-09 16:55:08 +02:00
'mysqli_connect' ,
2023-07-09 17:42:51 +02:00
'mysql_connect' ,
2023-07-09 16:55:08 +02:00
'fopen' ,
2023-07-09 17:23:54 +02:00
'fsockopen' ,
];
private const BANNED_FILE_FUNCTIONS = [
'amp\\file\\read' ,
'amp\\file\\write' ,
'amp\\file\\get' ,
'amp\\file\\put' ,
2023-07-09 16:55:08 +02:00
];
private const BANNED_CLASSES = [
PDO :: class ,
mysqli :: class ,
];
2023-07-10 19:56:44 +02:00
/**
* Perform static analysis on a certain event handler class , to make sure it satisfies some performance requirements .
*
* @ param class - string < EventHandler > $class Class name
*
* @ throws AssertionError If validation fails .
*/
final public static function validateEventHandler ( string $class ) : void
2023-07-09 17:23:54 +02:00
{
2023-07-09 16:55:08 +02:00
$file = read (( new ReflectionClass ( $class )) -> getFileName ());
$file = ( new ParserFactory ) -> create ( ParserFactory :: ONLY_PHP7 ) -> parse ( $file );
Assert :: notNull ( $file );
$traverser = new NodeTraverser ([ new NameResolver ()]);
$file = $traverser -> traverse ( $file );
2023-07-09 17:23:54 +02:00
$finder = new NodeFinder ;
/** @var DeclareDeclare|null $call */
$declare = $finder -> findFirstInstanceOf ( $file , DeclareDeclare :: class );
2023-07-10 19:56:44 +02:00
if ( $declare === null
|| $declare -> key -> name !== 'strict_types'
|| ! $declare -> value instanceof LNumber
|| $declare -> value -> value !== 1
) {
throw new AssertionError ( " An error occurred while analyzing plugin $class : for performance reasons, the first statement of a plugin must be declare(strict_types=1); " );
}
2023-07-09 16:55:08 +02:00
/** @var FuncCall $call */
2023-07-09 17:23:54 +02:00
foreach ( $finder -> findInstanceOf ( $file , FuncCall :: class ) as $call ) {
if ( ! $call -> name instanceof Name ) {
continue ;
}
$name = $call -> name -> toLowerString ();
if ( \in_array ( $name , self :: BANNED_FUNCTIONS , true )) {
2023-07-09 17:42:51 +02:00
throw new AssertionError ( " An error occurred while analyzing plugin $class : for performance reasons, plugins may not use the non-async blocking function $name ! " );
2023-07-09 16:55:08 +02:00
}
2023-07-09 17:23:54 +02:00
if ( \in_array ( $name , self :: BANNED_FILE_FUNCTIONS , true )) {
2023-07-09 17:42:51 +02:00
throw new AssertionError ( " An error occurred while analyzing plugin $class : for performance reasons, plugins may not use the file function $name , please use properties and __sleep to store plugin-related configuration in the session! " );
2023-07-09 17:23:54 +02:00
}
2023-07-09 16:55:08 +02:00
}
/** @var New_ $call */
2023-07-09 17:23:54 +02:00
foreach ( $finder -> findInstanceOf ( $file , New_ :: class ) as $new ) {
2023-07-09 16:55:08 +02:00
if ( $new -> class instanceof Name
2023-07-09 17:23:54 +02:00
&& \in_array ( $name = $new -> class -> toLowerString (), self :: BANNED_CLASSES , true )
2023-07-09 16:55:08 +02:00
) {
2023-07-09 17:42:51 +02:00
throw new AssertionError ( " An error occurred while analyzing plugin $class : for performance reasons, plugins may not use the non-async blocking class $name ! " );
2023-07-09 16:55:08 +02:00
}
}
2023-07-10 18:42:58 +02:00
if ( $finder -> findFirstInstanceOf ( $file , Include_ :: class )) {
throw new AssertionError ( " An error occurred while analyzing plugin $class : for performance reasons, plugins can only automatically include or require other files present in the plugins folder by triggering the PSR-4 autoloader (not by manually require()'ing them). " );
}
2023-07-09 16:55:08 +02:00
}
Documentation rework (#349)
* Documentation rework
* Apply fixes from StyleCI
* Documentation fixes
* Login as bot through web/cli API, allow using invite links in joinChannel, full invite links in importChatInvite and checkChatInvite, non-invite links in importChatInvite
* Apply fixes from StyleCI
* Logging fixes
* Build docs
* Add methods to modify start template, bugfix to logging and keyboard conversion
* Add TL documentator
* Document MTProto methods
* Documenting methods...
* 7% documented
* Bugfixes
* Update docs
* Update docs
* Simplify file management
* Implement automatic object conversion for media, and more awesome stuff
* Implement automatic object conversion for media, and more awesome stuff
* Implement event update handler and file upload/download callback
* Auto-detect mime type, duration, width and height of media
* Update docs
* Document new file functions
* Fix links
* Fix links
* Update bot.php to use event loop
* Implement webhook update handler and forking in main loop
* Build docs
* Better docs
* Fixes to secret chats
* Almost finished updating docs
* Bugfixes, implemented infinite loop for loop() method, almost finished docs
* Finish writing docs
* Add automatic documentation builder script
* Finished writing docs
2018-03-20 12:48:05 +01:00
}