mirror of https://github.com/procxx/kepka.git
				
				
				
			Support file reference refresh in Export.
This commit is contained in:
		
							parent
							
								
									64535251e8
								
							
						
					
					
						commit
						ae98e4ae44
					
				|  | @ -209,6 +209,32 @@ Utf8String FillLeft(const Utf8String &data, int length, char filler) { | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool RefreshFileReference(FileLocation &to, const FileLocation &from) { | ||||||
|  | 	if (to.dcId != from.dcId || to.data.type() != from.data.type()) { | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 	if (to.data.type() == mtpc_inputPhotoFileLocation) { | ||||||
|  | 		const auto &toData = to.data.c_inputPhotoFileLocation(); | ||||||
|  | 		const auto &fromData = from.data.c_inputPhotoFileLocation(); | ||||||
|  | 		if (toData.vid().v != fromData.vid().v | ||||||
|  | 			|| toData.vthumb_size().v != fromData.vthumb_size().v) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		to = from; | ||||||
|  | 		return true; | ||||||
|  | 	} else if (to.data.type() == mtpc_inputDocumentFileLocation) { | ||||||
|  | 		const auto &toData = to.data.c_inputDocumentFileLocation(); | ||||||
|  | 		const auto &fromData = from.data.c_inputDocumentFileLocation(); | ||||||
|  | 		if (toData.vid().v != fromData.vid().v | ||||||
|  | 			|| toData.vthumb_size().v != fromData.vthumb_size().v) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		to = from; | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| Image ParseMaxImage( | Image ParseMaxImage( | ||||||
| 		const MTPDphoto &photo, | 		const MTPDphoto &photo, | ||||||
| 		const QString &suggestedPath) { | 		const QString &suggestedPath) { | ||||||
|  |  | ||||||
|  | @ -60,6 +60,8 @@ struct FileLocation { | ||||||
| 	} | 	} | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | bool RefreshFileReference(FileLocation &to, const FileLocation &from); | ||||||
|  | 
 | ||||||
| struct File { | struct File { | ||||||
| 	enum class SkipReason { | 	enum class SkipReason { | ||||||
| 		None, | 		None, | ||||||
|  | @ -526,6 +528,12 @@ struct Message { | ||||||
| 	const Image &thumb() const; | 	const Image &thumb() const; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | struct FileOrigin { | ||||||
|  | 	int split = 0; | ||||||
|  | 	MTPInputPeer peer; | ||||||
|  | 	int32 messageId = 0; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| Message ParseMessage( | Message ParseMessage( | ||||||
| 	ParseMediaContext &context, | 	ParseMediaContext &context, | ||||||
| 	const MTPMessage &data, | 	const MTPMessage &data, | ||||||
|  |  | ||||||
|  | @ -181,6 +181,7 @@ struct ApiWrap::FileProcess { | ||||||
| 	FnMut<void(const QString &relativePath)> done; | 	FnMut<void(const QString &relativePath)> done; | ||||||
| 
 | 
 | ||||||
| 	Data::FileLocation location; | 	Data::FileLocation location; | ||||||
|  | 	Data::FileOrigin origin; | ||||||
| 	int offset = 0; | 	int offset = 0; | ||||||
| 	int size = 0; | 	int size = 0; | ||||||
| 
 | 
 | ||||||
|  | @ -400,6 +401,9 @@ auto ApiWrap::fileRequest(const Data::FileLocation &location, int offset) { | ||||||
| 		} else if (result.type() == qstr("LOCATION_INVALID") | 		} else if (result.type() == qstr("LOCATION_INVALID") | ||||||
| 			|| result.type() == qstr("VERSION_INVALID")) { | 			|| result.type() == qstr("VERSION_INVALID")) { | ||||||
| 			filePartUnavailable(); | 			filePartUnavailable(); | ||||||
|  | 		} else if (result.code() == 400 | ||||||
|  | 			&& result.type().startsWith(qstr("FILE_REFERENCE_"))) { | ||||||
|  | 			filePartRefreshReference(offset); | ||||||
| 		} else { | 		} else { | ||||||
| 			error(std::move(result)); | 			error(std::move(result)); | ||||||
| 		} | 		} | ||||||
|  | @ -696,6 +700,7 @@ void ApiWrap::requestOtherData( | ||||||
| 	_otherDataProcess->file.suggestedPath = suggestedPath; | 	_otherDataProcess->file.suggestedPath = suggestedPath; | ||||||
| 	loadFile( | 	loadFile( | ||||||
| 		_otherDataProcess->file, | 		_otherDataProcess->file, | ||||||
|  | 		Data::FileOrigin(), | ||||||
| 		[](FileProgress progress) { return true; }, | 		[](FileProgress progress) { return true; }, | ||||||
| 		[=](const QString &result) { otherDataDone(result); }); | 		[=](const QString &result) { otherDataDone(result); }); | ||||||
| } | } | ||||||
|  | @ -780,6 +785,7 @@ void ApiWrap::loadNextUserpic() { | ||||||
| 		; ++_userpicsProcess->fileIndex) { | 		; ++_userpicsProcess->fileIndex) { | ||||||
| 		const auto ready = processFileLoad( | 		const auto ready = processFileLoad( | ||||||
| 			list[_userpicsProcess->fileIndex].image.file, | 			list[_userpicsProcess->fileIndex].image.file, | ||||||
|  | 			Data::FileOrigin(), | ||||||
| 			[=](FileProgress value) { return loadUserpicProgress(value); }, | 			[=](FileProgress value) { return loadUserpicProgress(value); }, | ||||||
| 			[=](const QString &path) { loadUserpicDone(path); }); | 			[=](const QString &path) { loadUserpicDone(path); }); | ||||||
| 		if (!ready) { | 		if (!ready) { | ||||||
|  | @ -1382,6 +1388,24 @@ void ApiWrap::loadMessagesFiles(Data::MessagesSlice &&slice) { | ||||||
| 	loadNextMessageFile(); | 	loadNextMessageFile(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Data::Message *ApiWrap::currentFileMessage() const { | ||||||
|  | 	Expects(_chatProcess != nullptr); | ||||||
|  | 	Expects(_chatProcess->slice.has_value()); | ||||||
|  | 
 | ||||||
|  | 	return &_chatProcess->slice->list[_chatProcess->fileIndex]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Data::FileOrigin ApiWrap::currentFileMessageOrigin() const { | ||||||
|  | 	Expects(_chatProcess != nullptr); | ||||||
|  | 	Expects(_chatProcess->slice.has_value()); | ||||||
|  | 
 | ||||||
|  | 	auto result = Data::FileOrigin(); | ||||||
|  | 	result.messageId = currentFileMessage()->id; | ||||||
|  | 	result.peer = _chatProcess->info.input; | ||||||
|  | 	result.split = _chatProcess->info.splits[_chatProcess->localSplitIndex]; | ||||||
|  | 	return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ApiWrap::loadNextMessageFile() { | void ApiWrap::loadNextMessageFile() { | ||||||
| 	Expects(_chatProcess != nullptr); | 	Expects(_chatProcess != nullptr); | ||||||
| 	Expects(_chatProcess->slice.has_value()); | 	Expects(_chatProcess->slice.has_value()); | ||||||
|  | @ -1398,9 +1422,10 @@ void ApiWrap::loadNextMessageFile() { | ||||||
| 		}; | 		}; | ||||||
| 		const auto ready = processFileLoad( | 		const auto ready = processFileLoad( | ||||||
| 			list[_chatProcess->fileIndex].file(), | 			list[_chatProcess->fileIndex].file(), | ||||||
|  | 			currentFileMessageOrigin(), | ||||||
| 			fileProgress, | 			fileProgress, | ||||||
| 			[=](const QString &path) { loadMessageFileDone(path); }, | 			[=](const QString &path) { loadMessageFileDone(path); }, | ||||||
| 			&list[_chatProcess->fileIndex]); | 			currentFileMessage()); | ||||||
| 		if (!ready) { | 		if (!ready) { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
|  | @ -1409,9 +1434,10 @@ void ApiWrap::loadNextMessageFile() { | ||||||
| 		}; | 		}; | ||||||
| 		const auto thumbReady = processFileLoad( | 		const auto thumbReady = processFileLoad( | ||||||
| 			list[_chatProcess->fileIndex].thumb().file, | 			list[_chatProcess->fileIndex].thumb().file, | ||||||
|  | 			currentFileMessageOrigin(), | ||||||
| 			thumbProgress, | 			thumbProgress, | ||||||
| 			[=](const QString &path) { loadMessageThumbDone(path); }, | 			[=](const QString &path) { loadMessageThumbDone(path); }, | ||||||
| 			&list[_chatProcess->fileIndex]); | 			currentFileMessage()); | ||||||
| 		if (!thumbReady) { | 		if (!thumbReady) { | ||||||
| 			return; | 			return; | ||||||
| 		} | 		} | ||||||
|  | @ -1501,6 +1527,7 @@ void ApiWrap::finishMessages() { | ||||||
| 
 | 
 | ||||||
| bool ApiWrap::processFileLoad( | bool ApiWrap::processFileLoad( | ||||||
| 		Data::File &file, | 		Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin, | ||||||
| 		Fn<bool(FileProgress)> progress, | 		Fn<bool(FileProgress)> progress, | ||||||
| 		FnMut<void(QString)> done, | 		FnMut<void(QString)> done, | ||||||
| 		Data::Message *message) { | 		Data::Message *message) { | ||||||
|  | @ -1512,7 +1539,7 @@ bool ApiWrap::processFileLoad( | ||||||
| 	} else if (!file.location && file.content.isEmpty()) { | 	} else if (!file.location && file.content.isEmpty()) { | ||||||
| 		file.skipReason = SkipReason::Unavailable; | 		file.skipReason = SkipReason::Unavailable; | ||||||
| 		return true; | 		return true; | ||||||
| 	} else if (writePreloadedFile(file)) { | 	} else if (writePreloadedFile(file, origin)) { | ||||||
| 		return !file.relativePath.isEmpty(); | 		return !file.relativePath.isEmpty(); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | @ -1548,11 +1575,13 @@ bool ApiWrap::processFileLoad( | ||||||
| 		file.skipReason = SkipReason::FileSize; | 		file.skipReason = SkipReason::FileSize; | ||||||
| 		return true; | 		return true; | ||||||
| 	} | 	} | ||||||
| 	loadFile(file, std::move(progress), std::move(done)); | 	loadFile(file, origin, std::move(progress), std::move(done)); | ||||||
| 	return false; | 	return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool ApiWrap::writePreloadedFile(Data::File &file) { | bool ApiWrap::writePreloadedFile( | ||||||
|  | 		Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin) { | ||||||
| 	Expects(_settings != nullptr); | 	Expects(_settings != nullptr); | ||||||
| 
 | 
 | ||||||
| 	using namespace Output; | 	using namespace Output; | ||||||
|  | @ -1561,7 +1590,7 @@ bool ApiWrap::writePreloadedFile(Data::File &file) { | ||||||
| 		file.relativePath = *path; | 		file.relativePath = *path; | ||||||
| 		return true; | 		return true; | ||||||
| 	} else if (!file.content.isEmpty()) { | 	} else if (!file.content.isEmpty()) { | ||||||
| 		const auto process = prepareFileProcess(file); | 		const auto process = prepareFileProcess(file, origin); | ||||||
| 		if (const auto result = process->file.writeBlock(file.content)) { | 		if (const auto result = process->file.writeBlock(file.content)) { | ||||||
| 			file.relativePath = process->relativePath; | 			file.relativePath = process->relativePath; | ||||||
| 			_fileCache->save(file.location, file.relativePath); | 			_fileCache->save(file.location, file.relativePath); | ||||||
|  | @ -1575,13 +1604,14 @@ bool ApiWrap::writePreloadedFile(Data::File &file) { | ||||||
| 
 | 
 | ||||||
| void ApiWrap::loadFile( | void ApiWrap::loadFile( | ||||||
| 		const Data::File &file, | 		const Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin, | ||||||
| 		Fn<bool(FileProgress)> progress, | 		Fn<bool(FileProgress)> progress, | ||||||
| 		FnMut<void(QString)> done) { | 		FnMut<void(QString)> done) { | ||||||
| 	Expects(_fileProcess == nullptr); | 	Expects(_fileProcess == nullptr); | ||||||
| 	Expects(file.location.dcId != 0 | 	Expects(file.location.dcId != 0 | ||||||
| 		|| file.location.data.type() == mtpc_inputTakeoutFileLocation); | 		|| file.location.data.type() == mtpc_inputTakeoutFileLocation); | ||||||
| 
 | 
 | ||||||
| 	_fileProcess = prepareFileProcess(file); | 	_fileProcess = prepareFileProcess(file, origin); | ||||||
| 	_fileProcess->progress = std::move(progress); | 	_fileProcess->progress = std::move(progress); | ||||||
| 	_fileProcess->done = std::move(done); | 	_fileProcess->done = std::move(done); | ||||||
| 
 | 
 | ||||||
|  | @ -1598,7 +1628,9 @@ void ApiWrap::loadFile( | ||||||
| 	loadFilePart(); | 	loadFilePart(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| auto ApiWrap::prepareFileProcess(const Data::File &file) const | auto ApiWrap::prepareFileProcess( | ||||||
|  | 	const Data::File &file, | ||||||
|  | 	const Data::FileOrigin &origin) const | ||||||
| -> std::unique_ptr<FileProcess> { | -> std::unique_ptr<FileProcess> { | ||||||
| 	Expects(_settings != nullptr); | 	Expects(_settings != nullptr); | ||||||
| 
 | 
 | ||||||
|  | @ -1611,6 +1643,7 @@ auto ApiWrap::prepareFileProcess(const Data::File &file) const | ||||||
| 	result->relativePath = relativePath; | 	result->relativePath = relativePath; | ||||||
| 	result->location = file.location; | 	result->location = file.location; | ||||||
| 	result->size = file.size; | 	result->size = file.size; | ||||||
|  | 	result->origin = origin; | ||||||
| 	return result; | 	return result; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -1705,6 +1738,87 @@ void ApiWrap::filePartDone(int offset, const MTPupload_File &result) { | ||||||
| 	process->done(process->relativePath); | 	process->done(process->relativePath); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ApiWrap::filePartRefreshReference(int offset) { | ||||||
|  | 	Expects(_fileProcess != nullptr); | ||||||
|  | 
 | ||||||
|  | 	const auto &origin = _fileProcess->origin; | ||||||
|  | 	if (!origin.messageId) { | ||||||
|  | 		error("FILE_REFERENCE error for non-message file."); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 	if (origin.peer.type() == mtpc_inputPeerChannel | ||||||
|  | 		|| origin.peer.type() == mtpc_inputPeerChannelFromMessage) { | ||||||
|  | 		const auto channel = (origin.peer.type() == mtpc_inputPeerChannel) | ||||||
|  | 			? MTP_inputChannel( | ||||||
|  | 				origin.peer.c_inputPeerChannel().vchannel_id(), | ||||||
|  | 				origin.peer.c_inputPeerChannel().vaccess_hash()) | ||||||
|  | 			: MTP_inputChannelFromMessage( | ||||||
|  | 				origin.peer.c_inputPeerChannelFromMessage().vpeer(), | ||||||
|  | 				origin.peer.c_inputPeerChannelFromMessage().vmsg_id(), | ||||||
|  | 				origin.peer.c_inputPeerChannelFromMessage().vchannel_id()); | ||||||
|  | 		mainRequest(MTPchannels_GetMessages( | ||||||
|  | 			channel, | ||||||
|  | 			MTP_vector<MTPInputMessage>( | ||||||
|  | 				1, | ||||||
|  | 				MTP_inputMessageID(MTP_int(origin.messageId))) | ||||||
|  | 		)).fail([=](const RPCError &error) { | ||||||
|  | 			filePartUnavailable(); | ||||||
|  | 			return true; | ||||||
|  | 		}).done([=](const MTPmessages_Messages &result) { | ||||||
|  | 			filePartExtractReference(offset, result); | ||||||
|  | 		}).send(); | ||||||
|  | 	} else { | ||||||
|  | 		splitRequest(origin.split, MTPmessages_GetMessages( | ||||||
|  | 			MTP_vector<MTPInputMessage>( | ||||||
|  | 				1, | ||||||
|  | 				MTP_inputMessageID(MTP_int(origin.messageId))) | ||||||
|  | 		)).fail([=](const RPCError &error) { | ||||||
|  | 			filePartUnavailable(); | ||||||
|  | 			return true; | ||||||
|  | 		}).done([=](const MTPmessages_Messages &result) { | ||||||
|  | 			filePartExtractReference(offset, result); | ||||||
|  | 		}).send(); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ApiWrap::filePartExtractReference( | ||||||
|  | 		int offset, | ||||||
|  | 		const MTPmessages_Messages &result) { | ||||||
|  | 	Expects(_fileProcess != nullptr); | ||||||
|  | 
 | ||||||
|  | 	result.match([&](const MTPDmessages_messagesNotModified &data) { | ||||||
|  | 		error("Unexpected messagesNotModified received."); | ||||||
|  | 	}, [&](const auto &data) { | ||||||
|  | 		auto context = Data::ParseMediaContext(); | ||||||
|  | 		const auto messages = Data::ParseMessagesSlice( | ||||||
|  | 			context, | ||||||
|  | 			data.vmessages(), | ||||||
|  | 			data.vusers(), | ||||||
|  | 			data.vchats(), | ||||||
|  | 			_chatProcess->info.relativePath); | ||||||
|  | 		for (const auto &message : messages.list) { | ||||||
|  | 			if (message.id == _fileProcess->origin.messageId) { | ||||||
|  | 				const auto refresh1 = Data::RefreshFileReference( | ||||||
|  | 					_fileProcess->location, | ||||||
|  | 					message.file().location); | ||||||
|  | 				const auto refresh2 = Data::RefreshFileReference( | ||||||
|  | 					_fileProcess->location, | ||||||
|  | 					message.thumb().file.location); | ||||||
|  | 				if (refresh1 || refresh2) { | ||||||
|  | 					fileRequest( | ||||||
|  | 						_fileProcess->location, | ||||||
|  | 						offset | ||||||
|  | 					).done([=](const MTPupload_File &result) { | ||||||
|  | 						filePartDone(offset, result); | ||||||
|  | 					}).send(); | ||||||
|  | 					return; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		filePartUnavailable(); | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ApiWrap::filePartUnavailable() { | void ApiWrap::filePartUnavailable() { | ||||||
| 	Expects(_fileProcess != nullptr); | 	Expects(_fileProcess != nullptr); | ||||||
| 	Expects(!_fileProcess->requests.empty()); | 	Expects(!_fileProcess->requests.empty()); | ||||||
|  |  | ||||||
|  | @ -23,6 +23,7 @@ struct DialogsInfo; | ||||||
| struct DialogInfo; | struct DialogInfo; | ||||||
| struct MessagesSlice; | struct MessagesSlice; | ||||||
| struct Message; | struct Message; | ||||||
|  | struct FileOrigin; | ||||||
| } // namespace Data
 | } // namespace Data
 | ||||||
| 
 | 
 | ||||||
| namespace Output { | namespace Output { | ||||||
|  | @ -159,21 +160,33 @@ private: | ||||||
| 	void finishMessagesSlice(); | 	void finishMessagesSlice(); | ||||||
| 	void finishMessages(); | 	void finishMessages(); | ||||||
| 
 | 
 | ||||||
|  | 	[[nodiscard]] Data::Message *currentFileMessage() const; | ||||||
|  | 	[[nodiscard]] Data::FileOrigin currentFileMessageOrigin() const; | ||||||
|  | 
 | ||||||
| 	bool processFileLoad( | 	bool processFileLoad( | ||||||
| 		Data::File &file, | 		Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin, | ||||||
| 		Fn<bool(FileProgress)> progress, | 		Fn<bool(FileProgress)> progress, | ||||||
| 		FnMut<void(QString)> done, | 		FnMut<void(QString)> done, | ||||||
| 		Data::Message *message = nullptr); | 		Data::Message *message = nullptr); | ||||||
| 	std::unique_ptr<FileProcess> prepareFileProcess( | 	std::unique_ptr<FileProcess> prepareFileProcess( | ||||||
| 		const Data::File &file) const; | 		const Data::File &file, | ||||||
| 	bool writePreloadedFile(Data::File &file); | 		const Data::FileOrigin &origin) const; | ||||||
|  | 	bool writePreloadedFile( | ||||||
|  | 		Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin); | ||||||
| 	void loadFile( | 	void loadFile( | ||||||
| 		const Data::File &file, | 		const Data::File &file, | ||||||
|  | 		const Data::FileOrigin &origin, | ||||||
| 		Fn<bool(FileProgress)> progress, | 		Fn<bool(FileProgress)> progress, | ||||||
| 		FnMut<void(QString)> done); | 		FnMut<void(QString)> done); | ||||||
| 	void loadFilePart(); | 	void loadFilePart(); | ||||||
| 	void filePartDone(int offset, const MTPupload_File &result); | 	void filePartDone(int offset, const MTPupload_File &result); | ||||||
| 	void filePartUnavailable(); | 	void filePartUnavailable(); | ||||||
|  | 	void filePartRefreshReference(int offset); | ||||||
|  | 	void filePartExtractReference( | ||||||
|  | 		int offset, | ||||||
|  | 		const MTPmessages_Messages &result); | ||||||
| 
 | 
 | ||||||
| 	template <typename Request> | 	template <typename Request> | ||||||
| 	class RequestBuilder; | 	class RequestBuilder; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue