You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Comrad/komrade/backend/crypt.py

147 lines
4.0 KiB
Python

4 years ago
"""
Storage for both keys and data
"""
4 years ago
import os,sys; sys.path.append(os.path.abspath(os.path.join(os.path.abspath(os.path.join(os.path.dirname(__file__),'..')),'..')))
from komrade import *
4 years ago
from simplekv.fs import FilesystemStore
from simplekv.memory.redisstore import RedisStore
import redis
import hashlib,os
import zlib
4 years ago
4 years ago
LOG_GET_SET = True
4 years ago
4 years ago
class Crypt(Logger):
4 years ago
def __init__(self,name=None,fn=None,cell=None,init_d=None):
4 years ago
if not name and fn: name=os.path.basename(fn).replace('.','_')
self.name,self.fn,self.cell = name,fn,cell
self.store = FilesystemStore(self.fn)
4 years ago
if init_d:
for k,v in init_d.items():
4 years ago
try:
self.store.put(k,v)
except OSError as e:
self.log('!!',e)
self.log('!! key ->',k)
self.log('!! val ->',v)
raise KomradeException()
4 years ago
def log(self,*x):
if LOG_GET_SET:
super().log(*x)
4 years ago
def hash(self,binary_data):
return hashlib.sha256(binary_data).hexdigest()
# return zlib.adler32(binary_data)
def force_binary(self,k_b):
4 years ago
if k_b is None: return None
4 years ago
if type(k_b)==str: k_b=k_b.encode()
4 years ago
if type(k_b)!=bytes: k_b=k_b.decode()
4 years ago
return k_b
4 years ago
def package_key(self,k,prefix=''):
# self.log('k???',type(k),k)
4 years ago
if not k: return b''
# self.log('prefix???',type(prefix),prefix)
4 years ago
k_b = self.force_binary(k)
# self.log(type(k_b),k_b)
4 years ago
# k_s = k_b.decode()
# self.log(type(k_s),k_s)
# k_s2 = prefix + k_s
# self.log(type(k_s2),k_s2)
# k_b2 = k_s2.encode()
k_b2 = self.force_binary(prefix) + k_b
# self.log('k_b2',type(k_b2),k_b2)
4 years ago
# k_b = self.cell.encrypt(k_b)
4 years ago
# prefix_b = self.force_binary(prefix)
4 years ago
return k_b2
4 years ago
def package_val(self,k):
k_b = self.force_binary(k)
4 years ago
if self.cell is not None: k_b = self.cell.encrypt(k_b)
4 years ago
return k_b
def unpackage_val(self,k_b):
try:
4 years ago
if self.cell is not None: k_b = self.cell.decrypt(k_b)
4 years ago
except ThemisError:
4 years ago
pass
return k_b
4 years ago
4 years ago
def set(self,k,v,prefix=''):
# self.log('set() k -->',prefix,k)
4 years ago
k_b=self.package_key(k,prefix=prefix)
# self.log('set() k_b -->',k_b)
4 years ago
k_b_hash = self.hash(k_b)
# self.log('k_b_hash',type(k_b_hash),k_b_hash)
4 years ago
# self.log('set() v -->',v)
4 years ago
v_b=self.package_val(v)
4 years ago
self.log(f'set(\n\t{prefix}{k},\n\t{k_b}\n\t{k_b_hash}\n\t\n\t{v_b}\n)\n')
4 years ago
# stop
4 years ago
# stop
4 years ago
4 years ago
return self.store.put(k_b_hash,v_b)
4 years ago
def exists(self,k,prefix=''):
return bool(self.get(k,prefix=prefix))
4 years ago
def get(self,k,prefix=''):
# self.log('k1? -->',prefix,k)
4 years ago
k_b=self.package_key(k,prefix=prefix)
# self.log('k2? -->',k_b)
4 years ago
k_b_hash = self.hash(k_b)
# self.log('k_b_hash',type(k_b_hash),k_b_hash)
4 years ago
4 years ago
try:
4 years ago
v=self.store.get(k_b_hash)
4 years ago
except KeyError:
return None
# self.log('v? -->',v)
4 years ago
v_b=self.unpackage_val(v)
# self.log('v_b?',v_b)
# self.log('get()',k_b,'-->',v_b)
4 years ago
return v_b
class KeyCrypt(Crypt):
def __init__(self):
4 years ago
return super().__init__(name=PATH_CRYPT_CA_KEYS.replace('.','_'))
4 years ago
class DataCrypt(Crypt):
def __init__(self):
4 years ago
return super().__init__(name=PATH_CRYPT_CA_DATA.replace('.','_'))
4 years ago
4 years ago
from collections import defaultdict
4 years ago
class CryptMemory(Crypt):
def __init__(self):
self.data = defaultdict(None)
self.crypt = defaultdict(None)
4 years ago
self.cell = None
4 years ago
def set(self,k,v,prefix=''):
k_b=self.package_key(k,prefix=prefix)
v_b=self.package_val(v)
self.data[k]=v_b
self.crypt[k_b]=v_b
4 years ago
if __name__=='__main__':
crypt = Crypt('testt')
print(crypt.set('hellothere',b'ryan'))
# print(crypt.get(b'hello there'))