Major fixes and refactoring of wrapper APIs
This commit is contained in:
parent
46c4e222db
commit
b339a9d849
@ -1,6 +1,6 @@
|
|||||||
language: php
|
language: php
|
||||||
php:
|
php:
|
||||||
- '7.1'
|
- '7.4'
|
||||||
before_install:
|
before_install:
|
||||||
- echo "phar.readonly = 0" >> ~/.phpenv/versions/$(phpenv version-name)/etc/php.ini
|
- echo "phar.readonly = 0" >> ~/.phpenv/versions/$(phpenv version-name)/etc/php.ini
|
||||||
|
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
"vlucas/phpdotenv": "^3",
|
"vlucas/phpdotenv": "^3",
|
||||||
"erusev/parsedown": "^1.6",
|
"erusev/parsedown": "^1.6",
|
||||||
"rollbar/rollbar": "dev-master",
|
"rollbar/rollbar": "dev-master",
|
||||||
"ext-curl": "*",
|
|
||||||
"ext-mbstring": "*",
|
"ext-mbstring": "*",
|
||||||
"ext-json": "*",
|
"ext-json": "*",
|
||||||
"ext-xml": "*",
|
"ext-xml": "*",
|
||||||
@ -28,7 +27,8 @@
|
|||||||
"amphp/websocket-client": "dev-master",
|
"amphp/websocket-client": "dev-master",
|
||||||
"amphp/artax": "^3.0",
|
"amphp/artax": "^3.0",
|
||||||
"amphp/file": "^0.3.5",
|
"amphp/file": "^0.3.5",
|
||||||
"amphp/uri": "^0.1.4"
|
"amphp/uri": "^0.1.4",
|
||||||
|
"amphp/byte-stream": "dev-master#bc191a8 as 1.5"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
"phpdocumentor/reflection-docblock": "^3.1",
|
"phpdocumentor/reflection-docblock": "^3.1",
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
require 'vendor/autoload.php';
|
require 'vendor/autoload.php';
|
||||||
|
|
||||||
$MadelineProto = new \danog\MadelineProto\API('sessionf.madeline');
|
$MadelineProto = new \danog\MadelineProto\API('session.madeline');
|
||||||
$me = $MadelineProto->start();
|
$me = $MadelineProto->start();
|
||||||
|
|
||||||
$me = $MadelineProto->get_self();
|
$me = $MadelineProto->get_self();
|
||||||
|
@ -29,7 +29,7 @@ if (!isset($backtrace[0]["file"]) || !in_array(basename($backtrace[0]["file"]),
|
|||||||
die("madeline.phar cannot be required manually: use the automatic loader, instead: https://docs.madelineproto.xyz/docs/INSTALLATION.html#simple".PHP_EOL);
|
die("madeline.phar cannot be required manually: use the automatic loader, instead: https://docs.madelineproto.xyz/docs/INSTALLATION.html#simple".PHP_EOL);
|
||||||
}
|
}
|
||||||
if (isset($backtrace[1]["file"])) {
|
if (isset($backtrace[1]["file"])) {
|
||||||
chdir(dirname($backtrace[1]["file"]));
|
@chdir(dirname($backtrace[1]["file"]));
|
||||||
}
|
}
|
||||||
if ($contents = file_get_contents("https://phar.madelineproto.xyz/phar.php?v=new")) {
|
if ($contents = file_get_contents("https://phar.madelineproto.xyz/phar.php?v=new")) {
|
||||||
file_put_contents($backtrace[0]["file"], $contents);
|
file_put_contents($backtrace[0]["file"], $contents);
|
||||||
|
@ -92,7 +92,7 @@ class EventHandler extends \danog\MadelineProto\EventHandler
|
|||||||
|
|
||||||
if (file_exists('.env')) {
|
if (file_exists('.env')) {
|
||||||
echo 'Loading .env...'.PHP_EOL;
|
echo 'Loading .env...'.PHP_EOL;
|
||||||
$dotenv = new Dotenv\Dotenv(getcwd());
|
$dotenv = Dotenv\Dotenv::create(getcwd());
|
||||||
$dotenv->load();
|
$dotenv->load();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,6 +19,8 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto;
|
namespace danog\MadelineProto;
|
||||||
|
|
||||||
|
use Amp\Deferred;
|
||||||
|
|
||||||
class API extends APIFactory
|
class API extends APIFactory
|
||||||
{
|
{
|
||||||
use \danog\Serializable;
|
use \danog\Serializable;
|
||||||
@ -29,11 +31,21 @@ class API extends APIFactory
|
|||||||
public $API;
|
public $API;
|
||||||
public $getting_api_id = false;
|
public $getting_api_id = false;
|
||||||
public $my_telegram_org_wrapper;
|
public $my_telegram_org_wrapper;
|
||||||
|
public $asyncAPIPromise;
|
||||||
|
|
||||||
public function __magic_construct($params = [], $settings = [])
|
public function __magic_construct($params = [], $settings = [])
|
||||||
{
|
{
|
||||||
Magic::class_exists();
|
Magic::class_exists();
|
||||||
set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']);
|
set_error_handler(['\\danog\\MadelineProto\\Exception', 'ExceptionErrorHandler']);
|
||||||
|
$deferred = new Deferred;
|
||||||
|
$this->asyncAPIPromise = $deferred->promise();
|
||||||
|
$this->asyncAPIPromise->onResolve(function () {
|
||||||
|
$this->asyncAPIPromise = null;
|
||||||
|
});
|
||||||
|
$this->setInitPromise($this->__construct_async($params, $settings, $deferred));
|
||||||
|
}
|
||||||
|
public function __construct_async($params, $settings, $deferred)
|
||||||
|
{
|
||||||
if (is_string($params)) {
|
if (is_string($params)) {
|
||||||
$realpaths = Serialization::realpaths($params);
|
$realpaths = Serialization::realpaths($params);
|
||||||
$this->session = $realpaths['file'];
|
$this->session = $realpaths['file'];
|
||||||
@ -62,7 +74,7 @@ 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) {
|
||||||
@ -72,11 +84,11 @@ class API extends APIFactory
|
|||||||
if (defined('MADELINEPROTO_TEST') && MADELINEPROTO_TEST === 'pony') {
|
if (defined('MADELINEPROTO_TEST') && MADELINEPROTO_TEST === 'pony') {
|
||||||
throw $e;
|
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) {
|
||||||
@ -96,16 +108,15 @@ class API extends APIFactory
|
|||||||
|
|
||||||
if (isset($unserialized->API)) {
|
if (isset($unserialized->API)) {
|
||||||
$this->API = $unserialized->API;
|
$this->API = $unserialized->API;
|
||||||
$this->callFork((function () {
|
|
||||||
yield $this->API->asyncInitPromise;
|
|
||||||
$this->API->asyncInitPromise = null;
|
|
||||||
$this->APIFactory();
|
|
||||||
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
|
|
||||||
$pong = yield $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();
|
||||||
|
$deferred->resolve();
|
||||||
|
yield $this->API->initAsync();
|
||||||
|
$this->APIFactory();
|
||||||
|
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
|
||||||
|
$this->asyncInitPromise = null;
|
||||||
|
$pong = yield $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);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -113,34 +124,31 @@ class API extends APIFactory
|
|||||||
$params = $settings;
|
$params = $settings;
|
||||||
}
|
}
|
||||||
if (!isset($params['app_info']['api_id']) || !$params['app_info']['api_id']) {
|
if (!isset($params['app_info']['api_id']) || !$params['app_info']['api_id']) {
|
||||||
$app = $this->api_start();
|
$app = yield $this->api_start_async($params);
|
||||||
$params['app_info']['api_id'] = $app['api_id'];
|
$params['app_info']['api_id'] = $app['api_id'];
|
||||||
$params['app_info']['api_hash'] = $app['api_hash'];
|
$params['app_info']['api_hash'] = $app['api_hash'];
|
||||||
}
|
}
|
||||||
$this->API = new MTProto($params);
|
$this->API = new MTProto($params);
|
||||||
|
$this->APIFactory();
|
||||||
|
$deferred->resolve();
|
||||||
\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);
|
||||||
$this->callFork((function () {
|
yield $this->API->initAsync();
|
||||||
yield $this->API->asyncInitPromise;
|
$this->APIFactory();
|
||||||
$this->API->asyncInitPromise = null;
|
$this->asyncInitPromise = null;
|
||||||
$this->APIFactory();
|
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
|
||||||
\danog\MadelineProto\Logger::log('Ping...', Logger::ULTRA_VERBOSE);
|
$pong = yield $this->ping(['ping_id' => 3], ['async' => true]);
|
||||||
$pong = yield $this->ping(['ping_id' => 3], ['async' => true]);
|
\danog\MadelineProto\Logger::log('Pong: '.$pong['ping_id'], Logger::ULTRA_VERBOSE);
|
||||||
\danog\MadelineProto\Logger::log('Pong: ' . $pong['ping_id'], Logger::ULTRA_VERBOSE);
|
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['madelineproto_ready'], Logger::NOTICE);
|
||||||
\danog\MadelineProto\Logger::log(\danog\MadelineProto\Lang::$current_lang['madelineproto_ready'], Logger::NOTICE);
|
|
||||||
})());
|
|
||||||
$this->APIFactory();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function async($async)
|
public function async($async)
|
||||||
{
|
{
|
||||||
$this->async = $async;
|
$this->async = $async;
|
||||||
foreach ($this->API->get_methods_namespaced() as $pair) {
|
|
||||||
$namespace = key($pair);
|
|
||||||
$this->{$namespace}->async = $async;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($this->API->event_handler && class_exists($this->API->event_handler) && is_subclass_of($this->API->event_handler, '\danog\MadelineProto\EventHandler')) {
|
if ($this->API) {
|
||||||
$this->API->setEventHandler($this->API->event_handler);
|
if ($this->API->event_handler && class_exists($this->API->event_handler) && is_subclass_of($this->API->event_handler, '\danog\MadelineProto\EventHandler')) {
|
||||||
|
$this->API->setEventHandler($this->API->event_handler);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -154,7 +162,10 @@ class API extends APIFactory
|
|||||||
if (\danog\MadelineProto\Magic::$has_thread && is_object(\Thread::getCurrentThread()) || Magic::is_fork()) {
|
if (\danog\MadelineProto\Magic::$has_thread && is_object(\Thread::getCurrentThread()) || Magic::is_fork()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$this->serialize();
|
if ($this->asyncInitPromise) {
|
||||||
|
$this->init();
|
||||||
|
}
|
||||||
|
$this->wait($this->serialize());
|
||||||
//restore_error_handler();
|
//restore_error_handler();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,49 +174,6 @@ class API extends APIFactory
|
|||||||
return ['API', 'web_api_template', 'getting_api_id', 'my_telegram_org_wrapper'];
|
return ['API', 'web_api_template', 'getting_api_id', 'my_telegram_org_wrapper'];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function &__get($name)
|
|
||||||
{
|
|
||||||
if ($name === 'settings') {
|
|
||||||
$this->API->setdem = true;
|
|
||||||
|
|
||||||
return $this->API->settings;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->storage[$name];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __set($name, $value)
|
|
||||||
{
|
|
||||||
if ($name === 'settings') {
|
|
||||||
if ($this->API->phoneConfigWatcherId) {
|
|
||||||
$this->wait($this->API->phoneConfigWatcherId);
|
|
||||||
$this->API->phoneConfigWatcherId = null;
|
|
||||||
}
|
|
||||||
if (Magic::is_fork() && !Magic::$processed_fork) {
|
|
||||||
\danog\MadelineProto\Logger::log('Detected fork');
|
|
||||||
$this->API->reset_session();
|
|
||||||
foreach ($this->API->datacenter->sockets as $id => $datacenter) {
|
|
||||||
$this->API->close_and_reopen($id);
|
|
||||||
}
|
|
||||||
Magic::$processed_fork = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->__construct(array_replace_recursive($this->API->settings, $value));
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->storage[$name] = $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __isset($name)
|
|
||||||
{
|
|
||||||
return isset($this->API->storage[$name]);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __unset($name)
|
|
||||||
{
|
|
||||||
unset($this->API->storage[$name]);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function from_camel_case($input)
|
private function from_camel_case($input)
|
||||||
{
|
{
|
||||||
preg_match_all('!([A-Z][A-Z0-9]*(?=$|[A-Z][a-z0-9])|[A-Za-z][a-z0-9]+)!', $input, $matches);
|
preg_match_all('!([A-Z][A-Z0-9]*(?=$|[A-Z][a-z0-9])|[A-Za-z][a-z0-9]+)!', $input, $matches);
|
||||||
@ -221,10 +189,10 @@ class API extends APIFactory
|
|||||||
{
|
{
|
||||||
if ($this->API) {
|
if ($this->API) {
|
||||||
foreach ($this->API->get_method_namespaces() as $namespace) {
|
foreach ($this->API->get_method_namespaces() as $namespace) {
|
||||||
$this->{$namespace} = new APIFactory($namespace, $this->API);
|
$this->{$namespace} = new APIFactory($namespace, $this->API, $this->async);
|
||||||
}
|
}
|
||||||
$methods = get_class_methods($this->API);
|
$methods = get_class_methods($this->API);
|
||||||
foreach ($methods as $key => $method) {
|
foreach ($methods as $method) {
|
||||||
if ($method == 'method_call_async_read') {
|
if ($method == 'method_call_async_read') {
|
||||||
unset($methods[array_search('method_call', $methods)]);
|
unset($methods[array_search('method_call', $methods)]);
|
||||||
} elseif (stripos($method, 'async') !== false) {
|
} elseif (stripos($method, 'async') !== false) {
|
||||||
@ -266,6 +234,9 @@ class API extends APIFactory
|
|||||||
|
|
||||||
public function get_all_methods()
|
public function get_all_methods()
|
||||||
{
|
{
|
||||||
|
if ($this->asyncInitPromise) {
|
||||||
|
$this->init();
|
||||||
|
}
|
||||||
$methods = [];
|
$methods = [];
|
||||||
foreach ($this->API->methods->by_id as $method) {
|
foreach ($this->API->methods->by_id as $method) {
|
||||||
$methods[] = $method['method'];
|
$methods[] = $method['method'];
|
||||||
@ -274,16 +245,64 @@ class API extends APIFactory
|
|||||||
return array_merge($methods, get_class_methods($this->API));
|
return array_merge($methods, get_class_methods($this->API));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function serialize($params = null)
|
public function serialize($filename = null)
|
||||||
{
|
{
|
||||||
if ($params === null) {
|
return $this->callFork((function () use ($filename) {
|
||||||
$params = $this->session;
|
if ($filename === null) {
|
||||||
}
|
$filename = $this->session;
|
||||||
if (empty($params)) {
|
}
|
||||||
return;
|
if (empty($filename)) {
|
||||||
}
|
return;
|
||||||
Logger::log(\danog\MadelineProto\Lang::$current_lang['serializing_madelineproto']);
|
}
|
||||||
|
Logger::log(\danog\MadelineProto\Lang::$current_lang['serializing_madelineproto']);
|
||||||
|
|
||||||
return Serialization::serialize($params, $this);
|
if ($filename == '') {
|
||||||
|
throw new \danog\MadelineProto\Exception('Empty filename');
|
||||||
|
}
|
||||||
|
if (isset($this->API->setdem) && $this->API->setdem) {
|
||||||
|
$this->API->setdem = false;
|
||||||
|
$this->API->__construct($this->API->settings);
|
||||||
|
}
|
||||||
|
if ($this->API === null && !$this->getting_api_id) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if ($this->API && $this->API->asyncInitPromise) {
|
||||||
|
yield $this->API->initAsync();
|
||||||
|
}
|
||||||
|
$this->serialized = time();
|
||||||
|
$realpaths = Serialization::realpaths($filename);
|
||||||
|
if (!file_exists($realpaths['lockfile'])) {
|
||||||
|
touch($realpaths['lockfile']);
|
||||||
|
clearstatcache();
|
||||||
|
}
|
||||||
|
$realpaths['lockfile'] = fopen($realpaths['lockfile'], 'w');
|
||||||
|
\danog\MadelineProto\Logger::log('Waiting for exclusive lock of serialization lockfile...');
|
||||||
|
flock($realpaths['lockfile'], LOCK_EX);
|
||||||
|
\danog\MadelineProto\Logger::log('Lock acquired, serializing');
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (!$this->getting_api_id) {
|
||||||
|
$update_closure = $this->API->settings['updates']['callback'];
|
||||||
|
if ($this->API->settings['updates']['callback'] instanceof \Closure) {
|
||||||
|
$this->API->settings['updates']['callback'] = [$this->API, 'noop'];
|
||||||
|
}
|
||||||
|
$logger_closure = $this->API->settings['logger']['logger_param'];
|
||||||
|
if ($this->API->settings['logger']['logger_param'] instanceof \Closure) {
|
||||||
|
$this->API->settings['logger']['logger_param'] = [$this->API, 'noop'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$wrote = file_put_contents($realpaths['tempfile'], serialize($this));
|
||||||
|
rename($realpaths['tempfile'], $realpaths['file']);
|
||||||
|
} finally {
|
||||||
|
if (!$this->getting_api_id) {
|
||||||
|
$this->API->settings['updates']['callback'] = $update_closure;
|
||||||
|
$this->API->settings['logger']['logger_param'] = $logger_closure;
|
||||||
|
}
|
||||||
|
flock($realpaths['lockfile'], LOCK_UN);
|
||||||
|
fclose($realpaths['lockfile']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $wrote;
|
||||||
|
})());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,8 +20,9 @@
|
|||||||
namespace danog\MadelineProto;
|
namespace danog\MadelineProto;
|
||||||
|
|
||||||
use Amp\Promise;
|
use Amp\Promise;
|
||||||
|
use danog\MadelineProto\Async\AsyncConstruct;
|
||||||
|
|
||||||
class APIFactory
|
class APIFactory extends AsyncConstruct
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
* @internal this is a internal property generated by build_docs.php, don't change manually
|
* @internal this is a internal property generated by build_docs.php, don't change manually
|
||||||
@ -119,126 +120,121 @@ class APIFactory
|
|||||||
public $API;
|
public $API;
|
||||||
public $lua = false;
|
public $lua = false;
|
||||||
public $async = false;
|
public $async = false;
|
||||||
|
public $asyncAPIPromise;
|
||||||
|
|
||||||
protected $methods = [];
|
protected $methods = [];
|
||||||
|
|
||||||
public function __construct($namespace, $API)
|
public function __construct($namespace, $API, &$async)
|
||||||
{
|
{
|
||||||
$this->namespace = $namespace . '.';
|
$this->namespace = $namespace.'.';
|
||||||
$this->API = $API;
|
$this->API = $API;
|
||||||
|
$this->async = &$async;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __call($name, $arguments)
|
public function __call($name, $arguments)
|
||||||
{
|
{
|
||||||
|
$yielded = $this->__call_async($name, $arguments);
|
||||||
|
$async = $this->lua === false && (is_array(end($arguments)) && isset(end($arguments)['async']) ? end($arguments)['async'] : ($this->async && $name !== 'loop'));
|
||||||
|
if ($async) {
|
||||||
|
return $yielded;
|
||||||
|
}
|
||||||
|
if (!$this->lua) {
|
||||||
|
return $this->wait($yielded);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$yielded = $this->wait($yielded);
|
||||||
|
Lua::convert_objects($yielded);
|
||||||
|
|
||||||
|
return $yielded;
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __call_async($name, $arguments)
|
||||||
|
{
|
||||||
|
if ($this->asyncInitPromise) {
|
||||||
|
yield $this->initAsync();
|
||||||
|
}
|
||||||
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();
|
||||||
foreach ($this->API->datacenter->sockets as $id => $datacenter) {
|
foreach ($this->API->datacenter->sockets as $datacenter) {
|
||||||
$this->API->close_and_reopen($id);
|
yield $datacenter->reconnect();
|
||||||
}
|
}
|
||||||
Magic::$processed_fork = true;
|
Magic::$processed_fork = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->API->setdem) {
|
|
||||||
$this->API->setdem = false;
|
|
||||||
$this->API->__construct($this->API->settings);
|
|
||||||
}
|
|
||||||
//$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 ($this->API->setdem) {
|
||||||
if ($name !== 'accept_tos' && $name !== 'decline_tos') {
|
$this->API->setdem = false;
|
||||||
$this->API->check_tos();
|
$this->API->__construct($this->API->settings);
|
||||||
}*/
|
yield $this->API->initAsync();
|
||||||
|
}
|
||||||
|
if ($this->API->asyncInitPromise) {
|
||||||
|
yield $this->API->initAsync();
|
||||||
|
}
|
||||||
|
|
||||||
$lower_name = strtolower($name);
|
$lower_name = strtolower($name);
|
||||||
|
if ($this->namespace !== '' || !isset($this->methods[$lower_name])) {
|
||||||
|
$name = $this->namespace.$name;
|
||||||
|
$aargs = isset($arguments[1]) && is_array($arguments[1]) ? $arguments[1] : [];
|
||||||
|
$aargs['apifactory'] = true;
|
||||||
|
$aargs['datacenter'] = $this->API->datacenter->curdc;
|
||||||
|
$args = isset($arguments[0]) && is_array($arguments[0]) ? $arguments[0] : [];
|
||||||
|
|
||||||
if ($this->lua === false) {
|
return yield $this->API->method_call_async_read($name, $args, $aargs);
|
||||||
return $this->namespace !== '' || !isset($this->methods[$lower_name]) ? $this->__mtproto_call($this->namespace . $name, $arguments) : $this->__api_call($lower_name, $arguments);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
$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);
|
|
||||||
|
|
||||||
return $deserialized;
|
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\TL\Exception $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\NothingInTheSocketException $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\PTSException $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\SecurityException $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
} catch (\danog\MadelineProto\TL\Conversion\Exception $e) {
|
|
||||||
return ['error_code' => $e->getCode(), 'error' => $e->getMessage()];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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);
|
|
||||||
if (is_object($result) && ($result instanceof \Generator || $result instanceof Promise)) {
|
|
||||||
$async = is_array(end($arguments)) && isset(end($arguments)['async']) ? end($arguments)['async'] : ($this->async && $name !== 'loop');
|
|
||||||
if ($async) {
|
|
||||||
return $result;
|
|
||||||
} else {
|
|
||||||
return $this->wait($result);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __mtproto_call($name, $arguments)
|
|
||||||
{
|
|
||||||
$aargs = isset($arguments[1]) && is_array($arguments[1]) ? $arguments[1] : [];
|
|
||||||
$aargs['apifactory'] = true;
|
|
||||||
$args = isset($arguments[0]) && is_array($arguments[0]) ? $arguments[0] : [];
|
|
||||||
|
|
||||||
$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;
|
|
||||||
$aargs['datacenter'] = $this->API->datacenter->curdc;
|
|
||||||
return yield $this->API->method_call_async_read($name, $args, $aargs);
|
|
||||||
;
|
|
||||||
})());
|
|
||||||
} else {
|
|
||||||
$this->wait($this->API->asyncInitPromise);
|
|
||||||
$this->API->asyncInitPromise = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$aargs['datacenter'] = $this->API->datacenter->curdc;
|
|
||||||
$res = $this->API->method_call_async_read($name, $args, $aargs);
|
|
||||||
|
|
||||||
if ($async) {
|
|
||||||
return $res;
|
|
||||||
} else {
|
} else {
|
||||||
return $this->wait($res);
|
return yield $this->methods[$lower_name](...$arguments);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function &__get($name)
|
||||||
|
{
|
||||||
|
if ($this->asyncAPIPromise) {
|
||||||
|
$this->wait($this->asyncAPIPromise);
|
||||||
|
}
|
||||||
|
if ($name === 'settings') {
|
||||||
|
$this->API->setdem = true;
|
||||||
|
|
||||||
|
return $this->API->settings;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->API->storage[$name];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __set($name, $value)
|
||||||
|
{
|
||||||
|
if ($this->asyncAPIPromise) {
|
||||||
|
$this->wait($this->asyncAPIPromise);
|
||||||
|
}
|
||||||
|
if ($name === 'settings') {
|
||||||
|
if ($this->API->asyncInitPromise) {
|
||||||
|
$this->API->init();
|
||||||
|
}
|
||||||
|
return $this->API->__construct(array_replace_recursive($this->API->settings, $value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->API->storage[$name] = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __isset($name)
|
||||||
|
{
|
||||||
|
if ($this->asyncAPIPromise) {
|
||||||
|
$this->wait($this->asyncAPIPromise);
|
||||||
|
}
|
||||||
|
return isset($this->API->storage[$name]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __unset($name)
|
||||||
|
{
|
||||||
|
if ($this->asyncAPIPromise) {
|
||||||
|
$this->wait($this->asyncAPIPromise);
|
||||||
|
}
|
||||||
|
unset($this->API->storage[$name]);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
53
src/danog/MadelineProto/Async/AsyncConstruct.php
Normal file
53
src/danog/MadelineProto/Async/AsyncConstruct.php
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Async constructor abstract class.
|
||||||
|
*
|
||||||
|
* 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-2018 Daniil Gentili <daniil@daniil.it>
|
||||||
|
* @license https://opensource.org/licenses/AGPL-3.0 AGPLv3
|
||||||
|
*
|
||||||
|
* @link https://docs.madelineproto.xyz MadelineProto documentation
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace danog\MadelineProto\Async;
|
||||||
|
|
||||||
|
use danog\MadelineProto\Tools;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Async constructor class.
|
||||||
|
*
|
||||||
|
* Manages asynchronous construction and wakeup of classes
|
||||||
|
*
|
||||||
|
* @author Daniil Gentili <daniil@daniil.it>
|
||||||
|
*/
|
||||||
|
class AsyncConstruct
|
||||||
|
{
|
||||||
|
use Tools;
|
||||||
|
public $asyncInitPromise;
|
||||||
|
public function init()
|
||||||
|
{
|
||||||
|
if ($this->asyncInitPromise) {
|
||||||
|
$this->wait($this->asyncInitPromise);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function initAsync()
|
||||||
|
{
|
||||||
|
if ($this->asyncInitPromise) {
|
||||||
|
yield $this->asyncInitPromise;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function setInitPromise($promise)
|
||||||
|
{
|
||||||
|
$this->asyncInitPromise = $this->call($promise);
|
||||||
|
$this->asyncInitPromise->onResolve(function () {
|
||||||
|
$this->asyncInitPromise = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -94,7 +94,6 @@ class Connection
|
|||||||
return $this->ctx;
|
return $this->ctx;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Connect function.
|
* Connect function.
|
||||||
*
|
*
|
||||||
@ -157,7 +156,6 @@ class Connection
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public function sendMessage($message, $flush = true)
|
public function sendMessage($message, $flush = true)
|
||||||
{
|
{
|
||||||
$deferred = new Deferred();
|
$deferred = new Deferred();
|
||||||
@ -201,6 +199,7 @@ class Connection
|
|||||||
|
|
||||||
public function disconnect()
|
public function disconnect()
|
||||||
{
|
{
|
||||||
|
$this->API->logger->logger("Disconnecting from DC {$this->datacenter}");
|
||||||
$this->old = true;
|
$this->old = true;
|
||||||
foreach (['reader', 'writer', 'checker', 'waiter', 'updater'] as $loop) {
|
foreach (['reader', 'writer', 'checker', 'waiter', 'updater'] as $loop) {
|
||||||
if (isset($this->{$loop}) && $this->{$loop}) {
|
if (isset($this->{$loop}) && $this->{$loop}) {
|
||||||
@ -210,6 +209,7 @@ class Connection
|
|||||||
if ($this->stream) {
|
if ($this->stream) {
|
||||||
$this->stream->disconnect();
|
$this->stream->disconnect();
|
||||||
}
|
}
|
||||||
|
$this->API->logger->logger("Disconnected from DC {$this->datacenter}");
|
||||||
}
|
}
|
||||||
|
|
||||||
public function reconnect(): \Generator
|
public function reconnect(): \Generator
|
||||||
@ -226,12 +226,17 @@ class Connection
|
|||||||
|
|
||||||
$dc_config_number = isset($API->settings['connection_settings'][$datacenter]) ? $datacenter : 'all';
|
$dc_config_number = isset($API->settings['connection_settings'][$datacenter]) ? $datacenter : 'all';
|
||||||
$timeout = $API->settings['connection_settings'][$dc_config_number]['timeout'];
|
$timeout = $API->settings['connection_settings'][$dc_config_number]['timeout'];
|
||||||
|
$pfs = $API->settings['connection_settings'][$dc_config_number]['pfs'];
|
||||||
|
|
||||||
foreach ($this->new_outgoing as $message_id) {
|
foreach ($this->new_outgoing as $message_id) {
|
||||||
if (isset($this->outgoing_messages[$message_id]['sent'])
|
if (isset($this->outgoing_messages[$message_id]['sent'])
|
||||||
&& $this->outgoing_messages[$message_id]['sent'] + $timeout < time()
|
&& $this->outgoing_messages[$message_id]['sent'] + $timeout < time()
|
||||||
&& ($this->temp_auth_key === null) === $this->outgoing_messages[$message_id]['unencrypted']
|
&& ($this->temp_auth_key === null) === $this->outgoing_messages[$message_id]['unencrypted']
|
||||||
&& $this->outgoing_messages[$message_id]['_'] !== 'msgs_state_req'
|
&& $this->outgoing_messages[$message_id]['_'] !== 'msgs_state_req'
|
||||||
) {
|
) {
|
||||||
|
if ($pfs && !isset($this->temp_auth_key['bound']) && $this->outgoing_messages[$message_id]['_'] !== 'auth.bindTempAuthKey') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -239,6 +244,33 @@ class Connection
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getPendingCalls()
|
||||||
|
{
|
||||||
|
$API = $this->API;
|
||||||
|
$datacenter = $this->datacenter;
|
||||||
|
|
||||||
|
$dc_config_number = isset($API->settings['connection_settings'][$datacenter]) ? $datacenter : 'all';
|
||||||
|
$timeout = $API->settings['connection_settings'][$dc_config_number]['timeout'];
|
||||||
|
$pfs = $API->settings['connection_settings'][$dc_config_number]['pfs'];
|
||||||
|
|
||||||
|
$result = [];
|
||||||
|
foreach ($this->new_outgoing as $message_id) {
|
||||||
|
if (isset($this->outgoing_messages[$message_id]['sent'])
|
||||||
|
&& $this->outgoing_messages[$message_id]['sent'] + $timeout < time()
|
||||||
|
&& ($this->temp_auth_key === null) === $this->outgoing_messages[$message_id]['unencrypted']
|
||||||
|
&& $this->outgoing_messages[$message_id]['_'] !== 'msgs_state_req'
|
||||||
|
) {
|
||||||
|
if ($pfs && !isset($this->temp_auth_key['bound']) && $this->outgoing_messages[$message_id]['_'] !== 'auth.bindTempAuthKey') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$result[] = $message_id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
public function getName(): string
|
public function getName(): string
|
||||||
{
|
{
|
||||||
return __CLASS__;
|
return __CLASS__;
|
||||||
|
@ -38,6 +38,7 @@ use danog\MadelineProto\Stream\Transport\DefaultStream;
|
|||||||
use danog\MadelineProto\Stream\Transport\WssStream;
|
use danog\MadelineProto\Stream\Transport\WssStream;
|
||||||
use danog\MadelineProto\Stream\Transport\WsStream;
|
use danog\MadelineProto\Stream\Transport\WsStream;
|
||||||
use danog\MadelineProto\TL\Conversion\Exception;
|
use danog\MadelineProto\TL\Conversion\Exception;
|
||||||
|
use Amp\Artax\Cookie\ArrayCookieJar;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages datacenters.
|
* Manages datacenters.
|
||||||
@ -67,12 +68,13 @@ class DataCenter
|
|||||||
if ($socket instanceof Connection && !strpos($key, '_bk')) {
|
if ($socket instanceof Connection && !strpos($key, '_bk')) {
|
||||||
$this->API->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['dc_con_stop'], $key), \danog\MadelineProto\Logger::VERBOSE);
|
$this->API->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['dc_con_stop'], $key), \danog\MadelineProto\Logger::VERBOSE);
|
||||||
$socket->old = true;
|
$socket->old = true;
|
||||||
|
$socket->setExtra($this->API);
|
||||||
$socket->disconnect();
|
$socket->disconnect();
|
||||||
} else {
|
} else {
|
||||||
unset($this->sockets[$key]);
|
unset($this->sockets[$key]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->HTTPClient = new DefaultClient(null, new HttpSocketPool(new ProxySocketPool($this)));
|
$this->HTTPClient = new DefaultClient(new ArrayCookieJar, new HttpSocketPool(new ProxySocketPool($this)));
|
||||||
}
|
}
|
||||||
public function rawConnectAsync(string $uri, CancellationToken $token = null, ClientConnectContext $ctx = null): \Generator
|
public function rawConnectAsync(string $uri, CancellationToken $token = null, ClientConnectContext $ctx = null): \Generator
|
||||||
{
|
{
|
||||||
@ -170,10 +172,10 @@ class DataCenter
|
|||||||
default:
|
default:
|
||||||
throw new Exception(\danog\MadelineProto\Lang::$current_lang['protocol_invalid']);
|
throw new Exception(\danog\MadelineProto\Lang::$current_lang['protocol_invalid']);
|
||||||
}
|
}
|
||||||
if ($this->settings[$dc_config_number]['obfuscated'] && !in_array($default[1][0], [HttpsStream::getName(), HttpStream::getName()])) {
|
if ($this->settings[$dc_config_number]['obfuscated'] && !in_array($default[2][0], [HttpsStream::getName(), HttpStream::getName()])) {
|
||||||
$default = [[DefaultStream::getName(), []], [BufferedRawStream::getName(), []], [ObfuscatedStream::getName(), []], end($default)];
|
$default = [[DefaultStream::getName(), []], [BufferedRawStream::getName(), []], [ObfuscatedStream::getName(), []], end($default)];
|
||||||
}
|
}
|
||||||
if ($this->settings[$dc_config_number]['transport'] && !in_array($default[1][0], [HttpsStream::getName(), HttpStream::getName()])) {
|
if ($this->settings[$dc_config_number]['transport'] && !in_array($default[2][0], [HttpsStream::getName(), HttpStream::getName()])) {
|
||||||
switch ($this->settings[$dc_config_number]['transport']) {
|
switch ($this->settings[$dc_config_number]['transport']) {
|
||||||
case 'tcp':
|
case 'tcp':
|
||||||
if ($this->settings[$dc_config_number]['obfuscated']) {
|
if ($this->settings[$dc_config_number]['obfuscated']) {
|
||||||
|
@ -24,54 +24,10 @@ class EventHandler extends APIFactory
|
|||||||
public function __construct($MadelineProto)
|
public function __construct($MadelineProto)
|
||||||
{
|
{
|
||||||
$this->API = $MadelineProto->API;
|
$this->API = $MadelineProto->API;
|
||||||
$this->async = $MadelineProto->async;
|
$this->async = &$MadelineProto->async;
|
||||||
$this->methods = $MadelineProto->methods;
|
$this->methods = &$MadelineProto->methods;
|
||||||
foreach ($this->API->get_method_namespaces() as $namespace) {
|
foreach ($this->API->get_method_namespaces() as $namespace) {
|
||||||
$this->{$namespace} = new APIFactory($namespace, $this->API);
|
$this->{$namespace} = new APIFactory($namespace, $this->API, $this->async);
|
||||||
$this->{$namespace}->async = $MadelineProto->async;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function &__get($name)
|
|
||||||
{
|
|
||||||
if ($name === 'settings') {
|
|
||||||
$this->API->setdem = true;
|
|
||||||
|
|
||||||
return $this->API->settings;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->storage[$name];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __set($name, $value)
|
|
||||||
{
|
|
||||||
if ($name === 'settings') {
|
|
||||||
if ($this->API->phoneConfigWatcherId) {
|
|
||||||
$this->wait($this->API->phoneConfigWatcherId);
|
|
||||||
$this->API->phoneConfigWatcherId = null;
|
|
||||||
}
|
|
||||||
if (Magic::is_fork() && !Magic::$processed_fork) {
|
|
||||||
\danog\MadelineProto\Logger::log('Detected fork');
|
|
||||||
$this->API->reset_session();
|
|
||||||
foreach ($this->API->datacenter->sockets as $id => $datacenter) {
|
|
||||||
$this->API->close_and_reopen($id);
|
|
||||||
}
|
|
||||||
Magic::$processed_fork = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->__construct(array_replace_recursive($this->API->settings, $value));
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->API->storage[$name] = $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __isset($name)
|
|
||||||
{
|
|
||||||
return isset($this->API->storage[$name]);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __unset($name)
|
|
||||||
{
|
|
||||||
unset($this->API->storage[$name]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -26,11 +26,7 @@ class Exception extends \Exception
|
|||||||
|
|
||||||
public function __toString()
|
public function __toString()
|
||||||
{
|
{
|
||||||
$result = $this->file === 'MadelineProto' ? $this->message : '\\danog\\MadelineProto\\Exception'.($this->message !== '' ? ': ' : '').$this->message.' in '.$this->file.':'.$this->line.PHP_EOL.\danog\MadelineProto\Magic::$revision.PHP_EOL.'TL Trace (YOU ABSOLUTELY MUST READ THE TEXT BELOW):'.PHP_EOL.$this->getTLTrace();
|
return $this->file === 'MadelineProto' ? $this->message : '\\danog\\MadelineProto\\Exception'.($this->message !== '' ? ': ' : '').$this->message.' in '.$this->file.':'.$this->line.PHP_EOL.\danog\MadelineProto\Magic::$revision.PHP_EOL.'TL Trace (YOU ABSOLUTELY MUST READ THE TEXT BELOW):'.PHP_EOL.$this->getTLTrace();
|
||||||
if (php_sapi_name() !== 'cli') {
|
|
||||||
$result = str_replace(PHP_EOL, '<br>'.PHP_EOL, $result);
|
|
||||||
}
|
|
||||||
return $result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __construct($message = null, $code = 0, self $previous = null, $file = null, $line = null)
|
public function __construct($message = null, $code = 0, self $previous = null, $file = null, $line = null)
|
||||||
|
@ -38,7 +38,9 @@ class CheckLoop extends ResumableSignalLoop
|
|||||||
$this->startedLoop();
|
$this->startedLoop();
|
||||||
$API->logger->logger("Entered check loop in DC {$datacenter}", Logger::ULTRA_VERBOSE);
|
$API->logger->logger("Entered check loop in DC {$datacenter}", Logger::ULTRA_VERBOSE);
|
||||||
|
|
||||||
$timeout = $API->settings['connection_settings'][isset($API->settings['connection_settings'][$datacenter]) ? $datacenter : 'all']['timeout'];
|
$dc_config_number = isset($API->settings['connection_settings'][$datacenter]) ? $datacenter : 'all';
|
||||||
|
|
||||||
|
$timeout = $API->settings['connection_settings'][$dc_config_number]['timeout'];
|
||||||
while (true) {
|
while (true) {
|
||||||
while (empty($connection->new_outgoing)) {
|
while (empty($connection->new_outgoing)) {
|
||||||
if (yield $this->waitSignal($this->pause())) {
|
if (yield $this->waitSignal($this->pause())) {
|
||||||
@ -52,7 +54,7 @@ class CheckLoop extends ResumableSignalLoop
|
|||||||
if ($connection->hasPendingCalls()) {
|
if ($connection->hasPendingCalls()) {
|
||||||
$last_recv = $connection->get_max_id(true);
|
$last_recv = $connection->get_max_id(true);
|
||||||
if ($connection->temp_auth_key !== null) {
|
if ($connection->temp_auth_key !== null) {
|
||||||
$message_ids = array_values($connection->new_outgoing);
|
$message_ids = $connection->getPendingCalls();//array_values($connection->new_outgoing);
|
||||||
$deferred = new Deferred();
|
$deferred = new Deferred();
|
||||||
$deferred->promise()->onResolve(
|
$deferred->promise()->onResolve(
|
||||||
function ($e, $result) use ($message_ids, $API, $connection, $datacenter) {
|
function ($e, $result) use ($message_ids, $API, $connection, $datacenter) {
|
||||||
@ -109,6 +111,7 @@ class CheckLoop extends ResumableSignalLoop
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
$list = '';
|
$list = '';
|
||||||
|
// Don't edit this here pls
|
||||||
foreach ($message_ids as $message_id) {
|
foreach ($message_ids as $message_id) {
|
||||||
$list .= $connection->outgoing_messages[$message_id]['_'].', ';
|
$list .= $connection->outgoing_messages[$message_id]['_'].', ';
|
||||||
}
|
}
|
||||||
@ -126,7 +129,6 @@ class CheckLoop extends ResumableSignalLoop
|
|||||||
}
|
}
|
||||||
$connection->writer->resume();
|
$connection->writer->resume();
|
||||||
}
|
}
|
||||||
//$t = time();
|
|
||||||
if (yield $this->waitSignal($this->pause($timeout))) {
|
if (yield $this->waitSignal($this->pause($timeout))) {
|
||||||
$API->logger->logger("Exiting check loop in DC $datacenter");
|
$API->logger->logger("Exiting check loop in DC $datacenter");
|
||||||
$this->exitedLoop();
|
$this->exitedLoop();
|
||||||
|
@ -107,7 +107,7 @@ class ReadLoop extends SignalLoop
|
|||||||
}
|
}
|
||||||
$this->startedLoop();
|
$this->startedLoop();
|
||||||
if ($this->API->is_http($datacenter)) {
|
if ($this->API->is_http($datacenter)) {
|
||||||
$this->API->datacenter->sockets[$datacenter]->waiter->resume();
|
Loop::defer([$connection->waiter, 'resume']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -118,6 +118,7 @@ class ReadLoop extends SignalLoop
|
|||||||
$datacenter = $this->datacenter;
|
$datacenter = $this->datacenter;
|
||||||
$connection = $this->connection;
|
$connection = $this->connection;
|
||||||
if (isset($this->connection->old)) {
|
if (isset($this->connection->old)) {
|
||||||
|
$API->logger->logger("Not reading because connection is old");
|
||||||
throw new NothingInTheSocketException();
|
throw new NothingInTheSocketException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,7 +35,6 @@ class UpdateLoop extends ResumableSignalLoop
|
|||||||
{
|
{
|
||||||
$API = $this->API;
|
$API = $this->API;
|
||||||
$datacenter = $this->datacenter;
|
$datacenter = $this->datacenter;
|
||||||
$connection = $this->connection;
|
|
||||||
|
|
||||||
if (!$this->API->settings['updates']['handle_updates']) {
|
if (!$this->API->settings['updates']['handle_updates']) {
|
||||||
yield new Success(0);
|
yield new Success(0);
|
||||||
|
@ -18,13 +18,13 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto\Loop\Connection;
|
namespace danog\MadelineProto\Loop\Connection;
|
||||||
|
|
||||||
use Amp\Coroutine;
|
|
||||||
use Amp\Success;
|
use Amp\Success;
|
||||||
use danog\MadelineProto\Connection;
|
use danog\MadelineProto\Connection;
|
||||||
use danog\MadelineProto\Logger;
|
use danog\MadelineProto\Logger;
|
||||||
use danog\MadelineProto\Loop\Impl\ResumableSignalLoop;
|
use danog\MadelineProto\Loop\Impl\ResumableSignalLoop;
|
||||||
use danog\MadelineProto\MTProtoTools\Crypt;
|
use danog\MadelineProto\MTProtoTools\Crypt;
|
||||||
use danog\MadelineProto\Tools;
|
use danog\MadelineProto\Tools;
|
||||||
|
use danog\MadelineProto\Magic;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Socket write loop.
|
* Socket write loop.
|
||||||
@ -44,7 +44,7 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
|
|
||||||
$this->startedLoop();
|
$this->startedLoop();
|
||||||
$API->logger->logger("Entered write loop in DC {$datacenter}", Logger::ULTRA_VERBOSE);
|
$API->logger->logger("Entered write loop in DC {$datacenter}", Logger::ULTRA_VERBOSE);
|
||||||
|
|
||||||
$please_wait = false;
|
$please_wait = false;
|
||||||
while (true) {
|
while (true) {
|
||||||
if (empty($connection->pending_outgoing) || $please_wait) {
|
if (empty($connection->pending_outgoing) || $please_wait) {
|
||||||
@ -101,7 +101,7 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
|
|
||||||
$pad_length = -$length & 15;
|
$pad_length = -$length & 15;
|
||||||
$pad_length += 16 * $this->random_int($modulus = 16);
|
$pad_length += 16 * $this->random_int($modulus = 16);
|
||||||
|
|
||||||
$pad = $this->random($pad_length);
|
$pad = $this->random($pad_length);
|
||||||
$buffer = yield $connection->stream->getWriteBuffer(8 + 8 + 4 + $pad_length + $length);
|
$buffer = yield $connection->stream->getWriteBuffer(8 + 8 + 4 + $pad_length + $length);
|
||||||
|
|
||||||
@ -157,18 +157,16 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($API->is_http($datacenter) && !$has_http_wait) {
|
if ($API->is_http($datacenter) && !$has_http_wait) {
|
||||||
//$connection->pending_outgoing[$connection->pending_outgoing_key++] = ['_' => 'http_wait', 'serialized_body' => $this->API->serialize_object(['type' => ''], ['_' => 'http_wait', 'max_wait' => $API->settings['connection_settings'][$dc_config_number]['timeout'] * 1000 - 100, 'wait_after' => 0, 'max_delay' => 0], 'http_wait'), 'content_related' => true, 'unencrypted' => false, 'method' => true];
|
|
||||||
$connection->pending_outgoing[$connection->pending_outgoing_key++] = ['_' => 'http_wait', 'serialized_body' => yield $this->API->serialize_object_async(['type' => ''], ['_' => 'http_wait', 'max_wait' => 30000, 'wait_after' => 0, 'max_delay' => 1], 'http_wait'), 'content_related' => true, 'unencrypted' => false, 'method' => true];
|
$connection->pending_outgoing[$connection->pending_outgoing_key++] = ['_' => 'http_wait', 'serialized_body' => yield $this->API->serialize_object_async(['type' => ''], ['_' => 'http_wait', 'max_wait' => 30000, 'wait_after' => 0, 'max_delay' => 1], 'http_wait'), 'content_related' => true, 'unencrypted' => false, 'method' => true];
|
||||||
$connection->pending_outgoing_key %= Connection::PENDING_MAX;
|
$connection->pending_outgoing_key %= Connection::PENDING_MAX;
|
||||||
|
|
||||||
$has_http_wait = true;
|
$has_http_wait = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
$total_length = 0;
|
$total_length = 0;
|
||||||
$count = 0;
|
$count = 0;
|
||||||
ksort($connection->pending_outgoing);
|
ksort($connection->pending_outgoing);
|
||||||
|
$skipped = false;
|
||||||
foreach ($connection->pending_outgoing as $k => $message) {
|
foreach ($connection->pending_outgoing as $k => $message) {
|
||||||
if ($message['unencrypted']) {
|
if ($message['unencrypted']) {
|
||||||
continue;
|
continue;
|
||||||
@ -177,8 +175,9 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
unset($connection->pending_outgoing[$k]);
|
unset($connection->pending_outgoing[$k]);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if ($API->settings['connection_settings'][$dc_config_number]['pfs'] && !isset($connection->temp_auth_key['bound']) && !strpos($datacenter, 'cdn') && $message['_'] !== 'auth.bindTempAuthKey') {
|
if ($API->settings['connection_settings'][$dc_config_number]['pfs'] && !isset($connection->temp_auth_key['bound']) && !strpos($datacenter, 'cdn') && !in_array($message['_'], ['http_wait', 'auth.bindTempAuthKey']) && $message['method']) {
|
||||||
$API->logger->logger("Skipping {$message['_']} due to unbound keys in DC {$datacenter}");
|
$API->logger->logger("Skipping {$message['_']} due to unbound keys in DC {$datacenter}");
|
||||||
|
$skipped = true;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,7 +190,6 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
$MTmessage = ['_' => 'MTmessage', 'msg_id' => $message_id, 'body' => $body, 'seqno' => $connection->generate_out_seq_no($message['content_related'])];
|
$MTmessage = ['_' => 'MTmessage', 'msg_id' => $message_id, 'body' => $body, 'seqno' => $connection->generate_out_seq_no($message['content_related'])];
|
||||||
|
|
||||||
if (isset($message['method']) && $message['method'] && $message['_'] !== 'http_wait') {
|
if (isset($message['method']) && $message['method'] && $message['_'] !== 'http_wait') {
|
||||||
|
|
||||||
if ((!isset($connection->temp_auth_key['connection_inited']) || $connection->temp_auth_key['connection_inited'] === false) && $message['_'] !== 'auth.bindTempAuthKey') {
|
if ((!isset($connection->temp_auth_key['connection_inited']) || $connection->temp_auth_key['connection_inited'] === false) && $message['_'] !== 'auth.bindTempAuthKey') {
|
||||||
$API->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['write_client_info'], $message['_']), \danog\MadelineProto\Logger::NOTICE);
|
$API->logger->logger(sprintf(\danog\MadelineProto\Lang::$current_lang['write_client_info'], $message['_']), \danog\MadelineProto\Logger::NOTICE);
|
||||||
$MTmessage['body'] = yield $API->serialize_method_async(
|
$MTmessage['body'] = yield $API->serialize_method_async(
|
||||||
@ -201,15 +199,15 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
'query' => yield $API->serialize_method_async(
|
'query' => yield $API->serialize_method_async(
|
||||||
'initConnection',
|
'initConnection',
|
||||||
[
|
[
|
||||||
'api_id' => $API->settings['app_info']['api_id'],
|
'api_id' => $API->settings['app_info']['api_id'],
|
||||||
'api_hash' => $API->settings['app_info']['api_hash'],
|
'api_hash' => $API->settings['app_info']['api_hash'],
|
||||||
'device_model' => strpos($datacenter, 'cdn') === false ? $API->settings['app_info']['device_model'] : 'n/a',
|
'device_model' => strpos($datacenter, 'cdn') === false ? $API->settings['app_info']['device_model'] : 'n/a',
|
||||||
'system_version' => strpos($datacenter, 'cdn') === false ? $API->settings['app_info']['system_version'] : 'n/a',
|
'system_version' => strpos($datacenter, 'cdn') === false ? $API->settings['app_info']['system_version'] : 'n/a',
|
||||||
'app_version' => $API->settings['app_info']['app_version'],
|
'app_version' => $API->settings['app_info']['app_version'],
|
||||||
'system_lang_code' => $API->settings['app_info']['lang_code'],
|
'system_lang_code' => $API->settings['app_info']['lang_code'],
|
||||||
'lang_code' => $API->settings['app_info']['lang_code'],
|
'lang_code' => $API->settings['app_info']['lang_code'],
|
||||||
'lang_pack' => $API->settings['app_info']['lang_pack'],
|
'lang_pack' => $API->settings['app_info']['lang_pack'],
|
||||||
'query' => $MTmessage['body'],
|
'query' => $MTmessage['body'],
|
||||||
]
|
]
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
@ -230,12 +228,12 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* if ($API->settings['requests']['gzip_encode_if_gt'] !== -1 && ($l = strlen($MTmessage['body'])) > $API->settings['requests']['gzip_encode_if_gt']) {
|
/* if ($API->settings['requests']['gzip_encode_if_gt'] !== -1 && ($l = strlen($MTmessage['body'])) > $API->settings['requests']['gzip_encode_if_gt']) {
|
||||||
if (($g = strlen($gzipped = gzencode($MTmessage['body']))) < $l) {
|
if (($g = strlen($gzipped = gzencode($MTmessage['body']))) < $l) {
|
||||||
$MTmessage['body'] = yield $API->serialize_object_async(['type' => 'gzip_packed'], ['packed_data' => $gzipped], 'gzipped data');
|
$MTmessage['body'] = yield $API->serialize_object_async(['type' => 'gzip_packed'], ['packed_data' => $gzipped], 'gzipped data');
|
||||||
$API->logger->logger('Using GZIP compression for ' . $message['_'] . ', saved ' . ($l - $g) . ' bytes of data, reduced call size by ' . $g * 100 / $l . '%', \danog\MadelineProto\Logger::ULTRA_VERBOSE);
|
$API->logger->logger('Using GZIP compression for ' . $message['_'] . ', saved ' . ($l - $g) . ' bytes of data, reduced call size by ' . $g * 100 / $l . '%', \danog\MadelineProto\Logger::ULTRA_VERBOSE);
|
||||||
}
|
}
|
||||||
unset($gzipped);
|
unset($gzipped);
|
||||||
}*/
|
}*/
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$body_length = strlen($MTmessage['body']);
|
$body_length = strlen($MTmessage['body']);
|
||||||
@ -311,7 +309,7 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
|
|
||||||
if ($has_http_wait) {
|
if ($has_http_wait) {
|
||||||
$connection->last_http_wait = $sent;
|
$connection->last_http_wait = $sent;
|
||||||
} elseif ($API->isAltervista()) {
|
} elseif (Magic::$altervista) {
|
||||||
$connection->last_http_wait = PHP_INT_MAX;
|
$connection->last_http_wait = PHP_INT_MAX;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -332,8 +330,9 @@ class WriteLoop extends ResumableSignalLoop
|
|||||||
}
|
}
|
||||||
|
|
||||||
//if (!empty($connection->pending_outgoing)) $connection->select();
|
//if (!empty($connection->pending_outgoing)) $connection->select();
|
||||||
} while (!empty($connection->pending_outgoing));
|
} while (!empty($connection->pending_outgoing) && !$skipped);
|
||||||
|
|
||||||
$connection->pending_outgoing_key = 0;
|
$connection->pending_outgoing_key = 0;
|
||||||
|
return $skipped;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,11 +26,12 @@ use danog\MadelineProto\Stream\MTProtoTransport\HttpsStream;
|
|||||||
use danog\MadelineProto\Stream\MTProtoTransport\HttpStream;
|
use danog\MadelineProto\Stream\MTProtoTransport\HttpStream;
|
||||||
use danog\MadelineProto\TL\TLCallback;
|
use danog\MadelineProto\TL\TLCallback;
|
||||||
use danog\MadelineProto\MTProtoTools\CombinedUpdatesState;
|
use danog\MadelineProto\MTProtoTools\CombinedUpdatesState;
|
||||||
|
use danog\MadelineProto\Async\AsyncConstruct;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages all of the mtproto stuff.
|
* Manages all of the mtproto stuff.
|
||||||
*/
|
*/
|
||||||
class MTProto implements TLCallback
|
class MTProto extends AsyncConstruct implements TLCallback
|
||||||
{
|
{
|
||||||
use \danog\Serializable;
|
use \danog\Serializable;
|
||||||
use \danog\MadelineProto\MTProtoTools\AckHandler;
|
use \danog\MadelineProto\MTProtoTools\AckHandler;
|
||||||
@ -142,20 +143,18 @@ class MTProto implements TLCallback
|
|||||||
public $setdem = false;
|
public $setdem = false;
|
||||||
public $storage = [];
|
public $storage = [];
|
||||||
private $postpone_updates = false;
|
private $postpone_updates = false;
|
||||||
private $altervista = false;
|
|
||||||
private $supportUser = 0;
|
private $supportUser = 0;
|
||||||
public $referenceDatabase;
|
public $referenceDatabase;
|
||||||
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));
|
$this->setInitPromise($this->__construct_async($settings));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __async_construct($settings = [])
|
public function __construct_async($settings = [])
|
||||||
{
|
{
|
||||||
\danog\MadelineProto\Magic::class_exists();
|
\danog\MadelineProto\Magic::class_exists();
|
||||||
// Parse settings
|
// Parse settings
|
||||||
@ -215,7 +214,6 @@ class MTProto implements TLCallback
|
|||||||
}
|
}
|
||||||
yield $this->get_config_async([], ['datacenter' => $this->datacenter->curdc]);
|
yield $this->get_config_async([], ['datacenter' => $this->datacenter->curdc]);
|
||||||
$this->v = self::V;
|
$this->v = self::V;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __sleep()
|
public function __sleep()
|
||||||
@ -225,7 +223,7 @@ class MTProto implements TLCallback
|
|||||||
|
|
||||||
public function isAltervista()
|
public function isAltervista()
|
||||||
{
|
{
|
||||||
return $this->altervista;
|
return Magic::$altervista;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isInitingAuthorization()
|
public function isInitingAuthorization()
|
||||||
@ -236,9 +234,9 @@ class MTProto implements TLCallback
|
|||||||
public function __wakeup()
|
public function __wakeup()
|
||||||
{
|
{
|
||||||
$backtrace = debug_backtrace(DEBUG_BACKTRACE_PROVIDE_OBJECT, 3);
|
$backtrace = debug_backtrace(DEBUG_BACKTRACE_PROVIDE_OBJECT, 3);
|
||||||
$this->asyncInitPromise = $this->call($this->__async_wakeup($backtrace));
|
$this->setInitPromise($this->__wakeup_async($backtrace));
|
||||||
}
|
}
|
||||||
public function __async_wakeup($backtrace)
|
public function __wakeup_async($backtrace)
|
||||||
{
|
{
|
||||||
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']);
|
||||||
@ -267,7 +265,6 @@ class MTProto implements TLCallback
|
|||||||
$this->referenceDatabase = new ReferenceDatabase($this);
|
$this->referenceDatabase = new ReferenceDatabase($this);
|
||||||
}
|
}
|
||||||
$this->update_callbacks([$this, $this->referenceDatabase]);
|
$this->update_callbacks([$this, $this->referenceDatabase]);
|
||||||
$this->altervista = isset($_SERVER['SERVER_ADMIN']) && strpos($_SERVER['SERVER_ADMIN'], 'altervista.org');
|
|
||||||
|
|
||||||
$this->settings['connection_settings']['all']['ipv6'] = \danog\MadelineProto\Magic::$ipv6;
|
$this->settings['connection_settings']['all']['ipv6'] = \danog\MadelineProto\Magic::$ipv6;
|
||||||
/*if (isset($this->settings['pwr']['update_handler']) && $this->settings['pwr']['update_handler'] === $this->settings['updates']['callback']) {
|
/*if (isset($this->settings['pwr']['update_handler']) && $this->settings['pwr']['update_handler'] === $this->settings['updates']['callback']) {
|
||||||
@ -374,7 +371,7 @@ 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];
|
||||||
yield $this->__async_construct($settings);
|
yield $this->__construct_async($settings);
|
||||||
$force = true;
|
$force = true;
|
||||||
foreach ($this->secret_chats as $chat => $data) {
|
foreach ($this->secret_chats as $chat => $data) {
|
||||||
try {
|
try {
|
||||||
@ -548,7 +545,6 @@ class MTProto implements TLCallback
|
|||||||
$app_version = '4.9.1 (13613)';
|
$app_version = '4.9.1 (13613)';
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->altervista = isset($_SERVER['SERVER_ADMIN']) && strpos($_SERVER['SERVER_ADMIN'], 'altervista.org');
|
|
||||||
// Set default settings
|
// Set default settings
|
||||||
$default_settings = ['authorization' => [
|
$default_settings = ['authorization' => [
|
||||||
// Authorization settings
|
// Authorization settings
|
||||||
@ -615,7 +611,7 @@ class MTProto implements TLCallback
|
|||||||
// connection settings
|
// connection settings
|
||||||
'all' => [
|
'all' => [
|
||||||
// These settings will be applied on every datacenter that hasn't a custom settings subarray...
|
// These settings will be applied on every datacenter that hasn't a custom settings subarray...
|
||||||
'protocol' => $this->altervista ? 'http' : 'tcp_abridged',
|
'protocol' => Magic::$altervista ? 'http' : 'tcp_abridged',
|
||||||
// can be tcp_full, tcp_abridged, tcp_intermediate, http, https, obfuscated2, udp (unsupported)
|
// can be tcp_full, tcp_abridged, tcp_intermediate, http, https, obfuscated2, udp (unsupported)
|
||||||
'test_mode' => false,
|
'test_mode' => false,
|
||||||
// decides whether to connect to the main telegram servers or to the testing servers (deep telegram)
|
// decides whether to connect to the main telegram servers or to the testing servers (deep telegram)
|
||||||
@ -623,9 +619,9 @@ class MTProto implements TLCallback
|
|||||||
// decides whether to use ipv6, ipv6 attribute of API attribute of API class contains autodetected boolean
|
// decides whether to use ipv6, ipv6 attribute of API attribute of API class contains autodetected boolean
|
||||||
'timeout' => 2,
|
'timeout' => 2,
|
||||||
// timeout for sockets
|
// timeout for sockets
|
||||||
'proxy' => $this->altervista ? '\\HttpProxy' : '\\Socket',
|
'proxy' => Magic::$altervista ? '\\HttpProxy' : '\\Socket',
|
||||||
// The proxy class to use
|
// The proxy class to use
|
||||||
'proxy_extra' => $this->altervista ? ['address' => 'localhost', 'port' => 80] : [],
|
'proxy_extra' => Magic::$altervista ? ['address' => 'localhost', 'port' => 80] : [],
|
||||||
// Extra parameters to pass to the proxy class using setExtra
|
// Extra parameters to pass to the proxy class using setExtra
|
||||||
'obfuscated' => false,
|
'obfuscated' => false,
|
||||||
'transport' => 'tcp',
|
'transport' => 'tcp',
|
||||||
|
@ -790,15 +790,6 @@ trait PeerHandler
|
|||||||
{
|
{
|
||||||
$settings = isset($this->settings['connection_settings'][$this->datacenter->curdc]) ? $this->settings['connection_settings'][$this->datacenter->curdc] : $this->settings['connection_settings']['all'];
|
$settings = isset($this->settings['connection_settings'][$this->datacenter->curdc]) ? $this->settings['connection_settings'][$this->datacenter->curdc] : $this->settings['connection_settings']['all'];
|
||||||
if (!isset($this->settings['pwr']) || $this->settings['pwr']['pwr'] === false || $settings['test_mode']) {
|
if (!isset($this->settings['pwr']) || $this->settings['pwr']['pwr'] === false || $settings['test_mode']) {
|
||||||
/*
|
|
||||||
try {
|
|
||||||
if (isset($res['username'])) {
|
|
||||||
shell_exec('curl '.escapeshellarg('https://api.pwrtelegram.xyz/getchat?chat_id=@'.$res['username']).' -s -o /dev/null >/dev/null 2>/dev/null & ');
|
|
||||||
}
|
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
|
||||||
$this->logger->logger([$e->getMessage());
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!empty($res)) {
|
if (!empty($res)) {
|
||||||
|
File diff suppressed because one or more lines are too long
@ -18,6 +18,8 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto;
|
namespace danog\MadelineProto;
|
||||||
|
|
||||||
|
use Amp\Artax\Request;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Wrapper for my.telegram.org.
|
* Wrapper for my.telegram.org.
|
||||||
*/
|
*/
|
||||||
@ -25,119 +27,98 @@ class MyTelegramOrgWrapper
|
|||||||
{
|
{
|
||||||
private $logged = false;
|
private $logged = false;
|
||||||
private $hash = '';
|
private $hash = '';
|
||||||
|
private $token;
|
||||||
|
private $number;
|
||||||
|
private $creation_hash;
|
||||||
|
private $settings;
|
||||||
const MY_TELEGRAM_URL = 'https://my.telegram.org';
|
const MY_TELEGRAM_URL = 'https://my.telegram.org';
|
||||||
|
|
||||||
public function __construct($number)
|
public function __sleep()
|
||||||
{
|
{
|
||||||
if (!extension_loaded('curl')) {
|
return ['logged', 'hash', 'token', 'number', 'creation_hash', 'settings'];
|
||||||
throw new Exception(['extension', 'curl']);
|
}
|
||||||
|
public function __construct($settings)
|
||||||
|
{
|
||||||
|
if (!isset($settings['all'])) {
|
||||||
|
$settings['connection_settings'] = ['all' => [
|
||||||
|
// These settings will be applied on every datacenter that hasn't a custom settings subarray...
|
||||||
|
'protocol' => Magic::$altervista ? 'http' : 'tcp_abridged',
|
||||||
|
// can be tcp_full, tcp_abridged, tcp_intermediate, http, https, obfuscated2, udp (unsupported)
|
||||||
|
'test_mode' => false,
|
||||||
|
// decides whether to connect to the main telegram servers or to the testing servers (deep telegram)
|
||||||
|
'ipv6' => \danog\MadelineProto\Magic::$ipv6,
|
||||||
|
// decides whether to use ipv6, ipv6 attribute of API attribute of API class contains autodetected boolean
|
||||||
|
'timeout' => 2,
|
||||||
|
// timeout for sockets
|
||||||
|
'proxy' => Magic::$altervista ? '\\HttpProxy' : '\\Socket',
|
||||||
|
// The proxy class to use
|
||||||
|
'proxy_extra' => Magic::$altervista ? ['address' => 'localhost', 'port' => 80] : [],
|
||||||
|
// Extra parameters to pass to the proxy class using setExtra
|
||||||
|
'obfuscated' => false,
|
||||||
|
'transport' => 'tcp',
|
||||||
|
'pfs' => extension_loaded('gmp'),
|
||||||
|
],
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
$this->settings = $settings;
|
||||||
|
$this->__wakeup();
|
||||||
|
}
|
||||||
|
public function __wakeup()
|
||||||
|
{
|
||||||
|
$this->datacenter = new DataCenter(
|
||||||
|
new class($this->settings)
|
||||||
|
{
|
||||||
|
public function __construct($settings)
|
||||||
|
{
|
||||||
|
$this->logger = new Logger(
|
||||||
|
isset($settings['logger']['logger']) ? $settings['logger']['logger'] : php_sapi_name() === 'cli' ? 3 : 2,
|
||||||
|
isset($settings['logger']['logger_param']) ? $settings['logger']['logger_param'] : Magic::$script_cwd.'/MadelineProto.log',
|
||||||
|
isset($settings['logger']['logger_level']) ? $settings['logger']['logger_level'] : Logger::VERBOSE,
|
||||||
|
isset($settings['logger']['max_size']) ? $settings['logger']['max_size'] : 100 * 1024 * 1024);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
$this->settings['connection_settings']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
public function login_async($number)
|
||||||
|
{
|
||||||
$this->number = $number;
|
$this->number = $number;
|
||||||
$ch = curl_init();
|
$request = new Request(self::MY_TELEGRAM_URL.'/auth/send_password', 'POST');
|
||||||
|
$request = $request->withBody(http_build_query(['phone' => $number]));
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/auth/send_password');
|
$request = $request->withHeaders($this->getHeaders('origin'));
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query(['phone' => $number]));
|
$result = yield $response->getBody();
|
||||||
curl_setopt($ch, CURLOPT_POST, 1);
|
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
|
||||||
|
|
||||||
$headers = $this->get_headers('origin', []);
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error: '.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
$resulta = json_decode($result, true);
|
$resulta = json_decode($result, true);
|
||||||
|
|
||||||
if (!isset($resulta['random_hash'])) {
|
if (!isset($resulta['random_hash'])) {
|
||||||
throw new Exception($result);
|
throw new Exception($result);
|
||||||
}
|
}
|
||||||
$this->hash = $resulta['random_hash'];
|
$this->hash = $resulta['random_hash'];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public function complete_login_async($password)
|
||||||
* Function for generating curl request headers.
|
|
||||||
*/
|
|
||||||
private function get_headers($httpType, $cookies)
|
|
||||||
{
|
|
||||||
// Common header flags.
|
|
||||||
$headers = [];
|
|
||||||
$headers[] = 'Dnt: 1';
|
|
||||||
$headers[] = 'Connection: keep-alive';
|
|
||||||
$headers[] = 'Accept-Language: it-IT,it;q=0.8,en-US;q=0.6,en;q=0.4';
|
|
||||||
$headers[] = 'User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36';
|
|
||||||
|
|
||||||
// Add additional headers based on the type of request.
|
|
||||||
switch ($httpType) {
|
|
||||||
case 'origin':
|
|
||||||
$headers[] = 'Origin: '.self::MY_TELEGRAM_URL;
|
|
||||||
$headers[] = 'Accept-Encoding: gzip, deflate, br';
|
|
||||||
$headers[] = 'Content-Type: application/x-www-form-urlencoded; charset=UTF-8';
|
|
||||||
$headers[] = 'Accept: application/json, text/javascript, */*; q=0.01';
|
|
||||||
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL.'/auth';
|
|
||||||
$headers[] = 'X-Requested-With: XMLHttpRequest';
|
|
||||||
break;
|
|
||||||
case 'refer':
|
|
||||||
$headers[] = 'Accept-Encoding: gzip, deflate, sdch, br';
|
|
||||||
$headers[] = 'Upgrade-Insecure-Requests: 1';
|
|
||||||
$headers[] = 'Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8';
|
|
||||||
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL;
|
|
||||||
$headers[] = 'Cache-Control: max-age=0';
|
|
||||||
break;
|
|
||||||
case 'app':
|
|
||||||
$headers[] = 'Origin: '.self::MY_TELEGRAM_URL;
|
|
||||||
$headers[] = 'Accept-Encoding: gzip, deflate, br';
|
|
||||||
$headers[] = 'Content-Type: application/x-www-form-urlencoded; charset=UTF-8';
|
|
||||||
$headers[] = 'Accept: */*';
|
|
||||||
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL.'/apps';
|
|
||||||
$headers[] = 'X-Requested-With: XMLHttpRequest';
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add every cookie to the header.
|
|
||||||
foreach ($cookies as $cookie) {
|
|
||||||
$headers[] = 'Cookie: '.$cookie;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $headers;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function complete_login($password)
|
|
||||||
{
|
{
|
||||||
if ($this->logged) {
|
if ($this->logged) {
|
||||||
throw new Exception('Already logged in!');
|
throw new Exception('Already logged in!');
|
||||||
}
|
}
|
||||||
$ch = curl_init();
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/auth/login');
|
$request = new Request(self::MY_TELEGRAM_URL.'/auth/login', 'POST');
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$request = $request->withBody(http_build_query(['phone' => $this->number, 'random_hash' => $this->hash, 'password' => $password]));
|
||||||
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query(['phone' => $this->number, 'random_hash' => $this->hash, 'password' => $password]));
|
$request = $request->withHeaders($this->getHeaders('origin'));
|
||||||
curl_setopt($ch, CURLOPT_POST, 1);
|
$request = $request->withHeader('user-agent', 'Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13');
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
|
$result = yield $response->getBody();
|
||||||
|
|
||||||
$headers = $this->get_headers('origin', []);
|
switch ($result) {
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
|
|
||||||
curl_setopt($ch, CURLOPT_USERAGENT, 'Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13');
|
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
|
||||||
curl_setopt($ch, CURLOPT_HEADER, 1);
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error: '.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
|
|
||||||
list($response_headers, $response_content) = preg_split('/(\r\n){2}/', $result, 2);
|
|
||||||
switch ($response_content) {
|
|
||||||
case 'true':
|
case 'true':
|
||||||
//Logger::log(['Login OK'], Logger::VERBOSE);
|
//Logger::log(['Login OK'], Logger::VERBOSE);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Exception($response_content);
|
throw new Exception($result);
|
||||||
}
|
}
|
||||||
$this->token = explode(';', explode('stel_token=', $response_headers)[1])[0];
|
|
||||||
|
$this->token = explode(';', explode('stel_token=', $response->getHeader('Set-Cookie'))[1])[0];
|
||||||
|
|
||||||
return $this->logged = true;
|
return $this->logged = true;
|
||||||
}
|
}
|
||||||
@ -147,27 +128,17 @@ class MyTelegramOrgWrapper
|
|||||||
return $this->logged;
|
return $this->logged;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function has_app()
|
public function has_app_async()
|
||||||
{
|
{
|
||||||
if (!$this->logged) {
|
if (!$this->logged) {
|
||||||
throw new Exception('Not logged in!');
|
throw new Exception('Not logged in!');
|
||||||
}
|
}
|
||||||
$ch = curl_init();
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/apps');
|
$request = new Request(self::MY_TELEGRAM_URL.'/apps');
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$request = $request->withHeaders($this->getHeaders('refer'));
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
|
$result = yield $response->getBody();
|
||||||
|
|
||||||
$cookies = [];
|
|
||||||
array_push($cookies, 'stel_token='.$this->token);
|
|
||||||
$headers = $this->get_headers('refer', $cookies);
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error: '.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
$title = explode('</title>', explode('<title>', $result)[1])[0];
|
$title = explode('</title>', explode('<title>', $result)[1])[0];
|
||||||
switch ($title) {
|
switch ($title) {
|
||||||
case 'App configuration':return true;
|
case 'App configuration':return true;
|
||||||
@ -180,27 +151,16 @@ class MyTelegramOrgWrapper
|
|||||||
throw new Exception($title);
|
throw new Exception($title);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get_app()
|
public function get_app_async()
|
||||||
{
|
{
|
||||||
if (!$this->logged) {
|
if (!$this->logged) {
|
||||||
throw new Exception('Not logged in!');
|
throw new Exception('Not logged in!');
|
||||||
}
|
}
|
||||||
$ch = curl_init();
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/apps');
|
$request = new Request(self::MY_TELEGRAM_URL.'/apps');
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$request = $request->withHeaders($this->getHeaders('refer'));
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
|
$result = yield $response->getBody();
|
||||||
$cookies = [];
|
|
||||||
array_push($cookies, 'stel_token='.$this->token);
|
|
||||||
$headers = $this->get_headers('refer', $cookies);
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error: '.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
|
|
||||||
$cose = explode('<label for="app_id" class="col-md-4 text-right control-label">App api_id:</label>
|
$cose = explode('<label for="app_id" class="col-md-4 text-right control-label">App api_id:</label>
|
||||||
<div class="col-md-7">
|
<div class="col-md-7">
|
||||||
@ -216,58 +176,29 @@ class MyTelegramOrgWrapper
|
|||||||
return ['api_id' => (int) $api_id, 'api_hash' => $api_hash];
|
return ['api_id' => (int) $api_id, 'api_hash' => $api_hash];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function create_app($settings)
|
public function create_app_async($settings)
|
||||||
{
|
{
|
||||||
if (!$this->logged) {
|
if (!$this->logged) {
|
||||||
throw new Exception('Not logged in!');
|
throw new Exception('Not logged in!');
|
||||||
}
|
}
|
||||||
if ($this->has_app()) {
|
if (yield $this->has_app_async()) {
|
||||||
throw new Exception('The app was already created!');
|
throw new Exception('The app was already created!');
|
||||||
}
|
}
|
||||||
|
|
||||||
$ch = curl_init();
|
$request = new Request(self::MY_TELEGRAM_URL.'/apps/create', 'POST');
|
||||||
|
$request = $request->withHeaders($this->getHeaders('app'));
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/apps/create');
|
$request = $request->withBody(http_build_query(['hash' => $this->creation_hash, 'app_title' => $settings['app_title'], 'app_shortname' => $settings['app_shortname'], 'app_url' => $settings['app_url'], 'app_platform' => $settings['app_platform'], 'app_desc' => $settings['app_desc']]));
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query(['hash' => $this->creation_hash, 'app_title' => $settings['app_title'], 'app_shortname' => $settings['app_shortname'], 'app_url' => $settings['app_url'], 'app_platform' => $settings['app_platform'], 'app_desc' => $settings['app_desc']]));
|
$result = yield $response->getBody();
|
||||||
curl_setopt($ch, CURLOPT_POST, 1);
|
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
|
||||||
|
|
||||||
$cookies = [];
|
|
||||||
array_push($cookies, 'stel_token='.$this->token);
|
|
||||||
$headers = $this->get_headers('app', $cookies);
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error:'.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
|
|
||||||
if ($result) {
|
if ($result) {
|
||||||
throw new Exception($result);
|
throw new Exception($result);
|
||||||
}
|
}
|
||||||
|
|
||||||
$ch = curl_init();
|
$request = new Request(self::MY_TELEGRAM_URL.'/apps');
|
||||||
|
$request = $request->withHeaders($this->getHeaders('refer'));
|
||||||
curl_setopt($ch, CURLOPT_URL, self::MY_TELEGRAM_URL.'/apps');
|
$response = yield $this->datacenter->getHTTPClient()->request($request);
|
||||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
$result = yield $response->getBody();
|
||||||
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'GET');
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_ENCODING, 'gzip, deflate');
|
|
||||||
|
|
||||||
$cookies = [];
|
|
||||||
array_push($cookies, 'stel_token='.$this->token);
|
|
||||||
$headers = $this->get_headers('refer', $cookies);
|
|
||||||
|
|
||||||
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
|
|
||||||
|
|
||||||
$result = curl_exec($ch);
|
|
||||||
if (curl_errno($ch)) {
|
|
||||||
throw new Exception('Curl error:'.curl_error($ch));
|
|
||||||
}
|
|
||||||
curl_close($ch);
|
|
||||||
|
|
||||||
$title = explode('</title>', explode('<title>', $result)[1])[0];
|
$title = explode('</title>', explode('<title>', $result)[1])[0];
|
||||||
if ($title === 'Create new application') {
|
if ($title === 'Create new application') {
|
||||||
@ -289,4 +220,53 @@ class MyTelegramOrgWrapper
|
|||||||
|
|
||||||
return ['api_id' => (int) $api_id, 'api_hash' => $api_hash];
|
return ['api_id' => (int) $api_id, 'api_hash' => $api_hash];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function for generating curl request headers.
|
||||||
|
*/
|
||||||
|
private function getHeaders($httpType)
|
||||||
|
{
|
||||||
|
// Common header flags.
|
||||||
|
$headers = [];
|
||||||
|
$headers[] = 'Dnt: 1';
|
||||||
|
$headers[] = 'Connection: keep-alive';
|
||||||
|
$headers[] = 'Accept-Language: it-IT,it;q=0.8,en-US;q=0.6,en;q=0.4';
|
||||||
|
$headers[] = 'User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/57.0.2987.133 Safari/537.36';
|
||||||
|
|
||||||
|
// Add additional headers based on the type of request.
|
||||||
|
switch ($httpType) {
|
||||||
|
case 'origin':
|
||||||
|
$headers[] = 'Origin: '.self::MY_TELEGRAM_URL;
|
||||||
|
//$headers[] = 'Accept-Encoding: gzip, deflate, br';
|
||||||
|
$headers[] = 'Content-Type: application/x-www-form-urlencoded; charset=UTF-8';
|
||||||
|
$headers[] = 'Accept: application/json, text/javascript, */*; q=0.01';
|
||||||
|
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL.'/auth';
|
||||||
|
$headers[] = 'X-Requested-With: XMLHttpRequest';
|
||||||
|
break;
|
||||||
|
case 'refer':
|
||||||
|
//$headers[] = 'Accept-Encoding: gzip, deflate, sdch, br';
|
||||||
|
$headers[] = 'Upgrade-Insecure-Requests: 1';
|
||||||
|
$headers[] = 'Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8';
|
||||||
|
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL;
|
||||||
|
$headers[] = 'Cache-Control: max-age=0';
|
||||||
|
break;
|
||||||
|
case 'app':
|
||||||
|
$headers[] = 'Origin: '.self::MY_TELEGRAM_URL;
|
||||||
|
//$headers[] = 'Accept-Encoding: gzip, deflate, br';
|
||||||
|
$headers[] = 'Content-Type: application/x-www-form-urlencoded; charset=UTF-8';
|
||||||
|
$headers[] = 'Accept: */*';
|
||||||
|
$headers[] = 'Referer: '.self::MY_TELEGRAM_URL.'/apps';
|
||||||
|
$headers[] = 'X-Requested-With: XMLHttpRequest';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$final_headers = [];
|
||||||
|
foreach ($headers as $header) {
|
||||||
|
list($key, $value) = explode(':', $header, 2);
|
||||||
|
$final_headers[trim($key)] = trim($value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $final_headers;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -29,11 +29,6 @@ class Serialization
|
|||||||
echo $exception.PHP_EOL;
|
echo $exception.PHP_EOL;
|
||||||
|
|
||||||
return;
|
return;
|
||||||
foreach (self::$instances as $instance) {
|
|
||||||
if (isset($instance->session)) {
|
|
||||||
$instance->serialize();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function realpaths($file)
|
public static function realpaths($file)
|
||||||
@ -42,68 +37,4 @@ class Serialization
|
|||||||
|
|
||||||
return ['file' => $file, 'lockfile' => $file.'.lock', 'tempfile' => $file.'.temp.session'];
|
return ['file' => $file, 'lockfile' => $file.'.lock', 'tempfile' => $file.'.temp.session'];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Serialize API class.
|
|
||||||
*
|
|
||||||
* @param string $filename the dump file
|
|
||||||
* @param API $instance
|
|
||||||
* @param bool $force
|
|
||||||
*
|
|
||||||
* @return number
|
|
||||||
*/
|
|
||||||
public static function serialize($filename, $instance, $force = false)
|
|
||||||
{
|
|
||||||
if ($filename == '') {
|
|
||||||
throw new \danog\MadelineProto\Exception('Empty filename');
|
|
||||||
}
|
|
||||||
if (isset($instance->API->setdem) && $instance->API->setdem) {
|
|
||||||
$instance->API->setdem = false;
|
|
||||||
$instance->API->__construct($instance->API->settings);
|
|
||||||
}
|
|
||||||
if ($instance->API === null && !$instance->getting_api_id) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if ($instance->API && $instance->API->asyncInitPromise) {
|
|
||||||
return $instance->call((static function () use ($filename, $instance, $force) {
|
|
||||||
yield $instance->API->asyncInitPromise;
|
|
||||||
$instance->API->asyncInitPromise = null;
|
|
||||||
return self::serialize($filename, $instance, $force);
|
|
||||||
})());
|
|
||||||
}
|
|
||||||
$instance->serialized = time();
|
|
||||||
$realpaths = self::realpaths($filename);
|
|
||||||
if (!file_exists($realpaths['lockfile'])) {
|
|
||||||
touch($realpaths['lockfile']);
|
|
||||||
clearstatcache();
|
|
||||||
}
|
|
||||||
$realpaths['lockfile'] = fopen($realpaths['lockfile'], 'w');
|
|
||||||
\danog\MadelineProto\Logger::log('Waiting for exclusive lock of serialization lockfile...');
|
|
||||||
flock($realpaths['lockfile'], LOCK_EX);
|
|
||||||
\danog\MadelineProto\Logger::log('Lock acquired, serializing');
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (!$instance->getting_api_id) {
|
|
||||||
$update_closure = $instance->API->settings['updates']['callback'];
|
|
||||||
if ($instance->API->settings['updates']['callback'] instanceof \Closure) {
|
|
||||||
$instance->API->settings['updates']['callback'] = [$instance->API, 'noop'];
|
|
||||||
}
|
|
||||||
$logger_closure = $instance->API->settings['logger']['logger_param'];
|
|
||||||
if ($instance->API->settings['logger']['logger_param'] instanceof \Closure) {
|
|
||||||
$instance->API->settings['logger']['logger_param'] = [$instance->API, 'noop'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$wrote = file_put_contents($realpaths['tempfile'], serialize($instance));
|
|
||||||
rename($realpaths['tempfile'], $realpaths['file']);
|
|
||||||
} finally {
|
|
||||||
if (!$instance->getting_api_id) {
|
|
||||||
$instance->API->settings['updates']['callback'] = $update_closure;
|
|
||||||
$instance->API->settings['logger']['logger_param'] = $logger_closure;
|
|
||||||
}
|
|
||||||
flock($realpaths['lockfile'], LOCK_UN);
|
|
||||||
fclose($realpaths['lockfile']);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $wrote;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,6 @@ class BufferedRawStream implements \danog\MadelineProto\Stream\BufferedStreamInt
|
|||||||
protected $memory_stream;
|
protected $memory_stream;
|
||||||
private $append = '';
|
private $append = '';
|
||||||
private $append_after = 0;
|
private $append_after = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Asynchronously connect to a TCP/TLS server.
|
* Asynchronously connect to a TCP/TLS server.
|
||||||
*
|
*
|
||||||
@ -178,7 +177,6 @@ class BufferedRawStream implements \danog\MadelineProto\Stream\BufferedStreamInt
|
|||||||
$chunk = yield $this->read();
|
$chunk = yield $this->read();
|
||||||
if ($chunk === null) {
|
if ($chunk === null) {
|
||||||
$this->disconnect();
|
$this->disconnect();
|
||||||
|
|
||||||
throw new \danog\MadelineProto\NothingInTheSocketException();
|
throw new \danog\MadelineProto\NothingInTheSocketException();
|
||||||
}
|
}
|
||||||
fwrite($this->memory_stream, $chunk);
|
fwrite($this->memory_stream, $chunk);
|
||||||
|
@ -252,6 +252,7 @@ trait Tools
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
return $promise;
|
||||||
}
|
}
|
||||||
public function rethrow($e)
|
public function rethrow($e)
|
||||||
{
|
{
|
||||||
|
@ -19,30 +19,34 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto\Wrappers;
|
namespace danog\MadelineProto\Wrappers;
|
||||||
|
|
||||||
|
use function Amp\ByteStream\getStdin;
|
||||||
|
use function Amp\ByteStream\getStdout;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages simple logging in and out.
|
* Manages simple logging in and out.
|
||||||
*/
|
*/
|
||||||
trait ApiStart
|
trait ApiStart
|
||||||
{
|
{
|
||||||
public function api_start()
|
public function api_start_async($settings)
|
||||||
{
|
{
|
||||||
if (php_sapi_name() === 'cli') {
|
if (php_sapi_name() === 'cli') {
|
||||||
if (!function_exists('readline')) {
|
$stdin = getStdin();
|
||||||
$readline = function ($prompt = null) {
|
$stdout = getStdout();
|
||||||
if ($prompt) {
|
$readline = function ($prompt = null) use ($stdout, $stdin) {
|
||||||
echo $prompt;
|
if ($prompt) {
|
||||||
}
|
yield $stdout->write($prompt);
|
||||||
$fp = fopen('php://stdin', 'r');
|
}
|
||||||
$line = rtrim(fgets($fp, 1024));
|
static $lines = [''];
|
||||||
|
while (count($lines) < 2 && ($chunk = yield $stdin->read()) !== null) {
|
||||||
return $line;
|
$chunk = explode("\n", str_replace(["\r", "\n\n"], "\n", $chunk));
|
||||||
};
|
$lines[count($lines) - 1] .= array_shift($chunk);
|
||||||
} else {
|
$lines = array_merge($lines, $chunk);
|
||||||
$readline = 'readline';
|
}
|
||||||
}
|
return array_shift($lines);
|
||||||
|
};
|
||||||
echo 'You did not define a valid API ID/API hash. Do you want to define it now manually, or automatically? (m/a)
|
echo 'You did not define a valid API ID/API hash. Do you want to define it now manually, or automatically? (m/a)
|
||||||
Note that you can also provide the API parameters directly in the code using the settings: https://docs.madelineproto.xyz/docs/SETTINGS.html#settingsapp_infoapi_id'.PHP_EOL;
|
Note that you can also provide the API parameters directly in the code using the settings: https://docs.madelineproto.xyz/docs/SETTINGS.html#settingsapp_infoapi_id'.PHP_EOL;
|
||||||
if (strpos($res = $readline('Your choice (m/a): '), 'm') !== false) {
|
if (strpos(yield $readline('Your choice (m/a): '), 'm') !== false) {
|
||||||
echo '1) Login to my.telegram.org
|
echo '1) Login to my.telegram.org
|
||||||
2) Go to API development tools
|
2) Go to API development tools
|
||||||
3) App title: your app\'s name, can be anything
|
3) App title: your app\'s name, can be anything
|
||||||
@ -51,21 +55,22 @@ Note that you can also provide the API parameters directly in the code using the
|
|||||||
Platform: anything
|
Platform: anything
|
||||||
Description: Describe your app here
|
Description: Describe your app here
|
||||||
4) Click on create application'.PHP_EOL;
|
4) Click on create application'.PHP_EOL;
|
||||||
$app['api_id'] = $readline('5) Enter your API ID: ');
|
$app['api_id'] = yield $readline('5) Enter your API ID: ');
|
||||||
$app['api_hash'] = $readline('6) Enter your API hash: ');
|
$app['api_hash'] = yield $readline('6) Enter your API hash: ');
|
||||||
|
|
||||||
return $app;
|
return $app;
|
||||||
} else {
|
} else {
|
||||||
$this->my_telegram_org_wrapper = new \danog\MadelineProto\MyTelegramOrgWrapper($readline('Enter a phone number that is already registered on Telegram: '));
|
$this->my_telegram_org_wrapper = new \danog\MadelineProto\MyTelegramOrgWrapper($settings);
|
||||||
$this->my_telegram_org_wrapper->complete_login($readline('Enter the verification code you received in telegram: '));
|
yield $this->my_telegram_org_wrapper->login_async(yield $readline('Enter a phone number that is already registered on Telegram: '));
|
||||||
if (!$this->my_telegram_org_wrapper->has_app()) {
|
yield $this->my_telegram_org_wrapper->complete_login_async(yield $readline('Enter the verification code you received in telegram: '));
|
||||||
$app_title = $readline('Enter the app\'s name, can be anything: ');
|
if (!yield $this->my_telegram_org_wrapper->has_app_async()) {
|
||||||
$short_name = $readline('Enter the app\'s short name, can be anything: ');
|
$app_title = yield $readline('Enter the app\'s name, can be anything: ');
|
||||||
$url = $readline('Enter the app/website\'s URL, or t.me/yourusername: ');
|
$short_name = yield $readline('Enter the app\'s short name, can be anything: ');
|
||||||
$description = $readline('Describe your app: ');
|
$url = yield $readline('Enter the app/website\'s URL, or t.me/yourusername: ');
|
||||||
$app = $this->my_telegram_org_wrapper->create_app(['app_title' => $app_title, 'app_shortname' => $short_name, 'app_url' => $short_name, 'app_platform' => 'web', 'app_desc' => $description]);
|
$description = yield $readline('Describe your app: ');
|
||||||
|
$app = yield $this->my_telegram_org_wrapper->create_app_async(['app_title' => $app_title, 'app_shortname' => $short_name, 'app_url' => $url, 'app_platform' => 'web', 'app_desc' => $description]);
|
||||||
} else {
|
} else {
|
||||||
$app = $this->my_telegram_org_wrapper->get_app();
|
$app = yield $this->my_telegram_org_wrapper->get_app_async();
|
||||||
}
|
}
|
||||||
|
|
||||||
return $app;
|
return $app;
|
||||||
@ -80,69 +85,78 @@ Note that you can also provide the API parameters directly in the code using the
|
|||||||
|
|
||||||
return $app;
|
return $app;
|
||||||
} elseif (isset($_POST['phone_number'])) {
|
} elseif (isset($_POST['phone_number'])) {
|
||||||
$this->web_api_phone_login();
|
yield $this->web_api_phone_login_async($settings);
|
||||||
} else {
|
} else {
|
||||||
$this->web_api_echo();
|
yield $this->web_api_echo_async();
|
||||||
}
|
}
|
||||||
} elseif (!$this->my_telegram_org_wrapper->logged_in()) {
|
} elseif (!$this->my_telegram_org_wrapper->logged_in()) {
|
||||||
if (isset($_POST['code'])) {
|
if (isset($_POST['code'])) {
|
||||||
$this->web_api_complete_login();
|
yield $this->web_api_complete_login_async();
|
||||||
if ($this->my_telegram_org_wrapper->has_app()) {
|
if (yield $this->my_telegram_org_wrapper->has_app_async()) {
|
||||||
return $this->my_telegram_org_wrapper->get_app();
|
return yield $this->my_telegram_org_wrapper->get_app_async();
|
||||||
}
|
}
|
||||||
$this->web_api_echo();
|
yield $this->web_api_echo_async();
|
||||||
|
} else if (isset($_POST['api_id']) && isset($_POST['api_hash'])) {
|
||||||
|
$app['api_id'] = (int) $_POST['api_id'];
|
||||||
|
$app['api_hash'] = $_POST['api_hash'];
|
||||||
|
$this->getting_api_id = false;
|
||||||
|
|
||||||
|
return $app;
|
||||||
|
} elseif (isset($_POST['phone_number'])) {
|
||||||
|
yield $this->web_api_phone_login_async($settings);
|
||||||
} else {
|
} else {
|
||||||
$this->web_api_echo("You didn't provide a phone code!");
|
$this->my_telegram_org_wrapper = null;
|
||||||
|
yield $this->web_api_echo_async();
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (isset($_POST['app_title'], $_POST['app_shortname'], $_POST['app_url'], $_POST['app_platform'], $_POST['app_desc'])) {
|
if (isset($_POST['app_title'], $_POST['app_shortname'], $_POST['app_url'], $_POST['app_platform'], $_POST['app_desc'])) {
|
||||||
$app = $this->web_api_create_app();
|
$app = yield $this->web_api_create_app_async();
|
||||||
$this->getting_api_id = false;
|
$this->getting_api_id = false;
|
||||||
|
|
||||||
return $app;
|
return $app;
|
||||||
} else {
|
} else {
|
||||||
$this->web_api_echo("You didn't provide all of the required parameters!");
|
yield $this->web_api_echo_async("You didn't provide all of the required parameters!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
$this->asyncInitPromise = null;
|
||||||
exit;
|
exit;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function web_api_phone_login()
|
public function web_api_phone_login_async($settings)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$this->my_telegram_org_wrapper = new \danog\MadelineProto\MyTelegramOrgWrapper($_POST['phone_number']);
|
$this->my_telegram_org_wrapper = new \danog\MadelineProto\MyTelegramOrgWrapper($settings);
|
||||||
$this->web_api_echo();
|
yield $this->my_telegram_org_wrapper->login_async($_POST['phone_number']);
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
yield $this->web_api_echo_async();
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().'. Try again.');
|
} catch (\Throwable $e) {
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
yield $this->web_api_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().'. Try again.');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function web_api_complete_login()
|
public function web_api_complete_login_async()
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$this->my_telegram_org_wrapper->complete_login($_POST['code']);
|
yield $this->my_telegram_org_wrapper->complete_login_async($_POST['code']);
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_api_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_api_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function web_api_create_app()
|
public function web_api_create_app_async()
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$params = $_POST;
|
$params = $_POST;
|
||||||
unset($params['creating_app']);
|
unset($params['creating_app']);
|
||||||
$app = $this->my_telegram_org_wrapper->create_app($params);
|
$app = yield $this->my_telegram_org_wrapper->create_app_async($params);
|
||||||
|
|
||||||
return $app;
|
return $app;
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().' Try again.');
|
yield $this->web_api_echo_async('ERROR: '.$e->getMessage().' Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_api_echo('ERROR: '.$e->getMessage().' Try again.');
|
yield $this->web_api_echo_async('ERROR: '.$e->getMessage().' Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,8 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto\Wrappers;
|
namespace danog\MadelineProto\Wrappers;
|
||||||
|
|
||||||
|
use function Amp\ByteStream\getOutput;
|
||||||
|
|
||||||
trait ApiTemplates
|
trait ApiTemplates
|
||||||
{
|
{
|
||||||
private $web_api_template = '<!DOCTYPE html>
|
private $web_api_template = '<!DOCTYPE html>
|
||||||
@ -51,12 +53,13 @@ trait ApiTemplates
|
|||||||
$this->web_template = $template;
|
$this->web_template = $template;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function web_api_echo($message = '')
|
public function web_api_echo_async($message = '')
|
||||||
{
|
{
|
||||||
|
$stdout = getOutput();
|
||||||
if (!isset($this->my_telegram_org_wrapper)) {
|
if (!isset($this->my_telegram_org_wrapper)) {
|
||||||
if (isset($_POST['type'])) {
|
if (isset($_POST['type'])) {
|
||||||
if ($_POST['type'] === 'manual') {
|
if ($_POST['type'] === 'manual') {
|
||||||
echo $this->web_api_echo_template('Enter your API ID and API hash<br><b>'.$message.'</b><ol>
|
yield $stdout->write($this->web_api_echo_template('Enter your API ID and API hash<br><b>'.$message.'</b><ol>
|
||||||
<li>Login to my.telegram.org</li>
|
<li>Login to my.telegram.org</li>
|
||||||
<li>Go to API development tools</li>
|
<li>Go to API development tools</li>
|
||||||
<li>
|
<li>
|
||||||
@ -68,18 +71,21 @@ trait ApiTemplates
|
|||||||
</ul>
|
</ul>
|
||||||
</li>
|
</li>
|
||||||
<li>Click on create application</li>
|
<li>Click on create application</li>
|
||||||
</ol>', '<input type="string" name="api_id" placeholder="API ID" required/><input type="string" name="api_hash" placeholder="API hash" required/>');
|
</ol>', '<input type="string" name="api_id" placeholder="API ID" required/><input type="string" name="api_hash" placeholder="API hash" required/>'));
|
||||||
} else {
|
} else {
|
||||||
echo $this->web_api_echo_template('Enter your phone number<br><b>'.$message.'</b>', '<input type="text" name="phone_number" placeholder="Phone number" required/>');
|
yield $stdout->write($this->web_api_echo_template('Enter your phone number<br><b>'.$message.'</b>', '<input type="text" name="phone_number" placeholder="Phone number" required/>'));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
echo $this->web_api_echo_template('Do you want to enter the API id and the API hash manually or automatically?<br>Note that you can also provide it directly in the code using the <a href="https://docs.madelineproto.xyz/docs/SETTINGS.html#settingsapp_infoapi_id">settings</a>.<b>'.$message.'</b>', '<select name="type"><option value="automatic">Automatically</option><option value="manual">Manually</option></select>');
|
if ($message) {
|
||||||
|
$message = '<br><br>'.$message;
|
||||||
|
}
|
||||||
|
yield $stdout->write($this->web_api_echo_template('Do you want to enter the API id and the API hash manually or automatically?<br>Note that you can also provide it directly in the code using the <a href="https://docs.madelineproto.xyz/docs/SETTINGS.html#settingsapp_infoapi_id">settings</a>.<b>'.$message.'</b>', '<select name="type"><option value="automatic">Automatically</option><option value="manual">Manually</option></select>'));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!$this->my_telegram_org_wrapper->logged_in()) {
|
if (!$this->my_telegram_org_wrapper->logged_in()) {
|
||||||
echo $this->web_api_echo_template('Enter your code<br><b>'.$message.'</b>', '<input type="text" name="code" placeholder="Code" required/>');
|
yield $stdout->write($this->web_api_echo_template('Enter your code<br><b>'.$message.'</b>', '<input type="text" name="code" placeholder="Code" required/>'));
|
||||||
} else {
|
} else {
|
||||||
echo $this->web_api_echo_template(
|
yield $stdout->write($this->web_api_echo_template(
|
||||||
'Enter the API info<br><b>'.$message.'</b>',
|
'Enter the API info<br><b>'.$message.'</b>',
|
||||||
'<input type="hidden" name="creating_app" value="yes" required/>
|
'<input type="hidden" name="creating_app" value="yes" required/>
|
||||||
Enter the app name, can be anything: <br><input type="text" name="app_title" required/><br>
|
Enter the app name, can be anything: <br><input type="text" name="app_title" required/><br>
|
||||||
@ -111,7 +117,7 @@ trait ApiTemplates
|
|||||||
<input type="radio" name="app_platform" value="other"> Other (specify in description)
|
<input type="radio" name="app_platform" value="other"> Other (specify in description)
|
||||||
</label>
|
</label>
|
||||||
<br><br>Enter the app description, can be anything: <br><textarea name="app_desc" required></textarea><br><br>
|
<br><br>Enter the app description, can be anything: <br><textarea name="app_desc" required></textarea><br><br>
|
||||||
');
|
'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,9 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto\Wrappers;
|
namespace danog\MadelineProto\Wrappers;
|
||||||
|
|
||||||
|
use function Amp\ByteStream\getStdin;
|
||||||
|
use function Amp\ByteStream\getStdout;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Manages simple logging in and out.
|
* Manages simple logging in and out.
|
||||||
*/
|
*/
|
||||||
@ -30,29 +33,30 @@ trait Start
|
|||||||
return yield $this->get_self_async();
|
return yield $this->get_self_async();
|
||||||
}
|
}
|
||||||
if (php_sapi_name() === 'cli') {
|
if (php_sapi_name() === 'cli') {
|
||||||
if (!function_exists('readline')) {
|
$stdin = getStdin();
|
||||||
$readline = function ($prompt = null) {
|
$stdout = getStdout();
|
||||||
if ($prompt) {
|
$readline = function ($prompt = null) use ($stdout, $stdin) {
|
||||||
echo $prompt;
|
if ($prompt) {
|
||||||
}
|
yield $stdout->write($prompt);
|
||||||
$fp = fopen('php://stdin', 'r');
|
}
|
||||||
$line = rtrim(fgets($fp, 1024));
|
static $lines = [''];
|
||||||
|
while (count($lines) < 2 && ($chunk = yield $stdin->read()) !== null) {
|
||||||
return $line;
|
$chunk = explode("\n", str_replace(["\r", "\n\n"], "\n", $chunk));
|
||||||
};
|
$lines[count($lines) - 1] .= array_shift($chunk);
|
||||||
|
$lines = array_merge($lines, $chunk);
|
||||||
|
}
|
||||||
|
return array_shift($lines);
|
||||||
|
};
|
||||||
|
if (strpos(yield $readline('Do you want to login as user or bot (u/b)? '), 'b') !== false) {
|
||||||
|
yield $this->bot_login_async(yield $readline('Enter your bot token: '));
|
||||||
} else {
|
} else {
|
||||||
$readline = 'readline';
|
yield $this->phone_login_async(yield $readline('Enter your phone number: '));
|
||||||
}
|
$authorization = yield $this->complete_phone_login_async(yield $readline('Enter the phone code: '));
|
||||||
if (strpos($readline('Do you want to login as user or bot (u/b)? '), 'b') !== false) {
|
|
||||||
yield $this->bot_login_async($readline('Enter your bot token: '));
|
|
||||||
} else {
|
|
||||||
yield $this->phone_login_async($readline('Enter your phone number: '));
|
|
||||||
$authorization = yield $this->complete_phone_login_async($readline('Enter the phone code: '));
|
|
||||||
if ($authorization['_'] === 'account.password') {
|
if ($authorization['_'] === 'account.password') {
|
||||||
$authorization = yield $this->complete_2fa_login_async($readline('Please enter your password (hint '.$authorization['hint'].'): '));
|
$authorization = yield $this->complete_2fa_login_async(yield $readline('Please enter your password (hint '.$authorization['hint'].'): '));
|
||||||
}
|
}
|
||||||
if ($authorization['_'] === 'account.needSignup') {
|
if ($authorization['_'] === 'account.needSignup') {
|
||||||
$authorization = yield $this->complete_signup_async($readline('Please enter your first name: '), $readline('Please enter your last name (can be empty): '));
|
$authorization = yield $this->complete_signup_async(yield $readline('Please enter your first name: '), yield $readline('Please enter your last name (can be empty): '));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->serialize();
|
$this->serialize();
|
||||||
@ -65,25 +69,25 @@ trait Start
|
|||||||
} elseif (isset($_POST['token'])) {
|
} elseif (isset($_POST['token'])) {
|
||||||
yield $this->web_bot_login_async();
|
yield $this->web_bot_login_async();
|
||||||
} else {
|
} else {
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
}
|
}
|
||||||
} elseif ($this->authorized === self::WAITING_CODE) {
|
} elseif ($this->authorized === self::WAITING_CODE) {
|
||||||
if (isset($_POST['phone_code'])) {
|
if (isset($_POST['phone_code'])) {
|
||||||
yield $this->web_complete_phone_login_async();
|
yield $this->web_complete_phone_login_async();
|
||||||
} else {
|
} else {
|
||||||
$this->web_echo("You didn't provide a phone code!");
|
yield $this->web_echo_async("You didn't provide a phone code!");
|
||||||
}
|
}
|
||||||
} elseif ($this->authorized === self::WAITING_PASSWORD) {
|
} elseif ($this->authorized === self::WAITING_PASSWORD) {
|
||||||
if (isset($_POST['password'])) {
|
if (isset($_POST['password'])) {
|
||||||
yield $this->web_complete_2fa_login_async();
|
yield $this->web_complete_2fa_login_async();
|
||||||
} else {
|
} else {
|
||||||
$this->web_echo("You didn't provide the password!");
|
yield $this->web_echo_async("You didn't provide the password!");
|
||||||
}
|
}
|
||||||
} elseif ($this->authorized === self::WAITING_SIGNUP) {
|
} elseif ($this->authorized === self::WAITING_SIGNUP) {
|
||||||
if (isset($_POST['first_name'])) {
|
if (isset($_POST['first_name'])) {
|
||||||
yield $this->web_complete_signup_async();
|
yield $this->web_complete_signup_async();
|
||||||
} else {
|
} else {
|
||||||
$this->web_echo("You didn't provide the first name!");
|
yield $this->web_echo_async("You didn't provide the first name!");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ($this->authorized === self::LOGGED_IN) {
|
if ($this->authorized === self::LOGGED_IN) {
|
||||||
@ -99,11 +103,11 @@ trait Start
|
|||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
yield $this->phone_login_async($_POST['phone_number']);
|
yield $this->phone_login_async($_POST['phone_number']);
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,11 +115,11 @@ trait Start
|
|||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
yield $this->complete_phone_login_async($_POST['phone_code']);
|
yield $this->complete_phone_login_async($_POST['phone_code']);
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -123,11 +127,11 @@ trait Start
|
|||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
yield $this->complete_2fa_login_async($_POST['password']);
|
yield $this->complete_2fa_login_async($_POST['password']);
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -135,11 +139,11 @@ trait Start
|
|||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
yield $this->complete_signup_async($_POST['first_name'], isset($_POST['last_name']) ? $_POST['last_name'] : '');
|
yield $this->complete_signup_async($_POST['first_name'], isset($_POST['last_name']) ? $_POST['last_name'] : '');
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -147,11 +151,11 @@ trait Start
|
|||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
yield $this->bot_login_async($_POST['token']);
|
yield $this->bot_login_async($_POST['token']);
|
||||||
$this->web_echo();
|
yield $this->web_echo_async();
|
||||||
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
} catch (\danog\MadelineProto\RPCErrorException $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
} catch (\danog\MadelineProto\Exception $e) {
|
} catch (\danog\MadelineProto\Exception $e) {
|
||||||
$this->web_echo('ERROR: '.$e->getMessage().'. Try again.');
|
yield $this->web_echo_async('ERROR: '.$e->getMessage().'. Try again.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,33 +19,36 @@
|
|||||||
|
|
||||||
namespace danog\MadelineProto\Wrappers;
|
namespace danog\MadelineProto\Wrappers;
|
||||||
|
|
||||||
|
use function Amp\ByteStream\getOutput;
|
||||||
|
|
||||||
trait Templates
|
trait Templates
|
||||||
{
|
{
|
||||||
public function web_echo($message = '')
|
public function web_echo_async($message = '')
|
||||||
{
|
{
|
||||||
|
$stdout = getOutput();
|
||||||
switch ($this->authorized) {
|
switch ($this->authorized) {
|
||||||
case self::NOT_LOGGED_IN:
|
case self::NOT_LOGGED_IN:
|
||||||
if (isset($_POST['type'])) {
|
if (isset($_POST['type'])) {
|
||||||
if ($_POST['type'] === 'phone') {
|
if ($_POST['type'] === 'phone') {
|
||||||
echo $this->web_echo_template('Enter your phone number<br><b>'.$message.'</b>', '<input type="text" name="phone_number" placeholder="Phone number" required/>');
|
yield $stdout->write($this->web_echo_template('Enter your phone number<br><b>'.$message.'</b>', '<input type="text" name="phone_number" placeholder="Phone number" required/>'));
|
||||||
} else {
|
} else {
|
||||||
echo $this->web_echo_template('Enter your bot token<br><b>'.$message.'</b>', '<input type="text" name="token" placeholder="Bot token" required/>');
|
yield $stdout->write($this->web_echo_template('Enter your bot token<br><b>'.$message.'</b>', '<input type="text" name="token" placeholder="Bot token" required/>'));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
echo $this->web_echo_template('Do you want to login as user or bot?<br><b>'.$message.'</b>', '<select name="type"><option value="phone">User</option><option value="bot">Bot</option></select>');
|
yield $stdout->write($this->web_echo_template('Do you want to login as user or bot?<br><b>'.$message.'</b>', '<select name="type"><option value="phone">User</option><option value="bot">Bot</option></select>'));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case self::WAITING_CODE:
|
case self::WAITING_CODE:
|
||||||
echo $this->web_echo_template('Enter your code<br><b>'.$message.'</b>', '<input type="text" name="phone_code" placeholder="Phone code" required/>');
|
yield $stdout->write($this->web_echo_template('Enter your code<br><b>'.$message.'</b>', '<input type="text" name="phone_code" placeholder="Phone code" required/>'));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case self::WAITING_PASSWORD:
|
case self::WAITING_PASSWORD:
|
||||||
echo $this->web_echo_template('Enter your password<br><b>'.$message.'</b>', '<input type="password" name="password" placeholder="Hint: '.$this->authorization['hint'].'" required/>');
|
yield $stdout->write($this->web_echo_template('Enter your password<br><b>'.$message.'</b>', '<input type="password" name="password" placeholder="Hint: '.$this->authorization['hint'].'" required/>'));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case self::WAITING_SIGNUP:
|
case self::WAITING_SIGNUP:
|
||||||
echo $this->web_echo_template('Sign up please<br><b>'.$message.'</b>', '<input type="text" name="first_name" placeholder="First name" required/><input type="text" name="last_name" placeholder="Last name"/>');
|
yield $stdout->write($this->web_echo_template('Sign up please<br><b>'.$message.'</b>', '<input type="text" name="first_name" placeholder="First name" required/><input type="text" name="last_name" placeholder="Last name"/>'));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -43,7 +43,6 @@ $settings = json_decode(getenv('MTPROTO_SETTINGS'), true) ?: [];
|
|||||||
*/
|
*/
|
||||||
echo 'Loading MadelineProto...'.PHP_EOL;
|
echo 'Loading MadelineProto...'.PHP_EOL;
|
||||||
$MadelineProto = new \danog\MadelineProto\API(getcwd().'/testing.madeline', $settings);
|
$MadelineProto = new \danog\MadelineProto\API(getcwd().'/testing.madeline', $settings);
|
||||||
var_dump($MadelineProto->get_dialogs_full());
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$MadelineProto->get_self();
|
$MadelineProto->get_self();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user