| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | /*
 | 
					
						
							|  |  |  | This file is part of Telegram Desktop, | 
					
						
							|  |  |  | the official desktop application for the Telegram messaging service. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | For license and copyright information please follow this link: | 
					
						
							|  |  |  | https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #include "mtproto/dedicated_file_loader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-13 11:31:12 +03:00
										 |  |  | #include "mtproto/facade.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-06 12:37:12 +03:00
										 |  |  | #include "main/main_account.h" // Account::sessionChanges.
 | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | #include "main/main_session.h" // Session::account.
 | 
					
						
							| 
									
										
										
										
											2019-09-26 13:55:35 +03:00
										 |  |  | #include "core/application.h"
 | 
					
						
							|  |  |  | #include "base/call_delayed.h"
 | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace MTP { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<MTPInputChannel> ExtractChannel( | 
					
						
							|  |  |  | 		const MTPcontacts_ResolvedPeer &result) { | 
					
						
							|  |  |  | 	const auto &data = result.c_contacts_resolvedPeer(); | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	if (const auto peer = peerFromMTP(data.vpeer())) { | 
					
						
							|  |  |  | 		for (const auto &chat : data.vchats().v) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			if (chat.type() == mtpc_channel) { | 
					
						
							|  |  |  | 				const auto &channel = chat.c_channel(); | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 				if (peer == peerFromChannel(channel.vid())) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 					return MTP_inputChannel( | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 						channel.vid(), | 
					
						
							|  |  |  | 						MTP_long(channel.vaccess_hash().value_or_empty())); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return std::nullopt; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<DedicatedLoader::File> ParseFile( | 
					
						
							|  |  |  | 		const MTPmessages_Messages &result) { | 
					
						
							|  |  |  | 	const auto message = GetMessagesElement(result); | 
					
						
							|  |  |  | 	if (!message || message->type() != mtpc_message) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP file message not found.")); | 
					
						
							|  |  |  | 		return std::nullopt; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto &data = message->c_message(); | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	const auto media = data.vmedia(); | 
					
						
							|  |  |  | 	if (!media || media->type() != mtpc_messageMediaDocument) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		LOG(("Update Error: MTP file media not found.")); | 
					
						
							|  |  |  | 		return std::nullopt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	const auto &inner = media->c_messageMediaDocument(); | 
					
						
							|  |  |  | 	const auto document = inner.vdocument(); | 
					
						
							|  |  |  | 	if (!document || document->type() != mtpc_document) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		LOG(("Update Error: MTP file not found.")); | 
					
						
							|  |  |  | 		return std::nullopt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	const auto &fields = document->c_document(); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	const auto name = [&] { | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 		for (const auto &attribute : fields.vattributes().v) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			if (attribute.type() == mtpc_documentAttributeFilename) { | 
					
						
							|  |  |  | 				const auto &data = attribute.c_documentAttributeFilename(); | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 				return qs(data.vfile_name()); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return QString(); | 
					
						
							|  |  |  | 	}(); | 
					
						
							|  |  |  | 	if (name.isEmpty()) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP file name not found.")); | 
					
						
							|  |  |  | 		return std::nullopt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	const auto size = fields.vsize().v; | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	if (size <= 0) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP file size is invalid.")); | 
					
						
							|  |  |  | 		return std::nullopt; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto location = MTP_inputDocumentFileLocation( | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 		fields.vid(), | 
					
						
							|  |  |  | 		fields.vaccess_hash(), | 
					
						
							|  |  |  | 		fields.vfile_reference(), | 
					
						
							|  |  |  | 		MTP_string()); | 
					
						
							|  |  |  | 	return DedicatedLoader::File{ name, size, fields.vdc_id().v, location }; | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | WeakInstance::WeakInstance(base::weak_ptr<Main::Session> session) | 
					
						
							|  |  |  | : _session(session) | 
					
						
							| 
									
										
										
										
											2020-06-17 13:36:25 +04:00
										 |  |  | , _instance(_session ? &_session->account().mtp() : nullptr) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	if (!valid()) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	connect(_instance, &QObject::destroyed, this, [=] { | 
					
						
							|  |  |  | 		_instance = nullptr; | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 		_session = nullptr; | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		die(); | 
					
						
							|  |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	_session->account().sessionChanges( | 
					
						
							| 
									
										
										
										
											2019-07-24 13:45:24 +02:00
										 |  |  | 	) | rpl::filter([](Main::Session *session) { | 
					
						
							| 
									
										
										
										
											2019-06-06 12:37:12 +03:00
										 |  |  | 		return !session; | 
					
						
							|  |  |  | 	}) | rpl::start_with_next([=] { | 
					
						
							|  |  |  | 		die(); | 
					
						
							|  |  |  | 	}, _lifetime); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | base::weak_ptr<Main::Session> WeakInstance::session() const { | 
					
						
							|  |  |  | 	return _session; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | bool WeakInstance::valid() const { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	return (_session != nullptr); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | Instance *WeakInstance::instance() const { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	return _instance; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WeakInstance::die() { | 
					
						
							|  |  |  | 	for (const auto &[requestId, fail] : base::take(_requests)) { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 		if (_instance) { | 
					
						
							|  |  |  | 			_instance->cancel(requestId); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-01-13 17:28:05 +04:00
										 |  |  | 		fail(RPCError::Local( | 
					
						
							|  |  |  | 			"UNAVAILABLE", | 
					
						
							|  |  |  | 			"MTP instance is not available.")); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WeakInstance::removeRequest(mtpRequestId requestId) { | 
					
						
							|  |  |  | 	if (const auto i = _requests.find(requestId); i != end(_requests)) { | 
					
						
							|  |  |  | 		_requests.erase(i); | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WeakInstance::reportUnavailable( | 
					
						
							|  |  |  | 		Fn<void(const RPCError &error)> callback) { | 
					
						
							|  |  |  | 	InvokeQueued(this, [=] { | 
					
						
							| 
									
										
										
										
											2019-01-13 17:28:05 +04:00
										 |  |  | 		callback(RPCError::Local( | 
					
						
							|  |  |  | 			"UNAVAILABLE", | 
					
						
							|  |  |  | 			"MTP instance is not available.")); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WeakInstance::~WeakInstance() { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	if (_instance) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		for (const auto &[requestId, fail] : base::take(_requests)) { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 			_instance->cancel(requestId); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AbstractDedicatedLoader::AbstractDedicatedLoader( | 
					
						
							|  |  |  | 	const QString &filepath, | 
					
						
							|  |  |  | 	int chunkSize) | 
					
						
							|  |  |  | : _filepath(filepath) | 
					
						
							|  |  |  | , _chunkSize(chunkSize) { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::start() { | 
					
						
							|  |  |  | 	if (!validateOutput() | 
					
						
							|  |  |  | 		|| (!_output.isOpen() && !_output.open(QIODevice::Append))) { | 
					
						
							|  |  |  | 		QFile(_filepath).remove(); | 
					
						
							|  |  |  | 		threadSafeFailed(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	LOG(("Update Info: Starting loading '%1' from %2 offset." | 
					
						
							|  |  |  | 		).arg(_filepath | 
					
						
							|  |  |  | 		).arg(alreadySize())); | 
					
						
							|  |  |  | 	startLoading(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AbstractDedicatedLoader::alreadySize() const { | 
					
						
							|  |  |  | 	QMutexLocker lock(&_sizesMutex); | 
					
						
							|  |  |  | 	return _alreadySize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AbstractDedicatedLoader::totalSize() const { | 
					
						
							|  |  |  | 	QMutexLocker lock(&_sizesMutex); | 
					
						
							|  |  |  | 	return _totalSize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rpl::producer<QString> AbstractDedicatedLoader::ready() const { | 
					
						
							|  |  |  | 	return _ready.events(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | auto AbstractDedicatedLoader::progress() const -> rpl::producer<Progress> { | 
					
						
							|  |  |  | 	return _progress.events(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rpl::producer<> AbstractDedicatedLoader::failed() const { | 
					
						
							|  |  |  | 	return _failed.events(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::wipeFolder() { | 
					
						
							|  |  |  | 	QFileInfo info(_filepath); | 
					
						
							|  |  |  | 	const auto dir = info.dir(); | 
					
						
							|  |  |  | 	const auto all = dir.entryInfoList(QDir::Files); | 
					
						
							|  |  |  | 	for (auto i = all.begin(), e = all.end(); i != e; ++i) { | 
					
						
							|  |  |  | 		if (i->absoluteFilePath() != info.absoluteFilePath()) { | 
					
						
							|  |  |  | 			QFile::remove(i->absoluteFilePath()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AbstractDedicatedLoader::validateOutput() { | 
					
						
							|  |  |  | 	if (_filepath.isEmpty()) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	QFileInfo info(_filepath); | 
					
						
							|  |  |  | 	const auto dir = info.dir(); | 
					
						
							|  |  |  | 	if (!dir.exists()) { | 
					
						
							|  |  |  | 		dir.mkdir(dir.absolutePath()); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	_output.setFileName(_filepath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!info.exists()) { | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto fullSize = info.size(); | 
					
						
							|  |  |  | 	if (fullSize < _chunkSize || fullSize > kMaxFileSize) { | 
					
						
							|  |  |  | 		return _output.remove(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto goodSize = int((fullSize % _chunkSize) | 
					
						
							|  |  |  | 		? (fullSize - (fullSize % _chunkSize)) | 
					
						
							|  |  |  | 		: fullSize); | 
					
						
							|  |  |  | 	if (_output.resize(goodSize)) { | 
					
						
							|  |  |  | 		_alreadySize = goodSize; | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::threadSafeProgress(Progress progress) { | 
					
						
							|  |  |  | 	crl::on_main(this, [=] { | 
					
						
							|  |  |  | 		_progress.fire_copy(progress); | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::threadSafeReady() { | 
					
						
							|  |  |  | 	crl::on_main(this, [=] { | 
					
						
							|  |  |  | 		_ready.fire_copy(_filepath); | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::threadSafeFailed() { | 
					
						
							|  |  |  | 	crl::on_main(this, [=] { | 
					
						
							|  |  |  | 		_failed.fire({}); | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AbstractDedicatedLoader::writeChunk(bytes::const_span data, int totalSize) { | 
					
						
							|  |  |  | 	const auto size = data.size(); | 
					
						
							|  |  |  | 	if (size > 0) { | 
					
						
							|  |  |  | 		const auto written = _output.write(QByteArray::fromRawData( | 
					
						
							|  |  |  | 			reinterpret_cast<const char*>(data.data()), | 
					
						
							|  |  |  | 			size)); | 
					
						
							|  |  |  | 		if (written != size) { | 
					
						
							|  |  |  | 			threadSafeFailed(); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto progress = [&] { | 
					
						
							|  |  |  | 		QMutexLocker lock(&_sizesMutex); | 
					
						
							|  |  |  | 		if (!_totalSize) { | 
					
						
							|  |  |  | 			_totalSize = totalSize; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		_alreadySize += size; | 
					
						
							|  |  |  | 		return Progress { _alreadySize, _totalSize }; | 
					
						
							|  |  |  | 	}(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (progress.size > 0 && progress.already >= progress.size) { | 
					
						
							|  |  |  | 		_output.close(); | 
					
						
							|  |  |  | 		threadSafeReady(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		threadSafeProgress(progress); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | rpl::lifetime &AbstractDedicatedLoader::lifetime() { | 
					
						
							|  |  |  | 	return _lifetime; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DedicatedLoader::DedicatedLoader( | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	base::weak_ptr<Main::Session> session, | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	const QString &folder, | 
					
						
							|  |  |  | 	const File &file) | 
					
						
							|  |  |  | : AbstractDedicatedLoader(folder + '/' + file.name, kChunkSize) | 
					
						
							|  |  |  | , _size(file.size) | 
					
						
							|  |  |  | , _dcId(file.dcId) | 
					
						
							|  |  |  | , _location(file.location) | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | , _mtp(session) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	Expects(_size > 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DedicatedLoader::startLoading() { | 
					
						
							|  |  |  | 	if (!_mtp.valid()) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP is unavailable.")); | 
					
						
							|  |  |  | 		threadSafeFailed(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	LOG(("Update Info: Loading using MTP from '%1'.").arg(_dcId)); | 
					
						
							|  |  |  | 	_offset = alreadySize(); | 
					
						
							|  |  |  | 	writeChunk({}, _size); | 
					
						
							|  |  |  | 	sendRequest(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DedicatedLoader::sendRequest() { | 
					
						
							|  |  |  | 	if (_requests.size() >= kRequestsCount || _offset >= _size) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto offset = _offset; | 
					
						
							|  |  |  | 	_requests.push_back({ offset }); | 
					
						
							|  |  |  | 	_mtp.send( | 
					
						
							| 
									
										
										
										
											2019-08-01 10:22:49 +01:00
										 |  |  | 		MTPupload_GetFile( | 
					
						
							|  |  |  | 			MTP_flags(0), | 
					
						
							|  |  |  | 			_location, | 
					
						
							|  |  |  | 			MTP_int(offset), | 
					
						
							|  |  |  | 			MTP_int(kChunkSize)), | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		[=](const MTPupload_File &result) { gotPart(offset, result); }, | 
					
						
							|  |  |  | 		failHandler(), | 
					
						
							|  |  |  | 		MTP::updaterDcId(_dcId)); | 
					
						
							|  |  |  | 	_offset += kChunkSize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (_requests.size() < kRequestsCount) { | 
					
						
							| 
									
										
										
										
											2019-09-26 13:55:35 +03:00
										 |  |  | 		base::call_delayed(kNextRequestDelay, this, [=] { sendRequest(); }); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DedicatedLoader::gotPart(int offset, const MTPupload_File &result) { | 
					
						
							|  |  |  | 	Expects(!_requests.empty()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (result.type() == mtpc_upload_fileCdnRedirect) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP does not support cdn right now.")); | 
					
						
							|  |  |  | 		threadSafeFailed(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto &data = result.c_upload_file(); | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	if (data.vbytes().v.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		LOG(("Update Error: MTP empty part received.")); | 
					
						
							|  |  |  | 		threadSafeFailed(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto i = ranges::find( | 
					
						
							|  |  |  | 		_requests, | 
					
						
							|  |  |  | 		offset, | 
					
						
							|  |  |  | 		[](const Request &request) { return request.offset; }); | 
					
						
							|  |  |  | 	Assert(i != end(_requests)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 	i->bytes = data.vbytes().v; | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 	while (!_requests.empty() && !_requests.front().bytes.isEmpty()) { | 
					
						
							|  |  |  | 		writeChunk(bytes::make_span(_requests.front().bytes), _size); | 
					
						
							|  |  |  | 		_requests.pop_front(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sendRequest(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fn<void(const RPCError &)> DedicatedLoader::failHandler() { | 
					
						
							|  |  |  | 	return [=](const RPCError &error) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP load failed with '%1'" | 
					
						
							|  |  |  | 			).arg(QString::number(error.code()) + ':' + error.type())); | 
					
						
							|  |  |  | 		threadSafeFailed(); | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ResolveChannel( | 
					
						
							|  |  |  | 		not_null<MTP::WeakInstance*> mtp, | 
					
						
							|  |  |  | 		const QString &username, | 
					
						
							|  |  |  | 		Fn<void(const MTPInputChannel &channel)> done, | 
					
						
							|  |  |  | 		Fn<void()> fail) { | 
					
						
							|  |  |  | 	const auto failed = [&] { | 
					
						
							|  |  |  | 		LOG(("Dedicated MTP Error: Channel '%1' resolve failed." | 
					
						
							|  |  |  | 			).arg(username)); | 
					
						
							|  |  |  | 		fail(); | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	const auto session = mtp->session(); | 
					
						
							|  |  |  | 	if (!mtp->valid()) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		failed(); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct ResolveResult { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 		base::weak_ptr<Main::Session> session; | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 		MTPInputChannel channel; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	static std::map<QString, ResolveResult> ResolveCache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto i = ResolveCache.find(username); | 
					
						
							|  |  |  | 	if (i != end(ResolveCache)) { | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 		if (i->second.session.get() == session.get()) { | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			done(i->second.channel); | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ResolveCache.erase(i); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto doneHandler = [=](const MTPcontacts_ResolvedPeer &result) { | 
					
						
							|  |  |  | 		Expects(result.type() == mtpc_contacts_resolvedPeer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (const auto channel = ExtractChannel(result)) { | 
					
						
							|  |  |  | 			ResolveCache.emplace( | 
					
						
							|  |  |  | 				username, | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 				ResolveResult { session, *channel }); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			done(*channel); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			failed(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	const auto failHandler = [=](const RPCError &error) { | 
					
						
							|  |  |  | 		LOG(("Dedicated MTP Error: Resolve failed with '%1'" | 
					
						
							|  |  |  | 			).arg(QString::number(error.code()) + ':' + error.type())); | 
					
						
							|  |  |  | 		fail(); | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	mtp->send( | 
					
						
							|  |  |  | 		MTPcontacts_ResolveUsername(MTP_string(username)), | 
					
						
							|  |  |  | 		doneHandler, | 
					
						
							|  |  |  | 		failHandler); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<MTPMessage> GetMessagesElement( | 
					
						
							|  |  |  | 		const MTPmessages_Messages &list) { | 
					
						
							| 
									
										
										
										
											2019-01-15 15:57:45 +04:00
										 |  |  | 	return list.match([&](const MTPDmessages_messagesNotModified &) { | 
					
						
							|  |  |  | 		return std::optional<MTPMessage>(std::nullopt); | 
					
						
							|  |  |  | 	}, [&](const auto &data) { | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 		return data.vmessages().v.isEmpty() | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			? std::nullopt | 
					
						
							| 
									
										
										
										
											2019-07-05 15:38:38 +02:00
										 |  |  | 			: std::make_optional(data.vmessages().v[0]); | 
					
						
							| 
									
										
										
										
											2019-01-15 15:57:45 +04:00
										 |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void StartDedicatedLoader( | 
					
						
							|  |  |  | 		not_null<MTP::WeakInstance*> mtp, | 
					
						
							|  |  |  | 		const DedicatedLoader::Location &location, | 
					
						
							|  |  |  | 		const QString &folder, | 
					
						
							|  |  |  | 		Fn<void(std::unique_ptr<DedicatedLoader>)> ready) { | 
					
						
							|  |  |  | 	const auto doneHandler = [=](const MTPmessages_Messages &result) { | 
					
						
							|  |  |  | 		const auto file = ParseFile(result); | 
					
						
							|  |  |  | 		ready(file | 
					
						
							|  |  |  | 			? std::make_unique<MTP::DedicatedLoader>( | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 				mtp->session(), | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 				folder, | 
					
						
							|  |  |  | 				*file) | 
					
						
							|  |  |  | 			: nullptr); | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	const auto failHandler = [=](const RPCError &error) { | 
					
						
							|  |  |  | 		LOG(("Update Error: MTP check failed with '%1'" | 
					
						
							|  |  |  | 			).arg(QString::number(error.code()) + ':' + error.type())); | 
					
						
							|  |  |  | 		ready(nullptr); | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto [username, postId] = location; | 
					
						
							| 
									
										
										
										
											2020-06-10 17:24:41 +04:00
										 |  |  | 	ResolveChannel(mtp, username, [=, postId = postId]( | 
					
						
							| 
									
										
										
										
											2018-12-26 21:05:06 +04:00
										 |  |  | 			const MTPInputChannel &channel) { | 
					
						
							|  |  |  | 		mtp->send( | 
					
						
							|  |  |  | 			MTPchannels_GetMessages( | 
					
						
							|  |  |  | 				channel, | 
					
						
							|  |  |  | 				MTP_vector<MTPInputMessage>( | 
					
						
							|  |  |  | 					1, | 
					
						
							|  |  |  | 					MTP_inputMessageID(MTP_int(postId)))), | 
					
						
							|  |  |  | 			doneHandler, | 
					
						
							|  |  |  | 			failHandler); | 
					
						
							|  |  |  | 	}, [=] { ready(nullptr); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace MTP
 |