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