feat: get public key from attachment if it exists

This commit is contained in:
missytake 2025-04-10 21:34:02 +02:00
parent daef27d36a
commit ce4f9ed8f4
Signed by: missytake
GPG key ID: 04CC6658320518DF
4 changed files with 87 additions and 3 deletions

View file

@ -21,6 +21,7 @@ install_requires =
wkdhash wkdhash
requests requests
email-validator email-validator
PGPy
[options.packages.find] [options.packages.find]
where = src where = src

View file

@ -1,4 +1,5 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
import pgpy
from deltachat_rpc_client import events, run_bot_cli, EventType, Message from deltachat_rpc_client import events, run_bot_cli, EventType, Message
from email_validator import validate_email, EmailNotValidError from email_validator import validate_email, EmailNotValidError
@ -24,12 +25,20 @@ def command(event):
return snapshot.chat.send_text(snapshot.chat.account.get_qr_code()) return snapshot.chat.send_text(snapshot.chat.account.get_qr_code())
if snapshot.text == "Messages are guaranteed to be end-to-end encrypted from now on.": if snapshot.text == "Messages are guaranteed to be end-to-end encrypted from now on.":
return return
email = snapshot.text email = snapshot.text
public_key = None
if snapshot.get('file'):
public_key, email_from_uid = import_key_from_attachment(snapshot.file)
email = email if email else email_from_uid
try: try:
validate_email(email, check_deliverability=False) validate_email(email, check_deliverability=False)
except EmailNotValidError: except EmailNotValidError:
return snapshot.chat.send_text(HELP_MSG + f"\n\n{email} is not an email address :/") return snapshot.chat.send_text(HELP_MSG + f"\n\n{email} is not an email address :/")
if not public_key:
public_key = request_key_by_email(email) public_key = request_key_by_email(email)
if not public_key: if not public_key:
return snapshot.chat.send_text(f"Sorry, I could not find a key for {email}.") return snapshot.chat.send_text(f"Sorry, I could not find a key for {email}.")
@ -40,6 +49,31 @@ def command(event):
return snapshot.chat.send_file(vcard_path) return snapshot.chat.send_file(vcard_path)
def import_key_from_attachment(file_path: str) -> (str, str):
"""Import a PGP public key from an attachment.
:param file_path: the path to the attachment
:return: a tuple with the public key and a User ID
"""
try:
key = pgpy.PGPKey.from_file(file_path)[0]
except AttributeError:
return "", ""
except pgpy.errors.PGPError:
return "", ""
ascii_key = str(key)
keyparts = []
for line in ascii_key.splitlines():
if line != "" and not line.startswith("Comment: ") and not line.startswith("-----"):
keyparts.append(line)
uid = key.get_uid("").userid
try:
validate_email(uid, check_deliverability=False)
except EmailNotValidError:
uid = key.get_uid("").email
return "".join(keyparts), uid
def request_key_by_email(email) -> str: def request_key_by_email(email) -> str:
domain = email.split("@")[1] domain = email.split("@")[1]
public_key = request_from_wkd(email, domain) public_key = request_from_wkd(email, domain)

View file

