1
0
mirror of https://github.com/danog/MadelineProto.git synced 2025-01-12 02:38:17 +01:00
MadelineProto/src/AbstractAPI.php

94 lines
3.3 KiB
PHP

<?php
declare(strict_types=1);
/**
* APIFactory 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>
* @copyright 2016-2023 Daniil Gentili <daniil@daniil.it>
* @license https://opensource.org/licenses/AGPL-3.0 AGPLv3
* @link https://docs.madelineproto.xyz MadelineProto documentation
*/
namespace danog\MadelineProto;
use Amp\Future\UnhandledFutureError;
use Amp\SignalException;
use Revolt\EventLoop;
abstract class AbstractAPI extends InternalDoc
{
/**
* Start MadelineProto and the event handler (enables async).
*
* Also initializes error reporting, catching and reporting all errors surfacing from the event loop.
*
* @param string $eventHandler Event handler class name
*/
protected function startAndLoopInternal(string $eventHandler): void
{
$started = false;
$errors = [];
$prev = EventLoop::getErrorHandler();
EventLoop::setErrorHandler(
$cb = function (\Throwable $e) use (&$errors, &$started): void {
if ($e instanceof UnhandledFutureError) {
$e = $e->getPrevious();
}
if ($e instanceof SecurityException || $e instanceof SignalException) {
throw $e;
}
if (\str_starts_with($e->getMessage(), 'Could not connect to DC ')) {
throw $e;
}
$t = \time();
$errors = [$t => $errors[$t] ?? 0];
$errors[$t]++;
if ($errors[$t] > 10 && (!$this->wrapper->getAPI()->isInited() || !$started)) {
$this->wrapper->logger('More than 10 errors in a second and not inited, exiting!', Logger::FATAL_ERROR);
return;
}
echo $e;
$this->wrapper->logger((string) $e, Logger::FATAL_ERROR);
$this->report("Surfaced: $e");
}
);
try {
$this->startAndLoopLogic($eventHandler, $started);
} finally {
if (EventLoop::getErrorHandler() === $cb) {
EventLoop::setErrorHandler($prev);
}
}
}
abstract protected function reconnectFull(): bool;
protected function startAndLoopLogic(string $eventHandler, bool &$started): void
{
$this->start();
if (!$this->reconnectFull()) {
return;
}
$this->wrapper->getAPI()->setEventHandler($eventHandler);
$started = true;
/** @psalm-suppress TooFewArguments Always MTProto here */
$this->wrapper->getAPI()->loop();
}
/**
* Sleep function.
*/
public function __sleep(): array
{
return [];
}
}