Multiple bugfixes

This commit is contained in:
Daniil Gentili 2020-09-27 19:36:17 +02:00
parent 86fc8b6e6e
commit 5560b140f5
10 changed files with 39 additions and 84 deletions

View File

@ -189,7 +189,7 @@ class API extends InternalDoc
if ($settings instanceof SettingsIpc) { if ($settings instanceof SettingsIpc) {
$forceFull = $forceFull || $settings->getForceFull(); $forceFull = $forceFull || $settings->getForceFull();
} elseif ($settings instanceof Settings) { } elseif ($settings instanceof Settings) {
$forceFull = $forceFull || $settings->getSerialization()->getForceFull(); $forceFull = $forceFull || $settings->getIpc()->getForceFull();
} }
[$unserialized, $this->unlock] = yield Tools::timeoutWithDefault( [$unserialized, $this->unlock] = yield Tools::timeoutWithDefault(
@ -250,13 +250,15 @@ class API extends InternalDoc
$this->init(); $this->init();
if (!$this->oldInstance) { if (!$this->oldInstance) {
$this->logger->logger('Shutting down MadelineProto ('.static::class.')'); $this->logger->logger('Shutting down MadelineProto ('.static::class.')');
$this->destructing = true;
if ($this->API) { if ($this->API) {
$this->API->destructing = true; $this->API->destructing = true;
$this->API->unreference(); $this->API->unreference();
} }
$this->destructing = true;
if (isset($this->wrapper)) { if (isset($this->wrapper)) {
$this->logger->logger('Prompting final serialization...');
Tools::wait($this->wrapper->serialize()); Tools::wait($this->wrapper->serialize());
$this->logger->logger('Done final serialization!');
} }
if ($this->unlock) { if ($this->unlock) {
($this->unlock)(); ($this->unlock)();

View File

@ -34,7 +34,7 @@ class AsyncConstruct
/** /**
* Async init promise. * Async init promise.
* *
* @var Promise * @var Promise|null|boolean
*/ */
private $asyncInitPromise; private $asyncInitPromise;
/** /**
@ -93,10 +93,9 @@ class AsyncConstruct
$this->asyncInitPromise = Tools::call($promise); $this->asyncInitPromise = Tools::call($promise);
$this->asyncInitPromise->onResolve( $this->asyncInitPromise->onResolve(
function (?\Throwable $error, $result): void { function (?\Throwable $error, $result): void {
if ($error) { if (!$error) {
throw $error; $this->asyncInitPromise = null;
} }
$this->asyncInitPromise = null;
} }
); );
} }

View File

@ -1,59 +0,0 @@
<?php
/**
* CombinedEventHandler 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-2020 Daniil Gentili <daniil@daniil.it>
* @license https://opensource.org/licenses/AGPL-3.0 AGPLv3
*
* @link https://docs.madelineproto.xyz MadelineProto documentation
*/
namespace danog\MadelineProto;
abstract class CombinedEventHandler
{
private $CombinedAPI;
public function __construct($CombinedAPI)
{
$this->CombinedAPI = $CombinedAPI;
foreach ($CombinedAPI->instances as $path => $instance) {
$this->referenceInstance($path);
}
}
final public function __sleep()
{
$keys = \method_exists($this, '__magic_sleep') ? $this->__magic_sleep() : \get_object_vars($this);
unset($keys['CombinedAPI']);
if (isset($this->CombinedAPI) && $this->CombinedAPI instanceof CombinedAPI) {
foreach ($this->CombinedAPI->instance_paths as $path) {
unset($keys[$path]);
}
} else {
foreach ($keys as $key => $value) {
if ($value instanceof API && $key === $value->session) {
unset($keys[$key]);
}
}
}
return \array_keys($keys);
}
final public function referenceInstance($path)
{
$this->{$path} = $this->CombinedAPI->instances[$path];
}
final public function removeInstance($path)
{
if (isset($this->{$path})) {
unset($this->{$path});
}
}
}

View File

@ -18,7 +18,7 @@ final class IpcState
/** /**
* Exception. * Exception.
*/ */
private ?\Throwable $exception; private ?ExitFailure $exception;
/** /**
* Construct. * Construct.
* *
@ -29,7 +29,7 @@ final class IpcState
{ {
$this->startupTime = \microtime(true); $this->startupTime = \microtime(true);
$this->startupId = $startupId; $this->startupId = $startupId;
$this->exception = $exception; $this->exception = $exception ? new ExitFailure($exception) : null;
} }
/** /**
@ -59,6 +59,6 @@ final class IpcState
*/ */
public function getException(): ?\Throwable public function getException(): ?\Throwable
{ {
return $this->exception; return $this->exception ? $this->exception->getException() : null;
} }
} }

View File

@ -98,11 +98,11 @@ use danog\MadelineProto\Tools;
Magic::$script_cwd = $_GET['cwd'] ?? Magic::getcwd(); Magic::$script_cwd = $_GET['cwd'] ?? Magic::getcwd();
$API = new API($ipcPath, (new Ipc)->setForceFull(true)); $API = new API($ipcPath, (new Ipc)->setForceFull(true));
$API->init();
$API->initSelfRestart();
while (true) { while (true) {
try { try {
$API->init();
$API->initSelfRestart();
Tools::wait($session->storeIpcState(new IpcState($runnerId))); Tools::wait($session->storeIpcState(new IpcState($runnerId)));
Tools::wait(Server::waitShutdown()); Tools::wait(Server::waitShutdown());
return; return;
@ -112,10 +112,15 @@ use danog\MadelineProto\Tools;
} }
} }
} catch (\Throwable $e) { } catch (\Throwable $e) {
Logger::log("Got exception $e in IPC server, exiting...", Logger::FATAL_ERROR); Logger::log("$e", Logger::FATAL_ERROR);
Logger::log("Got exception in IPC server, exiting...", Logger::FATAL_ERROR);
$ipc = Tools::wait($session->getIpcState()); $ipc = Tools::wait($session->getIpcState());
if (!($ipc && $ipc->getStartupId() === $runnerId && !$ipc->getException())) { if (!($ipc && $ipc->getStartupId() === $runnerId && !$ipc->getException())) {
Logger::log("Reporting error!");
Tools::wait($session->storeIpcState(new IpcState($runnerId, $e))); Tools::wait($session->storeIpcState(new IpcState($runnerId, $e)));
Logger::log("Reported error!");
} else {
Logger::log("Not reporting error!");
} }
} }
} }

