Ida/pbkdf2 #2
| @ -24,7 +24,8 @@ algo_keywords = [ | |||||||
|     'aes-128-ctr', |     'aes-128-ctr', | ||||||
| ] | ] | ||||||
| hash_keywords = [ | hash_keywords = [ | ||||||
|     'scrypt' |     'scrypt', | ||||||
|  |     'pbkdf2' | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| default_kdfparams = { | default_kdfparams = { | ||||||
| 
					
					idaapayo marked this conversation as resolved
					
						
						
							Outdated
						
					
				 | |||||||
| @ -35,6 +36,12 @@ default_kdfparams = { | |||||||
|     'salt': os.urandom(32).hex(), |     'salt': os.urandom(32).hex(), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | pbkdf2_kdfparams = { | ||||||
| 
				
					
						kamikazechaser
						commented  Chnage this to  Chnage this to `default_pbkdf2_kdfparams` | |||||||
|  |     'c': 100000, | ||||||
|  |     'dklen': 32, | ||||||
|  |     'prf': 'sha256', | ||||||
|  |     'salt': os.urandom(32).hex(), | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| def to_mac(mac_key, ciphertext_bytes): | def to_mac(mac_key, ciphertext_bytes): | ||||||
|     h = sha3.keccak_256() |     h = sha3.keccak_256() | ||||||
| @ -53,11 +60,29 @@ class Hashes: | |||||||
|         r = int(kdfparams['r']) |         r = int(kdfparams['r']) | ||||||
|         salt = bytes.fromhex(kdfparams['salt']) |         salt = bytes.fromhex(kdfparams['salt']) | ||||||
| 
 | 
 | ||||||
|         return hashlib.scrypt(passphrase.encode('utf-8'), salt=salt,n=n, p=p, r=r, maxmem=1024*1024*1024, dklen=dklen) |         return hashlib.scrypt(passphrase.encode('utf-8'), salt=salt, n=n, p=p, r=r, maxmem=1024 * 1024 * 1024, | ||||||
|  |                               dklen=dklen) | ||||||
|  | 
 | ||||||
|  |     @staticmethod | ||||||
|  |     def from_pbkdf2(kdfparams=pbkdf2_kdfparams, passphrase=''): | ||||||
|  |         hashname = kdfparams['prf'] | ||||||
| 
					
					kamikazechaser marked this conversation as resolved
					
						
						
							Outdated
						
					
				 
				
					
						kamikazechaser
						commented  
 `prf` key-value from keystore files could also be `hmac-sha256`, which isn't a valid param for `hashlib.pbkdf2_hmac` default to `sha256` instead. | |||||||
|  |         pwd = passphrase.encode('utf-8') | ||||||
|  |         salt = bytes.fromhex(kdfparams['salt']) | ||||||
|  |         itr = int(kdfparams['c']) | ||||||
|  |         dklen = int(kdfparams['dklen']) | ||||||
| 
					
					kamikazechaser marked this conversation as resolved
					
						
						
							Outdated
						
					
				 
				
					
						kamikazechaser
						commented  Change this to  Change this to `hash_name="sha256"` | |||||||
|  | 
 | ||||||
|  |         derived_key = hashlib.pbkdf2_hmac( | ||||||
| 
				
					
						kamikazechaser
						commented  We can pass the params directly and return on on it to reduce verbosity. We can pass the params directly and return on on it to reduce verbosity. | |||||||
|  |             hash_name=hashname, | ||||||
|  |             password=pwd, | ||||||
|  |             salt=salt, | ||||||
|  |             iterations=itr, | ||||||
|  |             dklen=dklen | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         return derived_key | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Ciphers: | class Ciphers: | ||||||
| 
 |  | ||||||
|     aes_128_block_size = 1 << 7 |     aes_128_block_size = 1 << 7 | ||||||
|     aes_iv_len = 16 |     aes_iv_len = 16 | ||||||
| 
 | 
 | ||||||
| @ -68,7 +93,6 @@ class Ciphers: | |||||||
|         plaintext = cipher.decrypt(ciphertext) |         plaintext = cipher.decrypt(ciphertext) | ||||||
|         return plaintext |         return plaintext | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def encrypt_aes_128_ctr(plaintext, encryption_key, iv): |     def encrypt_aes_128_ctr(plaintext, encryption_key, iv): | ||||||
|         ctr = Counter.new(Ciphers.aes_128_block_size, initial_value=iv) |         ctr = Counter.new(Ciphers.aes_128_block_size, initial_value=iv) | ||||||
| @ -77,11 +101,19 @@ class Ciphers: | |||||||
|         return ciphertext |         return ciphertext | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def to_dict(private_key_bytes, passphrase=''): | def to_dict(private_key_bytes, kdf :str, passphrase=''): | ||||||
| 
				
					
						kamikazechaser
						commented  This is a potentially breaking change if this method is being used elsewhere. I suggest we default to  This is a potentially breaking change if this method is being used elsewhere. I suggest we default to `scrypt` for now. | |||||||
|     |  | ||||||
|     private_key = coincurve.PrivateKey(secret=private_key_bytes) |     private_key = coincurve.PrivateKey(secret=private_key_bytes) | ||||||
| 
 | 
 | ||||||
|  |     if kdf == 'scrypt': | ||||||
|         encryption_key = Hashes.from_scrypt(passphrase=passphrase) |         encryption_key = Hashes.from_scrypt(passphrase=passphrase) | ||||||
|  |         kdfparams = default_kdfparams | ||||||
|  | 
 | ||||||
|  |     elif kdf == 'pbkdf2': | ||||||
|  |         encryption_key = Hashes.from_pbkdf2(passphrase=passphrase) | ||||||
|  |         kdfparams = pbkdf2_kdfparams | ||||||
|  | 
 | ||||||
|  |     else: | ||||||
|  |         raise NotImplementedError("KDF not implemented: {0}".format(kdf)) | ||||||
| 
 | 
 | ||||||
|     address_hex = private_key_to_address(private_key) |     address_hex = private_key_to_address(private_key) | ||||||
|     iv_bytes = os.urandom(Ciphers.aes_iv_len) |     iv_bytes = os.urandom(Ciphers.aes_iv_len) | ||||||
| @ -96,8 +128,8 @@ def to_dict(private_key_bytes, passphrase=''): | |||||||
|         'cipherparams': { |         'cipherparams': { | ||||||
|             'iv': iv_bytes.hex(), |             'iv': iv_bytes.hex(), | ||||||
|         }, |         }, | ||||||
|         'kdf': 'scrypt', |         'kdf': kdf, | ||||||
|         'kdfparams': default_kdfparams, |         'kdfparams': kdfparams, | ||||||
|         'mac': mac.hex(), |         'mac': mac.hex(), | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -112,7 +144,6 @@ def to_dict(private_key_bytes, passphrase=''): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def from_dict(o, passphrase=''): | def from_dict(o, passphrase=''): | ||||||
| 
 |  | ||||||
|     cipher = o['crypto']['cipher'] |     cipher = o['crypto']['cipher'] | ||||||
|     if cipher not in algo_keywords: |     if cipher not in algo_keywords: | ||||||
|         raise NotImplementedError('cipher "{}" not implemented'.format(cipher)) |         raise NotImplementedError('cipher "{}" not implemented'.format(cipher)) | ||||||
| @ -145,7 +176,6 @@ def from_dict(o, passphrase=''): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def from_file(filepath, passphrase=''): | def from_file(filepath, passphrase=''): | ||||||
| 
 |  | ||||||
|     f = open(filepath, 'r') |     f = open(filepath, 'r') | ||||||
|     try: |     try: | ||||||
|         o = json.load(f) |         o = json.load(f) | ||||||
|  | |||||||
| @ -16,6 +16,10 @@ from funga.eth.keystore.keyfile import ( | |||||||
|         from_file, |         from_file, | ||||||
|         to_dict, |         to_dict, | ||||||
|         ) |         ) | ||||||
|  | # from testkeyfile import ( | ||||||
| 
					
					kamikazechaser marked this conversation as resolved
					
						
						
							Outdated
						
					
				 
				
					
						kamikazechaser
						commented  Remove these comments Remove these comments | |||||||
|  | #     from_file, | ||||||
|  | #     to_dict | ||||||
|  | # ) | ||||||
| from funga.eth.encoding import ( | from funga.eth.encoding import ( | ||||||
|         private_key_to_address, |         private_key_to_address, | ||||||
|         private_key_from_bytes, |         private_key_from_bytes, | ||||||
| @ -77,7 +81,7 @@ def main(): | |||||||
|         else: |         else: | ||||||
|             pk_bytes = os.urandom(32) |             pk_bytes = os.urandom(32) | ||||||
| 
				
					
						lash
						commented  ... and then you don't have to change this. ... and then you don't have to change this. | |||||||
|         pk = coincurve.PrivateKey(secret=pk_bytes) |         pk = coincurve.PrivateKey(secret=pk_bytes) | ||||||
|         o = to_dict(pk_bytes, passphrase) |         o = to_dict(pk_bytes, 'pbkdf2', passphrase) | ||||||
|         r = json.dumps(o) |         r = json.dumps(o) | ||||||
| 
 | 
 | ||||||
|     print(r)  |     print(r)  | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	
This should be renamed to
default_scrypt_paramsThoughts? @lash Not sure which naming convention you prefer