mirror of https://github.com/procxx/kepka.git
Apply file type/size restrictions in export.
This commit is contained in:
parent
df91b2bfeb
commit
8d52ca6be6
|
@ -142,6 +142,7 @@ void ParseAttributes(
|
||||||
}, [&](const MTPDdocumentAttributeAnimated &data) {
|
}, [&](const MTPDdocumentAttributeAnimated &data) {
|
||||||
result.isAnimated = true;
|
result.isAnimated = true;
|
||||||
}, [&](const MTPDdocumentAttributeSticker &data) {
|
}, [&](const MTPDdocumentAttributeSticker &data) {
|
||||||
|
result.isSticker = true;
|
||||||
result.stickerEmoji = ParseString(data.valt);
|
result.stickerEmoji = ParseString(data.valt);
|
||||||
}, [&](const MTPDdocumentAttributeVideo &data) {
|
}, [&](const MTPDdocumentAttributeVideo &data) {
|
||||||
if (data.is_round_message()) {
|
if (data.is_round_message()) {
|
||||||
|
|
|
@ -51,6 +51,12 @@ struct FileLocation {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct File {
|
struct File {
|
||||||
|
enum class SkipReason {
|
||||||
|
None,
|
||||||
|
Unavailable,
|
||||||
|
FileType,
|
||||||
|
FileSize,
|
||||||
|
};
|
||||||
FileLocation location;
|
FileLocation location;
|
||||||
int size = 0;
|
int size = 0;
|
||||||
QByteArray content;
|
QByteArray content;
|
||||||
|
@ -58,6 +64,7 @@ struct File {
|
||||||
QString suggestedPath;
|
QString suggestedPath;
|
||||||
|
|
||||||
QString relativePath;
|
QString relativePath;
|
||||||
|
SkipReason skipReason = SkipReason::None;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Image {
|
struct Image {
|
||||||
|
@ -89,6 +96,7 @@ struct Document {
|
||||||
Utf8String songTitle;
|
Utf8String songTitle;
|
||||||
int duration = 0;
|
int duration = 0;
|
||||||
|
|
||||||
|
bool isSticker = false;
|
||||||
bool isAnimated = false;
|
bool isAnimated = false;
|
||||||
bool isVideoMessage = false;
|
bool isVideoMessage = false;
|
||||||
bool isVoiceMessage = false;
|
bool isVoiceMessage = false;
|
||||||
|
|
|
@ -26,9 +26,8 @@ constexpr auto kChatsSliceLimit = 100;
|
||||||
constexpr auto kMessagesSliceLimit = 100;
|
constexpr auto kMessagesSliceLimit = 100;
|
||||||
constexpr auto kFileMaxSize = 1500 * 1024 * 1024;
|
constexpr auto kFileMaxSize = 1500 * 1024 * 1024;
|
||||||
|
|
||||||
bool WillLoadFile(const Data::File &file) {
|
bool FileIsAvailable(const Data::File &file) {
|
||||||
return file.relativePath.isEmpty()
|
return file.relativePath.isEmpty() && (file.location.dcId != 0);
|
||||||
&& (!file.content.isEmpty() || file.location.dcId != 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
@ -258,17 +257,16 @@ void ApiWrap::loadNextUserpic() {
|
||||||
Expects(_userpicsProcess != nullptr);
|
Expects(_userpicsProcess != nullptr);
|
||||||
Expects(_userpicsProcess->slice.has_value());
|
Expects(_userpicsProcess->slice.has_value());
|
||||||
|
|
||||||
const auto &list = _userpicsProcess->slice->list;
|
auto &list = _userpicsProcess->slice->list;
|
||||||
while (true) {
|
while (true) {
|
||||||
const auto index = ++_userpicsProcess->fileIndex;
|
const auto index = ++_userpicsProcess->fileIndex;
|
||||||
if (index >= list.size()) {
|
if (index >= list.size()) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
const auto &file = list[index].image.file;
|
const auto ready = processFileLoad(
|
||||||
if (WillLoadFile(file)) {
|
list[index].image.file,
|
||||||
loadFile(
|
[=](const QString &path) { loadUserpicDone(path); });
|
||||||
file,
|
if (!ready) {
|
||||||
[=](const QString &path) { loadUserpicDone(path); });
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -510,17 +508,16 @@ void ApiWrap::loadNextMessageFile() {
|
||||||
Expects(_dialogsProcess->single->slice.has_value());
|
Expects(_dialogsProcess->single->slice.has_value());
|
||||||
|
|
||||||
const auto process = _dialogsProcess->single.get();
|
const auto process = _dialogsProcess->single.get();
|
||||||
const auto &list = process->slice->list;
|
auto &list = process->slice->list;
|
||||||
while (true) {
|
while (true) {
|
||||||
const auto index = ++process->fileIndex;
|
const auto index = ++process->fileIndex;
|
||||||
if (index >= list.size()) {
|
if (index >= list.size()) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
const auto &file = list[index].file();
|
const auto ready = processFileLoad(
|
||||||
if (WillLoadFile(file)) {
|
list[index].file(),
|
||||||
loadFile(
|
[=](const QString &path) { loadMessageFileDone(path); });
|
||||||
file,
|
if (!ready) {
|
||||||
[=](const QString &path) { loadMessageFileDone(path); });
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -569,32 +566,92 @@ void ApiWrap::finishDialogs() {
|
||||||
base::take(_dialogsProcess)->finish();
|
base::take(_dialogsProcess)->finish();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ApiWrap::loadFile(const Data::File &file, FnMut<void(QString)> done) {
|
bool ApiWrap::processFileLoad(
|
||||||
Expects(_fileProcess == nullptr);
|
Data::File &file,
|
||||||
|
FnMut<void(QString)> done,
|
||||||
|
Data::Message *message) {
|
||||||
|
using SkipReason = Data::File::SkipReason;
|
||||||
|
|
||||||
|
if (!file.relativePath.isEmpty()) {
|
||||||
|
return true;
|
||||||
|
} else if (writePreloadedFile(file)) {
|
||||||
|
return true;
|
||||||
|
} else if (!FileIsAvailable(file)) {
|
||||||
|
file.skipReason = SkipReason::Unavailable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
using Type = MediaSettings::Type;
|
||||||
|
const auto type = message ? message->media.content.match(
|
||||||
|
[&](const Data::Document &data) {
|
||||||
|
if (data.isSticker) {
|
||||||
|
return Type::Sticker;
|
||||||
|
} else if (data.isVideoMessage) {
|
||||||
|
return Type::VideoMessage;
|
||||||
|
} else if (data.isVoiceMessage) {
|
||||||
|
return Type::VoiceMessage;
|
||||||
|
} else if (data.isAnimated) {
|
||||||
|
return Type::GIF;
|
||||||
|
} else if (data.isVideoFile) {
|
||||||
|
return Type::Video;
|
||||||
|
} else {
|
||||||
|
return Type::File;
|
||||||
|
}
|
||||||
|
}, [](const auto &data) {
|
||||||
|
return Type::Photo;
|
||||||
|
}) : Type::Photo;
|
||||||
|
|
||||||
|
if ((_settings->media.types & type) != type) {
|
||||||
|
file.skipReason = SkipReason::FileType;
|
||||||
|
return true;
|
||||||
|
} else if (file.size >= _settings->media.sizeLimit) {
|
||||||
|
file.skipReason = SkipReason::FileSize;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
loadFile(file, std::move(done));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ApiWrap::writePreloadedFile(Data::File &file) {
|
||||||
Expects(_settings != nullptr);
|
Expects(_settings != nullptr);
|
||||||
Expects(WillLoadFile(file));
|
|
||||||
|
|
||||||
using namespace Output;
|
using namespace Output;
|
||||||
const auto relativePath = File::PrepareRelativePath(
|
|
||||||
_settings->path,
|
|
||||||
file.suggestedPath);
|
|
||||||
_fileProcess = std::make_unique<FileProcess>(
|
|
||||||
_settings->path + relativePath);
|
|
||||||
_fileProcess->relativePath = relativePath;
|
|
||||||
_fileProcess->location = file.location;
|
|
||||||
_fileProcess->size = file.size;
|
|
||||||
_fileProcess->done = std::move(done);
|
|
||||||
|
|
||||||
if (!file.content.isEmpty()) {
|
if (!file.content.isEmpty()) {
|
||||||
|
const auto process = prepareFileProcess(file);
|
||||||
auto &output = _fileProcess->file;
|
auto &output = _fileProcess->file;
|
||||||
if (output.writeBlock(file.content) == File::Result::Success) {
|
if (output.writeBlock(file.content) == File::Result::Success) {
|
||||||
_fileProcess->done(relativePath);
|
file.relativePath = process->relativePath;
|
||||||
} else {
|
return true;
|
||||||
error(QString("Could not open '%1'.").arg(relativePath));
|
|
||||||
}
|
}
|
||||||
} else {
|
error(QString("Could not write '%1'.").arg(process->relativePath));
|
||||||
loadFilePart();
|
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ApiWrap::loadFile(const Data::File &file, FnMut<void(QString)> done) {
|
||||||
|
Expects(_fileProcess == nullptr);
|
||||||
|
Expects(FileIsAvailable(file));
|
||||||
|
|
||||||
|
_fileProcess = prepareFileProcess(file);
|
||||||
|
_fileProcess->done = std::move(done);
|
||||||
|
|
||||||
|
loadFilePart();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto ApiWrap::prepareFileProcess(const Data::File &file) const
|
||||||
|
-> std::unique_ptr<FileProcess> {
|
||||||
|
Expects(_settings != nullptr);
|
||||||
|
|
||||||
|
const auto relativePath = Output::File::PrepareRelativePath(
|
||||||
|
_settings->path,
|
||||||
|
file.suggestedPath);
|
||||||
|
auto result = std::make_unique<FileProcess>(
|
||||||
|
_settings->path + relativePath);
|
||||||
|
result->relativePath = relativePath;
|
||||||
|
result->location = file.location;
|
||||||
|
result->size = file.size;
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ApiWrap::loadFilePart() {
|
void ApiWrap::loadFilePart() {
|
||||||
|
|
|
@ -21,6 +21,7 @@ struct SessionsList;
|
||||||
struct DialogsInfo;
|
struct DialogsInfo;
|
||||||
struct DialogInfo;
|
struct DialogInfo;
|
||||||
struct MessagesSlice;
|
struct MessagesSlice;
|
||||||
|
struct Message;
|
||||||
} // namespace Data
|
} // namespace Data
|
||||||
|
|
||||||
struct Settings;
|
struct Settings;
|
||||||
|
@ -56,6 +57,10 @@ public:
|
||||||
~ApiWrap();
|
~ApiWrap();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
struct UserpicsProcess;
|
||||||
|
struct FileProcess;
|
||||||
|
struct DialogsProcess;
|
||||||
|
|
||||||
void startMainSession(FnMut<void()> done);
|
void startMainSession(FnMut<void()> done);
|
||||||
|
|
||||||
void handleUserpicsSlice(const MTPphotos_Photos &result);
|
void handleUserpicsSlice(const MTPphotos_Photos &result);
|
||||||
|
@ -64,8 +69,6 @@ private:
|
||||||
void loadUserpicDone(const QString &relativePath);
|
void loadUserpicDone(const QString &relativePath);
|
||||||
void finishUserpics();
|
void finishUserpics();
|
||||||
|
|
||||||
void requestSavedContacts();
|
|
||||||
|
|
||||||
void requestDialogsSlice();
|
void requestDialogsSlice();
|
||||||
void appendDialogsSlice(Data::DialogsInfo &&info);
|
void appendDialogsSlice(Data::DialogsInfo &&info);
|
||||||
void finishDialogsList();
|
void finishDialogsList();
|
||||||
|
@ -75,10 +78,18 @@ private:
|
||||||
void requestMessagesSlice();
|
void requestMessagesSlice();
|
||||||
void loadMessagesFiles(Data::MessagesSlice &&slice);
|
void loadMessagesFiles(Data::MessagesSlice &&slice);
|
||||||
void loadNextMessageFile();
|
void loadNextMessageFile();
|
||||||
|
|
||||||
void loadMessageFileDone(const QString &relativePath);
|
void loadMessageFileDone(const QString &relativePath);
|
||||||
void finishMessages();
|
void finishMessages();
|
||||||
void finishDialogs();
|
void finishDialogs();
|
||||||
|
|
||||||
|
bool processFileLoad(
|
||||||
|
Data::File &file,
|
||||||
|
FnMut<void(QString)> done,
|
||||||
|
Data::Message *message = nullptr);
|
||||||
|
std::unique_ptr<FileProcess> prepareFileProcess(
|
||||||
|
const Data::File &file) const;
|
||||||
|
bool writePreloadedFile(Data::File &file);
|
||||||
void loadFile(const Data::File &file, FnMut<void(QString)> done);
|
void loadFile(const Data::File &file, FnMut<void(QString)> done);
|
||||||
void loadFilePart();
|
void loadFilePart();
|
||||||
void filePartDone(int offset, const MTPupload_File &result);
|
void filePartDone(int offset, const MTPupload_File &result);
|
||||||
|
@ -99,13 +110,8 @@ private:
|
||||||
std::unique_ptr<Settings> _settings;
|
std::unique_ptr<Settings> _settings;
|
||||||
MTPInputUser _user = MTP_inputUserSelf();
|
MTPInputUser _user = MTP_inputUserSelf();
|
||||||
|
|
||||||
struct UserpicsProcess;
|
|
||||||
std::unique_ptr<UserpicsProcess> _userpicsProcess;
|
std::unique_ptr<UserpicsProcess> _userpicsProcess;
|
||||||
|
|
||||||
struct FileProcess;
|
|
||||||
std::unique_ptr<FileProcess> _fileProcess;
|
std::unique_ptr<FileProcess> _fileProcess;
|
||||||
|
|
||||||
struct DialogsProcess;
|
|
||||||
std::unique_ptr<DialogsProcess> _dialogsProcess;
|
std::unique_ptr<DialogsProcess> _dialogsProcess;
|
||||||
|
|
||||||
rpl::event_stream<RPCError> _errors;
|
rpl::event_stream<RPCError> _errors;
|
||||||
|
|
|
@ -171,6 +171,13 @@ QByteArray SerializeMessage(
|
||||||
const auto pushAction = [&](const QByteArray &action) {
|
const auto pushAction = [&](const QByteArray &action) {
|
||||||
push("Action", action);
|
push("Action", action);
|
||||||
};
|
};
|
||||||
|
const auto pushTTL = [&](
|
||||||
|
const QByteArray &label = "Self destruct period") {
|
||||||
|
if (const auto ttl = message.media.ttl) {
|
||||||
|
push(label, NumberToString(ttl) + " sec.");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
message.action.content.match([&](const ActionChatCreate &data) {
|
message.action.content.match([&](const ActionChatCreate &data) {
|
||||||
pushActor();
|
pushActor();
|
||||||
pushAction("Create group");
|
pushAction("Create group");
|
||||||
|
@ -299,11 +306,12 @@ QByteArray SerializeMessage(
|
||||||
}
|
}
|
||||||
|
|
||||||
message.media.content.match([&](const Photo &photo) {
|
message.media.content.match([&](const Photo &photo) {
|
||||||
|
pushTTL();
|
||||||
}, [&](const Document &data) {
|
}, [&](const Document &data) {
|
||||||
const auto pushPath = [&](const QByteArray &label) {
|
const auto pushPath = [&](const QByteArray &label) {
|
||||||
push(label, FormatFilePath(data.file));
|
push(label, FormatFilePath(data.file));
|
||||||
};
|
};
|
||||||
if (!data.stickerEmoji.isEmpty()) {
|
if (data.isSticker) {
|
||||||
pushPath("Sticker");
|
pushPath("Sticker");
|
||||||
push("Emoji", data.stickerEmoji);
|
push("Emoji", data.stickerEmoji);
|
||||||
} else if (data.isVideoMessage) {
|
} else if (data.isVideoMessage) {
|
||||||
|
@ -321,7 +329,7 @@ QByteArray SerializeMessage(
|
||||||
} else {
|
} else {
|
||||||
pushPath("File");
|
pushPath("File");
|
||||||
}
|
}
|
||||||
if (data.stickerEmoji.isEmpty()) {
|
if (!data.isSticker) {
|
||||||
push("Mime type", data.mime);
|
push("Mime type", data.mime);
|
||||||
}
|
}
|
||||||
if (data.duration) {
|
if (data.duration) {
|
||||||
|
@ -331,6 +339,7 @@ QByteArray SerializeMessage(
|
||||||
push("Width", NumberToString(data.width));
|
push("Width", NumberToString(data.width));
|
||||||
push("Height", NumberToString(data.height));
|
push("Height", NumberToString(data.height));
|
||||||
}
|
}
|
||||||
|
pushTTL();
|
||||||
}, [&](const ContactInfo &data) {
|
}, [&](const ContactInfo &data) {
|
||||||
push("Contact information", SerializeKeyValue({
|
push("Contact information", SerializeKeyValue({
|
||||||
{ "First name", data.firstName },
|
{ "First name", data.firstName },
|
||||||
|
@ -342,6 +351,7 @@ QByteArray SerializeMessage(
|
||||||
{ "Latitude", NumberToString(data.latitude) },
|
{ "Latitude", NumberToString(data.latitude) },
|
||||||
{ "Longitude", NumberToString(data.longitude) },
|
{ "Longitude", NumberToString(data.longitude) },
|
||||||
}) : QByteArray("(empty value)"));
|
}) : QByteArray("(empty value)"));
|
||||||
|
pushTTL("Live location period");
|
||||||
}, [&](const Venue &data) {
|
}, [&](const Venue &data) {
|
||||||
push("Place name", data.title);
|
push("Place name", data.title);
|
||||||
push("Address", data.address);
|
push("Address", data.address);
|
||||||
|
|
Loading…
Reference in New Issue