2019-06-28 18:55:29 +00:00
|
|
|
__filename__ = "posts.py"
|
|
|
|
__author__ = "Bob Mottram"
|
|
|
|
__credits__ = ['lamia']
|
|
|
|
__license__ = "AGPL3+"
|
|
|
|
__version__ = "0.0.1"
|
|
|
|
__maintainer__ = "Bob Mottram"
|
|
|
|
__email__ = "bob@freedombone.net"
|
|
|
|
__status__ = "Production"
|
|
|
|
|
2019-08-15 22:33:42 +00:00
|
|
|
# see https://tools.ietf.org/html/draft-cavage-http-signatures-06
|
|
|
|
|
2019-06-28 18:55:29 +00:00
|
|
|
from Crypto.PublicKey import RSA
|
|
|
|
from Crypto.Hash import SHA256
|
|
|
|
#from Crypto.Signature import PKCS1_v1_5
|
|
|
|
from Crypto.Signature import pkcs1_15
|
|
|
|
from requests.auth import AuthBase
|
|
|
|
import base64
|
|
|
|
import json
|
2019-08-15 09:08:18 +00:00
|
|
|
from time import gmtime, strftime
|
2019-06-28 18:55:29 +00:00
|
|
|
|
2019-07-03 09:40:27 +00:00
|
|
|
def signPostHeaders(privateKeyPem: str, nickname: str, domain: str, \
|
2019-07-02 20:54:22 +00:00
|
|
|
port: int,path: str, \
|
2019-07-03 19:00:03 +00:00
|
|
|
httpPrefix: str, messageBodyJson: {}) -> str:
|
2019-06-28 18:55:29 +00:00
|
|
|
"""Returns a raw signature string that can be plugged into a header and
|
|
|
|
used to verify the authenticity of an HTTP transmission.
|
|
|
|
"""
|
2019-07-01 09:31:02 +00:00
|
|
|
if port!=80 and port!=443:
|
|
|
|
domain=domain+':'+str(port)
|
|
|
|
|
2019-08-15 18:21:43 +00:00
|
|
|
dateStr=strftime("%a, %d %b %Y %H:%M:%S %Z", gmtime())
|
2019-07-04 14:36:29 +00:00
|
|
|
keyID = httpPrefix+'://'+domain+'/users/'+nickname+'#main-key'
|
2019-06-28 18:55:29 +00:00
|
|
|
if not messageBodyJson:
|
2019-08-15 21:34:25 +00:00
|
|
|
headers = {'(request-target)': f'post {path}','host': domain,'date': dateStr,'content-type': 'application/json'}
|
2019-06-28 18:55:29 +00:00
|
|
|
else:
|
2019-08-16 10:36:41 +00:00
|
|
|
messageBodyJsonStr=json.dumps(messageBodyJson)
|
2019-07-02 20:54:22 +00:00
|
|
|
bodyDigest = \
|
2019-08-16 10:36:41 +00:00
|
|
|
base64.b64encode(SHA256.new(messageBodyJsonStr.encode()).digest()).decode('utf-8')
|
2019-08-16 09:12:47 +00:00
|
|
|
headers = {'(request-target)': f'post {path}','host': domain,'date': dateStr,'digest': f'SHA-256={bodyDigest}','content-type': 'application/activity+json'}
|
2019-06-28 18:55:29 +00:00
|
|
|
privateKeyPem = RSA.import_key(privateKeyPem)
|
2019-08-15 21:34:25 +00:00
|
|
|
#headers.update({
|
|
|
|
# '(request-target)': f'post {path}',
|
|
|
|
#})
|
2019-06-28 18:55:29 +00:00
|
|
|
# build a digest for signing
|
|
|
|
signedHeaderKeys = headers.keys()
|
|
|
|
signedHeaderText = ''
|
|
|
|
for headerKey in signedHeaderKeys:
|
|
|
|
signedHeaderText += f'{headerKey}: {headers[headerKey]}\n'
|
2019-08-15 21:34:25 +00:00
|
|
|
#print(f'headerKey: {headerKey}: {headers[headerKey]}')
|
2019-06-28 18:55:29 +00:00
|
|
|
signedHeaderText = signedHeaderText.strip()
|
|
|
|
headerDigest = SHA256.new(signedHeaderText.encode('ascii'))
|
|
|
|
|
|
|
|
# Sign the digest
|
|
|
|
rawSignature = pkcs1_15.new(privateKeyPem).sign(headerDigest)
|
|
|
|
signature = base64.b64encode(rawSignature).decode('ascii')
|
|
|
|
|
|
|
|
# Put it into a valid HTTP signature format
|
|
|
|
signatureDict = {
|
|
|
|
'keyId': keyID,
|
|
|
|
'algorithm': 'rsa-sha256',
|
|
|
|
'headers': ' '.join(signedHeaderKeys),
|
|
|
|
'signature': signature
|
|
|
|
}
|
|
|
|
signatureHeader = ','.join(
|
|
|
|
[f'{k}="{v}"' for k, v in signatureDict.items()])
|
|
|
|
return signatureHeader
|
|
|
|
|
2019-07-03 09:40:27 +00:00
|
|
|
def createSignedHeader(privateKeyPem: str,nickname: str,domain: str,port: int, \
|
2019-07-03 19:00:03 +00:00
|
|
|
path: str,httpPrefix: str,withDigest: bool, \
|
2019-07-02 20:54:22 +00:00
|
|
|
messageBodyJson: {}) -> {}:
|
2019-07-01 09:31:02 +00:00
|
|
|
headerDomain=domain
|
|
|
|
|
2019-08-15 21:34:25 +00:00
|
|
|
if port:
|
|
|
|
if port!=80 and port!=443:
|
|
|
|
headerDomain=headerDomain+':'+str(port)
|
2019-07-01 09:31:02 +00:00
|
|
|
|
2019-08-15 18:21:43 +00:00
|
|
|
dateStr=strftime("%a, %d %b %Y %H:%M:%S %Z", gmtime())
|
2019-08-15 21:34:25 +00:00
|
|
|
path='/inbox'
|
2019-08-16 10:36:41 +00:00
|
|
|
print('Testing 123 '+str(withDigest))
|
2019-07-01 09:31:02 +00:00
|
|
|
if not withDigest:
|
2019-08-15 21:34:25 +00:00
|
|
|
headers = {'(request-target)': f'post {path}','host': headerDomain,'date': dateStr}
|
2019-08-16 10:36:41 +00:00
|
|
|
signatureHeader = \
|
|
|
|
signPostHeaders(privateKeyPem, nickname, domain, port, \
|
|
|
|
path, httpPrefix, None)
|
2019-07-01 09:31:02 +00:00
|
|
|
else:
|
|
|
|
messageBodyJsonStr=json.dumps(messageBodyJson)
|
2019-07-02 20:54:22 +00:00
|
|
|
bodyDigest = \
|
2019-08-16 09:56:58 +00:00
|
|
|
base64.b64encode(SHA256.new(messageBodyJsonStr.encode()).digest()).decode('utf-8')
|
2019-08-16 09:12:47 +00:00
|
|
|
headers = {'(request-target)': f'post {path}','host': headerDomain,'date': dateStr,'digest': f'SHA-256={bodyDigest}','content-type': 'application/activity+json'}
|
2019-08-16 10:36:41 +00:00
|
|
|
signatureHeader = \
|
|
|
|
signPostHeaders(privateKeyPem, nickname, domain, port, \
|
|
|
|
path, httpPrefix, messageBodyJson)
|
2019-07-01 09:31:02 +00:00
|
|
|
headers['signature'] = signatureHeader
|
|
|
|
return headers
|
|
|
|
|
2019-08-15 21:34:25 +00:00
|
|
|
def verifyPostHeaders(httpPrefix: str,publicKeyPem: str,headers: dict, \
|
|
|
|
path: str,GETmethod: bool, \
|
2019-07-02 20:54:22 +00:00
|
|
|
messageBodyJsonStr: str) -> bool:
|
2019-06-28 18:55:29 +00:00
|
|
|
"""Returns true or false depending on if the key that we plugged in here
|
|
|
|
validates against the headers, method, and path.
|
|
|
|
publicKeyPem - the public key from an rsa key pair
|
|
|
|
headers - should be a dictionary of request headers
|
|
|
|
path - the relative url that was requested from this site
|
|
|
|
GETmethod - GET or POST
|
2019-07-01 09:31:02 +00:00
|
|
|
messageBodyJsonStr - the received request body (used for digest)
|
2019-06-28 18:55:29 +00:00
|
|
|
"""
|
|
|
|
if GETmethod:
|
|
|
|
method='GET'
|
|
|
|
else:
|
|
|
|
method='POST'
|
|
|
|
|
|
|
|
publicKeyPem = RSA.import_key(publicKeyPem)
|
|
|
|
# Build a dictionary of the signature values
|
|
|
|
signatureHeader = headers['signature']
|
|
|
|
signatureDict = {
|
|
|
|
k: v[1:-1]
|
|
|
|
for k, v in [i.split('=', 1) for i in signatureHeader.split(',')]
|
|
|
|
}
|
2019-08-15 21:34:25 +00:00
|
|
|
#print('signatureHeader: '+str(signatureHeader))
|
|
|
|
#print('signatureDict: '+str(signatureDict))
|
2019-06-28 18:55:29 +00:00
|
|
|
|
|
|
|
# Unpack the signed headers and set values based on current headers and
|
|
|
|
# body (if a digest was included)
|
|
|
|
signedHeaderList = []
|
|
|
|
for signedHeader in signatureDict['headers'].split(' '):
|
|
|
|
if signedHeader == '(request-target)':
|
|
|
|
signedHeaderList.append(
|
|
|
|
f'(request-target): {method.lower()} {path}')
|
|
|
|
elif signedHeader == 'digest':
|
2019-07-02 20:54:22 +00:00
|
|
|
bodyDigest = \
|
2019-08-16 10:36:41 +00:00
|
|
|
base64.b64encode(SHA256.new(messageBodyJsonStr.encode()).digest()).decode('utf-8')
|
2019-06-28 18:55:29 +00:00
|
|
|
signedHeaderList.append(f'digest: SHA-256={bodyDigest}')
|
|
|
|
else:
|
2019-08-15 21:34:25 +00:00
|
|
|
if headers.get(signedHeader):
|
2019-08-15 17:09:17 +00:00
|
|
|
signedHeaderList.append(
|
|
|
|
f'{signedHeader}: {headers[signedHeader]}')
|
2019-08-15 21:34:25 +00:00
|
|
|
else:
|
|
|
|
signedHeaderCap=signedHeader.capitalize()
|
|
|
|
if headers.get(signedHeaderCap):
|
|
|
|
signedHeaderList.append(
|
|
|
|
f'{signedHeader}: {headers[signedHeaderCap]}')
|
2019-06-28 18:55:29 +00:00
|
|
|
|
2019-08-16 10:36:41 +00:00
|
|
|
#print('signedHeaderList: '+str(signedHeaderList))
|
2019-06-28 18:55:29 +00:00
|
|
|
# Now we have our header data digest
|
|
|
|
signedHeaderText = '\n'.join(signedHeaderList)
|
|
|
|
headerDigest = SHA256.new(signedHeaderText.encode('ascii'))
|
|
|
|
|
|
|
|
# Get the signature, verify with public key, return result
|
|
|
|
signature = base64.b64decode(signatureDict['signature'])
|
|
|
|
|
|
|
|
try:
|
|
|
|
pkcs1_15.new(publicKeyPem).verify(headerDigest, signature)
|
|
|
|
return True
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
return False
|