Skip to content

Commit

Permalink
much simpler keys
Browse files Browse the repository at this point in the history
Signed-off-by: Trishank Karthik Kuppusamy <[email protected]>
  • Loading branch information
trishankatdatadog committed Jul 1, 2020
1 parent 11d76e7 commit f2861bf
Show file tree
Hide file tree
Showing 3 changed files with 110 additions and 188 deletions.
20 changes: 10 additions & 10 deletions tests/proxy_server.py
Original file line number Diff line number Diff line change
Expand Up @@ -374,22 +374,22 @@ def parse_qsl(s):
req_header_text = "%s %s %s\n%s" % (req.command, req.path, req.request_version, req.headers)
res_header_text = "%s %d %s\n%s" % (res.response_version, res.status, res.reason, res.headers)

print with_color(33, req_header_text)
print(with_color(33, req_header_text))

u = urlparse.urlsplit(req.path)
if u.query:
query_text = parse_qsl(u.query)
print with_color(32, "==== QUERY PARAMETERS ====\n%s\n" % query_text)
print(with_color(32, "==== QUERY PARAMETERS ====\n%s\n" % query_text))

cookie = req.headers.get('Cookie', '')
if cookie:
cookie = parse_qsl(re.sub(r';\s*', '&', cookie))
print with_color(32, "==== COOKIE ====\n%s\n" % cookie)
print(with_color(32, "==== COOKIE ====\n%s\n" % cookie))

auth = req.headers.get('Authorization', '')
if auth.lower().startswith('basic'):
token = auth.split()[1].decode('base64')
print with_color(31, "==== BASIC AUTH ====\n%s\n" % token)
print(with_color(31, "==== BASIC AUTH ====\n%s\n" % token))

if req_body is not None:
req_body_text = None
Expand All @@ -412,14 +412,14 @@ def parse_qsl(s):
req_body_text = req_body

if req_body_text:
print with_color(32, "==== REQUEST BODY ====\n%s\n" % req_body_text)
print(with_color(32, "==== REQUEST BODY ====\n%s\n" % req_body_text))

print with_color(36, res_header_text)
print(with_color(36, res_header_text))

cookies = res.headers.getheaders('Set-Cookie')
if cookies:
cookies = '\n'.join(cookies)
print with_color(31, "==== SET-COOKIE ====\n%s\n" % cookies)
print(with_color(31, "==== SET-COOKIE ====\n%s\n" % cookies))

if res_body is not None:
res_body_text = None
Expand All @@ -440,12 +440,12 @@ def parse_qsl(s):
m = re.search(r'<title[^>]*>\s*([^<]+?)\s*</title>', res_body, re.I)
if m:
h = HTMLParser()
print with_color(32, "==== HTML TITLE ====\n%s\n" % h.unescape(m.group(1).decode('utf-8')))
print(with_color(32, "==== HTML TITLE ====\n%s\n" % h.unescape(m.group(1).decode('utf-8'))))
elif content_type.startswith('text/') and len(res_body) < 1024:
res_body_text = res_body

if res_body_text:
print with_color(32, "==== RESPONSE BODY ====\n%s\n" % res_body_text)
print(with_color(32, "==== RESPONSE BODY ====\n%s\n" % res_body_text))

