2017-07-31 19:17:10 +02:00
#!/usr/bin/env php
< ? php
2019-05-12 21:03:13 +02:00
2017-07-31 19:17:10 +02:00
/*
2019-05-31 12:18:10 +02:00
Copyright 2016 - 2019 Daniil Gentili
2017-07-31 19:17:10 +02:00
( https :// daniil . it )
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 />.
2019-03-29 20:25:42 +01:00
*/
2019-06-04 14:55:58 +02:00
set_include_path ( get_include_path () . ':' . realpath ( dirname ( __FILE__ ) . '/MadelineProto/' ));
2017-08-18 13:27:44 +02:00
2019-06-04 14:55:58 +02:00
if ( ! file_exists ( __DIR__ . '/vendor/autoload.php' )) {
echo 'You did not run composer update, using madeline.php' . PHP_EOL ;
2018-03-29 23:06:40 +02:00
if ( ! file_exists ( 'madeline.php' )) {
copy ( 'https://phar.madelineproto.xyz/madeline.php' , 'madeline.php' );
}
include 'madeline.php' ;
} else {
require_once 'vendor/autoload.php' ;
}
Merge alpha into master (async, huge bugfixes and more) (#546)
* Implement async and lots of bugfixes
* Implement more async
* Implement async, implement bugfixes for the connection module, for the datacenter module, huge bugfixes, huge perfomance improvements, media DCs for https, advanced selecting, custom var_dump, totally rewritten IOLoop and response mechanism, promises, improvements to the TL parser, custom mb_substr
* Apply fixes from StyleCI
* Bugfixes
* Apply fixes from StyleCI
* Bugfixes, implement combined promises
* Apply fixes from StyleCI
* Support passing method arguments as callable
* Starting to write async upload logic
* Apply fixes from StyleCI
* Start implementing async file upload
* Apply fixes from StyleCI
* bugfix
* Apply fixes from StyleCI
* Start rewriting connection module
* Add PHP file docblocks for all classes
* Start working on new async stream API
* Finish writing stream API
* More stream API fixes
* Apply fixes from StyleCI
* Rewrite DataCenter and Connection modules
* Clean up stream API documentation
* Fixes
* Apply fixes from StyleCI
* Add referenced parameter to get length of buffer to read in getReadBuffer API
* Moved all MessageHandler code in the Connection module, added a PHP version warning in the phar
* Start fixing reads
* Fix all protocol stream wrappers
* Apply fixes from StyleCI
* Implement disconnection, and remove end function
* Working async RPC
* Implement async file upload
* Bugfix
* Method recall bugfixes
* Bugfixes
* Trait bugfixes
* Fix FIFO buffer
* Bugfixes and speedtests
* Async logging
* Implement websocket streams
* Implement loop API, signal API, clean closing and start changing layer
* Small magna, websocket and HTTP fixes
* Clean up loop API
* Improved stack traces, 2FA and async
* Login fixes
* Added instructions for manual verification
* Small fixes
* More app info improvements
* More app info improvements
* TL and 2FA fixes
* Update to layer 89
* More bugfixes
* Implement broken media reporting
* Remove debug comments
* PHP 7.2 backwards compatibility
* Bugfixes
* Async key generation
* Some simplifications
* Transport fixes
* Cleanup
* async API
* Performance fixes
* Fixes to async API
* Bugfixes
* Implement one-time async loop
* Authorization and logging fixes
* Update to layer 91
* 7to5 fix
* Null coalesce conversion
* Implement socks5 proxy
* Implement HTTP proxy
* Fixes to HTTP proxy
* MTProxy and socks5 fixes
* Disable PHP 5 conversion
* Proxies have higher priority
* Avoid error handling in vendor
* Override composer dependencies
* Fix travis build
* Final composer fixes
* Proxy logic fixes
* Fix get_updates update handling
* Do not use parallel file driver if not supported
* Refactor loader and implement HTTP fixes
* Suppress errors in loader
* HTTP and authorization fixes
* HTTP fixes
* Improved peer management
* Use HTTP protocol on altervista
* Small bugfixes
* Minor fixes
* Docufix
* Docufix
* Legacy fixes
* Fix message queue
* Avoid updating if using MTProxy
* Improve logs and examples
* Trim final newlines while converting parse mode
* Reimplement noResponse flag
* Async combined event handler and APIFactory fixes
* Actually return config
* Case-insensitive methods
* Bugfix
* Apply fixes from StyleCI (#545)
* MTProxy fixes
* PHP 5 warning
* Improved PHP 5 warning
* Use <br> along with newlines in web logs
* Update docs
2018-12-26 20:51:14 +01:00
if ( ! file_exists ( 'songs.php' )) {
copy ( 'https://github.com/danog/MadelineProto/raw/master/songs.php' , 'songs.php' );
2017-07-31 19:17:10 +02:00
}
2019-06-04 14:55:58 +02:00
echo 'Deserializing MadelineProto from session.madeline...' . PHP_EOL ;
2017-08-13 18:52:32 +02:00
2018-03-29 23:04:50 +02:00
/* if ( ! isset ( $MadelineProto -> inputEncryptedFilePhoto ) && false ) {
2019-03-29 20:25:42 +01:00
$MadelineProto -> inputEncryptedFilePhoto = $MadelineProto -> upload_encrypted ( 'tests/faust.jpg' , 'fausticorn.jpg' ); // This gets an inputFile object with file name magic
$MadelineProto -> inputEncryptedFileGif = $MadelineProto -> upload_encrypted ( 'tests/pony.mp4' );
$MadelineProto -> inputEncryptedFileSticker = $MadelineProto -> upload_encrypted ( 'tests/lel.webp' );
$MadelineProto -> inputEncryptedFileDocument = $MadelineProto -> upload_encrypted ( 'tests/60' , 'magic' ); // This gets an inputFile object with file name magic
$MadelineProto -> inputEncryptedFileVideo = $MadelineProto -> upload_encrypted ( 'tests/swing.mp4' );
$MadelineProto -> inputEncryptedFileAudio = $MadelineProto -> upload_encrypted ( 'tests/mosconi.mp3' );
2018-03-29 23:04:50 +02:00
} */
2017-12-16 19:08:11 +01:00
2019-07-18 23:48:35 +02:00
use danog\MadelineProto\Loop\Impl\ResumableSignalLoop ;
class MessageLoop extends ResumableSignalLoop
{
const INTERVAL = 10 ;
private $timeout ;
private $call ;
public function __construct ( $API , $call , $timeout = self :: INTERVAL )
{
$this -> API = $API ;
$this -> call = $call ;
$this -> timeout = $timeout ;
}
public function loop ()
{
$MadelineProto = $this -> API ;
$logger = & $MadelineProto -> logger ;
while ( true ) {
$result = yield $this -> waitSignal ( $this -> pause ( $this -> timeout ));
if ( $result ) {
$logger -> logger ( " Got signal in $this , exiting " );
return ;
}
try {
yield $MadelineProto -> messages -> editMessage ([ 'id' => $this -> call -> mId , 'peer' => $this -> call -> getOtherID (), 'message' => 'Total running calls: ' . count ( $MadelineProto -> getEventHandler () -> calls ) . PHP_EOL . PHP_EOL . $this -> call -> getDebugString ()]);
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
$MadelineProto -> logger ( $e );
}
}
}
public function __toString () : string
{
return " VoIP message loop " . $this -> call -> getOtherId ();
}
}
class StatusLoop extends ResumableSignalLoop
{
const INTERVAL = 2 ;
private $timeout ;
private $call ;
public function __construct ( $API , $call , $timeout = self :: INTERVAL )
{
$this -> API = $API ;
$this -> call = $call ;
$this -> timeout = $timeout ;
}
public function loop ()
{
$MadelineProto = $this -> API ;
$logger = & $MadelineProto -> logger ;
$call = $this -> call ;
while ( true ) {
$result = yield $this -> waitSignal ( $this -> pause ( $this -> timeout ));
if ( $result ) {
$logger -> logger ( " Got signal in $this , exiting " );
return ;
}
if ( $call -> getCallState () === \danog\MadelineProto\VoIP :: CALL_STATE_ENDED ) {
try {
/* yield $this -> messages -> sendMedia ([
'reply_to_msg_id' => $this -> times [ $call -> getOtherID ()][ 1 ],
'peer' => $call -> getOtherID (), 'message' => 'Call statistics by @magnaluna' ,
'media' => [
'_' => 'inputMediaUploadedDocument' ,
'file' => " /tmp/stats " . $call -> getCallID ()[ 'id' ] . " .txt " ,
'attributes' => [
[ '_' => 'documentAttributeFilename' , 'file_name' => " stats " . $call -> getCallID ()[ 'id' ] . " .txt " ]
]
],
]); */
yield $MadelineProto -> messages -> sendMedia ([
'reply_to_msg_id' => $this -> call -> mId ,
'peer' => $call -> getOtherID (), 'message' => 'Debug info by @magnaluna' ,
'media' => [
'_' => 'inputMediaUploadedDocument' ,
'file' => '/tmp/logs' . $call -> getCallID ()[ 'id' ] . '.log' ,
'attributes' => [
[ '_' => 'documentAttributeFilename' , 'file_name' => 'logs' . $call -> getCallID ()[ 'id' ] . '.log' ],
],
],
]);
} catch ( \danog\MadelineProto\Exception $e ) {
$MadelineProto -> logger ( $e );
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
$MadelineProto -> logger ( $e );
} catch ( \danog\MadelineProto\Exception $e ) {
$MadelineProto -> logger ( $e );
}
@ unlink ( '/tmp/logs' . $call -> getCallID ()[ 'id' ] . '.log' );
@ unlink ( '/tmp/stats' . $call -> getCallID ()[ 'id' ] . '.txt' );
$MadelineProto -> getEventHandler () -> cleanUpCall ( $call -> getOtherID ());
return ;
}
}
}
public function __toString () : string
{
return " VoIP status loop " . $this -> call -> getOtherId ();
}
}
2018-03-29 23:04:50 +02:00
class EventHandler extends \danog\MadelineProto\EventHandler
2018-03-24 19:06:19 +01:00
{
2019-07-18 23:48:35 +02:00
const ADMINS = [ 101374607 ]; // @danogentili, creator of MadelineProto
private $messageLoops = [];
private $statusLoops = [];
private $programmed_call ;
private $my_users ;
2018-03-29 23:04:50 +02:00
public function configureCall ( $call )
{
include 'songs.php' ;
$call -> configuration [ 'enable_NS' ] = false ;
$call -> configuration [ 'enable_AGC' ] = false ;
$call -> configuration [ 'enable_AEC' ] = false ;
2019-06-04 14:55:58 +02:00
$call -> configuration [ 'log_file_path' ] = '/tmp/logs' . $call -> getCallID ()[ 'id' ] . '.log' ; // Default is /dev/null
2018-05-04 12:21:33 +02:00
//$call->configuration["stats_dump_file_path"] = "/tmp/stats".$call->getCallID()['id'].".txt"; // Default is /dev/null
2018-03-29 23:04:50 +02:00
$call -> parseConfig ();
$call -> playOnHold ( $songs );
2019-07-18 23:48:35 +02:00
if ( $call -> getCallState () === \danog\MadelineProto\VoIP :: CALL_STATE_INCOMING ) {
if ( $call -> accept () === false ) {
2019-07-29 16:48:25 +02:00
$this -> logger ( 'DID NOT ACCEPT A CALL' );
2019-07-18 23:48:35 +02:00
}
}
if ( $call -> getCallState () !== \danog\MadelineProto\VoIP :: CALL_STATE_ENDED ) {
$this -> calls [ $call -> getOtherID ()] = $call ;
try {
$call -> mId = yield $this -> messages -> sendMessage ([ 'peer' => $call -> getOtherID (), 'message' => 'Total running calls: ' . count ( $this -> calls ) . PHP_EOL . PHP_EOL . $call -> getDebugString ()])[ 'id' ];
} catch ( \Throwable $e ) {
$this -> logger ( $e );
}
$this -> messageLoops [ $call -> getOtherID ()] = new MessageLoop ( $this , $call );
$this -> statusLoops [ $call -> getOtherID ()] = new StatusLoop ( $this , $call );
$this -> messageLoops [ $call -> getOtherID ()] -> start ();
$this -> statusLoops [ $call -> getOtherID ()] -> start ();
}
//yield $this->messages->sendMessage(['message' => var_export($call->configuration, true), 'peer' => $call->getOtherID()]);
}
public function cleanUpCall ( $user )
{
if ( isset ( $this -> calls [ $user ])) {
unset ( $this -> calls [ $user ]);
}
if ( isset ( $this -> messageLoops [ $user ])) {
$this -> messageLoops [ $user ] -> signal ( true );
unset ( $this -> messageLoops [ $user ]);
}
if ( isset ( $this -> statusLoops [ $user ])) {
$this -> statusLoops [ $user ] -> signal ( true );
unset ( $this -> statusLoops [ $user ]);
}
}
public function makeCall ( $user )
{
try {
if ( isset ( $this -> calls [ $user ])) {
if ( $this -> calls [ $user ] -> getCallState () === \danog\MadelineProto\VoIP :: CALL_STATE_ENDED ) {
yield $this -> cleanUpCall ( $user );
} else {
yield $this -> messages -> sendMessage ([ 'peer' => $user , 'message' => " I'm already in a call with you! " ]);
return ;
}
}
yield $this -> configureCall ( yield $this -> requestCall ( $user ));
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
try {
if ( $e -> rpc === 'USER_PRIVACY_RESTRICTED' ) {
$e = 'Please disable call privacy settings to make me call you' ;
} /* elseif ( strpos ( $e -> rpc , 'FLOOD_WAIT_' ) === 0 ) {
$t = str_replace ( 'FLOOD_WAIT_' , '' , $e -> rpc );
$this -> programmed_call [] = [ $user , time () + 1 + $t ];
$e = " I'll call you back in $t seconds. \n You can also call me right now. " ;
} */
yield $this -> messages -> sendMessage ([ 'peer' => $user , 'message' => ( string ) $e ]);
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
}
} catch ( \Throwable $e ) {
yield $this -> messages -> sendMessage ([ 'peer' => $user , 'message' => ( string ) $e ]);
}
2018-03-24 16:22:13 +01:00
}
2018-03-29 23:04:50 +02:00
public function handleMessage ( $chat_id , $from_id , $message )
{
try {
if ( ! isset ( $this -> my_users [ $from_id ]) || $message === '/start' ) {
$this -> my_users [ $from_id ] = true ;
$message = '/call' ;
2019-07-18 23:48:35 +02:00
yield $this -> messages -> sendMessage ([ 'no_webpage' => true , 'peer' => $chat_id , 'message' => " Hi, I'm @magnaluna the webradio.
2017-09-18 11:05:28 +02:00
Call _me_ to listen to some ** awesome ** music , or send / call to make _me_ call _you_ ( don ' t forget to disable call privacy settings ! ) .
You can also program a phone call with / program :
/ program 29 August 2018 - call me the 29 th of august 2018
/ program + 1 hour 30 minutes - call me in one hour and thirty minutes
/ program next Thursday - call me next Thursday at midnight
Send / start to see this message again .
I also provide advanced stats during calls !
I ' m a userbot powered by @ MadelineProto , created by @ danogentili .
2018-03-30 12:53:51 +02:00
Source code : https :// github . com / danog / MadelineProto
2018-03-29 23:04:50 +02:00
Propic art by @ magnaluna on [ deviantart ]( https :// magnaluna . deviantart . com ) . " , 'parse_mode' => 'Markdown']);
}
if ( ! isset ( $this -> calls [ $from_id ]) && $message === '/call' ) {
2019-07-18 23:48:35 +02:00
yield $this -> makeCall ( $from_id );
2018-03-29 23:04:50 +02:00
}
if ( strpos ( $message , '/program' ) === 0 ) {
$time = strtotime ( str_replace ( '/program ' , '' , $message ));
if ( $time === false ) {
2019-07-18 23:48:35 +02:00
yield $this -> messages -> sendMessage ([ 'peer' => $chat_id , 'message' => 'Invalid time provided' ]);
} else if ( $time - time () <= 0 ) {
yield $this -> messages -> sendMessage ([ 'peer' => $chat_id , 'message' => 'Invalid time provided' ]);
2018-03-29 23:04:50 +02:00
} else {
2019-07-18 23:48:35 +02:00
yield $this -> messages -> sendMessage ([ 'peer' => $chat_id , 'message' => 'OK' ]);
2018-03-29 23:04:50 +02:00
$this -> programmed_call [] = [ $from_id , $time ];
2019-07-18 23:48:35 +02:00
$key = count ( $this -> programmed_call ) - 1 ;
yield $this -> sleep ( $time - time ());
yield $this -> makeCall ( $from_id );
unset ( $this -> programmed_call [ $key ]);
2018-03-29 23:04:50 +02:00
}
}
2019-07-18 23:48:35 +02:00
if ( $message === '/broadcast' && in_array ( self :: ADMINS , $from_id )) {
2018-03-29 23:05:15 +02:00
$time = time () + 100 ;
2018-03-30 18:03:16 +02:00
$message = explode ( ' ' , $message , 2 );
2018-03-30 18:02:34 +02:00
unset ( $message [ 0 ]);
2018-03-30 18:03:16 +02:00
$message = implode ( ' ' , $message );
2019-07-18 23:48:35 +02:00
$params = [ 'multiple' => true ];
foreach ( yield $this -> get_dialogs () as $peer ) {
$params [] = [ 'peer' => $peer , 'message' => $message ];
2018-03-29 23:04:50 +02:00
}
2019-07-18 23:48:35 +02:00
yield $this -> messages -> sendMessage ( $params );
2018-03-29 23:04:50 +02:00
}
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
try {
if ( $e -> rpc === 'USER_PRIVACY_RESTRICTED' ) {
$e = 'Please disable call privacy settings to make me call you' ;
2019-07-18 23:48:35 +02:00
} /* elseif ( strpos ( $e -> rpc , 'FLOOD_WAIT_' ) === 0 ) {
2018-03-29 23:04:50 +02:00
$t = str_replace ( 'FLOOD_WAIT_' , '' , $e -> rpc );
2019-07-18 23:48:35 +02:00
$e = " Too many people used the /call function. I'll be able to call you in $t seconds. \n You can also call me right now " ;
} */
yield $this -> messages -> sendMessage ([ 'peer' => $chat_id , 'message' => ( string ) $e ]);
2018-03-29 23:04:50 +02:00
} catch ( \danog\MadelineProto\RPCErrorException $e ) {
}
2019-07-18 23:48:35 +02:00
$this -> logger ( $e );
2018-03-29 23:04:50 +02:00
} catch ( \danog\MadelineProto\Exception $e ) {
2019-07-18 23:48:35 +02:00
$this -> logger ( $e );
2018-03-29 23:04:50 +02:00
}
}
public function onUpdateNewMessage ( $update )
{
if ( $update [ 'message' ][ 'out' ] || $update [ 'message' ][ 'to_id' ][ '_' ] !== 'peerUser' || ! isset ( $update [ 'message' ][ 'from_id' ])) {
return ;
}
2019-07-18 23:48:35 +02:00
$this -> logger -> logger ( $update );
$chat_id = $from_id = yield $this -> get_info ( $update )[ 'bot_api_id' ];
$message = $update [ 'message' ][ 'message' ] ? ? '' ;
yield $this -> handleMessage ( $chat_id , $from_id , $message );
2018-03-29 23:04:50 +02:00
}
2018-03-29 23:05:15 +02:00
2018-03-29 23:04:50 +02:00
public function onUpdateNewEncryptedMessage ( $update )
{
return ;
2019-07-18 23:48:35 +02:00
$chat_id = yield $this -> get_info ( $update )[ 'InputEncryptedChat' ];
$from_id = yield $this -> get_secret_chat ( $chat_id )[ 'user_id' ];
2018-03-29 23:04:50 +02:00
$message = isset ( $update [ 'message' ][ 'decrypted_message' ][ 'message' ]) ? $update [ 'message' ][ 'decrypted_message' ][ 'message' ] : '' ;
2019-07-18 23:48:35 +02:00
yield $this -> handleMessage ( $chat_id , $from_id , $message );
2018-03-29 23:04:50 +02:00
}
2018-03-29 23:05:15 +02:00
public function onUpdateEncryption ( $update )
{
2018-03-29 23:04:50 +02:00
return ;
2018-03-29 23:05:15 +02:00
try {
if ( $update [ 'chat' ][ '_' ] !== 'encryptedChat' ) {
return ;
}
2019-07-18 23:48:35 +02:00
$chat_id = yield $this -> get_info ( $update )[ 'InputEncryptedChat' ];
$from_id = yield $this -> get_secret_chat ( $chat_id )[ 'user_id' ];
2018-03-29 23:05:15 +02:00
$message = '' ;
} catch ( \danog\MadelineProto\Exception $e ) {
return ;
}
2019-07-18 23:48:35 +02:00
yield $this -> handleMessage ( $chat_id , $from_id , $message );
2018-03-29 23:04:50 +02:00
}
2018-03-29 23:05:15 +02:00
2018-03-29 23:04:50 +02:00
public function onUpdatePhoneCall ( $update )
{
2018-03-29 23:05:15 +02:00
if ( is_object ( $update [ 'phone_call' ]) && isset ( $update [ 'phone_call' ] -> madeline ) && $update [ 'phone_call' ] -> getCallState () === \danog\MadelineProto\VoIP :: CALL_STATE_INCOMING ) {
2019-07-18 23:48:35 +02:00
yield $this -> configureCall ( $update [ 'phone_call' ]);
2018-03-29 23:05:15 +02:00
}
2018-03-29 23:04:50 +02:00
}
2018-04-17 23:52:28 +02:00
2019-07-18 23:48:35 +02:00
/* public function onAny ( $update )
2018-04-17 23:51:54 +02:00
{
2019-07-18 23:48:35 +02:00
$this -> logger -> logger ( $update );
} */
2018-04-17 23:52:28 +02:00
2019-07-18 23:48:35 +02:00
public function __construct ( $API )
2018-03-29 23:04:50 +02:00
{
2019-07-18 23:48:35 +02:00
parent :: __construct ( $API );
$this -> programmed_call = [];
foreach ( $this -> programmed_call as $key => list ( $user , $time )) {
continue ;
$sleepTime = $time <= time () ? 0 : $time - time ();
$this -> callFork (( function () use ( $sleepTime , $key , $user ) {
yield $this -> sleep ( $sleepTime );
yield $this -> makeCall ( $user );
2018-03-29 23:05:15 +02:00
unset ( $this -> programmed_call [ $key ]);
2019-07-18 23:48:35 +02:00
})());
2018-03-29 23:04:50 +02:00
}
}
2019-07-18 23:48:35 +02:00
public function __sleep ()
{
return [ 'programmed_call' , 'my_users' ];
}
2018-03-24 19:06:19 +01:00
}
2019-03-29 20:25:42 +01:00
2019-06-04 14:55:58 +02:00
if ( ! class_exists ( '\\danog\\MadelineProto\\VoIPServerConfig' )) {
die ( 'Install the libtgvoip extension: https://voip.madelineproto.xyz' . PHP_EOL );
}
2019-03-30 00:41:16 +01:00
2019-03-29 20:25:42 +01:00
\danog\MadelineProto\VoIPServerConfig :: update (
[
2019-06-04 14:55:58 +02:00
'audio_init_bitrate' => 100 * 1000 ,
'audio_max_bitrate' => 100 * 1000 ,
'audio_min_bitrate' => 10 * 1000 ,
2019-03-29 20:25:42 +01:00
'audio_congestion_window' => 4 * 1024 ,
]
);
2019-07-29 16:48:25 +02:00
$MadelineProto = new \danog\MadelineProto\API ( 'session.madeline' , [ 'secret_chats' => [ 'accept_chats' => false ], 'logger' => [ 'logger' => 3 , 'logger_level' => 5 , 'logger_param' => getcwd () . '/MadelineProto.log' ], 'updates' => [ 'getdifference_interval' => 10 ], 'serialization' => [ 'serialization_interval' => 30 , 'cleanup_before_serialization' => true ], 'flood_timeout' => [ 'wait_if_lt' => 86400 ]]);
2019-07-18 23:48:35 +02:00
foreach ([ 'calls' , 'programmed_call' , 'my_users' ] as $key ) {
if ( isset ( $MadelineProto -> API -> storage [ $key ])) {
unset ( $MadelineProto -> API -> storage [ $key ]);
2018-06-29 18:20:07 +02:00
}
}
2019-09-13 17:10:08 +02:00
$MadelineProto -> async ( true );
$MadelineProto -> loop ( function () use ( $MadelineProto ) {
yield $MadelineProto -> start ();
yield $MadelineProto -> setEventHandler ( '\EventHandler' );
});
2018-03-29 23:04:50 +02:00
$MadelineProto -> loop ();