@ -0,0 +1,30 @@
-----BEGIN PGP PUBLIC KEY BLOCK-----
mDMEXcLIEBYJKwYBBAHaRw8BAQdAmlYU7TEgGL3eq2WXC95tQtZYHjpJOCjb7qq3
vJd1lG60Im1pc3N5dGFrZSA8bWlzc3l0YWtlQHN5c3RlbWxpLm9yZz6IkAQTFggA
OAIbAwULCQgHAgYVCgkICwIEFgIDAQIeAQIXgBYhBEX/6kI9pCw7MYVUWQTMZlgy
BRjfBQJnDuTLAAoJEATMZlgyBRjfeUUA/0P0E/quL71dn5Zjc4ewsnykT1GODazG
mk+xprSwAvEyAPwJy+uJgJz5pSFdmi2lGRrOERmKUu8AdQ/M7dSm6kt3AIiQBBMW
CAA4AhsDBQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAFiEERf/qQj2kLDsxhVRZBMxm
WDIFGN8FAmcO5FkACgkQBMxmWDIFGN+0EAD+PVALMtR4PDB4aaxpJ5L1p6mGmq1l
b8wqZtdAyHK9+7EA/jPU12/e368B6OHknY1YzGxQxyPETcL8hUS26CU6wnUEtBZt
aXNzeXRha2VAc3lzdGVtbGkub3JniJAEExYIADgCGwMFCwkIBwIGFQoJCAsCBBYC
AwECHgECF4AWIQRF/+pCPaQsOzGFVFkEzGZYMgUY3wUCZw7kywAKCRAEzGZYMgUY
37s7AP4uOSeC9cwDKnPnov7L6Sp0axUNncX+n5sjepkPpwgVIQD/Zt85kzYrodvs
x4QdolweWDFrH9DFxTsTSw2GWIIE6Q2IlgQTFggAPhYhBEX/6kI9pCw7MYVUWQTM
ZlgyBRjfBQJdwsgQAhsDBQkJZgGABQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAAAoJ
EATMZlgyBRjfXhwA/0kfgkVlieZJuKX3lOPGPPPupygP0+noJmr/TSGU48PXAQCB
JOsiJ7MWY/9jrfQFvL6/nMrohvRbWTrlr2zYeKGFCYiQBBMWCAA4AhsDBQsJCAcC
BhUKCQgLAgQWAgMBAh4BAheAFiEERf/qQj2kLDsxhVRZBMxmWDIFGN8FAmcO5FkA
CgkQBMxmWDIFGN/eWQEA915dJBj2g87jAe/6lgTl3r7iKPymUDq1JF5qqGd44q0B
AImC4KepwfwD0opn3ufWAgEWcnvGF01M8s+gBtgTUEAHuDgEXcLIEBIKKwYBBAGX
VQEFAQEHQGTy3kLa+rSeHhK35fDN/k46zZFh+LDZQ0a2552FuPJ6AwEIB4h4BBgW
CAAgAhsMFiEERf/qQj2kLDsxhVRZBMxmWDIFGN8FAmcO5P8ACgkQBMxmWDIFGN/j
JgEA5+ESV8PtiUcwxml7GpTGwbIv8GoDA1YlPcUeku/S20QA/iVvzlf8Oj5Wvhet
8VMzU37wWsFJ2n6aAyM2WmOrPVoPuDMEXcLKyhYJKwYBBAHaRw8BAQdAtUIjrOUm
kTbVVkwIXeAkC/s3Z1wrW2/+KIgkQbbi38WIeAQYFggAIAIbIBYhBEX/6kI9pCw7
MYVUWQTMZlgyBRjfBQJnDuULAAoJEATMZlgyBRjf19ABALR4jIhWK/5e87V3+xuX
9R0MAyBDztZjb8nfXJ3HWK65AQC6S7JybxebS+jHSavTIF0nuaaqBXZx3me1edqw
VVBADw==
=Zqcr
-----END PGP PUBLIC KEY BLOCK-----

View file

