2020-04-03 18:52:18 +00:00
|
|
|
__filename__ = "pgp.py"
|
|
|
|
__author__ = "Bob Mottram"
|
|
|
|
__license__ = "AGPL3+"
|
2021-01-26 10:07:42 +00:00
|
|
|
__version__ = "1.2.0"
|
2020-04-03 18:52:18 +00:00
|
|
|
__maintainer__ = "Bob Mottram"
|
|
|
|
__email__ = "bob@freedombone.net"
|
|
|
|
__status__ = "Production"
|
2019-12-17 20:44:18 +00:00
|
|
|
|
2021-03-11 20:33:45 +00:00
|
|
|
import os
|
2021-03-11 17:15:32 +00:00
|
|
|
import subprocess
|
2021-03-11 20:33:45 +00:00
|
|
|
from pathlib import Path
|
2021-03-11 19:13:41 +00:00
|
|
|
from person import getActorJson
|
2021-03-11 17:15:32 +00:00
|
|
|
|
2019-12-17 20:44:18 +00:00
|
|
|
|
|
|
|
def getEmailAddress(actorJson: {}) -> str:
|
|
|
|
"""Returns the email address for the given actor
|
|
|
|
"""
|
|
|
|
if not actorJson.get('attachment'):
|
|
|
|
return ''
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('email'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('value'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
2019-12-17 20:44:18 +00:00
|
|
|
continue
|
|
|
|
if '@' not in propertyValue['value']:
|
|
|
|
continue
|
|
|
|
if '.' not in propertyValue['value']:
|
|
|
|
continue
|
|
|
|
return propertyValue['value']
|
|
|
|
return ''
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
|
2019-12-17 20:44:18 +00:00
|
|
|
def getPGPpubKey(actorJson: {}) -> str:
|
|
|
|
"""Returns PGP public key for the given actor
|
|
|
|
"""
|
|
|
|
if not actorJson.get('attachment'):
|
|
|
|
return ''
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('pgp'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('value'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
2019-12-17 20:44:18 +00:00
|
|
|
continue
|
2021-03-12 09:50:08 +00:00
|
|
|
if not containsPGPPublicKey(propertyValue['value']):
|
2019-12-17 20:44:18 +00:00
|
|
|
continue
|
|
|
|
return propertyValue['value']
|
|
|
|
return ''
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
|
2020-07-06 09:52:06 +00:00
|
|
|
def getPGPfingerprint(actorJson: {}) -> str:
|
|
|
|
"""Returns PGP fingerprint for the given actor
|
|
|
|
"""
|
|
|
|
if not actorJson.get('attachment'):
|
|
|
|
return ''
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('openpgp'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('value'):
|
|
|
|
continue
|
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
|
|
|
continue
|
|
|
|
if len(propertyValue['value']) < 10:
|
|
|
|
continue
|
|
|
|
return propertyValue['value']
|
|
|
|
return ''
|
|
|
|
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
def setEmailAddress(actorJson: {}, emailAddress: str) -> None:
|
2019-12-17 20:44:18 +00:00
|
|
|
"""Sets the email address for the given actor
|
|
|
|
"""
|
2020-07-06 10:25:18 +00:00
|
|
|
notEmailAddress = False
|
|
|
|
if '@' not in emailAddress:
|
|
|
|
notEmailAddress = True
|
|
|
|
if '.' not in emailAddress:
|
|
|
|
notEmailAddress = True
|
2020-12-12 15:31:28 +00:00
|
|
|
if '<' in emailAddress:
|
|
|
|
notEmailAddress = True
|
2020-07-06 10:25:18 +00:00
|
|
|
if emailAddress.startswith('@'):
|
|
|
|
notEmailAddress = True
|
|
|
|
|
2019-12-17 20:44:18 +00:00
|
|
|
if not actorJson.get('attachment'):
|
2020-04-03 18:52:18 +00:00
|
|
|
actorJson['attachment'] = []
|
2019-12-17 20:44:18 +00:00
|
|
|
|
2019-12-17 23:35:59 +00:00
|
|
|
# remove any existing value
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyFound = None
|
2019-12-17 23:35:59 +00:00
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('email'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyFound = propertyValue
|
2019-12-17 23:35:59 +00:00
|
|
|
break
|
|
|
|
if propertyFound:
|
|
|
|
actorJson['attachment'].remove(propertyFound)
|
2020-07-06 10:25:18 +00:00
|
|
|
if notEmailAddress:
|
2019-12-17 20:44:18 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('email'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
2019-12-17 20:44:18 +00:00
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyValue['value'] = emailAddress
|
2019-12-17 20:44:18 +00:00
|
|
|
return
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
newEmailAddress = {
|
2019-12-17 20:44:18 +00:00
|
|
|
"name": "Email",
|
|
|
|
"type": "PropertyValue",
|
|
|
|
"value": emailAddress
|
|
|
|
}
|
|
|
|
actorJson['attachment'].append(newEmailAddress)
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
|
|
|
|
def setPGPpubKey(actorJson: {}, PGPpubKey: str) -> None:
|
2019-12-17 20:44:18 +00:00
|
|
|
"""Sets a PGP public key for the given actor
|
|
|
|
"""
|
2020-07-06 09:52:06 +00:00
|
|
|
removeKey = False
|
|
|
|
if not PGPpubKey:
|
|
|
|
removeKey = True
|
|
|
|
else:
|
2021-03-12 09:50:08 +00:00
|
|
|
if not containsPGPPublicKey(PGPpubKey):
|
2020-07-06 09:52:06 +00:00
|
|
|
removeKey = True
|
2020-12-12 15:31:28 +00:00
|
|
|
if '<' in PGPpubKey:
|
|
|
|
removeKey = True
|
2020-07-06 09:52:06 +00:00
|
|
|
|
2019-12-17 20:44:18 +00:00
|
|
|
if not actorJson.get('attachment'):
|
2020-04-03 18:52:18 +00:00
|
|
|
actorJson['attachment'] = []
|
2019-12-17 20:44:18 +00:00
|
|
|
|
2019-12-17 23:35:59 +00:00
|
|
|
# remove any existing value
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyFound = None
|
2019-12-17 23:35:59 +00:00
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('pgp'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyFound = propertyValue
|
2019-12-17 23:35:59 +00:00
|
|
|
break
|
|
|
|
if propertyFound:
|
|
|
|
actorJson['attachment'].remove(propertyValue)
|
2020-07-06 09:52:06 +00:00
|
|
|
if removeKey:
|
2019-12-17 20:44:18 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('pgp'):
|
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
2019-12-17 20:44:18 +00:00
|
|
|
continue
|
2020-04-03 18:52:18 +00:00
|
|
|
propertyValue['value'] = PGPpubKey
|
2019-12-17 20:44:18 +00:00
|
|
|
return
|
|
|
|
|
2020-04-03 18:52:18 +00:00
|
|
|
newPGPpubKey = {
|
2019-12-17 20:44:18 +00:00
|
|
|
"name": "PGP",
|
|
|
|
"type": "PropertyValue",
|
|
|
|
"value": PGPpubKey
|
|
|
|
}
|
|
|
|
actorJson['attachment'].append(newPGPpubKey)
|
2020-07-06 09:52:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
def setPGPfingerprint(actorJson: {}, fingerprint: str) -> None:
|
|
|
|
"""Sets a PGP fingerprint for the given actor
|
|
|
|
"""
|
|
|
|
removeFingerprint = False
|
|
|
|
if not fingerprint:
|
|
|
|
removeFingerprint = True
|
|
|
|
else:
|
|
|
|
if len(fingerprint) < 10:
|
|
|
|
removeFingerprint = True
|
|
|
|
|
|
|
|
if not actorJson.get('attachment'):
|
|
|
|
actorJson['attachment'] = []
|
|
|
|
|
|
|
|
# remove any existing value
|
|
|
|
propertyFound = None
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('openpgp'):
|
|
|
|
continue
|
|
|
|
propertyFound = propertyValue
|
|
|
|
break
|
|
|
|
if propertyFound:
|
|
|
|
actorJson['attachment'].remove(propertyValue)
|
|
|
|
if removeFingerprint:
|
|
|
|
return
|
|
|
|
|
|
|
|
for propertyValue in actorJson['attachment']:
|
|
|
|
if not propertyValue.get('name'):
|
|
|
|
continue
|
|
|
|
if not propertyValue.get('type'):
|
|
|
|
continue
|
|
|
|
if not propertyValue['name'].lower().startswith('openpgp'):
|
|
|
|
continue
|
|
|
|
if propertyValue['type'] != 'PropertyValue':
|
|
|
|
continue
|
|
|
|
propertyValue['value'] = fingerprint.strip()
|
|
|
|
return
|
|
|
|
|
|
|
|
newPGPfingerprint = {
|
|
|
|
"name": "OpenPGP",
|
|
|
|
"type": "PropertyValue",
|
|
|
|
"value": fingerprint
|
|
|
|
}
|
|
|
|
actorJson['attachment'].append(newPGPfingerprint)
|
2021-03-11 14:23:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
def extractPGPPublicKey(content: str) -> str:
|
|
|
|
"""Returns the PGP key from the given text
|
|
|
|
"""
|
|
|
|
startBlock = '--BEGIN PGP PUBLIC KEY BLOCK--'
|
|
|
|
endBlock = '--END PGP PUBLIC KEY BLOCK--'
|
2021-03-11 17:15:32 +00:00
|
|
|
if startBlock not in content:
|
2021-03-11 14:23:03 +00:00
|
|
|
return None
|
2021-03-11 17:15:32 +00:00
|
|
|
if endBlock not in content:
|
2021-03-11 14:23:03 +00:00
|
|
|
return None
|
|
|
|
if '\n' not in content:
|
|
|
|
return None
|
|
|
|
linesList = content.split('\n')
|
|
|
|
extracting = False
|
|
|
|
publicKey = ''
|
|
|
|
for line in linesList:
|
|
|
|
if not extracting:
|
|
|
|
if startBlock in line:
|
|
|
|
extracting = True
|
|
|
|
else:
|
|
|
|
if endBlock in line:
|
|
|
|
publicKey += line
|
|
|
|
break
|
|
|
|
if extracting:
|
|
|
|
publicKey += line + '\n'
|
|
|
|
return publicKey
|
2021-03-11 17:15:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _pgpImportPubKey(recipientPubKey: str) -> str:
|
|
|
|
""" Import the given public key
|
|
|
|
"""
|
|
|
|
# do a dry run
|
|
|
|
cmdImportPubKey = \
|
|
|
|
'echo "' + recipientPubKey + '" | gpg --dry-run --import 2> /dev/null'
|
|
|
|
proc = subprocess.Popen([cmdImportPubKey],
|
|
|
|
stdout=subprocess.PIPE, shell=True)
|
|
|
|
(importResult, err) = proc.communicate()
|
|
|
|
if err:
|
|
|
|
return None
|
|
|
|
|
|
|
|
# this time for real
|
|
|
|
cmdImportPubKey = \
|
|
|
|
'echo "' + recipientPubKey + '" | gpg --import 2> /dev/null'
|
|
|
|
proc = subprocess.Popen([cmdImportPubKey],
|
|
|
|
stdout=subprocess.PIPE, shell=True)
|
|
|
|
(importResult, err) = proc.communicate()
|
|
|
|
if err:
|
|
|
|
return None
|
|
|
|
|
|
|
|
# get the key id
|
|
|
|
cmdImportPubKey = \
|
|
|
|
'echo "' + recipientPubKey + '" | gpg --show-keys'
|
|
|
|
proc = subprocess.Popen([cmdImportPubKey],
|
|
|
|
stdout=subprocess.PIPE, shell=True)
|
|
|
|
(importResult, err) = proc.communicate()
|
|
|
|
if not importResult:
|
|
|
|
return None
|
|
|
|
importResult = importResult.decode('utf-8').split('\n')
|
|
|
|
keyId = ''
|
|
|
|
for line in importResult:
|
|
|
|
if line.startswith('pub'):
|
|
|
|
continue
|
|
|
|
elif line.startswith('uid'):
|
|
|
|
continue
|
|
|
|
elif line.startswith('sub'):
|
|
|
|
continue
|
|
|
|
keyId = line.strip()
|
|
|
|
break
|
|
|
|
return keyId
|
|
|
|
|
|
|
|
|
2021-03-11 20:33:45 +00:00
|
|
|
def _pgpEncrypt(content: str, recipientPubKey: str) -> str:
|
2021-03-11 17:15:32 +00:00
|
|
|
""" Encrypt using your default pgp key to the given recipient
|
|
|
|
"""
|
|
|
|
keyId = _pgpImportPubKey(recipientPubKey)
|
|
|
|
if not keyId:
|
|
|
|
return None
|
|
|
|
|
|
|
|
cmdEncrypt = \
|
|
|
|
'echo "' + content + '" | gpg --encrypt --armor --recipient ' + \
|
|
|
|
keyId + ' 2> /dev/null'
|
|
|
|
proc = subprocess.Popen([cmdEncrypt],
|
|
|
|
stdout=subprocess.PIPE, shell=True)
|
|
|
|
(encryptResult, err) = proc.communicate()
|
|
|
|
if not encryptResult:
|
|
|
|
return None
|
|
|
|
encryptResult = encryptResult.decode('utf-8')
|
2021-03-12 09:50:08 +00:00
|
|
|
if not isPGPEncrypted(encryptResult):
|
2021-03-11 17:15:32 +00:00
|
|
|
return None
|
|
|
|
return encryptResult
|
|
|
|
|
|
|
|
|
2021-03-11 20:33:45 +00:00
|
|
|
def _getPGPPublicKeyFromActor(handle: str, actorJson=None) -> str:
|
2021-03-11 19:13:41 +00:00
|
|
|
"""Searches tags on the actor to see if there is any PGP
|
|
|
|
public key specified
|
|
|
|
"""
|
2021-03-11 20:33:45 +00:00
|
|
|
if not actorJson:
|
|
|
|
actorJson = getActorJson(handle, False, False, True)
|
2021-03-11 19:13:41 +00:00
|
|
|
if not actorJson:
|
|
|
|
return None
|
|
|
|
if not actorJson.get('attachment'):
|
|
|
|
return None
|
|
|
|
if not isinstance(actorJson['attachment'], list):
|
|
|
|
return None
|
2021-03-11 20:33:45 +00:00
|
|
|
# search through the tags on the actor
|
2021-03-11 19:13:41 +00:00
|
|
|
for tag in actorJson['attachment']:
|
2021-03-11 20:33:45 +00:00
|
|
|
if not isinstance(tag, dict):
|
|
|
|
continue
|
2021-03-11 19:13:41 +00:00
|
|
|
if not tag.get('value'):
|
|
|
|
continue
|
2021-03-11 20:35:48 +00:00
|
|
|
if not isinstance(tag['value'], str):
|
|
|
|
continue
|
2021-03-12 09:50:08 +00:00
|
|
|
if containsPGPPublicKey(tag['value']):
|
|
|
|
return tag['value']
|
2021-03-11 19:13:41 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
2021-03-11 20:33:45 +00:00
|
|
|
def hasLocalPGPkey() -> bool:
|
|
|
|
"""Returns true if there is a local .gnupg directory
|
|
|
|
"""
|
|
|
|
homeDir = str(Path.home())
|
|
|
|
gpgDir = homeDir + '/.gnupg'
|
2021-03-11 20:48:02 +00:00
|
|
|
if os.path.isdir(gpgDir):
|
2021-03-11 20:33:45 +00:00
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
def pgpEncryptToActor(content: str, toHandle: str) -> str:
|
|
|
|
"""PGP encrypt a message to the given actor or handle
|
|
|
|
"""
|
|
|
|
# get the actor and extract the pgp public key from it
|
|
|
|
recipientPubKey = _getPGPPublicKeyFromActor(toHandle)
|
|
|
|
if not recipientPubKey:
|
|
|
|
return None
|
|
|
|
# encrypt using the recipient public key
|
|
|
|
return _pgpEncrypt(content, recipientPubKey)
|
|
|
|
|
|
|
|
|
2021-03-12 09:50:08 +00:00
|
|
|
def isPGPEncrypted(content: str) -> bool:
|
|
|
|
"""Returns true if the given content is PGP encrypted
|
|
|
|
"""
|
|
|
|
if '--BEGIN PGP MESSAGE--' in content:
|
|
|
|
if '--END PGP MESSAGE--' in content:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
def containsPGPPublicKey(content: str) -> bool:
|
|
|
|
"""Returns true if the given content contains a PGP public key
|
|
|
|
"""
|
|
|
|
if '--BEGIN PGP PUBLIC KEY BLOCK--' in content:
|
|
|
|
if '--END PGP PUBLIC KEY BLOCK--' in content:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2021-03-11 19:13:41 +00:00
|
|
|
def pgpDecrypt(content: str, fromHandle: str) -> str:
|
2021-03-11 17:15:32 +00:00
|
|
|
""" Encrypt using your default pgp key to the given recipient
|
2021-03-11 19:13:41 +00:00
|
|
|
fromHandle can be a handle or actor url
|
2021-03-11 17:15:32 +00:00
|
|
|
"""
|
2021-03-12 09:50:08 +00:00
|
|
|
if not isPGPEncrypted(content):
|
2021-03-11 17:15:32 +00:00
|
|
|
return content
|
|
|
|
|
|
|
|
# if the public key is also included within the message then import it
|
2021-03-12 09:50:08 +00:00
|
|
|
if containsPGPPublicKey(content):
|
2021-03-11 17:15:32 +00:00
|
|
|
pubKey = extractPGPPublicKey(content)
|
2021-03-11 19:13:41 +00:00
|
|
|
else:
|
|
|
|
pubKey = _getPGPPublicKeyFromActor(content, fromHandle)
|
|
|
|
if pubKey:
|
|
|
|
_pgpImportPubKey(pubKey)
|
2021-03-11 17:15:32 +00:00
|
|
|
|
|
|
|
cmdDecrypt = \
|
|
|
|
'echo "' + content + '" | gpg --decrypt --armor 2> /dev/null'
|
|
|
|
proc = subprocess.Popen([cmdDecrypt],
|
|
|
|
stdout=subprocess.PIPE, shell=True)
|
|
|
|
(decryptResult, err) = proc.communicate()
|
|
|
|
if not decryptResult:
|
|
|
|
return content
|
2021-03-11 20:33:45 +00:00
|
|
|
decryptResult = decryptResult.decode('utf-8').strip()
|
2021-03-11 17:15:32 +00:00
|
|
|
return decryptResult
|