Added file upload/download wrappers, decided to store full chat/user/channel constructors in $API->chats, fixed bugs
This commit is contained in:
parent
f54fb0b6b3
commit
20d9b0fa6d
32
README.md
32
README.md
@ -199,7 +199,7 @@ $MadelineProto->update_settings($settings);
|
||||
### Handling updates
|
||||
|
||||
When an update is received, the update callback function (see settings) is called. By default, the get_updates_update_handler MadelineProto method is called. This method stores all incoming updates into an array (its size limit is specified by the updates\_array\_limit parameter in the settings) and can be fetched by running the `get_updates` method.
|
||||
This method accepts an array of options as the first parameter, and returns an array of updates. Example:
|
||||
This method accepts an array of options as the first parameter, and returns an array of updates (an array containing the update id and an object of type [Update](https://daniil.it/MadelineProto/API_docs/types/Updates.html)). Example:
|
||||
|
||||
```
|
||||
$MadelineProto = new \danog\MadelineProto\API();
|
||||
@ -314,6 +314,36 @@ array(3) {
|
||||
To specify a custom callback change the correct value in the settings. The specified callable must accept one parameter for the update.
|
||||
|
||||
|
||||
### Uploading and downloading files
|
||||
|
||||
MadelineProto provides wrapper methods to upload and download files.
|
||||
|
||||
Every method described in this section accepts a last optional paramater with a callable function that will be called during the upload/download using the first parameter to pass a floating point number indicating the upload/download status in percentage.
|
||||
|
||||
The upload method returns an [InputFile](https://daniil.it/MadelineProto/API_docs/types/InputFile.html) object that must be used to generate an [InputMedia](https://daniil.it/MadelineProto/API_docs/types/InputMedia.html) object, that can be later sent using the [sendmedia method](https://daniil.it/MadelineProto/API_docs/methods/messages_sendMedia.html).
|
||||
|
||||
|
||||
```
|
||||
$inputFile = $MadelineProto->upload('file', 'optional new file name.ext');
|
||||
// Generate an inputMedia object and store it in $inputMedia, see testing.php
|
||||
$MadelineProto->messages->sendMedia(['peer' => '@pwrtelegramgroup', 'media' => $inputMedia]);
|
||||
```
|
||||
|
||||
|
||||
See testing.php for more examples.
|
||||
|
||||
|
||||
There are multiple download methods that allow you to download a file to a directory, to a file or to a stream.
|
||||
The first parameter of these functions must always be a [messageMediaPhoto](https://daniil.it/MadelineProto/API_docs/constructors/messageMediaPhoto.html) or a [messageMediaDocument](https://daniil.it/MadelineProto/API_docs/constructors/messageMediaDocument.html) object. These objects are usually received in updates, see `bot.php` for examples
|
||||
|
||||
|
||||
```
|
||||
$output_file_name = $MadelineProto->download_to_dir($message_media, '/tmp/dldir');
|
||||
$custom_output_file_name = $MadelineProto->download_to_file($message_media, '/tmp/dldir/customname.ext');
|
||||
$stream = fopen('php://output', 'w'); // Stream to browser like with echo
|
||||
$MadelineProto->download_to_stream($message_media, $stream);
|
||||
```
|
||||
|
||||
|
||||
### Calling mtproto methods and available wrappers
|
||||
|
||||
|
13
bot.php
13
bot.php
@ -19,8 +19,17 @@ while (true) {
|
||||
case 'updateNewMessage':
|
||||
if ($update['update']['message']['out']) continue;
|
||||
$res = json_encode($update, JSON_PRETTY_PRINT);
|
||||
if ($res == '') $res =var_export($update, true);
|
||||
try { $MadelineProto->messages->sendMessage(['peer' => $update['update']['message']['from_id'], 'message' => $res, 'reply_to_msg_id' => $update['update']['message']['id'], 'entities' => [['_' => 'messageEntityPre', 'offset' => 0, 'length' => strlen($res), 'language' => 'json', ]]]); } catch (\danog\MadelineProto\RPCErrorException $e) { $MadelineProto->messages->sendMessage(['peer' => '@danogentili', 'message' => $e->getCode().': '.$e->getMessage().PHP_EOL.$e->getTraceAsString()]); }
|
||||
if ($res == '') $res = var_export($update, true);
|
||||
try {
|
||||
$MadelineProto->messages->sendMessage(['peer' => $update['update']['message']['from_id'], 'message' => $res, 'reply_to_msg_id' => $update['update']['message']['id'], 'entities' => [['_' => 'messageEntityPre', 'offset' => 0, 'length' => strlen($res), 'language' => 'json', ]]]);
|
||||
} catch (\danog\MadelineProto\RPCErrorException $e) { $MadelineProto->messages->sendMessage(['peer' => '@danogentili', 'message' => $e->getCode().': '.$e->getMessage().PHP_EOL.$e->getTraceAsString()]); }
|
||||
try {
|
||||
if (isset($update['update']['message']['media'])) {
|
||||
$time = time();
|
||||
$file = $MadelineProto->download_to_dir($update['update']['message']['media'], '/tmp');
|
||||
$MadelineProto->messages->sendMessage(['peer' => $update['update']['message']['from_id'], 'message' => 'Downloaded to '.$file.' in '.(time() - $time).' seconds', 'reply_to_msg_id' => $update['update']['message']['id'], 'entities' => [['_' => 'messageEntityPre', 'offset' => 0, 'length' => strlen($res), 'language' => 'json', ]]]);
|
||||
}
|
||||
} catch (\danog\MadelineProto\RPCErrorException $e) { $MadelineProto->messages->sendMessage(['peer' => '@danogentili', 'message' => $e->getCode().': '.$e->getMessage().PHP_EOL.$e->getTraceAsString()]); }
|
||||
}
|
||||
}
|
||||
echo 'Wrote '.\danog\MadelineProto\Serialization::serialize('bot.madeline', $MadelineProto).' bytes'.PHP_EOL;
|
||||
|
9
composer.lock
generated
9
composer.lock
generated
@ -4,7 +4,6 @@
|
||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
|
||||
"This file is @generated automatically"
|
||||
],
|
||||
"hash": "fc800fb5f8bf5490499819d3b4b2330d",
|
||||
"content-hash": "3a10bc147a48cd4573bcc9654d3be153",
|
||||
"packages": [
|
||||
{
|
||||
@ -58,7 +57,7 @@
|
||||
"struct",
|
||||
"unpack"
|
||||
],
|
||||
"time": "2016-11-14 15:09:50"
|
||||
"time": "2016-11-14T15:09:50+00:00"
|
||||
},
|
||||
{
|
||||
"name": "paragonie/constant_time_encoding",
|
||||
@ -119,7 +118,7 @@
|
||||
"hex2bin",
|
||||
"rfc4648"
|
||||
],
|
||||
"time": "2016-07-11 20:32:06"
|
||||
"time": "2016-07-11T20:32:06+00:00"
|
||||
},
|
||||
{
|
||||
"name": "paragonie/random_compat",
|
||||
@ -167,7 +166,7 @@
|
||||
"pseudorandom",
|
||||
"random"
|
||||
],
|
||||
"time": "2016-11-07 23:38:38"
|
||||
"time": "2016-11-07T23:38:38+00:00"
|
||||
},
|
||||
{
|
||||
"name": "phpseclib/phpseclib",
|
||||
@ -259,7 +258,7 @@
|
||||
"x.509",
|
||||
"x509"
|
||||
],
|
||||
"time": "2016-10-04 00:57:04"
|
||||
"time": "2016-10-04T00:57:04+00:00"
|
||||
}
|
||||
],
|
||||
"packages-dev": [],
|
||||
|
@ -203,7 +203,7 @@ $MadelineProto->update_settings($settings);
|
||||
### Handling updates
|
||||
|
||||
When an update is received, the update callback function (see settings) is called. By default, the get_updates_update_handler MadelineProto method is called. This method stores all incoming updates into an array (its size limit is specified by the updates\_array\_limit parameter in the settings) and can be fetched by running the `get_updates` method.
|
||||
This method accepts an array of options as the first parameter, and returns an array of updates. Example:
|
||||
This method accepts an array of options as the first parameter, and returns an array of updates (an array containing the update id and an object of type [Update](https://daniil.it/MadelineProto/API_docs/types/Updates.html)). Example:
|
||||
|
||||
```
|
||||
$MadelineProto = new \danog\MadelineProto\API();
|
||||
@ -318,6 +318,36 @@ array(3) {
|
||||
To specify a custom callback change the correct value in the settings. The specified callable must accept one parameter for the update.
|
||||
|
||||
|
||||
### Uploading and downloading files
|
||||
|
||||
MadelineProto provides wrapper methods to upload and download files.
|
||||
|
||||
Every method described in this section accepts a last optional paramater with a callable function that will be called during the upload/download using the first parameter to pass a floating point number indicating the upload/download status in percentage.
|
||||
|
||||
The upload method returns an [InputFile](https://daniil.it/MadelineProto/API_docs/types/InputFile.html) object that must be used to generate an [InputMedia](https://daniil.it/MadelineProto/API_docs/types/InputMedia.html) object, that can be later sent using the [sendmedia method](https://daniil.it/MadelineProto/API_docs/methods/messages_sendMedia.html).
|
||||
|
||||
|
||||
```
|
||||
$inputFile = $MadelineProto->upload('file', 'optional new file name.ext');
|
||||
// Generate an inputMedia object and store it in $inputMedia, see testing.php
|
||||
$MadelineProto->messages->sendMedia(['peer' => '@pwrtelegramgroup', 'media' => $inputMedia]);
|
||||
```
|
||||
|
||||
|
||||
See testing.php for more examples.
|
||||
|
||||
|
||||
There are multiple download methods that allow you to download a file to a directory, to a file or to a stream.
|
||||
The first parameter of these functions must always be a [messageMediaPhoto](https://daniil.it/MadelineProto/API_docs/constructors/messageMediaPhoto.html) or a [messageMediaDocument](https://daniil.it/MadelineProto/API_docs/constructors/messageMediaDocument.html) object. These objects are usually received in updates, see `bot.php` for examples
|
||||
|
||||
|
||||
```
|
||||
$output_file_name = $MadelineProto->download_to_dir($message_media, '/tmp/dldir');
|
||||
$custom_output_file_name = $MadelineProto->download_to_file($message_media, '/tmp/dldir/customname.ext');
|
||||
$stream = fopen('php://output', 'w'); // Stream to browser like with echo
|
||||
$MadelineProto->download_to_stream($message_media, $stream);
|
||||
```
|
||||
|
||||
|
||||
### Calling mtproto methods and available wrappers
|
||||
|
||||
|
BIN
mosconi.mp3
Normal file
BIN
mosconi.mp3
Normal file
Binary file not shown.
@ -16,6 +16,7 @@ class API extends APIFactory
|
||||
{
|
||||
use \danog\MadelineProto\Wrappers\Login;
|
||||
use \danog\MadelineProto\Wrappers\PeerHandler;
|
||||
use \danog\MadelineProto\Wrappers\FilesHandler;
|
||||
use \danog\MadelineProto\Wrappers\SettingsManager;
|
||||
|
||||
public $API;
|
||||
|
@ -73,11 +73,13 @@ trait CallHandler
|
||||
throw new \danog\MadelineProto\RPCErrorException($server_answer['error_message'], $server_answer['error_code']);
|
||||
break;
|
||||
}
|
||||
/*
|
||||
case 420:
|
||||
$seconds = preg_replace('/[^0-9]+/', '', $server_answer['error_message']);
|
||||
\danog\MadelineProto\Logger::log('Flood, waiting '.$seconds.' seconds...');
|
||||
sleep($seconds);
|
||||
throw new \danog\MadelineProto\Exception('Re-executing query...');
|
||||
*/
|
||||
default:
|
||||
throw new \danog\MadelineProto\RPCErrorException($server_answer['error_message'], $server_answer['error_code']);
|
||||
break;
|
||||
|
@ -24,8 +24,9 @@ trait PeerHandler
|
||||
foreach ($users as $key => $user) {
|
||||
switch ($user['_']) {
|
||||
case 'user':
|
||||
if (!isset($this->chats[$user['id']]) || $this->chats[$user['id']] !== $user) {
|
||||
$this->chats[$user['id']] = $user;
|
||||
if (!isset($this->chats[$user['id']]) || $this->chats[$user['id']]['user'] !== $user) {
|
||||
//$this->method_call('users.getFullUser', ['id' => $user]);
|
||||
$this->chats[$user['id']] = ['_' => 'userFull', 'user' => $user];
|
||||
$this->should_serialize = true;
|
||||
}
|
||||
case 'userEmpty':
|
||||
@ -43,18 +44,21 @@ trait PeerHandler
|
||||
switch ($chat['_']) {
|
||||
case 'chat':
|
||||
case 'chatEmpty':
|
||||
if (!isset($this->chats[-$chat['id']]) || $this->chats[-$chat['id']] !== $chat) {
|
||||
if (!isset($this->chats[-$chat['id']]) || $this->chats[-$chat['id']]['chat'] !== $chat) {
|
||||
//$this->method_call('messages.getFullChat', ['chat_id' => $chat['id']]);
|
||||
$this->chats[-$chat['id']] = ['_' => 'chatFull', 'chat' => $chat];
|
||||
$this->should_serialize = true;
|
||||
$this->chats[-$chat['id']] = $chat;
|
||||
|
||||
}
|
||||
|
||||
case 'chatForbidden':
|
||||
case 'channelEmpty':
|
||||
break;
|
||||
case 'channel':
|
||||
if (!isset($this->chats[(int) ('-100'.$chat['id'])]) || $this->chats[(int) ('-100'.$chat['id'])] !== $chat) {
|
||||
if (!isset($this->chats[(int) ('-100'.$chat['id'])]) || $this->chats[(int) ('-100'.$chat['id'])]['channel'] !== $chat) {
|
||||
$this->chats[(int)('-100'.$chat['id'])] = ['_' => 'channelFull', 'channel' => $chat];
|
||||
$this->should_serialize = true;
|
||||
$this->chats[(int) ('-100'.$chat['id'])] = $chat;
|
||||
//$this->method_call('channels.getFullChannel', ['channel' => $chat]);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
@ -62,7 +66,6 @@ trait PeerHandler
|
||||
break;
|
||||
}
|
||||
}
|
||||
$this->should_serialize = true;
|
||||
}
|
||||
|
||||
public function peer_isset($id)
|
||||
@ -111,13 +114,16 @@ trait PeerHandler
|
||||
{
|
||||
if (is_array($id)) {
|
||||
switch ($id['_']) {
|
||||
case 'inputPeerSelf':
|
||||
case 'inputUserSelf':
|
||||
case 'inputPeerSelf':
|
||||
$id = $this->datacenter->authorization['user']['id'];
|
||||
break;
|
||||
case 'user':
|
||||
$id = $id['id'];
|
||||
break;
|
||||
case 'userFull':
|
||||
$id = $id['user']['id'];
|
||||
break;
|
||||
case 'inputPeerUser':
|
||||
case 'inputUser':
|
||||
case 'peerUser':
|
||||
@ -127,6 +133,9 @@ trait PeerHandler
|
||||
case 'chat':
|
||||
$id = -$id['id'];
|
||||
break;
|
||||
case 'chatFull':
|
||||
$id = -$id['chat']['id'];
|
||||
break;
|
||||
case 'inputPeerChat':
|
||||
case 'peerChat':
|
||||
$id = -$id['chat_id'];
|
||||
@ -135,6 +144,10 @@ trait PeerHandler
|
||||
case 'channel':
|
||||
$id = '-100'.$id['id'];
|
||||
break;
|
||||
case 'channelFull':
|
||||
$id = '-100'.$id['channel']['id'];
|
||||
break;
|
||||
|
||||
case 'inputPeerChannel':
|
||||
case 'inputChannel':
|
||||
case 'peerChannel':
|
||||
@ -161,16 +174,16 @@ trait PeerHandler
|
||||
if (isset($this->chats[$id])) {
|
||||
return $this->gen_all($this->chats[$id]);
|
||||
}
|
||||
// if ($recursive) {
|
||||
// }
|
||||
throw new \danog\MadelineProto\Exception("Couldn't find peer by provided chat id ".$id);
|
||||
}
|
||||
$id = str_replace('@', '', $id);
|
||||
foreach ($this->chats as $chat) {
|
||||
if (isset($chat['username']) && strtolower($chat['username']) == strtolower($id)) {
|
||||
foreach (['user', 'chat', 'channel'] as $wut) {
|
||||
if (isset($chat[$wut]['username']) && strtolower($chat[$wut]['username']) == strtolower($id)) {
|
||||
return $this->gen_all($chat);
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($recursive) {
|
||||
$this->resolve_username($id);
|
||||
|
||||
@ -183,32 +196,35 @@ trait PeerHandler
|
||||
{
|
||||
$res = [$this->constructors->find_by_predicate($constructor['_'])['type'] => $constructor];
|
||||
switch ($constructor['_']) {
|
||||
case 'user':
|
||||
if ($constructor['self']) {
|
||||
case 'userFull':
|
||||
$res['User'] = $constructor['user'];
|
||||
if ($constructor['user']['self']) {
|
||||
$res['InputPeer'] = ['_' => 'inputPeerSelf'];
|
||||
$res['InputUser'] = ['_' => 'inputUserSelf'];
|
||||
} elseif (isset($constructor['access_hash'])) {
|
||||
$res['InputPeer'] = ['_' => 'inputPeerUser', 'user_id' => $constructor['id'], 'access_hash' => $constructor['access_hash']];
|
||||
$res['InputUser'] = ['_' => 'inputUser', 'user_id' => $constructor['id'], 'access_hash' => $constructor['access_hash']];
|
||||
} elseif (isset($constructor['user']['access_hash'])) {
|
||||
$res['InputPeer'] = ['_' => 'inputPeerUser', 'user_id' => $constructor['user']['id'], 'access_hash' => $constructor['user']['access_hash']];
|
||||
$res['InputUser'] = ['_' => 'inputUser', 'user_id' => $constructor['user']['id'], 'access_hash' => $constructor['user']['access_hash']];
|
||||
}
|
||||
$res['Peer'] = ['_' => 'peerUser', 'user_id' => $constructor['id']];
|
||||
$res['user_id'] = $constructor['id'];
|
||||
$res['bot_api_id'] = $constructor['id'];
|
||||
$res['Peer'] = ['_' => 'peerUser', 'user_id' => $constructor['user']['id']];
|
||||
$res['user_id'] = $constructor['user']['id'];
|
||||
$res['bot_api_id'] = $constructor['user']['id'];
|
||||
break;
|
||||
case 'chat':
|
||||
$res['InputPeer'] = ['_' => 'inputPeerChat', 'chat_id' => $constructor['id']];
|
||||
$res['Peer'] = ['_' => 'peerChat', 'chat_id' => $constructor['id']];
|
||||
$res['chat_id'] = $constructor['id'];
|
||||
$res['bot_api_id'] = -$constructor['id'];
|
||||
case 'chatFull':
|
||||
$res['Chat'] = $constructor['chat'];
|
||||
$res['InputPeer'] = ['_' => 'inputPeerChat', 'chat_id' => $constructor['chat']['id']];
|
||||
$res['Peer'] = ['_' => 'peerChat', 'chat_id' => $constructor['chat']['id']];
|
||||
$res['chat_id'] = $constructor['chat']['id'];
|
||||
$res['bot_api_id'] = -$constructor['chat']['id'];
|
||||
break;
|
||||
case 'channel':
|
||||
if (isset($constructor['access_hash'])) {
|
||||
$res['InputPeer'] = ['_' => 'inputPeerChannel', 'channel_id' => $constructor['id'], 'access_hash' => $constructor['access_hash']];
|
||||
$res['InputChannel'] = ['_' => 'inputChannel', 'channel_id' => $constructor['id'], 'access_hash' => $constructor['access_hash']];
|
||||
case 'channelFull':
|
||||
$res['Channel'] = $constructor['channel'];
|
||||
if (isset($constructor['channel']['access_hash'])) {
|
||||
$res['InputPeer'] = ['_' => 'inputPeerChannel', 'channel_id' => $constructor['channel']['id'], 'access_hash' => $constructor['channel']['access_hash']];
|
||||
$res['InputChannel'] = ['_' => 'inputChannel', 'channel_id' => $constructor['channel']['id'], 'access_hash' => $constructor['channel']['access_hash']];
|
||||
}
|
||||
$res['Peer'] = ['_' => 'peerChannel', 'channel_id' => $constructor['id']];
|
||||
$res['channel_id'] = $constructor['id'];
|
||||
$res['bot_api_id'] = (int) ('-100'.$constructor['id']);
|
||||
$res['Peer'] = ['_' => 'peerChannel', 'channel_id' => $constructor['channel']['id']];
|
||||
$res['channel_id'] = $constructor['channel']['id'];
|
||||
$res['bot_api_id'] = (int) ('-100'.$constructor['channel']['id']);
|
||||
break;
|
||||
default:
|
||||
throw new \danog\MadelineProto\Exception('Invalid constructor given '.var_export($constructor, true));
|
||||
|
@ -82,8 +82,30 @@ trait ResponseHandler
|
||||
if (isset($response['result']['users'])) {
|
||||
$this->add_users($response['result']['users']);
|
||||
}
|
||||
if (isset($response['result']['_']) && $this->constructors->find_by_predicate($response['result']['_'])['type'] == 'Update') {
|
||||
if (isset($response['result']['chats'])) {
|
||||
$this->add_chats($response['result']['chats']);
|
||||
}
|
||||
if (isset($response['result']['_'])) {
|
||||
switch ($this->constructors->find_by_predicate($response['result']['_'])['type']) {
|
||||
case 'Update':
|
||||
$this->handle_update($response['result']);
|
||||
break;
|
||||
|
||||
case 'userFull':
|
||||
$this->chats[$response['result']['user']['id']] = $response['result'];
|
||||
$this->should_serialize = true;
|
||||
break;
|
||||
|
||||
case 'chatFull':
|
||||
$this->chats[-$response['result']['chat']['id']] = $response['result'];
|
||||
$this->should_serialize = true;
|
||||
break;
|
||||
|
||||
case 'channelFull':
|
||||
$this->chats[(int)('-100'.$response['result']['channel']['id'])] = $response['result'];
|
||||
$this->should_serialize = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
switch ($response['_']) {
|
||||
case 'msgs_ack':
|
||||
|
@ -83,7 +83,6 @@ trait UpdateHandler
|
||||
}
|
||||
$difference = $this->method_call('updates.getChannelDifference', ['channel' => $input, 'filter' => ['_' => 'channelMessagesFilterEmpty'], 'pts' => $this->get_channel_state($channel)['pts'], 'limit' => 30]);
|
||||
\danog\MadelineProto\Logger::log('Got '.$difference['_']);
|
||||
$this->get_channel_state($channel)['sync_loading'] = false;
|
||||
switch ($difference['_']) {
|
||||
case 'updates.channelDifferenceEmpty':
|
||||
$this->set_channel_state($channel, $difference);
|
||||
@ -105,6 +104,7 @@ trait UpdateHandler
|
||||
throw new \danog\MadelineProto\Exception('Unrecognized update difference received: '.var_export($difference, true));
|
||||
break;
|
||||
}
|
||||
$this->get_channel_state($channel)['sync_loading'] = false;
|
||||
}
|
||||
|
||||
public function set_update_state($data)
|
||||
@ -138,7 +138,6 @@ trait UpdateHandler
|
||||
|
||||
$difference = $this->method_call('updates.getDifference', ['pts' => $this->get_update_state()['pts'], 'date' => $this->get_update_state()['date'], 'qts' => -1]);
|
||||
\danog\MadelineProto\Logger::log('Got '.$difference['_']);
|
||||
$this->get_update_state()['sync_loading'] = false;
|
||||
switch ($difference['_']) {
|
||||
case 'updates.differenceEmpty':
|
||||
$this->set_update_state($difference);
|
||||
@ -158,6 +157,7 @@ trait UpdateHandler
|
||||
throw new \danog\MadelineProto\Exception('Unrecognized update difference received: '.var_export($difference, true));
|
||||
break;
|
||||
}
|
||||
$this->get_update_state()['sync_loading'] = false;
|
||||
}
|
||||
|
||||
public function get_updates_state()
|
||||
@ -203,13 +203,13 @@ trait UpdateHandler
|
||||
} else {
|
||||
$cur_state = &$this->get_channel_state($channel_id, (isset($update['pts']) ? $update['pts'] : 0) - (isset($update['pts_count']) ? $update['pts_count'] : 0));
|
||||
}
|
||||
|
||||
/*
|
||||
if ($cur_state['sync_loading']) {
|
||||
\danog\MadelineProto\Logger::log('Sync loading, not handling update');
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
*/
|
||||
switch ($update['_']) {
|
||||
case 'updateChannelTooLong':
|
||||
$this->get_channel_difference($channel_id);
|
||||
@ -235,7 +235,7 @@ trait UpdateHandler
|
||||
|
||||
return false;
|
||||
}
|
||||
if ($message['from_id'] == $this->datacenter->authorization['user']['id']) { $message['out'] = true; }
|
||||
if (isset($message['from_id']) && $message['from_id'] == $this->datacenter->authorization['user']['id']) { $message['out'] = true; }
|
||||
break;
|
||||
default:
|
||||
if ($channel_id !== false && !$this->peer_isset('channel#'.$channel_id)) {
|
||||
@ -266,7 +266,7 @@ trait UpdateHandler
|
||||
if ($update['pts'] > $cur_state['pts']) {
|
||||
$cur_state['pts'] = $update['pts'];
|
||||
$pop_pts = true;
|
||||
} elseif (isset($update['pts_count']) && $update['pts_count']) {
|
||||
} elseif (isset($update['pts_count'])) {
|
||||
\danog\MadelineProto\Logger::log('Duplicate update. current pts: '.$cur_state['pts'].' + pts count: '.(isset($update['pts_count']) ? $update['pts_count'] : 0).' = new pts: '.$new_pts.'. update pts: '.$update['pts'].' <= current pts '.$cur_state['pts'].', channel id: '.$channel_id, $update);
|
||||
|
||||
return false;
|
||||
|
600
src/danog/MadelineProto/Wrappers/FilesHandler.php
Normal file
600
src/danog/MadelineProto/Wrappers/FilesHandler.php
Normal file
@ -0,0 +1,600 @@
|
||||
<?php
|
||||
/*
|
||||
Copyright 2016 Daniil Gentili
|
||||
(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/>.
|
||||
*/
|
||||
|
||||
namespace danog\MadelineProto\Wrappers;
|
||||
|
||||
/**
|
||||
* Manages file upload and download
|
||||
*/
|
||||
trait FilesHandler
|
||||
{
|
||||
public $all_mimes = array (
|
||||
'png' =>
|
||||
array (
|
||||
0 => 'image/png',
|
||||
1 => 'image/x-png',
|
||||
),
|
||||
'bmp' =>
|
||||
array (
|
||||
0 => 'image/bmp',
|
||||
1 => 'image/x-bmp',
|
||||
2 => 'image/x-bitmap',
|
||||
3 => 'image/x-xbitmap',
|
||||
4 => 'image/x-win-bitmap',
|
||||
5 => 'image/x-windows-bmp',
|
||||
6 => 'image/ms-bmp',
|
||||
7 => 'image/x-ms-bmp',
|
||||
8 => 'application/bmp',
|
||||
9 => 'application/x-bmp',
|
||||
10 => 'application/x-win-bitmap',
|
||||
),
|
||||
'gif' =>
|
||||
array (
|
||||
0 => 'image/gif',
|
||||
),
|
||||
'jpeg' =>
|
||||
array (
|
||||
0 => 'image/jpeg',
|
||||
1 => 'image/pjpeg',
|
||||
),
|
||||
'xspf' =>
|
||||
array (
|
||||
0 => 'application/xspf+xml',
|
||||
),
|
||||
'vlc' =>
|
||||
array (
|
||||
0 => 'application/videolan',
|
||||
),
|
||||
'wmv' =>
|
||||
array (
|
||||
0 => 'video/x-ms-wmv',
|
||||
1 => 'video/x-ms-asf',
|
||||
),
|
||||
'au' =>
|
||||
array (
|
||||
0 => 'audio/x-au',
|
||||
),
|
||||
'ac3' =>
|
||||
array (
|
||||
0 => 'audio/ac3',
|
||||
),
|
||||
'flac' =>
|
||||
array (
|
||||
0 => 'audio/x-flac',
|
||||
),
|
||||
'ogg' =>
|
||||
array (
|
||||
0 => 'audio/ogg',
|
||||
1 => 'video/ogg',
|
||||
2 => 'application/ogg',
|
||||
),
|
||||
'kmz' =>
|
||||
array (
|
||||
0 => 'application/vnd.google-earth.kmz',
|
||||
),
|
||||
'kml' =>
|
||||
array (
|
||||
0 => 'application/vnd.google-earth.kml+xml',
|
||||
),
|
||||
'rtx' =>
|
||||
array (
|
||||
0 => 'text/richtext',
|
||||
),
|
||||
'rtf' =>
|
||||
array (
|
||||
0 => 'text/rtf',
|
||||
),
|
||||
'jar' =>
|
||||
array (
|
||||
0 => 'application/java-archive',
|
||||
1 => 'application/x-java-application',
|
||||
2 => 'application/x-jar',
|
||||
),
|
||||
'zip' =>
|
||||
array (
|
||||
0 => 'application/x-zip',
|
||||
1 => 'application/zip',
|
||||
2 => 'application/x-zip-compressed',
|
||||
3 => 'application/s-compressed',
|
||||
4 => 'multipart/x-zip',
|
||||
),
|
||||
'7zip' =>
|
||||
array (
|
||||
0 => 'application/x-compressed',
|
||||
),
|
||||
'xml' =>
|
||||
array (
|
||||
0 => 'application/xml',
|
||||
1 => 'text/xml',
|
||||
),
|
||||
'svg' =>
|
||||
array (
|
||||
0 => 'image/svg+xml',
|
||||
),
|
||||
'3g2' =>
|
||||
array (
|
||||
0 => 'video/3gpp2',
|
||||
),
|
||||
'3gp' =>
|
||||
array (
|
||||
0 => 'video/3gp',
|
||||
1 => 'video/3gpp',
|
||||
),
|
||||
'mp4' =>
|
||||
array (
|
||||
0 => 'video/mp4',
|
||||
),
|
||||
'm4a' =>
|
||||
array (
|
||||
0 => 'audio/x-m4a',
|
||||
),
|
||||
'f4v' =>
|
||||
array (
|
||||
0 => 'video/x-f4v',
|
||||
),
|
||||
'flv' =>
|
||||
array (
|
||||
0 => 'video/x-flv',
|
||||
),
|
||||
'webm' =>
|
||||
array (
|
||||
0 => 'video/webm',
|
||||
),
|
||||
'aac' =>
|
||||
array (
|
||||
0 => 'audio/x-acc',
|
||||
),
|
||||
'm4u' =>
|
||||
array (
|
||||
0 => 'application/vnd.mpegurl',
|
||||
),
|
||||
'pdf' =>
|
||||
array (
|
||||
0 => 'application/pdf',
|
||||
1 => 'application/octet-stream',
|
||||
),
|
||||
'pptx' =>
|
||||
array (
|
||||
0 => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
|
||||
),
|
||||
'ppt' =>
|
||||
array (
|
||||
0 => 'application/powerpoint',
|
||||
1 => 'application/vnd.ms-powerpoint',
|
||||
2 => 'application/vnd.ms-office',
|
||||
3 => 'application/msword',
|
||||
),
|
||||
'docx' =>
|
||||
array (
|
||||
0 => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
|
||||
),
|
||||
'xlsx' =>
|
||||
array (
|
||||
0 => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
|
||||
1 => 'application/vnd.ms-excel',
|
||||
),
|
||||
'xl' =>
|
||||
array (
|
||||
0 => 'application/excel',
|
||||
),
|
||||
'xls' =>
|
||||
array (
|
||||
0 => 'application/msexcel',
|
||||
1 => 'application/x-msexcel',
|
||||
2 => 'application/x-ms-excel',
|
||||
3 => 'application/x-excel',
|
||||
4 => 'application/x-dos_ms_excel',
|
||||
5 => 'application/xls',
|
||||
6 => 'application/x-xls',
|
||||
),
|
||||
'xsl' =>
|
||||
array (
|
||||
0 => 'text/xsl',
|
||||
),
|
||||
'mpeg' =>
|
||||
array (
|
||||
0 => 'video/mpeg',
|
||||
),
|
||||
'mov' =>
|
||||
array (
|
||||
0 => 'video/quicktime',
|
||||
),
|
||||
'avi' =>
|
||||
array (
|
||||
0 => 'video/x-msvideo',
|
||||
1 => 'video/msvideo',
|
||||
2 => 'video/avi',
|
||||
3 => 'application/x-troff-msvideo',
|
||||
),
|
||||
'movie' =>
|
||||
array (
|
||||
0 => 'video/x-sgi-movie',
|
||||
),
|
||||
'log' =>
|
||||
array (
|
||||
0 => 'text/x-log',
|
||||
),
|
||||
'txt' =>
|
||||
array (
|
||||
0 => 'text/plain',
|
||||
),
|
||||
'css' =>
|
||||
array (
|
||||
0 => 'text/css',
|
||||
),
|
||||
'html' =>
|
||||
array (
|
||||
0 => 'text/html',
|
||||
),
|
||||
'wav' =>
|
||||
array (
|
||||
0 => 'audio/x-wav',
|
||||
1 => 'audio/wave',
|
||||
2 => 'audio/wav',
|
||||
),
|
||||
'xhtml' =>
|
||||
array (
|
||||
0 => 'application/xhtml+xml',
|
||||
),
|
||||
'tar' =>
|
||||
array (
|
||||
0 => 'application/x-tar',
|
||||
),
|
||||
'tgz' =>
|
||||
array (
|
||||
0 => 'application/x-gzip-compressed',
|
||||
),
|
||||
'psd' =>
|
||||
array (
|
||||
0 => 'application/x-photoshop',
|
||||
1 => 'image/vnd.adobe.photoshop',
|
||||
),
|
||||
'exe' =>
|
||||
array (
|
||||
0 => 'application/x-msdownload',
|
||||
),
|
||||
'js' =>
|
||||
array (
|
||||
0 => 'application/x-javascript',
|
||||
),
|
||||
'mp3' =>
|
||||
array (
|
||||
0 => 'audio/mpeg',
|
||||
1 => 'audio/mpg',
|
||||
2 => 'audio/mpeg3',
|
||||
3 => 'audio/mp3',
|
||||
),
|
||||
'rar' =>
|
||||
array (
|
||||
0 => 'application/x-rar',
|
||||
1 => 'application/rar',
|
||||
2 => 'application/x-rar-compressed',
|
||||
),
|
||||
'gzip' =>
|
||||
array (
|
||||
0 => 'application/x-gzip',
|
||||
),
|
||||
'hqx' =>
|
||||
array (
|
||||
0 => 'application/mac-binhex40',
|
||||
1 => 'application/mac-binhex',
|
||||
2 => 'application/x-binhex40',
|
||||
3 => 'application/x-mac-binhex40',
|
||||
),
|
||||
'cpt' =>
|
||||
array (
|
||||
0 => 'application/mac-compactpro',
|
||||
),
|
||||
'bin' =>
|
||||
array (
|
||||
0 => 'application/macbinary',
|
||||
1 => 'application/mac-binary',
|
||||
2 => 'application/x-binary',
|
||||
3 => 'application/x-macbinary',
|
||||
),
|
||||
'oda' =>
|
||||
array (
|
||||
0 => 'application/oda',
|
||||
),
|
||||
'ai' =>
|
||||
array (
|
||||
0 => 'application/postscript',
|
||||
),
|
||||
'smil' =>
|
||||
array (
|
||||
0 => 'application/smil',
|
||||
),
|
||||
'mif' =>
|
||||
array (
|
||||
0 => 'application/vnd.mif',
|
||||
),
|
||||
'wbxml' =>
|
||||
array (
|
||||
0 => 'application/wbxml',
|
||||
),
|
||||
'wmlc' =>
|
||||
array (
|
||||
0 => 'application/wmlc',
|
||||
),
|
||||
'dcr' =>
|
||||
array (
|
||||
0 => 'application/x-director',
|
||||
),
|
||||
'dvi' =>
|
||||
array (
|
||||
0 => 'application/x-dvi',
|
||||
),
|
||||
'gtar' =>
|
||||
array (
|
||||
0 => 'application/x-gtar',
|
||||
),
|
||||
'php' =>
|
||||
array (
|
||||
0 => 'application/x-httpd-php',
|
||||
1 => 'application/php',
|
||||
2 => 'application/x-php',
|
||||
3 => 'text/php',
|
||||
4 => 'text/x-php',
|
||||
5 => 'application/x-httpd-php-source',
|
||||
),
|
||||
'swf' =>
|
||||
array (
|
||||
0 => 'application/x-shockwave-flash',
|
||||
),
|
||||
'sit' =>
|
||||
array (
|
||||
0 => 'application/x-stuffit',
|
||||
),
|
||||
'z' =>
|
||||
array (
|
||||
0 => 'application/x-compress',
|
||||
),
|
||||
'mid' =>
|
||||
array (
|
||||
0 => 'audio/midi',
|
||||
),
|
||||
'aif' =>
|
||||
array (
|
||||
0 => 'audio/x-aiff',
|
||||
1 => 'audio/aiff',
|
||||
),
|
||||
'ram' =>
|
||||
array (
|
||||
0 => 'audio/x-pn-realaudio',
|
||||
),
|
||||
'rpm' =>
|
||||
array (
|
||||
0 => 'audio/x-pn-realaudio-plugin',
|
||||
),
|
||||
'ra' =>
|
||||
array (
|
||||
0 => 'audio/x-realaudio',
|
||||
),
|
||||
'rv' =>
|
||||
array (
|
||||
0 => 'video/vnd.rn-realvideo',
|
||||
),
|
||||
'jp2' =>
|
||||
array (
|
||||
0 => 'image/jp2',
|
||||
1 => 'video/mj2',
|
||||
2 => 'image/jpx',
|
||||
3 => 'image/jpm',
|
||||
),
|
||||
'tiff' =>
|
||||
array (
|
||||
0 => 'image/tiff',
|
||||
),
|
||||
'eml' =>
|
||||
array (
|
||||
0 => 'message/rfc822',
|
||||
),
|
||||
'pem' =>
|
||||
array (
|
||||
0 => 'application/x-x509-user-cert',
|
||||
1 => 'application/x-pem-file',
|
||||
),
|
||||
'p10' =>
|
||||
array (
|
||||
0 => 'application/x-pkcs10',
|
||||
1 => 'application/pkcs10',
|
||||
),
|
||||
'p12' =>
|
||||
array (
|
||||
0 => 'application/x-pkcs12',
|
||||
),
|
||||
'p7a' =>
|
||||
array (
|
||||
0 => 'application/x-pkcs7-signature',
|
||||
),
|
||||
'p7c' =>
|
||||
array (
|
||||
0 => 'application/pkcs7-mime',
|
||||
1 => 'application/x-pkcs7-mime',
|
||||
),
|
||||
'p7r' =>
|
||||
array (
|
||||
0 => 'application/x-pkcs7-certreqresp',
|
||||
),
|
||||
'p7s' =>
|
||||
array (
|
||||
0 => 'application/pkcs7-signature',
|
||||
),
|
||||
'crt' =>
|
||||
array (
|
||||
0 => 'application/x-x509-ca-cert',
|
||||
1 => 'application/pkix-cert',
|
||||
),
|
||||
'crl' =>
|
||||
array (
|
||||
0 => 'application/pkix-crl',
|
||||
1 => 'application/pkcs-crl',
|
||||
),
|
||||
'pgp' =>
|
||||
array (
|
||||
0 => 'application/pgp',
|
||||
),
|
||||
'gpg' =>
|
||||
array (
|
||||
0 => 'application/gpg-keys',
|
||||
),
|
||||
'rsa' =>
|
||||
array (
|
||||
0 => 'application/x-pkcs7',
|
||||
),
|
||||
'ics' =>
|
||||
array (
|
||||
0 => 'text/calendar',
|
||||
),
|
||||
'zsh' =>
|
||||
array (
|
||||
0 => 'text/x-scriptzsh',
|
||||
),
|
||||
'cdr' =>
|
||||
array (
|
||||
0 => 'application/cdr',
|
||||
1 => 'application/coreldraw',
|
||||
2 => 'application/x-cdr',
|
||||
3 => 'application/x-coreldraw',
|
||||
4 => 'image/cdr',
|
||||
5 => 'image/x-cdr',
|
||||
6 => 'zz-application/zz-winassoc-cdr',
|
||||
),
|
||||
'wma' =>
|
||||
array (
|
||||
0 => 'audio/x-ms-wma',
|
||||
),
|
||||
'vcf' =>
|
||||
array (
|
||||
0 => 'text/x-vcard',
|
||||
),
|
||||
'srt' =>
|
||||
array (
|
||||
0 => 'text/srt',
|
||||
),
|
||||
'vtt' =>
|
||||
array (
|
||||
0 => 'text/vtt',
|
||||
),
|
||||
'ico' =>
|
||||
array (
|
||||
0 => 'image/x-icon',
|
||||
1 => 'image/x-ico',
|
||||
2 => 'image/vnd.microsoft.icon',
|
||||
),
|
||||
'csv' =>
|
||||
array (
|
||||
0 => 'text/x-comma-separated-values',
|
||||
1 => 'text/comma-separated-values',
|
||||
2 => 'application/vnd.msexcel',
|
||||
),
|
||||
'json' =>
|
||||
array (
|
||||
0 => 'application/json',
|
||||
1 => 'text/json',
|
||||
),
|
||||
);
|
||||
public function upload($file, $file_name = '', $cb = null) {
|
||||
if (!file_exists($file)) throw new \danog\MadelineProto\Exception('Given file does not exist!');
|
||||
if (empty($file_name)) $file_name = basename($file);
|
||||
$file_size = filesize($file);
|
||||
if ($file_size > 1500 * 1024 * 1024) throw new \danog\MadelineProto\Exception('Given file is too big!');
|
||||
if ($cb === null) $cb = function ($percent) { \danog\MadelineProto\Logger::log('Upload status: '.$percent.'%'); };
|
||||
$part_size = 512 * 1024;
|
||||
$part_total_num = (int)ceil($file_size/$part_size);
|
||||
$part_num = 0;
|
||||
$method = $file_size > 10 * 1024 * 1024 ? 'upload.saveBigFilePart' : 'upload.saveFilePart';
|
||||
$constructor = $file_size > 10 * 1024 * 1024 ? 'inputFileBig' : 'inputFile';
|
||||
$file_id = \danog\PHP\Struct::unpack('<q', \phpseclib\Crypt\Random::string(8))[0];
|
||||
$f = fopen($file, 'r');
|
||||
fseek($f, 0);
|
||||
while (ftell($f) !== $file_size) {
|
||||
if (!$this->API->method_call($method, ['file_id' => $file_id, 'file_part' => $part_num++, 'file_total_parts' => $part_total_num, 'bytes' => stream_get_contents($f, $part_size)])) throw new \danog\MadelineProto\Exception('An error occurred while uploading file part '.$part_num);
|
||||
$cb(ftell($f) * 100 / $file_size);
|
||||
}
|
||||
fclose($f);
|
||||
return ['_' => $constructor, 'id' => $file_id, 'parts' => $part_total_num, 'name' => $file_name, 'md5_checksum' => md5_file($file)];
|
||||
}
|
||||
|
||||
public function get_extension_from_mime($mime) {
|
||||
foreach ($this->all_mimes as $key => $value) {
|
||||
if(array_search($mime,$value) !== false) return '.'.$key;
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
public function get_download_info($message_media) {
|
||||
$res = [];
|
||||
switch ($message_media['_']) {
|
||||
case 'messageMediaPhoto':
|
||||
$res['caption'] = $message_media['caption'];
|
||||
$res['ext'] = '.jpg';
|
||||
$photo = end($message_media['photo']['sizes']);
|
||||
$res['name'] = $photo['location']['volume_id'].'_'.$photo['location']['local_id'];
|
||||
$res['size'] = $photo['size'];
|
||||
$res['InputFileLocation'] = ['_' => 'inputFileLocation', 'volume_id' => $photo['location']['volume_id'], 'local_id' => $photo['location']['local_id'], 'secret' => $photo['location']['secret']];
|
||||
return $res;
|
||||
|
||||
case 'messageMediaDocument':
|
||||
$res['caption'] = $message_media['caption'];
|
||||
foreach ($message_media['document']['attributes'] as $attribute) {
|
||||
switch ($attribute['_']) {
|
||||
case 'documentAttributeFilename':
|
||||
$pathinfo = pathinfo($attribute['file_name']);
|
||||
$res['ext'] = '.'.$pathinfo['extension'];
|
||||
$res['name'] = $pathinfo['filename'];
|
||||
break;
|
||||
|
||||
case 'documentAttributeAudio':
|
||||
$audio = $attribute;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (isset($audio) && isset($audio['title']) && !isset($res['name'])) {
|
||||
$res['name'] = $audio['title'];
|
||||
if (isset($audio['performer'])) $res['name'] .= ' - '.$audio['performer'];
|
||||
}
|
||||
if (!isset($res['ext'])) $res['ext'] = $this->get_extension_from_mime($message_media['document']['mime_type']);
|
||||
if (!isset($res['name'])) $res['name'] = $message_media['document']['id'];
|
||||
$res['name'] .= '_'.$message_media['document']['id'];
|
||||
$res['size'] = $message_media['document']['size'];
|
||||
|
||||
$res['InputFileLocation'] = ['_' => 'inputDocumentFileLocation', 'id' => $message_media['document']['id'], 'access_hash' => $message_media['document']['access_hash'], 'version' => $message_media['document']['version']];
|
||||
return $res;
|
||||
|
||||
default:
|
||||
throw \danog\MadelineProto\Exception('Invalid constructor provided: '.$message_media['_']);
|
||||
}
|
||||
}
|
||||
public function download_to_dir($message_media, $dir, $cb = null) {
|
||||
$info = $this->get_download_info($message_media);
|
||||
return $this->download_to_file($message_media, $dir.'/'.$info['name'].$info['ext'], $cb);
|
||||
}
|
||||
|
||||
public function download_to_file($message_media, $file, $cb = null) {
|
||||
if (!file_exists($file)) touch($file);
|
||||
$stream = fopen($file, 'w');
|
||||
fseek($stream, filesize($file));
|
||||
$this->download_to_stream($message_media, $stream, $cb);
|
||||
return $file;
|
||||
}
|
||||
public function download_to_stream($message_media, $stream, $cb = null) {
|
||||
if ($cb === null) $cb = function ($percent) { \danog\MadelineProto\Logger::log('Download status: '.$percent.'%'); };
|
||||
$info = $this->get_download_info($message_media);
|
||||
$offset = ftell($stream);
|
||||
$part_size = 512 * 1024;
|
||||
while (fwrite($stream, $this->API->method_call('upload.getFile', ['location' => $info['InputFileLocation'], 'offset' => $offset += $part_size, 'limit' => $part_size])['bytes']) != false) {
|
||||
$cb(ftell($stream) * 100 / $info['size']);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
}
|
39
testing.php
39
testing.php
@ -46,9 +46,48 @@ $flutter = 'https://storage.pwrtelegram.xyz/pwrtelegrambot/document/file_6570.mp
|
||||
$mention = $MadelineProto->get_info('@danogentili'); // Returns an array with all of the constructors that can be extracted from a username or an id
|
||||
$mention = $mention['user_id']; // Selects only the numeric user id
|
||||
|
||||
$media = [];
|
||||
|
||||
// Photo uploaded as document
|
||||
$inputFile = $MadelineProto->upload('faust.jpg', 'fausticorn.jpg'); // This gets an inputFile object with file name magic
|
||||
$media['document_photo'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('faust.jpg'), 'caption' => 'This file was uploaded using MadelineProto', 'attributes' => [['_' => 'documentAttributeImageSize', 'w' => 1280, 'h' => 914]]];
|
||||
|
||||
// Photo
|
||||
$media['photo'] = ['_' => 'inputMediaUploadedPhoto', 'file' => $inputFile, 'mime_type' => mime_content_type('faust.jpg'), 'caption' => 'This photo was uploaded using MadelineProto'];
|
||||
|
||||
// GIF
|
||||
$inputFile = $MadelineProto->upload('pony.mp4');
|
||||
$media['gif'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('pony.mp4'), 'caption' => 'test', 'attributes' => [['_' => 'documentAttributeAnimated']]];
|
||||
|
||||
// Sticker
|
||||
$inputFile = $MadelineProto->upload('lel.webp');
|
||||
$media['sticker'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('lel.webp'), 'caption' => 'test', 'attributes' => [['_' => 'documentAttributeSticker', 'alt' => 'LEL', 'stickerset' => ['_' => 'inputStickerSetEmpty']]]];
|
||||
|
||||
// Video
|
||||
$inputFile = $MadelineProto->upload('swing.mp4');
|
||||
$media['video'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('swing.mp4'), 'caption' => 'test', 'attributes' => [['_' => 'documentAttributeVideo', 'duration' => 5, 'w' => 1280, 'h' => 720]]];
|
||||
|
||||
// audio
|
||||
$inputFile = $MadelineProto->upload('mosconi.mp3');
|
||||
$media['audio'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('mosconi.mp3'), 'caption' => 'test', 'attributes' => [['_' => 'documentAttributeAudio', 'voice' => false, 'duration' => 1, 'title' => 'AH NON LO SO', 'performer' => 'IL DIO GERARDO MOSCONI']]];
|
||||
|
||||
// voice
|
||||
$media['voice'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => mime_content_type('mosconi.mp3'), 'caption' => 'test', 'attributes' => [['_' => 'documentAttributeAudio', 'voice' => true, 'duration' => 1, 'title' => 'AH NON LO SO', 'performer' => 'IL DIO GERARDO MOSCONI']]];
|
||||
|
||||
// Document
|
||||
$time = time();
|
||||
$inputFile = $MadelineProto->upload('60', 'magic'); // This gets an inputFile object with file name magic
|
||||
var_dump(time() - $time);
|
||||
$media['document'] = ['_' => 'inputMediaUploadedDocument', 'file' => $inputFile, 'mime_type' => 'magic/magic', 'caption' => 'This file was uploaded using MadelineProto', 'attributes' => [['_' => 'documentAttributeFilename', 'file_name' => 'magic.magic']]];
|
||||
|
||||
|
||||
|
||||
foreach (['@pwrtelegramgroup', '@pwrtelegramgroupita'] as $peer) {
|
||||
$sentMessage = $MadelineProto->messages->sendMessage(['peer' => $peer, 'message' => $message, 'entities' => [['_' => 'inputMessageEntityMentionName', 'offset' => 0, 'length' => strlen($message), 'user_id' => $mention]]]);
|
||||
\danog\MadelineProto\Logger::log($sentMessage);
|
||||
foreach ($media as $type => $inputMedia) {
|
||||
\danog\MadelineProto\Logger::log($MadelineProto->messages->sendMedia(['peer' => $peer, 'media' => $inputMedia]));
|
||||
}
|
||||
}
|
||||
|
||||
sleep(5);
|
||||
|
Loading…
Reference in New Issue
Block a user