aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorsaturneric <[email protected]>2024-11-28 11:02:54 +0000
committersaturneric <[email protected]>2024-11-28 11:02:54 +0000
commit0a88a9a46ca6140415c6a908721760d55237dd30 (patch)
tree309097bfd8eaea6543eff563f997659103492706
parentrefactor: improve code structure (diff)
downloadModules-0a88a9a46ca6140415c6a908721760d55237dd30.tar.gz
Modules-0a88a9a46ca6140415c6a908721760d55237dd30.zip
feat: make email operations more comfortable
-rw-r--r--.gitmodules2
-rw-r--r--src/m_email/EMAilHelper.cpp170
-rw-r--r--src/m_email/EMailBasicGpgOpera.cpp308
-rw-r--r--src/m_email/EMailBasicGpgOpera.h15
-rw-r--r--src/m_email/EMailHelper.h49
-rw-r--r--src/m_email/EMailMetaDataDialog.cpp8
-rw-r--r--src/m_email/EMailModel.h7
-rw-r--r--src/m_email/EMailModule.cpp148
m---------src/m_email/vmime0
9 files changed, 585 insertions, 122 deletions
diff --git a/.gitmodules b/.gitmodules
index 83943df..57e7597 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -1,3 +1,3 @@
[submodule "src/m_email/vmime"]
path = src/m_email/vmime
- url = https://github.com/kisli/vmime.git
+ url = https://git.bktus.com/GpgFrontend/vmime.git
diff --git a/src/m_email/EMAilHelper.cpp b/src/m_email/EMAilHelper.cpp
index 7d19dad..ff9c0b7 100644
--- a/src/m_email/EMAilHelper.cpp
+++ b/src/m_email/EMAilHelper.cpp
@@ -64,7 +64,7 @@ auto ExtractFieldValue(const vmime::shared_ptr<vmime::header>& header,
auto field_value = field->getValue();
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 {};
}
@@ -81,7 +81,7 @@ auto ExtractFieldValueMailBox(const vmime::shared_ptr<vmime::header>& header,
auto field_value = field->getValue<vmime::mailbox>();
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 {};
}
@@ -99,7 +99,7 @@ auto ExtractFieldValueAddressList(
auto field_value = field->getValue<vmime::addressList>();
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 {};
}
@@ -114,6 +114,8 @@ auto ExtractFieldValueAddressList(
auto ExtractFieldValueText(const vmime::shared_ptr<vmime::header>& header,
const QString& field_name) -> QString {
+ if (!header->hasField(field_name.toStdString())) return {};
+
auto field = header->getField(field_name.toStdString());
if (!field) {
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>();
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 {};
}
@@ -139,7 +141,7 @@ auto ExtractFieldValueDateTime(const vmime::shared_ptr<vmime::header>& header,
auto field_value = field->getValue<vmime::datetime>();
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 {};
}
@@ -167,4 +169,162 @@ auto ParseEmailString(const QString& input, QString& name,
}
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;
} \ No newline at end of file
diff --git a/src/m_email/EMailBasicGpgOpera.cpp b/src/m_email/EMailBasicGpgOpera.cpp
index bf60d1c..10988a6 100644
--- a/src/m_email/EMailBasicGpgOpera.cpp
+++ b/src/m_email/EMailBasicGpgOpera.cpp
@@ -49,90 +49,9 @@ auto EncryptEMLData(int channel, const QStringList& keys,
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.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;
auto ret = GFGpgEncryptData(channel, QListToCharArray(keys), keys.size(),
- QDUP(plaintext_eml_data), 1, &enc_result);
+ QDUP(body_data), 1, &enc_result);
if (ret != 0) {
eml_data = "Encryption Failed";
@@ -260,7 +179,7 @@ auto EncryptEMLData(int channel, const QStringList& keys,
encrypted_data.toStdString());
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);
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();
@@ -461,13 +382,222 @@ auto SignEMLData(int channel, const QString& key,
auto mime_part_part_body = mime_part->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));
+ mime_part_body_content->setData(body_data.toBase64().toStdString());
mime_part_part_body->setContents(mime_part_body_content);
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(
+ 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(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);
@@ -503,7 +633,7 @@ auto SignEMLData(int channel, const QString& key,
signature.toStdString());
signature_part_body->setContents(signature_part_body_content);
- eml_data = Q_SC(msg->generate(vmime::lineLengthLimits::infinite));
+ eml_data = Q_SC(message->generate(vmime::lineLengthLimits::convenient));
FLOG_DEBUG("EML Data: %1", eml_data);
diff --git a/src/m_email/EMailBasicGpgOpera.h b/src/m_email/EMailBasicGpgOpera.h
index 06753d3..4a6f5dc 100644
--- a/src/m_email/EMailBasicGpgOpera.h
+++ b/src/m_email/EMailBasicGpgOpera.h
@@ -56,4 +56,17 @@ auto EncryptEMLData(int channel, const QStringList& keys,
*/
auto SignEMLData(int channel, const QString& key,
const EMailMetaData& meta_data, const QByteArray& body_data,
- QString& eml_data) -> int; \ No newline at end of file
+ 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; \ No newline at end of file
diff --git a/src/m_email/EMailHelper.h b/src/m_email/EMailHelper.h
index a4a7b77..6f82b9c 100644
--- a/src/m_email/EMailHelper.h
+++ b/src/m_email/EMailHelper.h
@@ -31,12 +31,7 @@
#include <QDateTime>
#include <QString>
-// vmime
-#define VMIME_STATIC
-#include <vmime/vmime.hpp>
-// vmime extra
-#include <vmime/contentDispositionField.hpp>
-#include <vmime/contentTypeField.hpp>
+#include "EMailModel.h"
auto inline Q_SC(const std::string& s) -> QString {
return QString::fromStdString(s);
@@ -112,4 +107,44 @@ auto ExtractFieldValueDateTime(const vmime::shared_ptr<vmime::header>& header,
* @return false
*/
auto ParseEmailString(const QString& input, QString& name,
- QString& email) -> bool; \ No newline at end of file
+ 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; \ No newline at end of file
diff --git a/src/m_email/EMailMetaDataDialog.cpp b/src/m_email/EMailMetaDataDialog.cpp
index bcabe48..3c8d137 100644
--- a/src/m_email/EMailMetaDataDialog.cpp
+++ b/src/m_email/EMailMetaDataDialog.cpp
@@ -118,9 +118,11 @@ void EMailMetaDataDialog::slot_parse_eml_meta_data() {
auto subject = ui_->subjectEdit->text();
auto to = raw_to.split(';', Qt::SkipEmptyParts);
- auto cc = raw_cc.split(';', Qt::SkipEmptyParts);
- auto bcc = raw_bcc.split(';', Qt::SkipEmptyParts);
-
+ auto cc = raw_cc.trimmed().isEmpty() ? QStringList()
+ : raw_cc.split(';', Qt::SkipEmptyParts);
+ auto bcc = raw_bcc.trimmed().isEmpty()
+ ? QStringList()
+ : raw_bcc.split(';', Qt::SkipEmptyParts);
QString name;
QString email;
diff --git a/src/m_email/EMailModel.h b/src/m_email/EMailModel.h
index 454645f..791bb17 100644
--- a/src/m_email/EMailModel.h
+++ b/src/m_email/EMailModel.h
@@ -31,6 +31,13 @@
#include <QString>
#include <QStringList>
+// vmime
+#define VMIME_STATIC
+#include <vmime/vmime.hpp>
+// vmime extra
+#include <vmime/contentDispositionField.hpp>
+#include <vmime/contentTypeField.hpp>
+
struct EMailMetaData {
QString from;
QStringList to;
diff --git a/src/m_email/EMailModule.cpp b/src/m_email/EMailModule.cpp
index 9e76689..e00e276 100644
--- a/src/m_email/EMailModule.cpp
+++ b/src/m_email/EMailModule.cpp
@@ -430,6 +430,29 @@ REGISTER_EVENT_HANDLER(EMAIL_SIGN_EML_DATA, [](const MEvent& event) -> int {
r_dialog->SetKeys({sign_key});
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);
QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
[=](const EMailMetaData& meta_data) {
@@ -469,6 +492,29 @@ REGISTER_EVENT_HANDLER(EMAIL_ENCRYPT_EML_DATA, [](const MEvent& event) -> int {
auto body_data =
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* r_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);
- QObject::connect(r_dialog, &EMailMetaDataDialog::SignalEMLMetaData, r_dialog,
- [=](const EMailMetaData& meta_data) {
- QString eml_data;
- auto 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::SignalEMLMetaData, r_dialog,
+ [=](const EMailMetaData& meta_data) {
+ QString eml_data;
+ QString plain_text_eml_data;
+
+ auto ret = BuildPlainTextEML(meta_data, body_data, plain_text_eml_data);
+
+ if (ret != 0) {
+ CB_ERR(event, -1, "Build PlainText EML Data Failed");
+ }
+
+ 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(
r_dialog, &EMailMetaDataDialog::SignalNoEMLMetaData, r_dialog,
@@ -505,6 +560,67 @@ REGISTER_EVENT_HANDLER(EMAIL_ENCRYPT_EML_DATA, [](const MEvent& event) -> int {
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 GFUnregisterModule() -> int {
diff --git a/src/m_email/vmime b/src/m_email/vmime
-Subproject 43b262bd8c79fbaadc9522eaa04a40427782e0c
+Subproject 193a9ee85b1a491955a7181250ebf0296ccc5d0