communex.compat.storage
Data storage compatible with the classic commune
library.
1""" 2Data storage compatible with the *classic* `commune` library. 3""" 4 5# TODO: encryption 6 7import base64 8import hashlib 9import json 10import os.path 11import time 12from typing import Any 13 14from nacl.secret import SecretBox 15from nacl.utils import random 16 17from communex.errors import PasswordNotProvidedError 18from communex.util import ensure_parent_dir_exists 19 20# from cryptography.fernet import Fernet 21# from cryptography.hazmat.primitives import hashes 22# from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC 23 24 25COMMUNE_HOME = "~/.commune" 26""" 27COMMUNE_HOME 28 29 classic commune data storage home directory. 30""" 31 32 33def _derive_key(password: str): 34 # Derive a 256-bit key from the password using Blake2b 35 key = hashlib.blake2b(password.encode(), digest_size=32).digest() 36 return key 37 38 39def _encrypt_data(password: str, data: Any) -> str: 40 key = _derive_key(password) 41 box = SecretBox(key) 42 nonce = random(SecretBox.NONCE_SIZE) 43 raw = json.dumps(data).encode() 44 ciphertext = box.encrypt(raw, nonce).ciphertext 45 encrypted = nonce + ciphertext 46 decoded_data = base64.b64encode(encrypted).decode() 47 return decoded_data 48 49 50def _decrypt_data(password: str, data: str) -> Any: 51 key = _derive_key(password) 52 box = SecretBox(key) 53 encrypted = base64.b64decode(data.encode()) 54 nonce = encrypted[: SecretBox.NONCE_SIZE] 55 ciphertext = encrypted[SecretBox.NONCE_SIZE :] 56 raw = box.decrypt(ciphertext, nonce) 57 return json.loads(raw.decode()) 58 59 60def classic_load( 61 path: str, mode: str = "json", password: str | None = None 62) -> Any: 63 """ 64 Load data from commune data storage. 65 66 Args: 67 path: Data storage file path. 68 mode: Data storage mode. 69 70 Returns: 71 Data loaded from the data storage. 72 73 Todo: 74 * Other serialization modes support. Only json mode is supported now. 75 76 Raises: 77 NotImplementedError: See Todo. 78 AssertionError: Raised when the data is not in the classic format. 79 """ 80 if mode != "json": 81 raise NotImplementedError( 82 "Our commune data storage only supports json mode" 83 ) 84 85 full_path = os.path.expanduser(os.path.join(COMMUNE_HOME, path)) 86 with open(full_path, "r") as file: 87 body = json.load(file) 88 89 if body["encrypted"] and password is None: 90 raise PasswordNotProvidedError( 91 "Data is encrypted but no password provided" 92 ) 93 if body["encrypted"] and password is not None: 94 content = _decrypt_data(password, body["data"]) 95 else: 96 content = body["data"] 97 98 assert isinstance(body, dict) 99 assert isinstance(body["timestamp"], int) 100 assert isinstance(content, (dict, list, tuple, set, float, str, int)) 101 return content # type: ignore 102 103 104def classic_put( 105 path: str, value: Any, mode: str = "json", password: str | None = None 106): 107 """ 108 Put data into commune data storage. 109 110 Args: 111 path: Data storage path. 112 value: Data to store. 113 mode: Data storage mode. 114 encrypt: Whether to encrypt the data. 115 116 Todo: 117 * Other serialization modes support. Only json mode is supported now. 118 119 Raises: 120 NotImplementedError: See Todo. 121 TypeError: Raised when value is not a valid type. 122 FileExistsError: Raised when the file already exists. 123 """ 124 if mode != "json": 125 raise NotImplementedError( 126 "Our commune data storage only supports json mode" 127 ) 128 if not isinstance(value, (dict, list, tuple, set, float, str, int)): 129 raise TypeError( 130 f"Invalid type for commune data storage value: {type(value)}" 131 ) 132 timestamp = int(time.time()) 133 134 full_path = os.path.expanduser(os.path.join(COMMUNE_HOME, path)) 135 136 if os.path.exists(full_path): 137 raise FileExistsError( 138 f"Commune data storage file already exists: {full_path}" 139 ) 140 141 ensure_parent_dir_exists(full_path) 142 143 if password: 144 value = _encrypt_data(password, value) 145 encrypted = True 146 else: 147 encrypted = False 148 149 with open(full_path, "w") as file: 150 json.dump( 151 {"data": value, "encrypted": encrypted, "timestamp": timestamp}, 152 file, 153 indent=4, 154 )
COMMUNE_HOME =
'~/.commune'
COMMUNE_HOME
classic commune data storage home directory.
def
classic_load(path: str, mode: str = 'json', password: str | None = None) -> Any:
61def classic_load( 62 path: str, mode: str = "json", password: str | None = None 63) -> Any: 64 """ 65 Load data from commune data storage. 66 67 Args: 68 path: Data storage file path. 69 mode: Data storage mode. 70 71 Returns: 72 Data loaded from the data storage. 73 74 Todo: 75 * Other serialization modes support. Only json mode is supported now. 76 77 Raises: 78 NotImplementedError: See Todo. 79 AssertionError: Raised when the data is not in the classic format. 80 """ 81 if mode != "json": 82 raise NotImplementedError( 83 "Our commune data storage only supports json mode" 84 ) 85 86 full_path = os.path.expanduser(os.path.join(COMMUNE_HOME, path)) 87 with open(full_path, "r") as file: 88 body = json.load(file) 89 90 if body["encrypted"] and password is None: 91 raise PasswordNotProvidedError( 92 "Data is encrypted but no password provided" 93 ) 94 if body["encrypted"] and password is not None: 95 content = _decrypt_data(password, body["data"]) 96 else: 97 content = body["data"] 98 99 assert isinstance(body, dict) 100 assert isinstance(body["timestamp"], int) 101 assert isinstance(content, (dict, list, tuple, set, float, str, int)) 102 return content # type: ignore
Load data from commune data storage.
Arguments:
- path: Data storage file path.
- mode: Data storage mode.
Returns:
Data loaded from the data storage.
Todo:
- Other serialization modes support. Only json mode is supported now.
Raises:
- NotImplementedError: See Todo.
- AssertionError: Raised when the data is not in the classic format.
def
classic_put( path: str, value: Any, mode: str = 'json', password: str | None = None):
105def classic_put( 106 path: str, value: Any, mode: str = "json", password: str | None = None 107): 108 """ 109 Put data into commune data storage. 110 111 Args: 112 path: Data storage path. 113 value: Data to store. 114 mode: Data storage mode. 115 encrypt: Whether to encrypt the data. 116 117 Todo: 118 * Other serialization modes support. Only json mode is supported now. 119 120 Raises: 121 NotImplementedError: See Todo. 122 TypeError: Raised when value is not a valid type. 123 FileExistsError: Raised when the file already exists. 124 """ 125 if mode != "json": 126 raise NotImplementedError( 127 "Our commune data storage only supports json mode" 128 ) 129 if not isinstance(value, (dict, list, tuple, set, float, str, int)): 130 raise TypeError( 131 f"Invalid type for commune data storage value: {type(value)}" 132 ) 133 timestamp = int(time.time()) 134 135 full_path = os.path.expanduser(os.path.join(COMMUNE_HOME, path)) 136 137 if os.path.exists(full_path): 138 raise FileExistsError( 139 f"Commune data storage file already exists: {full_path}" 140 ) 141 142 ensure_parent_dir_exists(full_path) 143 144 if password: 145 value = _encrypt_data(password, value) 146 encrypted = True 147 else: 148 encrypted = False 149 150 with open(full_path, "w") as file: 151 json.dump( 152 {"data": value, "encrypted": encrypted, "timestamp": timestamp}, 153 file, 154 indent=4, 155 )
Put data into commune data storage.
Arguments:
- path: Data storage path.
- value: Data to store.
- mode: Data storage mode.
- encrypt: Whether to encrypt the data.
Todo:
- Other serialization modes support. Only json mode is supported now.
Raises:
- NotImplementedError: See Todo.
- TypeError: Raised when value is not a valid type.
- FileExistsError: Raised when the file already exists.