View File

@ -504,8 +504,8 @@ class MTProto extends AsyncConstruct implements TLCallback
if (!$this->session || $this->session instanceof MemoryArray) { if (!$this->session || $this->session instanceof MemoryArray) {
return $data; return $data;
} }
yield $this->session['data'] = \serialize($data); yield $this->session->offsetSet('data', $data);
var_dump($this); var_dump("Saved!");
return $this->session; return $this->session;
} }
@ -981,7 +981,7 @@ class MTProto extends AsyncConstruct implements TLCallback
while (yield $iterator->advance()) { while (yield $iterator->advance()) {
[$id, $full] = $iterator->getCurrent(); [$id, $full] = $iterator->getCurrent();
if (isset($full['full'], $full['last_update'])) { if (isset($full['full'], $full['last_update'])) {
$this->full_chats[$id] = ['full' => $full['full'], 'last_update' => $full['last_update']]; yield $this->full_chats->offsetSet($id, ['full' => $full['full'], 'last_update' => $full['last_update']]);
} }
} }
@ -1152,6 +1152,8 @@ class MTProto extends AsyncConstruct implements TLCallback
foreach ($this->datacenter->getDataCenterConnections() as $datacenter) { foreach ($this->datacenter->getDataCenterConnections() as $datacenter) {
$datacenter->disconnect(); $datacenter->disconnect();
} }
$this->logger->logger("Unreferenced instance");
} }
/** /**
* Destructor. * Destructor.

View File

@ -130,7 +130,7 @@ trait PeerHandler
$user['access_hash'] = $existingChat['access_hash']; $user['access_hash'] = $existingChat['access_hash'];
} }
} }
$this->chats[$user['id']] = $user; yield $this->chats->offsetSet($user['id'], $user);
$this->cachePwrChat($user['id'], false, true); $this->cachePwrChat($user['id'], false, true);
} }
$this->cacheChatUsername($user['id'], $user); $this->cacheChatUsername($user['id'], $user);
@ -159,7 +159,7 @@ trait PeerHandler
$existingChat = yield $this->chats[-$chat['id']]; $existingChat = yield $this->chats[-$chat['id']];
if (!$existingChat || $existingChat !== $chat) { if (!$existingChat || $existingChat !== $chat) {
$this->logger->logger("Updated chat -{$chat['id']}", \danog\MadelineProto\Logger::ULTRA_VERBOSE); $this->logger->logger("Updated chat -{$chat['id']}", \danog\MadelineProto\Logger::ULTRA_VERBOSE);
$this->chats[-$chat['id']] = $chat; yield $this->chats->offsetSet(-$chat['id'], $chat);
$this->cachePwrChat(-$chat['id'], $this->getSettings()->getPeer()->getFullFetch(), true); $this->cachePwrChat(-$chat['id'], $this->getSettings()->getPeer()->getFullFetch(), true);
} }
$this->cacheChatUsername(-$chat['id'], $chat); $this->cacheChatUsername(-$chat['id'], $chat);
@ -197,7 +197,7 @@ trait PeerHandler
} }
$chat = $newchat; $chat = $newchat;
} }
$this->chats[$bot_api_id] = $chat; yield $this->chats->offsetSet($bot_api_id, $chat);
$fullChat = yield $this->full_chats[$bot_api_id]; $fullChat = yield $this->full_chats[$bot_api_id];
if ($this->getSettings()->getPeer()->getFullFetch() && (!$fullChat || $fullChat['full']['participants_count'] !== (yield from $this->getFullInfo($bot_api_id))['full']['participants_count'])) { if ($this->getSettings()->getPeer()->getFullFetch() && (!$fullChat || $fullChat['full']['participants_count'] !== (yield from $this->getFullInfo($bot_api_id))['full']['participants_count'])) {
$this->cachePwrChat($bot_api_id, $this->getSettings()->getPeer()->getFullFetch(), true); $this->cachePwrChat($bot_api_id, $this->getSettings()->getPeer()->getFullFetch(), true);

View File

@ -58,6 +58,10 @@ class Magic
* @var boolean * @var boolean
*/ */
public static $isFork = false; public static $isFork = false;
/**
* Whether this is an IPC worker
*/
public static bool $isIpcWorker = false;
/** /**
* Whether we can get our PID. * Whether we can get our PID.
* *
@ -230,6 +234,7 @@ class Magic
// Setup error reporting // Setup error reporting
\set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']); \set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']);
\set_exception_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionHandler']); \set_exception_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionHandler']);
self::$isIpcWorker = defined(\MADELINE_WORKER_TYPE::class) ? \MADELINE_WORKER_TYPE === 'madeline-ipc' : false;
if (PHP_SAPI !== 'cli' && PHP_SAPI !== 'phpdbg') { if (PHP_SAPI !== 'cli' && PHP_SAPI !== 'phpdbg') {
try { try {
\error_reporting(E_ALL); \error_reporting(E_ALL);
@ -261,11 +266,11 @@ class Magic
\pcntl_signal(SIGINT, SIG_DFL); \pcntl_signal(SIGINT, SIG_DFL);
Loop::unreference(Loop::onSignal(SIGINT, static function () { Loop::unreference(Loop::onSignal(SIGINT, static function () {
Logger::log('Got sigint', Logger::FATAL_ERROR); Logger::log('Got sigint', Logger::FATAL_ERROR);
Magic::shutdown(1); Magic::shutdown(self::$isIpcWorker ? 0 : 1);
})); }));
Loop::unreference(Loop::onSignal(SIGTERM, static function () { Loop::unreference(Loop::onSignal(SIGTERM, static function () {
Logger::log('Got sigterm', Logger::FATAL_ERROR); Logger::log('Got sigterm', Logger::FATAL_ERROR);
Magic::shutdown(1); Magic::shutdown(self::$isIpcWorker ? 0 : 1);
})); }));
} catch (\Throwable $e) { } catch (\Throwable $e) {
} }

View File

@ -195,7 +195,10 @@ abstract class Serialization
if ($unserialized instanceof DriverArray) { if ($unserialized instanceof DriverArray) {
Logger::log("Extracting session from database..."); Logger::log("Extracting session from database...");
yield from $unserialized->initConnection($unserialized->dbSettings); yield from $unserialized->initConnection($unserialized->dbSettings);
$unserialized = \unserialize(yield $unserialized['data']); $unserialized = yield $unserialized['data'];
if (!$unserialized) {
throw new Exception("Could not extract session from database!");
}
} }
} else { } else {
$unserialized = yield from self::legacyUnserialize($session->getLegacySessionPath()); $unserialized = yield from self::legacyUnserialize($session->getLegacySessionPath());

View File

@ -60,13 +60,11 @@ $count = \count(Lang::$lang[$lang_code]);
$curcount = 0; $curcount = 0;
\ksort(Lang::$current_lang); \ksort(Lang::$current_lang);
foreach (Lang::$current_lang as $key => $value) { foreach (Lang::$current_lang as $key => $value) {
if (!isset(Lang::$lang[$lang_code][$key])) { if (!(Lang::$lang[$lang_code][$key] ?? '')) {
Lang::$lang[$lang_code][$key] = $value; Lang::$lang[$lang_code][$key] = $value;
} }
if (Lang::$lang[$lang_code][$key] === $value && ( if (Lang::$lang[$lang_code][$key] === '' || (Lang::$lang[$lang_code][$key] === Lang::$lang['en'][$key] && Lang::$lang[$lang_code][$key] !== 'Bot')) {
$lang_code !== 'en' || $value == ''
)) {
$value = Lang::$lang[$lang_code][$key]; $value = Lang::$lang[$lang_code][$key];
if (\in_array($key, ['v_error', 'v_tgerror'])) { if (\in_array($key, ['v_error', 'v_tgerror'])) {
$value = \hex2bin($value); $value = \hex2bin($value);