feat: make email operations more comfortable
This commit is contained in:
parent
22558e2948
commit
0a88a9a46c
2
.gitmodules
vendored
2
.gitmodules
vendored
@ -1,3 +1,3 @@
|
|||||||
[submodule "src/m_email/vmime"]
|
[submodule "src/m_email/vmime"]
|
||||||
path = src/m_email/vmime
|
path = src/m_email/vmime
|
||||||
url = https://github.com/kisli/vmime.git
|
url = https://git.bktus.com/GpgFrontend/vmime.git
|
||||||
|
@ -64,7 +64,7 @@ auto ExtractFieldValue(const vmime::shared_ptr<vmime::header>& header,
|
|||||||
|
|
||||||
auto field_value = field->getValue();
|
auto field_value = field->getValue();
|
||||||
if (!field_value) {
|
if (!field_value) {
|
||||||
FLOG_WARN("cannot get '%s' Field Value from header", field_name);
|
FLOG_WARN("cannot get '%1' Field Value from header", field_name);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -81,7 +81,7 @@ auto ExtractFieldValueMailBox(const vmime::shared_ptr<vmime::header>& header,
|
|||||||
|
|
||||||
auto field_value = field->getValue<vmime::mailbox>();
|
auto field_value = field->getValue<vmime::mailbox>();
|
||||||
if (!field_value) {
|
if (!field_value) {
|
||||||
FLOG_WARN("cannot get '%s' Field Value from header", field_name);
|
FLOG_WARN("cannot get '%1' Field Value from header", field_name);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,7 +99,7 @@ auto ExtractFieldValueAddressList(
|
|||||||
|
|
||||||
auto field_value = field->getValue<vmime::addressList>();
|
auto field_value = field->getValue<vmime::addressList>();
|
||||||
if (!field_value) {
|
if (!field_value) {
|
||||||
FLOG_WARN("cannot get '%s' Field Value from header", field_name);
|
FLOG_WARN("cannot get '%1' Field Value from header", field_name);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -114,6 +114,8 @@ auto ExtractFieldValueAddressList(
|
|||||||
|
|
||||||
auto ExtractFieldValueText(const vmime::shared_ptr<vmime::header>& header,
|
auto ExtractFieldValueText(const vmime::shared_ptr<vmime::header>& header,
|
||||||
const QString& field_name) -> QString {
|
const QString& field_name) -> QString {
|
||||||
|
if (!header->hasField(field_name.toStdString())) return {};
|
||||||
|
|
||||||
auto field = header->getField(field_name.toStdString());
|
auto field = header->getField(field_name.toStdString());
|
||||||
if (!field) {
|
if (!field) {
|
||||||
FLOG_WARN("cannot get '%1' Field from header", field_name);
|
FLOG_WARN("cannot get '%1' Field from header", field_name);
|
||||||
@ -122,7 +124,7 @@ auto ExtractFieldValueText(const vmime::shared_ptr<vmime::header>& header,
|
|||||||
|
|
||||||
auto field_value = field->getValue<vmime::text>();
|
auto field_value = field->getValue<vmime::text>();
|
||||||
if (!field_value) {
|
if (!field_value) {
|
||||||
FLOG_WARN("cannot get '%s' Field Value from header", field_name);
|
FLOG_WARN("cannot get '%1' Field Value from header", field_name);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -139,7 +141,7 @@ auto ExtractFieldValueDateTime(const vmime::shared_ptr<vmime::header>& header,
|
|||||||
|
|
||||||
auto field_value = field->getValue<vmime::datetime>();
|
auto field_value = field->getValue<vmime::datetime>();
|
||||||
if (!field_value) {
|
if (!field_value) {
|
||||||
FLOG_WARN("cannot get '%s' Field Value from header", field_name);
|
FLOG_WARN("cannot get '%1' Field Value from header", field_name);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,4 +169,162 @@ auto ParseEmailString(const QString& input, QString& name,
|
|||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto EncodeBase64WithLineBreaks(const QByteArray& data,
|
||||||
|
int line_length) -> QString {
|
||||||
|
// Get the Base64 encoded data
|
||||||
|
QByteArray base64_data = data.toBase64();
|
||||||
|
|
||||||
|
// Split the base64 data into lines of the given line length
|
||||||
|
QStringList lines;
|
||||||
|
for (int i = 0; i < base64_data.size(); i += line_length) {
|
||||||
|
lines.append(base64_data.mid(i, line_length));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Join lines with CRLF
|
||||||
|
return lines.join("\r\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
auto CheckIfEMLMessage(const QByteArray& data,
|
||||||
|
vmime::shared_ptr<vmime::message>& message) -> bool {
|
||||||
|
vmime::string vmime_data(data.constData(), data.size());
|
||||||
|
|
||||||
|
message = vmime::make_shared<vmime::message>();
|
||||||
|
try {
|
||||||
|
message->parse(vmime_data);
|
||||||
|
return message->getParsedLength() != 0 && !message->getHeader()->isEmpty();
|
||||||
|
} catch (const vmime::exception& e) {
|
||||||
|
FLOG_DEBUG("error occurred when parsing vmime data: %1", e.what());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto BuildPlainTextEML(const EMailMetaData& meta_data,
|
||||||
|
const QByteArray& body_data, QString& eml_data) -> int {
|
||||||
|
auto from = meta_data.from;
|
||||||
|
auto recipient_list = meta_data.to;
|
||||||
|
auto cc_list = meta_data.cc;
|
||||||
|
auto bcc_list = meta_data.bcc;
|
||||||
|
auto subject = meta_data.subject;
|
||||||
|
|
||||||
|
QString name;
|
||||||
|
QString email;
|
||||||
|
|
||||||
|
try {
|
||||||
|
vmime::messageBuilder plaintext_msg_builder;
|
||||||
|
|
||||||
|
if (ParseEmailString(from, name, email)) {
|
||||||
|
plaintext_msg_builder.setExpeditor(
|
||||||
|
vmime::mailbox(vmime::text(name.toStdString()), email.toStdString()));
|
||||||
|
} else {
|
||||||
|
plaintext_msg_builder.setExpeditor(vmime::mailbox(from.toStdString()));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const QString& recipient : recipient_list) {
|
||||||
|
auto trimmed_recipient = recipient.trimmed();
|
||||||
|
if (ParseEmailString(trimmed_recipient, name, email)) {
|
||||||
|
plaintext_msg_builder.getRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
||||||
|
email.toStdString()));
|
||||||
|
} else {
|
||||||
|
plaintext_msg_builder.getRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(
|
||||||
|
trimmed_recipient.toStdString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const QString& recipient : cc_list) {
|
||||||
|
auto trimmed_recipient = recipient.trimmed();
|
||||||
|
if (ParseEmailString(trimmed_recipient, name, email)) {
|
||||||
|
plaintext_msg_builder.getCopyRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
||||||
|
email.toStdString()));
|
||||||
|
} else {
|
||||||
|
plaintext_msg_builder.getCopyRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(
|
||||||
|
trimmed_recipient.toStdString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const QString& recipient : bcc_list) {
|
||||||
|
auto trimmed_recipient = recipient.trimmed();
|
||||||
|
if (ParseEmailString(trimmed_recipient, name, email)) {
|
||||||
|
plaintext_msg_builder.getBlindCopyRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
||||||
|
email.toStdString()));
|
||||||
|
} else {
|
||||||
|
plaintext_msg_builder.getBlindCopyRecipients().appendAddress(
|
||||||
|
vmime::make_shared<vmime::mailbox>(
|
||||||
|
trimmed_recipient.toStdString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
plaintext_msg_builder.setSubject(vmime::text(subject.toStdString()));
|
||||||
|
|
||||||
|
vmime::shared_ptr<vmime::message> plaintext_msg =
|
||||||
|
plaintext_msg_builder.construct();
|
||||||
|
|
||||||
|
auto plaintext_msg_header = plaintext_msg->getHeader();
|
||||||
|
|
||||||
|
auto plaintext_msg_content_type_header_field =
|
||||||
|
plaintext_msg_header->getField<vmime::contentTypeField>(
|
||||||
|
vmime::fields::CONTENT_TYPE);
|
||||||
|
plaintext_msg_content_type_header_field->setValue("text/plain");
|
||||||
|
plaintext_msg_content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>("charset", "UTF-8"));
|
||||||
|
plaintext_msg_content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>("format", "flowed"));
|
||||||
|
|
||||||
|
auto plaintext_msg_content_trans_encode_field =
|
||||||
|
plaintext_msg_header->getField(
|
||||||
|
vmime::fields::CONTENT_TRANSFER_ENCODING);
|
||||||
|
plaintext_msg_content_trans_encode_field->setValue("base64");
|
||||||
|
|
||||||
|
auto plaintext_msg_body = plaintext_msg->getBody();
|
||||||
|
|
||||||
|
auto mime_part_body_content =
|
||||||
|
vmime::make_shared<vmime::stringContentHandler>();
|
||||||
|
mime_part_body_content->setData(body_data.toStdString());
|
||||||
|
plaintext_msg_body->setContents(mime_part_body_content);
|
||||||
|
|
||||||
|
eml_data =
|
||||||
|
Q_SC(plaintext_msg->generate(vmime::lineLengthLimits::convenient));
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
} catch (const vmime::exception& e) {
|
||||||
|
eml_data = QString("VMIME Error: %1").arg(e.what());
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto GetMetaData(QByteArray& data, EMailMetaData& meta_data) {}
|
||||||
|
|
||||||
|
auto GetEMLMetaData(vmime::shared_ptr<vmime::message>& message,
|
||||||
|
EMailMetaData& meta_data) -> int {
|
||||||
|
auto header = message->getHeader();
|
||||||
|
|
||||||
|
auto from_field_value_text =
|
||||||
|
ExtractFieldValueMailBox(header, vmime::fields::FROM);
|
||||||
|
auto to_field_value_text =
|
||||||
|
ExtractFieldValueAddressList(header, vmime::fields::TO);
|
||||||
|
auto cc_field_value_text =
|
||||||
|
ExtractFieldValueAddressList(header, vmime::fields::CC);
|
||||||
|
auto bcc_field_value_text =
|
||||||
|
ExtractFieldValueAddressList(header, vmime::fields::BCC);
|
||||||
|
auto date_field_value =
|
||||||
|
ExtractFieldValueDateTime(header, vmime::fields::DATE);
|
||||||
|
auto subject_field_value_text =
|
||||||
|
ExtractFieldValueText(header, vmime::fields::SUBJECT);
|
||||||
|
auto reply_to_field_value_text =
|
||||||
|
ExtractFieldValueMailBox(header, vmime::fields::REPLY_TO);
|
||||||
|
auto organization_text =
|
||||||
|
ExtractFieldValueText(header, vmime::fields::ORGANIZATION);
|
||||||
|
|
||||||
|
meta_data.from = from_field_value_text;
|
||||||
|
meta_data.to = to_field_value_text.split(',');
|
||||||
|
meta_data.cc = cc_field_value_text.split(',');
|
||||||
|
meta_data.bcc = bcc_field_value_text.split(',');
|
||||||
|
meta_data.subject = subject_field_value_text;
|
||||||
|
return 0;
|
||||||
}
|
}
|
@ -49,90 +49,9 @@ auto EncryptEMLData(int channel, const QStringList& keys,
|
|||||||
QString email;
|
QString email;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
vmime::messageBuilder plaintext_msg_builder;
|
|
||||||
|
|
||||||
if (ParseEmailString(from, name, email)) {
|
|
||||||
plaintext_msg_builder.setExpeditor(
|
|
||||||
vmime::mailbox(vmime::text(name.toStdString()), email.toStdString()));
|
|
||||||
} else {
|
|
||||||
plaintext_msg_builder.setExpeditor(vmime::mailbox(from.toStdString()));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const QString& recipient : recipient_list) {
|
|
||||||
auto trimmed_recipient = recipient.trimmed();
|
|
||||||
if (ParseEmailString(trimmed_recipient, name, email)) {
|
|
||||||
plaintext_msg_builder.getRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
|
||||||
email.toStdString()));
|
|
||||||
} else {
|
|
||||||
plaintext_msg_builder.getRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(
|
|
||||||
trimmed_recipient.toStdString()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const QString& recipient : cc_list) {
|
|
||||||
auto trimmed_recipient = recipient.trimmed();
|
|
||||||
if (ParseEmailString(trimmed_recipient, name, email)) {
|
|
||||||
plaintext_msg_builder.getCopyRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
|
||||||
email.toStdString()));
|
|
||||||
} else {
|
|
||||||
plaintext_msg_builder.getCopyRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(
|
|
||||||
trimmed_recipient.toStdString()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const QString& recipient : bcc_list) {
|
|
||||||
auto trimmed_recipient = recipient.trimmed();
|
|
||||||
if (ParseEmailString(trimmed_recipient, name, email)) {
|
|
||||||
plaintext_msg_builder.getBlindCopyRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(vmime::text(name.toStdString()),
|
|
||||||
email.toStdString()));
|
|
||||||
} else {
|
|
||||||
plaintext_msg_builder.getBlindCopyRecipients().appendAddress(
|
|
||||||
vmime::make_shared<vmime::mailbox>(
|
|
||||||
trimmed_recipient.toStdString()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
plaintext_msg_builder.setSubject(vmime::text(subject.toStdString()));
|
|
||||||
|
|
||||||
vmime::shared_ptr<vmime::message> plaintext_msg =
|
|
||||||
plaintext_msg_builder.construct();
|
|
||||||
|
|
||||||
auto plaintext_msg_header = plaintext_msg->getHeader();
|
|
||||||
|
|
||||||
auto plaintext_msg_content_type_header_field =
|
|
||||||
plaintext_msg_header->getField<vmime::contentTypeField>(
|
|
||||||
vmime::fields::CONTENT_TYPE);
|
|
||||||
plaintext_msg_content_type_header_field->setValue("text/plain");
|
|
||||||
plaintext_msg_content_type_header_field->appendParameter(
|
|
||||||
vmime::make_shared<vmime::parameter>("charset", "UTF-8"));
|
|
||||||
plaintext_msg_content_type_header_field->appendParameter(
|
|
||||||
vmime::make_shared<vmime::parameter>("format", "flowed"));
|
|
||||||
|
|
||||||
auto plaintext_msg_content_trans_encode_field =
|
|
||||||
plaintext_msg_header->getField(
|
|
||||||
vmime::fields::CONTENT_TRANSFER_ENCODING);
|
|
||||||
plaintext_msg_content_trans_encode_field->setValue("base64");
|
|
||||||
|
|
||||||
auto plaintext_msg_body = plaintext_msg->getBody();
|
|
||||||
|
|
||||||
auto mime_part_body_content =
|
|
||||||
vmime::make_shared<vmime::stringContentHandler>();
|
|
||||||
mime_part_body_content->setData(
|
|
||||||
body_data.toBase64().toStdString(),
|
|
||||||
vmime::encoding(vmime::encodingTypes::BASE64));
|
|
||||||
plaintext_msg_body->setContents(mime_part_body_content);
|
|
||||||
|
|
||||||
auto plaintext_eml_data =
|
|
||||||
Q_SC(plaintext_msg->generate(vmime::lineLengthLimits::infinite));
|
|
||||||
|
|
||||||
GFGpgEncryptionResult* enc_result = nullptr;
|
GFGpgEncryptionResult* enc_result = nullptr;
|
||||||
auto ret = GFGpgEncryptData(channel, QListToCharArray(keys), keys.size(),
|
auto ret = GFGpgEncryptData(channel, QListToCharArray(keys), keys.size(),
|
||||||
QDUP(plaintext_eml_data), 1, &enc_result);
|
QDUP(body_data), 1, &enc_result);
|
||||||
|
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
eml_data = "Encryption Failed";
|
eml_data = "Encryption Failed";
|
||||||
@ -260,7 +179,7 @@ auto EncryptEMLData(int channel, const QStringList& keys,
|
|||||||
encrypted_data.toStdString());
|
encrypted_data.toStdString());
|
||||||
encrypted_data_body->setContents(encrypted_data_content);
|
encrypted_data_body->setContents(encrypted_data_content);
|
||||||
|
|
||||||
eml_data = Q_SC(msg->generate(vmime::lineLengthLimits::infinite));
|
eml_data = Q_SC(msg->generate(vmime::lineLengthLimits::convenient));
|
||||||
FLOG_DEBUG("EML Data: %1", eml_data);
|
FLOG_DEBUG("EML Data: %1", eml_data);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -336,7 +255,9 @@ auto SignEMLData(int channel, const QString& key,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
msg_builder.setSubject(vmime::text(subject.toStdString()));
|
if (!subject.isEmpty()) {
|
||||||
|
msg_builder.setSubject(vmime::text(subject.toStdString()));
|
||||||
|
}
|
||||||
|
|
||||||
vmime::shared_ptr<vmime::message> msg = msg_builder.construct();
|
vmime::shared_ptr<vmime::message> msg = msg_builder.construct();
|
||||||
|
|
||||||
@ -461,13 +382,11 @@ auto SignEMLData(int channel, const QString& key,
|
|||||||
auto mime_part_part_body = mime_part->getBody();
|
auto mime_part_part_body = mime_part->getBody();
|
||||||
auto mime_part_body_content =
|
auto mime_part_body_content =
|
||||||
vmime::make_shared<vmime::stringContentHandler>();
|
vmime::make_shared<vmime::stringContentHandler>();
|
||||||
mime_part_body_content->setData(
|
mime_part_body_content->setData(body_data.toBase64().toStdString());
|
||||||
body_data.toBase64().toStdString(),
|
|
||||||
vmime::encoding(vmime::encodingTypes::BASE64));
|
|
||||||
mime_part_part_body->setContents(mime_part_body_content);
|
mime_part_part_body->setContents(mime_part_body_content);
|
||||||
|
|
||||||
auto container_raw_data =
|
auto container_raw_data =
|
||||||
Q_SC(container_part->generate(vmime::lineLengthLimits::infinite));
|
Q_SC(container_part->generate(vmime::lineLengthLimits::convenient));
|
||||||
|
|
||||||
auto container_raw_data_hash = QCryptographicHash::hash(
|
auto container_raw_data_hash = QCryptographicHash::hash(
|
||||||
container_raw_data.toLatin1(), QCryptographicHash::Sha1);
|
container_raw_data.toLatin1(), QCryptographicHash::Sha1);
|
||||||
@ -503,7 +422,218 @@ auto SignEMLData(int channel, const QString& key,
|
|||||||
signature.toStdString());
|
signature.toStdString());
|
||||||
signature_part_body->setContents(signature_part_body_content);
|
signature_part_body->setContents(signature_part_body_content);
|
||||||
|
|
||||||
eml_data = Q_SC(msg->generate(vmime::lineLengthLimits::infinite));
|
eml_data = Q_SC(msg->generate(vmime::lineLengthLimits::convenient));
|
||||||
|
|
||||||
|
FLOG_DEBUG("EML Data: %1", eml_data);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
} catch (const vmime::exception& e) {
|
||||||
|
eml_data = QString("VMIME Error: %1").arg(e.what());
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
eml_data = QString("Unknown Error: %1");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto AppendSignToEMLData(int channel, const QString& key,
|
||||||
|
const vmime::shared_ptr<vmime::message>& message,
|
||||||
|
QString& eml_data) -> int {
|
||||||
|
try {
|
||||||
|
auto header = message->getHeader();
|
||||||
|
|
||||||
|
auto backup_body_component = message->getBody()->clone();
|
||||||
|
|
||||||
|
std::shared_ptr<vmime::body> backup_body =
|
||||||
|
std::static_pointer_cast<vmime::body>(backup_body_component);
|
||||||
|
|
||||||
|
auto backup_content_type_header_field_component =
|
||||||
|
header->getField<vmime::headerField>(vmime::fields::CONTENT_TYPE)
|
||||||
|
->clone();
|
||||||
|
|
||||||
|
std::shared_ptr<vmime::headerField> backup_content_type_header_field =
|
||||||
|
std::static_pointer_cast<vmime::headerField>(
|
||||||
|
backup_content_type_header_field_component);
|
||||||
|
|
||||||
|
auto backup_content_trans_encode_field_component =
|
||||||
|
header
|
||||||
|
->getField<vmime::headerField>(
|
||||||
|
vmime::fields::CONTENT_TRANSFER_ENCODING)
|
||||||
|
->clone();
|
||||||
|
|
||||||
|
std::shared_ptr<vmime::headerField>
|
||||||
|
backup_content_trans_encode_header_field =
|
||||||
|
std::static_pointer_cast<vmime::headerField>(
|
||||||
|
backup_content_trans_encode_field_component);
|
||||||
|
|
||||||
|
FLOG_DEBUG("Content-Transfer-Encoding Header Data: %1",
|
||||||
|
backup_content_trans_encode_header_field->generate());
|
||||||
|
|
||||||
|
// no Content-Transfer-Encoding
|
||||||
|
header->removeField(
|
||||||
|
header->getField(vmime::fields::CONTENT_TRANSFER_ENCODING));
|
||||||
|
|
||||||
|
FLOG_DEBUG("Backup Content-Type Header Data: %1",
|
||||||
|
backup_content_type_header_field->generate());
|
||||||
|
FLOG_DEBUG("Backup Content-Transfer-Encoding Header Data: %1",
|
||||||
|
backup_content_trans_encode_header_field->generate());
|
||||||
|
|
||||||
|
auto content_type_header_field =
|
||||||
|
header->getField<vmime::contentTypeField>(vmime::fields::CONTENT_TYPE);
|
||||||
|
content_type_header_field->setValue("multipart/signed");
|
||||||
|
auto body_boundary = vmime::body::generateRandomBoundaryString();
|
||||||
|
content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>("protocol",
|
||||||
|
"application/pgp-signature"));
|
||||||
|
content_type_header_field->setBoundary(body_boundary);
|
||||||
|
|
||||||
|
auto root_body_part = vmime::make_shared<vmime::body>();
|
||||||
|
auto container_part = vmime::make_shared<vmime::bodyPart>();
|
||||||
|
auto mime_part = vmime::make_shared<vmime::bodyPart>();
|
||||||
|
auto public_key_part = vmime::make_shared<vmime::bodyPart>();
|
||||||
|
auto signature_part = vmime::make_shared<vmime::bodyPart>();
|
||||||
|
|
||||||
|
root_body_part->appendPart(container_part);
|
||||||
|
root_body_part->appendPart(signature_part);
|
||||||
|
message->setBody(root_body_part);
|
||||||
|
|
||||||
|
root_body_part->setPrologText(
|
||||||
|
"This is an OpenPGP/MIME signed message (RFC 4880 and 3156)");
|
||||||
|
|
||||||
|
auto container_boundary = vmime::body::generateRandomBoundaryString();
|
||||||
|
auto container_part_header = container_part->getHeader();
|
||||||
|
auto container_part_content_ttype_header_field =
|
||||||
|
container_part_header->getField<vmime::contentTypeField>(
|
||||||
|
vmime::fields::CONTENT_TYPE);
|
||||||
|
container_part_content_ttype_header_field->setValue("multipart/mixed");
|
||||||
|
container_part_content_ttype_header_field->setBoundary(container_boundary);
|
||||||
|
|
||||||
|
auto container_part_body = container_part->getBody();
|
||||||
|
|
||||||
|
container_part_body->appendPart(mime_part);
|
||||||
|
container_part_body->appendPart(public_key_part);
|
||||||
|
|
||||||
|
auto public_key_part_header = public_key_part->getHeader();
|
||||||
|
|
||||||
|
auto public_key_name = QString("OpenPGP_0x%1.asc").arg(key.toUpper());
|
||||||
|
auto public_key_part_content_type_header_field =
|
||||||
|
public_key_part_header->getField<vmime::contentTypeField>(
|
||||||
|
vmime::fields::CONTENT_TYPE);
|
||||||
|
public_key_part_content_type_header_field->setValue("application/pgp-keys");
|
||||||
|
public_key_part_content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>("name",
|
||||||
|
public_key_name.toStdString()));
|
||||||
|
|
||||||
|
auto public_key_part_content_desc_header_field =
|
||||||
|
public_key_part_header->getField(vmime::fields::CONTENT_DESCRIPTION);
|
||||||
|
public_key_part_content_desc_header_field->setValue("OpenPGP public key");
|
||||||
|
|
||||||
|
auto public_key_part_content_trans_encode_field =
|
||||||
|
public_key_part_header->getField(
|
||||||
|
vmime::fields::CONTENT_TRANSFER_ENCODING);
|
||||||
|
public_key_part_content_trans_encode_field->setValue("quoted-printable");
|
||||||
|
|
||||||
|
auto public_key_part_content_disp_header_field =
|
||||||
|
public_key_part_header->getField<vmime::contentDispositionField>(
|
||||||
|
vmime::fields::CONTENT_DISPOSITION);
|
||||||
|
public_key_part_content_disp_header_field->setValue("attachment");
|
||||||
|
public_key_part_content_disp_header_field->setFilename(
|
||||||
|
vmime::word(public_key_name.toStdString()));
|
||||||
|
|
||||||
|
auto signature_part_header = signature_part->getHeader();
|
||||||
|
|
||||||
|
auto signature_part_content_type_header_field =
|
||||||
|
signature_part_header->getField<vmime::contentTypeField>(
|
||||||
|
vmime::fields::CONTENT_TYPE);
|
||||||
|
signature_part_content_type_header_field->setValue(
|
||||||
|
"application/pgp-signature");
|
||||||
|
signature_part_content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>("name", "OpenPGP_signature.asc"));
|
||||||
|
|
||||||
|
auto signature_part_content_desc_header_field =
|
||||||
|
signature_part_header->getField(vmime::fields::CONTENT_DESCRIPTION);
|
||||||
|
signature_part_content_desc_header_field->setValue(
|
||||||
|
"OpenPGP digital signature");
|
||||||
|
|
||||||
|
auto signature_part_content_disp_header_field =
|
||||||
|
signature_part_header->getField<vmime::contentDispositionField>(
|
||||||
|
vmime::fields::CONTENT_DISPOSITION);
|
||||||
|
signature_part_content_disp_header_field->setValue("attachment");
|
||||||
|
signature_part_content_disp_header_field->setFilename(
|
||||||
|
vmime::word({"OpenPGP_signature.asc"}));
|
||||||
|
|
||||||
|
auto public_key = UDUP(GFGpgPublicKey(channel, QDUP(key), 1));
|
||||||
|
if (public_key.isEmpty()) {
|
||||||
|
eml_data = "Get Public Key of Sign Key Failed";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto public_key_part_part_body = public_key_part->getBody();
|
||||||
|
auto public_key_part_body_content =
|
||||||
|
vmime::make_shared<vmime::stringContentHandler>();
|
||||||
|
public_key_part_body_content->setData(
|
||||||
|
public_key.toLatin1().replace('\n', "\r\n").toStdString(),
|
||||||
|
vmime::encoding(vmime::encodingTypes::QUOTED_PRINTABLE));
|
||||||
|
public_key_part_part_body->setContents(public_key_part_body_content);
|
||||||
|
|
||||||
|
auto mime_part_header = mime_part->getHeader();
|
||||||
|
|
||||||
|
auto mime_part_content_trans_encode_field =
|
||||||
|
mime_part_header->getField<vmime::headerField>(
|
||||||
|
vmime::fields::CONTENT_TRANSFER_ENCODING);
|
||||||
|
mime_part_header->replaceField(mime_part_content_trans_encode_field,
|
||||||
|
backup_content_trans_encode_header_field);
|
||||||
|
|
||||||
|
auto mime_part_content_type_header_field =
|
||||||
|
mime_part_header->getField<vmime::contentTypeField>(
|
||||||
|
vmime::fields::CONTENT_TYPE);
|
||||||
|
mime_part_header->replaceField(mime_part_content_type_header_field,
|
||||||
|
backup_content_type_header_field);
|
||||||
|
|
||||||
|
mime_part->setBody(backup_body);
|
||||||
|
|
||||||
|
auto container_raw_data =
|
||||||
|
Q_SC(container_part->generate(vmime::lineLengthLimits::convenient));
|
||||||
|
|
||||||
|
container_raw_data.replace("\r\n", "\n");
|
||||||
|
container_raw_data.replace("\n", "\r\n");
|
||||||
|
|
||||||
|
auto container_raw_data_hash = QCryptographicHash::hash(
|
||||||
|
container_raw_data.toLatin1(), QCryptographicHash::Sha1);
|
||||||
|
FLOG_DEBUG("raw content of signature hash: %1",
|
||||||
|
container_raw_data_hash.toHex());
|
||||||
|
|
||||||
|
FLOG_DEBUG("MIME Raw Data For Signature: %1", container_raw_data);
|
||||||
|
FLOG_DEBUG("Signature Channel: %1, Sign Key: %2", channel, key);
|
||||||
|
|
||||||
|
GFGpgSignResult* s;
|
||||||
|
auto ret = GFGpgSignData(channel, QListToCharArray({key}), 1,
|
||||||
|
QDUP(container_raw_data), 1, 1, &s);
|
||||||
|
|
||||||
|
if (ret != 0) {
|
||||||
|
eml_data = "Sign Failed";
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto signature = UDUP(s->signature);
|
||||||
|
auto hash_algo = UDUP(s->hash_algo);
|
||||||
|
|
||||||
|
GFFreeMemory(s);
|
||||||
|
|
||||||
|
FLOG_DEBUG("Hash Algo: %1 Signature Data: %2", hash_algo, signature);
|
||||||
|
content_type_header_field->appendParameter(
|
||||||
|
vmime::make_shared<vmime::parameter>(
|
||||||
|
"micalg",
|
||||||
|
QString("pgp-%1").arg(hash_algo.toLower()).toStdString()));
|
||||||
|
|
||||||
|
auto signature_part_body = signature_part->getBody();
|
||||||
|
auto signature_part_body_content =
|
||||||
|
vmime::make_shared<vmime::stringContentHandler>(
|
||||||
|
signature.toStdString());
|
||||||
|
signature_part_body->setContents(signature_part_body_content);
|
||||||
|
|
||||||
|
eml_data = Q_SC(message->generate(vmime::lineLengthLimits::convenient));
|
||||||
|
|
||||||
FLOG_DEBUG("EML Data: %1", eml_data);
|
FLOG_DEBUG("EML Data: %1", eml_data);
|
||||||
|
|
||||||
|
@ -56,4 +56,17 @@ auto EncryptEMLData(int channel, const QStringList& keys,
|
|||||||
*/
|
*/
|
||||||
auto SignEMLData(int channel, const QString& key,
|
auto SignEMLData(int channel, const QString& key,
|
||||||
const EMailMetaData& meta_data, const QByteArray& body_data,
|
const EMailMetaData& meta_data, const QByteArray& body_data,
|
||||||
QString& eml_data) -> int;
|
QString& eml_data) -> int;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief
|
||||||
|
*
|
||||||
|
* @param channel
|
||||||
|
* @param key
|
||||||
|
* @param message
|
||||||
|
* @param eml_data
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
auto AppendSignToEMLData(int channel, const QString& key,
|
||||||
|
const vmime::shared_ptr<vmime::message>& message,
|
||||||
|
QString& eml_data) -> int;
|
@ -31,12 +31,7 @@
|
|||||||
#include <QDateTime>
|
#include <QDateTime>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
// vmime
|
#include "EMailModel.h"
|
||||||
#define VMIME_STATIC
|
|
||||||
#include <vmime/vmime.hpp>
|
|
||||||
// vmime extra
|
|
||||||
#include <vmime/contentDispositionField.hpp>
|
|
||||||
#include <vmime/contentTypeField.hpp>
|
|
||||||
|
|
||||||
auto inline Q_SC(const std::string& s) -> QString {
|
auto inline Q_SC(const std::string& s) -> QString {
|
||||||
return QString::fromStdString(s);
|
return QString::fromStdString(s);
|
||||||
@ -112,4 +107,44 @@ auto ExtractFieldValueDateTime(const vmime::shared_ptr<vmime::header>& header,
|
|||||||
* @return false
|
* @return false
|
||||||
*/
|
*/
|
||||||
auto ParseEmailString(const QString& input, QString& name,
|
auto ParseEmailString(const QString& input, QString& name,
|
||||||
QString& email) -> bool;
|
QString& email) -> bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief
|
||||||
|
*
|
||||||
|
* @param data
|
||||||
|
* @param lineLength
|
||||||
|
* @return QString
|
||||||
|
*/
|
||||||
|
auto EncodeBase64WithLineBreaks(const QByteArray& data,
|
||||||
|
int lineLength = 76) -> QString;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief
|
||||||
|
*
|
||||||
|
* @param data
|
||||||
|
* @return true
|
||||||
|
* @return false
|
||||||
|
*/
|
||||||
|
auto CheckIfEMLMessage(const QByteArray& data,
|
||||||
|
vmime::shared_ptr<vmime::message>& message) -> bool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief
|
||||||
|
*
|
||||||
|
* @param meta_data
|
||||||
|
* @param eml_data
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
auto BuildPlainTextEML(const EMailMetaData& meta_data,
|
||||||
|
const QByteArray& body_data, QString& eml_data) -> int;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief
|
||||||
|
*
|
||||||
|
* @param body_data
|
||||||
|
* @param meta_data
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
auto GetEMLMetaData(vmime::shared_ptr<vmime::message>& message,
|
||||||
|
EMailMetaData& meta_data) -> int;
|
@ -118,9 +118,11 @@ void EMailMetaDataDialog::slot_parse_eml_meta_data() {
|
|||||||
auto subject = ui_->subjectEdit->text();
|
auto subject = ui_->subjectEdit->text();
|
||||||
|
|
||||||
auto to = raw_to.split(';', Qt::SkipEmptyParts);
|
auto to = raw_to.split(';', Qt::SkipEmptyParts);
|
||||||
auto cc = raw_cc.split(';', Qt::SkipEmptyParts);
|
auto cc = raw_cc.trimmed().isEmpty() ? QStringList()
|
||||||
auto bcc = raw_bcc.split(';', Qt::SkipEmptyParts);
|
: raw_cc.split(';', Qt::SkipEmptyParts);
|
||||||
|
auto bcc = raw_bcc.trimmed().isEmpty()
|
||||||
|
? QStringList()
|
||||||
|
: raw_bcc.split(';', Qt::SkipEmptyParts);
|
||||||
QString name;
|
QString name;
|
||||||
QString email;
|
QString email;
|
||||||
|
|
||||||
|
@ -31,6 +31,13 @@
|
|||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
|
|
||||||
|
// vmime
|
||||||
|
#define VMIME_STATIC
|
||||||
|
#include <vmime/vmime.hpp>
|
||||||
|
// vmime extra
|
||||||
|
#include <vmime/contentDispositionField.hpp>
|
||||||
|
#include <vmime/contentTypeField.hpp>
|
||||||
|
|
||||||
struct EMailMetaData {
|
struct EMailMetaData {
|
||||||
QString from;
|
QString from;
|
||||||
QStringList to;
|
QStringList to;
|
||||||
|
@ -430,6 +430,29 @@ REGISTER_EVENT_HANDLER(EMAIL_SIGN_EML_DATA, [](const MEvent& event) -> int {
|
|||||||
r_dialog->SetKeys({sign_key});
|
r_dialog->SetKeys({sign_key});
|
||||||
r_dialog->SetBodyData({body_data});
|
r_dialog->SetBodyData({body_data});
|
||||||
|
|
||||||
|
vmime::shared_ptr<vmime::message> message;
|
||||||
|
if (CheckIfEMLMessage(body_data, message)) {
|
||||||
|
EMailMetaData meta_data;
|
||||||
|
auto ret = GetEMLMetaData(message, meta_data);
|
||||||
|
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -1, "Get MetaData From EML Data Failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
QString eml_data;
|
||||||
|
ret = AppendSignToEMLData(channel, sign_key, message, eml_data);
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -2, eml_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
CB(event, GFGetModuleID(),
|
||||||
|
{
|
||||||
|
{"ret", QString::number(0)},
|
||||||
|
{"eml_data", eml_data},
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
GFUIShowDialog(dialog, nullptr);
|
GFUIShowDialog(dialog, nullptr);
|
||||||
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
|
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
|
||||||
[=](const EMailMetaData& meta_data) {
|
[=](const EMailMetaData& meta_data) {
|
||||||
@ -469,6 +492,29 @@ REGISTER_EVENT_HANDLER(EMAIL_ENCRYPT_EML_DATA, [](const MEvent& event) -> int {
|
|||||||
auto body_data =
|
auto body_data =
|
||||||
QByteArray::fromBase64(QString(event["body_data"]).toLatin1());
|
QByteArray::fromBase64(QString(event["body_data"]).toLatin1());
|
||||||
|
|
||||||
|
vmime::shared_ptr<vmime::message> message;
|
||||||
|
if (CheckIfEMLMessage(body_data, message)) {
|
||||||
|
EMailMetaData meta_data;
|
||||||
|
auto ret = GetEMLMetaData(message, meta_data);
|
||||||
|
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -1, "Get MetaData From EML Data Failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
QString eml_data;
|
||||||
|
ret = EncryptEMLData(channel, encrypt_keys, meta_data, body_data, eml_data);
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -2, eml_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
CB(event, GFGetModuleID(),
|
||||||
|
{
|
||||||
|
{"ret", QString::number(0)},
|
||||||
|
{"eml_data", eml_data},
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
auto* dialog = GUI_OBJECT(CreateEMailMetaDataDialog, 1);
|
auto* dialog = GUI_OBJECT(CreateEMailMetaDataDialog, 1);
|
||||||
auto* r_dialog =
|
auto* r_dialog =
|
||||||
qobject_cast<EMailMetaDataDialog*>(static_cast<QObject*>(dialog));
|
qobject_cast<EMailMetaDataDialog*>(static_cast<QObject*>(dialog));
|
||||||
@ -481,22 +527,31 @@ REGISTER_EVENT_HANDLER(EMAIL_ENCRYPT_EML_DATA, [](const MEvent& event) -> int {
|
|||||||
|
|
||||||
GFUIShowDialog(dialog, nullptr);
|
GFUIShowDialog(dialog, nullptr);
|
||||||
|
|
||||||
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
|
QObject::connect(
|
||||||
[=](const EMailMetaData& meta_data) {
|
r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
|
||||||
QString eml_data;
|
[=](const EMailMetaData& meta_data) {
|
||||||
auto ret = EncryptEMLData(channel, encrypt_keys, meta_data,
|
QString eml_data;
|
||||||
body_data, eml_data);
|
QString plain_text_eml_data;
|
||||||
if (ret != 0) {
|
|
||||||
CB_ERR(event, -2, eml_data);
|
|
||||||
}
|
|
||||||
|
|
||||||
CB(event, GFGetModuleID(),
|
auto ret = BuildPlainTextEML(meta_data, body_data, plain_text_eml_data);
|
||||||
{
|
|
||||||
{"ret", QString::number(0)},
|
if (ret != 0) {
|
||||||
{"eml_data", eml_data},
|
CB_ERR(event, -1, "Build PlainText EML Data Failed");
|
||||||
});
|
}
|
||||||
return 0;
|
|
||||||
});
|
ret = EncryptEMLData(channel, encrypt_keys, meta_data,
|
||||||
|
plain_text_eml_data.toLatin1(), eml_data);
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -2, eml_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
CB(event, GFGetModuleID(),
|
||||||
|
{
|
||||||
|
{"ret", QString::number(0)},
|
||||||
|
{"eml_data", eml_data},
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
|
||||||
QObject::connect(
|
QObject::connect(
|
||||||
r_dialog, &EMailMetaDataDialog::SignalNoEMLMetaData, r_dialog,
|
r_dialog, &EMailMetaDataDialog::SignalNoEMLMetaData, r_dialog,
|
||||||
@ -505,6 +560,67 @@ REGISTER_EVENT_HANDLER(EMAIL_ENCRYPT_EML_DATA, [](const MEvent& event) -> int {
|
|||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
REGISTER_EVENT_HANDLER(
|
||||||
|
EMAIL_ENCRYPT_SIGN_EML_DATA, [](const MEvent& event) -> int {
|
||||||
|
if (event["body_data"].isEmpty()) CB_ERR(event, -1, "body_data is empty");
|
||||||
|
if (event["channel"].isEmpty()) CB_ERR(event, -1, "channel is empty");
|
||||||
|
if (event["encrypt_keys"].isEmpty())
|
||||||
|
CB_ERR(event, -1, "encrypt_keys is empty");
|
||||||
|
if (event["sign_key"].isEmpty()) CB_ERR(event, -1, "sign_key is empty");
|
||||||
|
|
||||||
|
auto channel = event.value("channel", "0").toInt();
|
||||||
|
auto sign_key = event.value("sign_key", "");
|
||||||
|
auto encrypt_keys = event.value("encrypt_keys", "").split(';');
|
||||||
|
|
||||||
|
FLOG_DEBUG("eml encrypt keys: %1", encrypt_keys.join(';'));
|
||||||
|
FLOG_DEBUG("eml sign key: %1", sign_key);
|
||||||
|
|
||||||
|
auto body_data =
|
||||||
|
QByteArray::fromBase64(QString(event["body_data"]).toLatin1());
|
||||||
|
|
||||||
|
auto* dialog = GUI_OBJECT(CreateEMailMetaDataDialog, 1);
|
||||||
|
auto* r_dialog =
|
||||||
|
qobject_cast<EMailMetaDataDialog*>(static_cast<QObject*>(dialog));
|
||||||
|
if (r_dialog == nullptr)
|
||||||
|
CB_ERR(event, -1, "convert dialog to r_dialog failed");
|
||||||
|
|
||||||
|
r_dialog->SetChannel(channel);
|
||||||
|
r_dialog->SetKeys(encrypt_keys);
|
||||||
|
r_dialog->SetBodyData({body_data});
|
||||||
|
|
||||||
|
GFUIShowDialog(dialog, nullptr);
|
||||||
|
|
||||||
|
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData,
|
||||||
|
r_dialog, [=](const EMailMetaData& meta_data) {
|
||||||
|
QString eml_data;
|
||||||
|
auto ret = SignEMLData(channel, sign_key, meta_data,
|
||||||
|
body_data, eml_data);
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -2, eml_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = EncryptEMLData(channel, encrypt_keys, meta_data,
|
||||||
|
body_data, eml_data);
|
||||||
|
if (ret != 0) {
|
||||||
|
CB_ERR(event, -2, eml_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
CB(event, GFGetModuleID(),
|
||||||
|
{
|
||||||
|
{"ret", QString::number(0)},
|
||||||
|
{"eml_data", eml_data},
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
|
||||||
|
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalNoEMLMetaData,
|
||||||
|
r_dialog, [=](const QString& error_string) {
|
||||||
|
CB_ERR(event, -1, error_string);
|
||||||
|
});
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
|
||||||
auto GFDeactivateModule() -> int { return 0; }
|
auto GFDeactivateModule() -> int { return 0; }
|
||||||
|
|
||||||
auto GFUnregisterModule() -> int {
|
auto GFUnregisterModule() -> int {
|
||||||
|
@ -1 +1 @@
|
|||||||
Subproject commit 43b262bd8c79fbaadc9522eaa04a40427782e0c6
|
Subproject commit 193a9ee85b1a491955a7181250ebf0296ccc5d0d
|
Loading…
Reference in New Issue
Block a user