. * * @author Daniil Gentili * @copyright 2016-2023 Daniil Gentili * @license https://opensource.org/licenses/AGPL-3.0 AGPLv3 * @link https://docs.madelineproto.xyz MadelineProto documentation */ namespace danog\MadelineProto; use Amp\DeferredFuture; use Amp\Future; use Amp\Sync\LocalMutex; use AssertionError; use Closure; use danog\MadelineProto\Db\DbPropertiesTrait; use danog\MadelineProto\EventHandler\Filter\Combinator\FiltersAnd; use danog\MadelineProto\EventHandler\Filter\Filter; use danog\MadelineProto\EventHandler\Handler; use danog\MadelineProto\EventHandler\Update; use Fiber; use Generator; use ReflectionAttribute; use ReflectionClass; use ReflectionMethod; use Revolt\EventLoop; use Webmozart\Assert\Assert; use function Amp\File\isDirectory; use function Amp\File\isFile; use function Amp\File\listFiles; /** * Event handler. */ abstract class EventHandler extends AbstractAPI { use DbPropertiesTrait { DbPropertiesTrait::initDb as private internalInitDb; } private static bool $includingPlugins = false; /** * Start MadelineProto and the event handler. * * Also initializes error reporting, catching and reporting all errors surfacing from the event loop. * * @param string $session Session name * @param ?SettingsAbstract $settings Settings */ final public static function startAndLoop(string $session, ?SettingsAbstract $settings = null): void { if (self::$includingPlugins) { Fiber::suspend(static::class); throw new AssertionError("Unreachable!"); } $settings ??= new SettingsEmpty; $API = new API($session, $settings); $API->startAndLoopInternal(static::class); } /** * Start MadelineProto as a bot and the event handler. * * Also initializes error reporting, catching and reporting all errors surfacing from the event loop. * * @param string $session Session name * @param string $token Bot token * @param ?SettingsAbstract $settings Settings */ final public static function startAndLoopBot(string $session, string $token, ?SettingsAbstract $settings = null): void { if (self::$includingPlugins) { Fiber::suspend(static::class); throw new AssertionError("Unreachable!"); } $settings ??= new SettingsEmpty; $API = new API($session, $settings); $API->botLogin($token); $API->startAndLoopInternal(static::class); } /** @internal */ final protected function reconnectFull(): bool { return true; } /** * Whether the event handler was started. */ private bool $startedInternal = false; private ?LocalMutex $startMutex = null; private ?DeferredFuture $startDeferred = null; /** * Start method handler. * * @internal */ final public function internalStart(APIWrapper $MadelineProto, array $pluginsPrev, array &$pluginsNew, bool $main = true): ?array { if ($this->startedInternal) { return null; } $this->startMutex ??= new LocalMutex; $this->startDeferred ??= new DeferredFuture; $startDeferred = $this->startDeferred; $lock = $this->startMutex->acquire(); try { $this->wrapper = $MadelineProto; $this->exportNamespaces(); if (isset(static::$dbProperties)) { $this->internalInitDb($this->wrapper->getAPI()); } if ($main) { $this->setReportPeers(Tools::call($this->getReportPeers())->await()); } if (\method_exists($this, 'onStart')) { $r = $this->onStart(); if ($r instanceof Generator) { $r = Tools::consumeGenerator($r); } if ($r instanceof Future) { $r = $r->await(); } } if ($main) { $this->setReportPeers(Tools::call($this->getReportPeers())->await()); } $constructors = $this->getTL()->getConstructors(); $methods = []; $handlers = []; $has_any = false; $basic_handler = static function (array $update, Closure $closure): void { $r = $closure($update); if ($r instanceof Generator) { Tools::consumeGenerator($r); } }; foreach ((new ReflectionClass($this))->getMethods(ReflectionMethod::IS_PUBLIC) as $methodRefl) { $method = $methodRefl->getName(); if ($method === 'onAny') { $has_any = true; continue; } $closure = $this->$method(...); $method_name = \lcfirst(\substr($method, 2)); if (($constructor = $constructors->findByPredicate($method_name)) && $constructor['type'] === 'Update') { $methods[$method_name] = [ function (array $update) use ($basic_handler, $closure): void { EventLoop::queue($basic_handler, $update, $closure); } ]; continue; } if (!($handler = $methodRefl->getAttributes(Handler::class))) { continue; } $filter = $methodRefl->getAttributes( Filter::class, ReflectionAttribute::IS_INSTANCEOF )[0] ?? null; if (!$filter) { continue; } $filter = new FiltersAnd( $filter->newInstance(), Filter::fromReflectionType($methodRefl->getParameters()[0]->getType()) ); $filter = $filter->initialize($this) ?? $filter; $handlers []= function (Update $update) use ($closure, $filter): void { if ($filter->apply($update)) { EventLoop::queue($closure, $update); } }; } if ($has_any) { $onAny = $this->onAny(...); foreach ($constructors->by_id as $constructor) { if ($constructor['type'] === 'Update' && !isset($methods[$constructor['predicate']])) { $methods[$constructor['predicate']] = [$onAny]; } } } $plugins = $this->internalGetPlugins(); foreach ($plugins as $class) { $plugin = $pluginsPrev[$class] ?? $pluginsNew[$class] ?? new $class; $pluginsNew[$class] = $plugin; [$newMethods, $newHandlers] = $plugin->internalStart($MadelineProto, $pluginsPrev, $pluginsNew, false) ?? []; foreach ($newMethods as $update => $method) { $methods[$update] ??= []; $methods[$update][] = $method; } $handlers = \array_merge($handler, $newHandlers); } $this->startedInternal = true; return [$methods, $handlers]; } finally { $this->startDeferred = null; $startDeferred->complete(); $lock->release(); } } /** * @internal */ final public function waitForInternalStart(): ?Future { if (!$this->startedInternal && !$this->startDeferred) { $this->startDeferred = new DeferredFuture; } return $this->startDeferred?->getFuture(); } /** * Get peers where to send error reports. * * @return string|int|array */ public function getReportPeers() { return []; } /** * Obtain a path or a list of paths that will be recursively searched for plugins. * * Plugin filenames end with .plugin.php, and will be included automatically. * * @return non-empty-string|non-empty-list|null */ public function getPluginPaths(): string|array|null { return null; } /** * Obtain a list of plugin event handlers to use, in addition with those found by getPluginPath. * * @return array> */ public function getPlugins(): array { return []; } /** * Obtain a list of plugin event handlers. */ private function internalGetPlugins(): array { $paths = $this->getPluginPaths(); if (\is_string($paths)) { $paths = [$paths]; } elseif ($paths === null) { $paths = []; } $plugins = \array_values($this->getPlugins()); $recurse = static function (string $path) use (&$recurse, &$plugins): void { foreach (listFiles($path) as $file) { if (isDirectory($file)) { $recurse($file); } elseif (isFile($file) && \str_ends_with($file, ".plugin.php")) { $f = new Fiber(fn () => require $file); $plugins []= $f->start(); } } }; try { self::$includingPlugins = true; \array_map($recurse, $paths); } finally { self::$includingPlugins = false; } $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!"); } return $plugins; } }