def request_handler(self, req, req_body):
pass
Expand Down Expand Up @@ -492,7 +492,7 @@ def test(HandlerClass=ProxyRequestHandler, ServerClass=ThreadingHTTPServer, prot
httpd = ServerClass(server_address, HandlerClass)

sa = httpd.socket.getsockname()
print "Serving HTTP Proxy on", sa[0], "port", sa[1], "..."
print("Serving HTTP Proxy on", sa[0], "port", sa[1], "...")
httpd.serve_forever()


Expand Down
212 changes: 68 additions & 144 deletions tuf/api/keys.py
Original file line number Diff line number Diff line change
@@ -1,169 +1,93 @@
# Borrowed from https://github.com/cnabio/signy/blob/afba301697df456b363790dc16483408b626a8af/scripts/in-toto/keys.py
# TODO:
# * Make a storage/provider-agnostic (e.g., filesystem, HSM) key management API, like securesystemslib.storage.

# Imports.

# 1st-party.
import os
import shutil

# 2nd-party.
from typing import Any, Dict, List, Optional

from abc import ABC, abstractmethod
from enum import Enum, unique
from typing import Any, List, Optional

import logging
import os

# 3rd-party.
from securesystemslib.interface import (
generate_and_write_ed25519_keypair,
get_password,
import_ecdsa_privatekey_from_file,
import_ed25519_privatekey_from_file,
import_ed25519_publickey_from_file,
import_rsa_privatekey_from_file,
)
from securesystemslib.keys import (
create_signature,
verify_signature,
)

# Utility classes.
# Generic classes.

class Threshold:
@unique
class Algorithm(Enum):
ECDSA = import_ecdsa_privatekey_from_file
ED25519 = import_ed25519_privatekey_from_file
RSA = import_rsa_privatekey_from_file

def __init__(self, m: int = 1, n: int = 1):
assert m > 0, f'{m} <= 0'
assert n > 0, f'{n} <= 0'
assert m <= n, f'{m} > {n}'
self.m = m
self.n = n
class Threshold:

class Keypath:
def __init__(self, min_: int = 1, max_: int = 1):
assert min_ > 0, f'{min_} <= 0'
assert max_ > 0, f'{max_} <= 0'
assert min_ <= max_, f'{min_} > {max_}'
self.min = min_
self.max = max_

def __init__(self, private: str, public: str):
assert os.path.isfile(private), private
assert os.path.isfile(public), public
self.private = private
self.public = public
class Key(ABC):

class Key:
@abstractmethod
def __init__(self) -> None:
raise NotImplementedError()

def __init__(self, path: str, obj: Any):
self.path = path
self.obj = obj
@property
@abstractmethod
def keyid(self) -> str:
raise NotImplementedError()

class Keypair:
@abstractmethod
def sign(self, signed: str) -> str:
raise NotImplementedError()

def __init__(self, private: Key, public: Key):
self.private = private
self.public = public
@abstractmethod
def verify(self, signed: str, signature: str) -> bool:
raise NotImplementedError()

Keypairs = List[Keypair]
Keys = List[Key]

class Keyring:
class KeyRing:

def __init__(self, threshold: Threshold, keypairs: Keypairs):
if len(keypairs) >= threshold.m:
logging.warning(f'{len(keypairs)} >= {threshold.m}')
if len(keypairs) <= threshold.n:
logging.warning(f'{len(keypairs)} <= {threshold.n}')
def __init__(self, threshold: Threshold, keys: Keys):
if len(keys) >= threshold.min:
logging.warning(f'{len(keys)} >= {threshold.min}')
if len(keys) <= threshold.max:
logging.warning(f'{len(keys)} <= {threshold.max}')
self.threshold = threshold
self.keypairs = keypairs
self.keys = keys

# Specific types of keys, such as those in RAM, or on HSMs (TODO).

class RAMKey(Key):

def __init__(self, obj: Any) -> None: # pylint: disable=super-init-not-called
self.__obj = obj

def keyid(self) -> str:
return self.__obj['keyid']

def sign(self, signed: str) -> str:
return create_signature(self.__obj, signed)

def verify(self, signed: str, signature: str) -> bool:
return verify_signature(self.__obj, signature, signed)

# Useful for securesytemslib.
KeyDict = Dict[str, Any]

# Utility functions.

def get_new_private_keypath(keystore_dir: str, rolename: str, i : int = 1) -> str:
return os.path.join(keystore_dir, f'{rolename}_ed25519_key_{i}')

def get_public_keypath(private_keypath: str) -> str:
# this is the tuf filename convention at the time of writing.
return f'{private_keypath}.pub'

def get_private_keys_from_keyring(keyring: Keyring) -> KeyDict:
privkeys = {}

for keypair in keyring.keypairs:
privkey = keypair.private.obj
keyid = privkey['keyid']
assert keyid not in privkeys
privkeys[keyid] = privkey

return privkeys

def get_public_keys_from_keyring(keyring: Keyring) -> KeyDict:
pubkeys = {}

for keypair in keyring.keypairs:
pubkey = keypair.public.obj
keyid = pubkey['keyid']
assert keyid not in pubkeys
pubkeys[keyid] = pubkey

return pubkeys

def write_keypair(keystore_dir: str, rolename: str, i: int = 1, n: int = 1, passphrase: Optional[str] = None) -> Keypath:
private_keypath = get_new_private_keypath(keystore_dir, rolename, i)
assert not os.path.isfile(private_keypath)
public_keypath = get_public_keypath(private_keypath)
assert not os.path.isfile(public_keypath)

# Make the keystore directory, WR-only by self, if not already there.
os.makedirs(keystore_dir, mode=0o700, exist_ok=True)

# FIXME: do not assume Ed25519
generate_and_write_ed25519_keypair(private_keypath, password=passphrase)

return Keypath(private_keypath, public_keypath)

def read_keypair(keypath: Keypath, passphrase: Optional[str] = None) -> Keypair:
private_keypath = keypath.private
private_key_obj = import_ed25519_privatekey_from_file(keypath.private, password=passphrase)
private_key = Key(private_keypath, private_key_obj)

# and its corresponding public key.
public_keypath = keypath.public
public_key_obj = import_ed25519_publickey_from_file(keypath.public)
public_key = Key(public_keypath, public_key_obj)

return Keypair(private_key, public_key)

def rename_keys_to_match_keyid(keystore_dir: str, keypair: Keypair) -> None:
'''
<Purpose>
Rename public / private keys to match their keyid, so that it is easy
to later find public keys on the repository, or private keys on disk.
Also see https://github.com/theupdateframework/tuf/issues/573
'''

keyid = keypair.public.obj['keyid']

# Rename the private key filename to match the keyid.
assert os.path.exists(keystore_dir), keystore_dir
new_private_keypath = os.path.join(keystore_dir, keyid)
# Move the key to the new filename.
assert not os.path.isfile(new_private_keypath), new_private_keypath
shutil.move(keypair.private.path, new_private_keypath)
# Update the path to the key.
keypair.private.path = new_private_keypath

# Rename the public key filename to match the keyid.
new_public_keypath = get_public_keypath(new_private_keypath)
# Move the key to the new filename.
assert not os.path.isfile(new_public_keypath), new_public_keypath
shutil.move(keypair.public.path, new_public_keypath)
# Update the path to the key.
keypair.public.path = new_public_keypath

def write_and_read_new_keys(keystore_dir: str, rolename: str, threshold: Threshold) -> Keyring:
keypairs = []

for i in range(1, threshold.n + 1):
print(f'Writing key {i}/{threshold.n} for the "{rolename}" rolename...')
passphrase = get_password(
prompt='Please enter a NON-EMPTY passphrase to ENCRYPT this key: ',
confirm=True
)
keypath = write_keypair(keystore_dir, rolename, i, threshold.n, passphrase)
keypair = read_keypair(keypath, passphrase)
# Rename the private and public keys to match the keyid instead.
# Why? So that we know how to find keys later on repository / disk.
rename_keys_to_match_keyid(keystore_dir, keypair)
keypairs.append(keypair)
print()

return Keyring(threshold, tuple(keypairs))
def read_key(filename: str, algorithm: str, passphrase: Optional[str] = None) -> Key:
handler = Algorithm[algorithm]
obj = handler(filename, password=passphrase)
return RAMKey(obj)
Loading

0 comments on commit f2861bf

Please sign in to comment.