fix: OpenSSH private-key fingerprint fallback + clean error on missing password
get_rsa_key_fingerprint(is_private=True) only loaded PEM private keys, so an OpenSSH-format private key raised — unlike decrypt_aes_key_with_rsa, which already had the fallback. mirrored it: on a PEM load failure, an OPENSSH-marked key is loaded via load_ssh_private_key. also normalized the encrypted-key-without-password case: cryptography raises TypeError there, which now becomes a clear ValueError('private key is encrypted but no password was provided') in both methods instead of leaking the raw TypeError.
Signed-off-by: disqualifier <dev@disqualifier.me>
This commit is contained in:
parent
16205e810a
commit
5de8b5d736
@ -11,13 +11,13 @@ and storage-agnostic.
|
|||||||
`requirements.txt`:
|
`requirements.txt`:
|
||||||
|
|
||||||
```
|
```
|
||||||
envelope_crypto @ git+ssh://git@git.rethinkstudios.io/rethink-public/envelope_crypto.git@v0.1.1
|
envelope_crypto @ git+ssh://git@git.rethinkstudios.io/rethink-public/envelope_crypto.git@v0.1.2
|
||||||
```
|
```
|
||||||
|
|
||||||
Direct:
|
Direct:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
pip install "envelope_crypto @ git+ssh://git@git.rethinkstudios.io/rethink-public/envelope_crypto.git@v0.1.1"
|
pip install "envelope_crypto @ git+ssh://git@git.rethinkstudios.io/rethink-public/envelope_crypto.git@v0.1.2"
|
||||||
```
|
```
|
||||||
|
|
||||||
Requires `cryptography` (pulled transitively).
|
Requires `cryptography` (pulled transitively).
|
||||||
|
|||||||
@ -4,7 +4,7 @@ build-backend = "hatchling.build"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "envelope_crypto"
|
name = "envelope_crypto"
|
||||||
version = "0.1.1"
|
version = "0.1.2"
|
||||||
description = "Envelope encryption (RSA-OAEP wrapped AES-256-GCM) for dict records — config-free, storage-agnostic, installable."
|
description = "Envelope encryption (RSA-OAEP wrapped AES-256-GCM) for dict records — config-free, storage-agnostic, installable."
|
||||||
requires-python = ">=3.10"
|
requires-python = ">=3.10"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
|||||||
@ -155,7 +155,10 @@ class EnvelopeCrypto:
|
|||||||
|
|
||||||
for an encrypted private key (is_private=True), pass its `password`; an
|
for an encrypted private key (is_private=True), pass its `password`; an
|
||||||
unencrypted key ignores it. fingerprinting always uses the public half, so a
|
unencrypted key ignores it. fingerprinting always uses the public half, so a
|
||||||
private and its public key produce the same fingerprint.
|
private and its public key produce the same fingerprint. PEM and OpenSSH
|
||||||
|
private-key formats are both accepted (mirrors decrypt_aes_key_with_rsa). an
|
||||||
|
encrypted key with no/wrong password raises ValueError with a clear message
|
||||||
|
(cryptography raises TypeError for the missing-password case — normalized here).
|
||||||
"""
|
"""
|
||||||
if is_file:
|
if is_file:
|
||||||
with open(key_path_or_data, "rb") as key_file:
|
with open(key_path_or_data, "rb") as key_file:
|
||||||
@ -168,9 +171,18 @@ class EnvelopeCrypto:
|
|||||||
)
|
)
|
||||||
|
|
||||||
if is_private:
|
if is_private:
|
||||||
private_key = serialization.load_pem_private_key(
|
pw = password.encode() if password else None
|
||||||
key_data, password=password.encode() if password else None
|
try:
|
||||||
)
|
private_key = serialization.load_pem_private_key(key_data, password=pw)
|
||||||
|
except ValueError as error:
|
||||||
|
if b"BEGIN OPENSSH PRIVATE KEY" in key_data:
|
||||||
|
private_key = serialization.load_ssh_private_key(key_data, password=pw)
|
||||||
|
else:
|
||||||
|
raise error
|
||||||
|
except TypeError as error:
|
||||||
|
raise ValueError(
|
||||||
|
"private key is encrypted but no password was provided"
|
||||||
|
) from error
|
||||||
public_key = private_key.public_key()
|
public_key = private_key.public_key()
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
@ -223,17 +235,18 @@ class EnvelopeCrypto:
|
|||||||
"""unwrap an AES key with an RSA private key"""
|
"""unwrap an AES key with an RSA private key"""
|
||||||
with open(rsa_private_key_path, "rb") as key_file:
|
with open(rsa_private_key_path, "rb") as key_file:
|
||||||
key_data = key_file.read()
|
key_data = key_file.read()
|
||||||
|
pw = password.encode() if password else None
|
||||||
try:
|
try:
|
||||||
private_key = serialization.load_pem_private_key(
|
private_key = serialization.load_pem_private_key(key_data, password=pw)
|
||||||
key_data, password=password.encode() if password else None
|
|
||||||
)
|
|
||||||
except ValueError as error:
|
except ValueError as error:
|
||||||
if b"BEGIN OPENSSH PRIVATE KEY" in key_data:
|
if b"BEGIN OPENSSH PRIVATE KEY" in key_data:
|
||||||
private_key = serialization.load_ssh_private_key(
|
private_key = serialization.load_ssh_private_key(key_data, password=pw)
|
||||||
key_data, password=password.encode() if password else None
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
raise error
|
raise error
|
||||||
|
except TypeError as error:
|
||||||
|
raise ValueError(
|
||||||
|
"private key is encrypted but no password was provided"
|
||||||
|
) from error
|
||||||
|
|
||||||
wrapped = base64.b64decode(encrypted_key_base64)
|
wrapped = base64.b64decode(encrypted_key_base64)
|
||||||
aes_key = private_key.decrypt(
|
aes_key = private_key.decrypt(
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user