Async constructor

This commit is contained in:
Daniil Gentili 2019-05-02 21:30:50 +02:00
parent 2e4be09279
commit 406a2d8aad
7 changed files with 1420 additions and 37 deletions

2
docs

@ -1 +1 @@
Subproject commit 0fcff8e5a31af300511949c4dbafc7be6c0d4dd7 Subproject commit 6d9ffce34a50e6400dba4a516c78062e5dc2f346

View File

@ -62,14 +62,17 @@ class API extends APIFactory
class_exists('\\Volatile'); class_exists('\\Volatile');
$tounserialize = str_replace('O:26:"danog\\MadelineProto\\Button":', 'O:35:"danog\\MadelineProto\\TL\\Types\\Button":', $tounserialize); $tounserialize = str_replace('O:26:"danog\\MadelineProto\\Button":', 'O:35:"danog\\MadelineProto\\TL\\Types\\Button":', $tounserialize);
foreach (['RSA', 'TL\\TLMethod', 'TL\\TLConstructor', 'MTProto', 'API', 'DataCenter', 'Connection', 'TL\\Types\\Button', 'TL\\Types\\Bytes', 'APIFactory'] as $class) { foreach (['RSA', 'TL\\TLMethod', 'TL\\TLConstructor', 'MTProto', 'API', 'DataCenter', 'Connection', 'TL\\Types\\Button', 'TL\\Types\\Bytes', 'APIFactory'] as $class) {
class_exists('\\danog\\MadelineProto\\'.$class); class_exists('\\danog\\MadelineProto\\' . $class);
} }
$unserialized = \danog\Serialization::unserialize($tounserialize); $unserialized = \danog\Serialization::unserialize($tounserialize);
} catch (\danog\MadelineProto\Exception $e) { } catch (\danog\MadelineProto\Exception $e) {
if ($e->getFile() === 'MadelineProto' && $e->getLine() === 1) {
throw $e;
}
class_exists('\\Volatile'); class_exists('\\Volatile');
$tounserialize = str_replace('O:26:"danog\\MadelineProto\\Button":', 'O:35:"danog\\MadelineProto\\TL\\Types\\Button":', $tounserialize); $tounserialize = str_replace('O:26:"danog\\MadelineProto\\Button":', 'O:35:"danog\\MadelineProto\\TL\\Types\\Button":', $tounserialize);
foreach (['RSA', 'TL\\TLMethod', 'TL\\TLConstructor', 'MTProto', 'API', 'DataCenter', 'Connection', 'TL\\Types\\Button', 'TL\\Types\\Bytes', 'APIFactory'] as $class) { foreach (['RSA', 'TL\\TLMethod', 'TL\\TLConstructor', 'MTProto', 'API', 'DataCenter', 'Connection', 'TL\\Types\\Button', 'TL\\Types\\Bytes', 'APIFactory'] as $class) {
class_exists('\\danog\\MadelineProto\\'.$class); class_exists('\\danog\\MadelineProto\\' . $class);
} }
Logger::log((string) $e, Logger::ERROR); Logger::log((string) $e, Logger::ERROR);
if (strpos($e->getMessage(), "Erroneous data format for unserializing 'phpseclib\\Math\\BigInteger'") === 0) { if (strpos($e->getMessage(), "Erroneous data format for unserializing 'phpseclib\\Math\\BigInteger'") === 0) {
@ -89,6 +92,15 @@ class API extends APIFactory
if (isset($unserialized->API)) { if (isset($unserialized->API)) {
$this->API = $unserialized->API; $this->API = $unserialized->API;
$promise = $this->call(function () {
yield $this->API->asyncInitPromise;
$this->API->asyncInitPromise = null;
$this->APIFactory();
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
$pong = $this->ping(['ping_id' => 3], ['async' => true]);
\danog\MadelineProto\Logger::log('Pong: ' . $pong['ping_id'], Logger::ULTRA_VERBOSE);
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['madelineproto_ready'], Logger::NOTICE);
});
$this->APIFactory(); $this->APIFactory();
return; return;
@ -103,11 +115,16 @@ class API extends APIFactory
} }
$this->API = new MTProto($params); $this->API = new MTProto($params);
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['apifactory_start'], Logger::VERBOSE); \danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['apifactory_start'], Logger::VERBOSE);
$promise = $this->call(function () {
yield $this->API->asyncInitPromise;
$this->API->asyncInitPromise = null;
$this->APIFactory();
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
$pong = $this->ping(['ping_id' => 3], ['async' => true]);
\danog\MadelineProto\Logger::log('Pong: ' . $pong['ping_id'], Logger::ULTRA_VERBOSE);
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['madelineproto_ready'], Logger::NOTICE);
});
$this->APIFactory(); $this->APIFactory();
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
$pong = $this->ping(['ping_id' => 3]);
\danog\MadelineProto\Logger::log('Pong: '.$pong['ping_id'], Logger::ULTRA_VERBOSE);
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['madelineproto_ready'], Logger::NOTICE);
} }
public function async($async) public function async($async)
@ -156,6 +173,10 @@ class API extends APIFactory
public function __set($name, $value) public function __set($name, $value)
{ {
if ($name === 'settings') { if ($name === 'settings') {
if ($this->API->phoneConfigWatcherId) {
$this->wait($this->API->phoneConfigWatcherId);
$this->API->phoneConfigWatcherId = null;
}
if (Magic::is_fork() && !Magic::$processed_fork) { if (Magic::is_fork() && !Magic::$processed_fork) {
\danog\MadelineProto\Logger::log('Detected fork'); \danog\MadelineProto\Logger::log('Detected fork');
$this->API->reset_session(); $this->API->reset_session();

View File

@ -124,7 +124,7 @@ class APIFactory
public function __construct($namespace, $API) public function __construct($namespace, $API)
{ {
$this->namespace = $namespace.'.'; $this->namespace = $namespace . '.';
$this->API = $API; $this->API = $API;
} }
@ -143,23 +143,24 @@ class APIFactory
$this->API->setdem = false; $this->API->setdem = false;
$this->API->__construct($this->API->settings); $this->API->__construct($this->API->settings);
} }
$this->API->get_config([], ['datacenter' => $this->API->datacenter->curdc]); //$this->API->get_config([], ['datacenter' => $this->API->datacenter->curdc]);
if (isset($this->session) && !is_null($this->session) && time() - $this->serialized > $this->API->settings['serialization']['serialization_interval']) { if (isset($this->session) && !is_null($this->session) && time() - $this->serialized > $this->API->settings['serialization']['serialization_interval']) {
Logger::log("Didn't serialize in a while, doing that now..."); Logger::log("Didn't serialize in a while, doing that now...");
$this->serialize($this->session); $this->serialize($this->session);
} }
/*
if ($name !== 'accept_tos' && $name !== 'decline_tos') { if ($name !== 'accept_tos' && $name !== 'decline_tos') {
$this->API->check_tos(); $this->API->check_tos();
} }*/
$lower_name = strtolower($name); $lower_name = strtolower($name);
if ($this->lua === false) { if ($this->lua === false) {
return $this->namespace !== '' || !isset($this->methods[$lower_name]) ? $this->__mtproto_call($this->namespace.$name, $arguments) : $this->__api_call($lower_name, $arguments); return $this->namespace !== '' || !isset($this->methods[$lower_name]) ? $this->__mtproto_call($this->namespace . $name, $arguments) : $this->__api_call($lower_name, $arguments);
} }
try { try {
$deserialized = $this->namespace !== '' || !isset($this->methods[$lower_name]) ? $this->__mtproto_call($this->namespace.$name, $arguments) : $this->__api_call($lower_name, $arguments); $deserialized = $this->namespace !== '' || !isset($this->methods[$lower_name]) ? $this->__mtproto_call($this->namespace . $name, $arguments) : $this->__api_call($lower_name, $arguments);
Lua::convert_objects($deserialized); Lua::convert_objects($deserialized);
@ -183,10 +184,23 @@ class APIFactory
public function __api_call($name, $arguments) public function __api_call($name, $arguments)
{ {
if ($this->API->asyncInitPromise) {
$async = is_array(end($arguments)) && isset(end($arguments)['async']) ? end($arguments)['async'] : ($this->async && $name !== 'loop');
if ($async) {
return $this->call(function () use ($name, $arguments) {
yield $this->API->asyncInitPromise;
$this->API->asyncInitPromise = null;
return yield $this->methods[$name](...$arguments);
});
} else {
$this->wait($this->API->asyncInitPromise);
$this->API->asyncInitPromise = null;
}
}
$result = $this->methods[$name](...$arguments); $result = $this->methods[$name](...$arguments);
if (is_object($result) && ($result instanceof \Generator || $result instanceof Promise)) { if (is_object($result) && ($result instanceof \Generator || $result instanceof Promise)) {
$async = is_array(end($arguments)) && isset(end($arguments)['async']) ? end($arguments)['async'] : $this->async; $async = is_array(end($arguments)) && isset(end($arguments)['async']) ? end($arguments)['async'] : ($this->async && $name !== 'loop');
if ($async && ($name !== 'loop' || isset(end($arguments)['async']))) { if ($async) {
return $result; return $result;
} else { } else {
return $this->wait($result); return $this->wait($result);
@ -204,6 +218,20 @@ class APIFactory
$args = isset($arguments[0]) && is_array($arguments[0]) ? $arguments[0] : []; $args = isset($arguments[0]) && is_array($arguments[0]) ? $arguments[0] : [];
$async = isset(end($arguments)['async']) ? end($arguments)['async'] : $this->async; $async = isset(end($arguments)['async']) ? end($arguments)['async'] : $this->async;
if ($this->API->asyncInitPromise) {
if ($async) {
return $this->call(function () use ($name, $args, $aargs) {
yield $this->API->asyncInitPromise;
$this->API->asyncInitPromise = null;
return yield $this->API->method_call_async_read($name, $args, $aargs);
;
});
} else {
$this->wait($this->API->asyncInitPromise);
$this->API->asyncInitPromise = null;
}
}
$res = $this->API->method_call_async_read($name, $args, $aargs); $res = $this->API->method_call_async_read($name, $args, $aargs);
if ($async) { if ($async) {

View File

@ -137,7 +137,6 @@ class MTProto implements TLCallback
private $msg_ids = []; private $msg_ids = [];
private $v = 0; private $v = 0;
private $dialog_params = ['_' => 'MadelineProto.dialogParams', 'limit' => 0, 'offset_date' => 0, 'offset_id' => 0, 'offset_peer' => ['_' => 'inputPeerEmpty'], 'count' => 0]; private $dialog_params = ['_' => 'MadelineProto.dialogParams', 'limit' => 0, 'offset_date' => 0, 'offset_id' => 0, 'offset_peer' => ['_' => 'inputPeerEmpty'], 'count' => 0];
private $ipv6 = false;
public $run_workers = false; public $run_workers = false;
public $setdem = false; public $setdem = false;
public $storage = []; public $storage = [];
@ -148,7 +147,14 @@ class MTProto implements TLCallback
public $update_deferred; public $update_deferred;
public $phoneConfigWatcherId; public $phoneConfigWatcherId;
public $asyncInitPromise;
public function __magic_construct($settings = []) public function __magic_construct($settings = [])
{
$this->asyncInitPromise = $this->call($this->__async_construct($settings));
}
public function __async_construct($settings = [])
{ {
\danog\MadelineProto\Magic::class_exists(); \danog\MadelineProto\Magic::class_exists();
// Parse settings // Parse settings
@ -185,11 +191,11 @@ class MTProto implements TLCallback
*/ */
$this->logger->logger(\danog\MadelineProto\Lang::$current_lang['TL_translation'], Logger::ULTRA_VERBOSE); $this->logger->logger(\danog\MadelineProto\Lang::$current_lang['TL_translation'], Logger::ULTRA_VERBOSE);
$this->construct_TL($this->settings['tl_schema']['src'], [$this, $this->referenceDatabase]); $this->construct_TL($this->settings['tl_schema']['src'], [$this, $this->referenceDatabase]);
$this->connect_to_all_dcs(); yield $this->connect_to_all_dcs_async();
$this->datacenter->curdc = 2; $this->datacenter->curdc = 2;
if ((!isset($this->authorization['user']['bot']) || !$this->authorization['user']['bot']) && $this->datacenter->sockets[$this->datacenter->curdc]->temp_auth_key !== null) { if ((!isset($this->authorization['user']['bot']) || !$this->authorization['user']['bot']) && $this->datacenter->sockets[$this->datacenter->curdc]->temp_auth_key !== null) {
try { try {
$nearest_dc = $this->method_call('help.getNearestDc', [], ['datacenter' => $this->datacenter->curdc]); $nearest_dc = yield $this->method_call_async_read('help.getNearestDc', [], ['datacenter' => $this->datacenter->curdc]);
$this->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['nearest_dc'], $nearest_dc['country'], $nearest_dc['nearest_dc']), Logger::NOTICE); $this->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['nearest_dc'], $nearest_dc['country'], $nearest_dc['nearest_dc']), Logger::NOTICE);
if ($nearest_dc['nearest_dc'] != $nearest_dc['this_dc']) { if ($nearest_dc['nearest_dc'] != $nearest_dc['this_dc']) {
$this->settings['connection_settings']['default_dc'] = $this->datacenter->curdc = (int) $nearest_dc['nearest_dc']; $this->settings['connection_settings']['default_dc'] = $this->datacenter->curdc = (int) $nearest_dc['nearest_dc'];
@ -200,10 +206,9 @@ class MTProto implements TLCallback
} }
} }
} }
$this->get_config([], ['datacenter' => $this->datacenter->curdc]); yield $this->get_config_async([], ['datacenter' => $this->datacenter->curdc]);
$this->v = self::V; $this->v = self::V;
return $this->settings;
} }
public function __sleep() public function __sleep()
@ -222,6 +227,10 @@ class MTProto implements TLCallback
} }
public function __wakeup() public function __wakeup()
{
$this->asyncInitPromise = $this->call($this->__async_wakeup());
}
public function __async_wakeup()
{ {
set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']); set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']);
set_exception_handler(['\\danog\\MadelineProto\\Serialization', 'serialize_all']); set_exception_handler(['\\danog\\MadelineProto\\Serialization', 'serialize_all']);
@ -356,12 +365,12 @@ class MTProto implements TLCallback
$this->reset_session(true, true); $this->reset_session(true, true);
$this->config = ['expires' => -1]; $this->config = ['expires' => -1];
$this->dh_config = ['version' => 0]; $this->dh_config = ['version' => 0];
$this->__construct($settings); yield $this->__async_construct($settings);
$force = true; $force = true;
foreach ($this->secret_chats as $chat => $data) { foreach ($this->secret_chats as $chat => $data) {
try { try {
if (isset($this->secret_chats[$chat]) && $this->secret_chats[$chat]['InputEncryptedChat'] !== null) { if (isset($this->secret_chats[$chat]) && $this->secret_chats[$chat]['InputEncryptedChat'] !== null) {
$this->notify_layer($chat); yield $this->notify_layer_async($chat);
} }
} catch (\danog\MadelineProto\RPCErrorException $e) { } catch (\danog\MadelineProto\RPCErrorException $e) {
} }
@ -372,7 +381,7 @@ class MTProto implements TLCallback
$this->channels_state = []; $this->channels_state = [];
$this->got_state = false; $this->got_state = false;
} }
$this->connect_to_all_dcs(); yield $this->connect_to_all_dcs_async();
foreach ($this->calls as $id => $controller) { foreach ($this->calls as $id => $controller) {
if (!is_object($controller)) { if (!is_object($controller)) {
unset($this->calls[$id]); unset($this->calls[$id]);
@ -383,15 +392,15 @@ class MTProto implements TLCallback
$controller->setMadeline($this); $controller->setMadeline($this);
} }
} }
if ($this->get_self()) { if (yield $this->get_self_async()) {
$this->authorized = self::LOGGED_IN; $this->authorized = self::LOGGED_IN;
} }
if ($this->authorized === self::LOGGED_IN) { if ($this->authorized === self::LOGGED_IN) {
$this->get_cdn_config($this->datacenter->curdc); yield $this->get_cdn_config_async($this->datacenter->curdc);
$this->setup_logger(); $this->setup_logger();
} }
if ($this->authorized === self::LOGGED_IN && !$this->authorization['user']['bot'] && $this->settings['peer']['cache_all_peers_on_startup']) { if ($this->authorized === self::LOGGED_IN && !$this->authorization['user']['bot'] && $this->settings['peer']['cache_all_peers_on_startup']) {
$this->get_dialogs($force); yield $this->get_dialogs_async($force);
} }
if ($this->authorized === self::LOGGED_IN && $this->settings['updates']['handle_updates'] && !$this->updates_state['sync_loading']) { if ($this->authorized === self::LOGGED_IN && $this->settings['updates']['handle_updates'] && !$this->updates_state['sync_loading']) {
$this->logger->logger(\danog\MadelineProto\Lang::$current_lang['getupdates_deserialization'], Logger::NOTICE); $this->logger->logger(\danog\MadelineProto\Lang::$current_lang['getupdates_deserialization'], Logger::NOTICE);
@ -709,7 +718,7 @@ class MTProto implements TLCallback
if (!is_array($settings)) { if (!is_array($settings)) {
$settings = []; $settings = [];
} }
$settings = array_replace_recursive($this->array_cast_recursive($default_settings, true), $this->array_cast_recursive($settings, true)); $settings = array_replace_recursive($default_settings, $settings);
if (isset(Lang::$lang[$settings['app_info']['lang_code']])) { if (isset(Lang::$lang[$settings['app_info']['lang_code']])) {
Lang::$current_lang = &Lang::$lang[$settings['app_info']['lang_code']]; Lang::$current_lang = &Lang::$lang[$settings['app_info']['lang_code']];
} }
@ -816,11 +825,6 @@ class MTProto implements TLCallback
} }
// Connects to all datacenters and if necessary creates authorization keys, binds them and writes client info // Connects to all datacenters and if necessary creates authorization keys, binds them and writes client info
public function connect_to_all_dcs()
{
return $this->wait($this->connect_to_all_dcs_async());
}
public function connect_to_all_dcs_async(): \Generator public function connect_to_all_dcs_async(): \Generator
{ {
$this->datacenter->__construct($this, $this->settings['connection'], $this->settings['connection_settings']); $this->datacenter->__construct($this, $this->settings['connection'], $this->settings['connection_settings']);
@ -836,14 +840,17 @@ class MTProto implements TLCallback
} }
yield $dcs; yield $dcs;
yield $this->init_authorization_async(); yield $this->init_authorization_async();
if (!$this->phoneConfigWatcherId) $this->phoneConfigWatcherId = Loop::repeat(24 * 3600 * 1000, [$this, 'get_phone_config_async']); if (!$this->phoneConfigWatcherId) {
$this->phoneConfigWatcherId = Loop::repeat(24 * 3600 * 1000, [$this, 'get_phone_config_async']);
}
yield $this->get_phone_config_async(); yield $this->get_phone_config_async();
$this->logger->logger("Started phone config fetcher"); $this->logger->logger("Started phone config fetcher");
} }
public function get_phone_config_async($watcherId = null) public function get_phone_config_async($watcherId = null)
{ {
if ($this->authorized === self::LOGGED_IN && class_exists('\\danog\\MadelineProto\\VoIPServerConfig') && !$this->authorization['user']['bot']) { if ($this->authorized === self::LOGGED_IN && class_exists('\\danog\\MadelineProto\\VoIPServerConfig') && !$this->authorization['user']['bot']) {
$this->logger->logger("Fetching phone config..."); $this->logger->logger("Fetching phone config...");
VoIPServerConfig::updateDefault(yield $this->method_call_async_read('phone.getCallConfig', [], ['datacenter' => $this->settings['connection_settings']['default_dc']])); VoIPServerConfig::updateDefault(yield $this->method_call_async_read('phone.getCallConfig', [], ['datacenter' => $this->settings['connection_settings']['default_dc']]));
} else { } else {
@ -868,13 +875,17 @@ class MTProto implements TLCallback
} }
public function get_cdn_config($datacenter) public function get_cdn_config($datacenter)
{
return $this->wait($this->get_cdn_config_async($datacenter));
}
public function get_cdn_config_async($datacenter)
{ {
/* /*
* *********************************************************************** * ***********************************************************************
* Fetch RSA keys for CDN datacenters * Fetch RSA keys for CDN datacenters
*/ */
try { try {
foreach ($this->method_call('help.getCdnConfig', [], ['datacenter' => $datacenter])['public_keys'] as $curkey) { foreach ((yield $this->method_call_async_read('help.getCdnConfig', [], ['datacenter' => $datacenter]))['public_keys'] as $curkey) {
$tempkey = new \danog\MadelineProto\RSA($curkey['public_key']); $tempkey = new \danog\MadelineProto\RSA($curkey['public_key']);
$this->rsa_keys[$tempkey->fp] = $tempkey; $this->rsa_keys[$tempkey->fp] = $tempkey;
} }

View File

@ -110,7 +110,11 @@ trait AuthKeyHandler
public function notify_layer($chat) public function notify_layer($chat)
{ {
$this->method_call('messages.sendEncryptedService', ['peer' => $chat, 'message' => ['_' => 'decryptedMessageService', 'action' => ['_' => 'decryptedMessageActionNotifyLayer', 'layer' => $this->encrypted_layer]]], ['datacenter' => $this->datacenter->curdc]); return $this->wait($this->notify_layer_async($chat));
}
public function notify_layer_async($chat)
{
yield $this->method_call_async_read('messages.sendEncryptedService', ['peer' => $chat, 'message' => ['_' => 'decryptedMessageService', 'action' => ['_' => 'decryptedMessageActionNotifyLayer', 'layer' => $this->encrypted_layer]]], ['datacenter' => $this->datacenter->curdc]);
} }
protected $temp_rekeyed_secret_chats = []; protected $temp_rekeyed_secret_chats = [];

File diff suppressed because it is too large Load Diff

View File

@ -62,7 +62,7 @@ trait Events
$this->settings['updates']['handle_updates'] = true; $this->settings['updates']['handle_updates'] = true;
$this->settings['updates']['run_callback'] = true; $this->settings['updates']['run_callback'] = true;
if (isset($this->datacenter->sockets[$this->settings['connection_settings']['default_dc']]->updater)) { if (isset($this->datacenter->sockets[$this->settings['connection_settings']['default_dc']]->updater) && !$this->asyncInitPromise) {
$this->datacenter->sockets[$this->settings['connection_settings']['default_dc']]->updater->start(); $this->datacenter->sockets[$this->settings['connection_settings']['default_dc']]->updater->start();
} }
} }