@ -2,7 +2,7 @@ import os
from email_validator import validate_email, EmailNotValidError from email_validator import validate_email, EmailNotValidError
import pytest import pytest
from keyserver_bot.hooks import request_key_by_email, delete_data from keyserver_bot.hooks import request_key_by_email, delete_data, import_key_from_attachment
from deltachat_rpc_client.pytestplugin import acfactory from deltachat_rpc_client.pytestplugin import acfactory
@ -60,3 +60,22 @@ def test_validate_email(email, valid):
validate_email(email, check_deliverability=False) validate_email(email, check_deliverability=False)
except EmailNotValidError: except EmailNotValidError:
assert not valid assert not valid
@pytest.mark.parametrize(
("file_path", "result"),
[
(
"tests/assets/missytake-pub.asc",
(
"mDMEXcLIEBYJKwYBBAHaRw8BAQdAmlYU7TEgGL3eq2WXC95tQtZYHjpJOCjb7qq3vJd1lG60Fm1pc3N5dGFrZUBzeXN0ZW1saS5vcmeIlgQTFggAPhYhBEX/6kI9pCw7MYVUWQTMZlgyBRjfBQJdwsgQAhsDBQkJZgGABQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAAAoJEATMZlgyBRjfXhwA/0kfgkVlieZJuKX3lOPGPPPupygP0+noJmr/TSGU48PXAQCBJOsiJ7MWY/9jrfQFvL6/nMrohvRbWTrlr2zYeKGFCYiQBBMWCAA4AhsDBQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAFiEERf/qQj2kLDsxhVRZBMxmWDIFGN8FAmcO5FkACgkQBMxmWDIFGN/eWQEA915dJBj2g87jAe/6lgTl3r7iKPymUDq1JF5qqGd44q0BAImC4KepwfwD0opn3ufWAgEWcnvGF01M8s+gBtgTUEAHiJAEExYIADgCGwMFCwkIBwIGFQoJCAsCBBYCAwECHgECF4AWIQRF/+pCPaQsOzGFVFkEzGZYMgUY3wUCZw7kywAKCRAEzGZYMgUY37s7AP4uOSeC9cwDKnPnov7L6Sp0axUNncX+n5sjepkPpwgVIQD/Zt85kzYrodvsx4QdolweWDFrH9DFxTsTSw2GWIIE6Q20Im1pc3N5dGFrZSA8bWlzc3l0YWtlQHN5c3RlbWxpLm9yZz6IkAQTFggAOAIbAwULCQgHAgYVCgkICwIEFgIDAQIeAQIXgBYhBEX/6kI9pCw7MYVUWQTMZlgyBRjfBQJnDuRZAAoJEATMZlgyBRjftBAA/j1QCzLUeDwweGmsaSeS9aephpqtZW/MKmbXQMhyvfuxAP4z1Ndv3t+vAejh5J2NWMxsUMcjxE3C/IVEtuglOsJ1BIiQBBMWCAA4AhsDBQsJCAcCBhUKCQgLAgQWAgMBAh4BAheAFiEERf/qQj2kLDsxhVRZBMxmWDIFGN8FAmcO5MsACgkQBMxmWDIFGN95RQD/Q/QT+q4vvV2flmNzh7CyfKRPUY4NrMaaT7GmtLAC8TIA/AnL64mAnPmlIV2aLaUZGs4RGYpS7wB1D8zt1KbqS3cAuDgEXcLIEBIKKwYBBAGXVQEFAQEHQGTy3kLa+rSeHhK35fDN/k46zZFh+LDZQ0a2552FuPJ6AwEIB4h4BBgWCAAgAhsMFiEERf/qQj2kLDsxhVRZBMxmWDIFGN8FAmcO5P8ACgkQBMxmWDIFGN/jJgEA5+ESV8PtiUcwxml7GpTGwbIv8GoDA1YlPcUeku/S20QA/iVvzlf8Oj5Wvhet8VMzU37wWsFJ2n6aAyM2WmOrPVoPuDMEXcLKyhYJKwYBBAHaRw8BAQdAtUIjrOUmkTbVVkwIXeAkC/s3Z1wrW2/+KIgkQbbi38WIeAQYFggAIAIbIBYhBEX/6kI9pCw7MYVUWQTMZlgyBRjfBQJnDuULAAoJEATMZlgyBRjf19ABALR4jIhWK/5e87V3+xuX9R0MAyBDztZjb8nfXJ3HWK65AQC6S7JybxebS+jHSavTIF0nuaaqBXZx3me1edqwVVBADw===wNTN",
"missytake@systemli.org",
),
),
("tests/test_hooks.py", ("", "")),
],
)
def test_import_key_from_attachment(file_path, result):
imported = import_key_from_attachment(file_path)
assert result[0] == imported[0]
assert result[1] == imported[1]