Source code for mincepy.types
from abc import ABCMeta, abstractmethod
import datetime
import uuid
try: # Python3
from hashlib import blake2b
except ImportError: # Python < 3.6
from pyblake2 import blake2b
import mincepy
from . import depositors
__all__ = 'Savable', 'Comparable', 'Object', 'SavableObject', 'PRIMITIVE_TYPES'
# The primitives that all archive types must support
PRIMITIVE_TYPES = (bool, int, float, str, dict, list, type(None), bytes, uuid.UUID,
datetime.datetime)
def is_primitive(obj):
return isinstance(obj, PRIMITIVE_TYPES)
[docs]class Savable(metaclass=ABCMeta):
"""Interface for an object that can save an load its instance state"""
TYPE_ID = None
LATEST_MIGRATION = None # type: mincepy.ObjectMigration
def __init__(self):
assert self.TYPE_ID is not None, "Must set the TYPE_ID for an object to be savable"
[docs] @abstractmethod
def save_instance_state(self, saver: depositors.Saver):
"""Save the instance state of an object, should return a saved instance"""
[docs] @abstractmethod
def load_instance_state(self, saved_state, loader: depositors.Loader):
"""Take the given object and load the instance state into it"""
[docs]class Comparable(metaclass=ABCMeta):
"""Interface for an object that can be compared and hashed"""
@abstractmethod
def __eq__(self, other) -> bool:
"""Determine if two objects are equal"""
[docs] @abstractmethod
def yield_hashables(self, hasher):
"""Produce a hash representing the value"""
[docs]class Object(Comparable, metaclass=ABCMeta):
"""A simple object that is comparable"""
[docs]class SavableObject(Object, Savable, metaclass=ABCMeta):
"""A class that is both savable and comparable"""
_historian = None
def __init__(self):
super().__init__()
mincepy.process.CreatorsRegistry.created(self)
class Equator:
def __init__(self, equators=tuple()):
self._equators = list(equators)
def do_hash(*args):
hasher = blake2b(digest_size=32)
for arg in args:
hasher.update(arg)
return hasher.hexdigest()
self._hasher = do_hash
def add_equator(self, equator):
self._equators.append(equator)
def remove_equator(self, equator):
self._equators.reverse()
try:
self._equators.remove(equator)
except ValueError:
raise ValueError("Unknown equator '{}'".format(equator))
finally:
self._equators.reverse()
def get_equator(self, obj):
# Iterate in reversed order i.e. the latest added should be used preferentially
for equator in reversed(self._equators):
if isinstance(obj, equator.TYPE):
return equator
raise TypeError("Don't know how to compare '{}' types, no type equator set".format(
type(obj)))
def yield_hashables(self, obj):
try:
equator = self.get_equator(obj)
except TypeError:
# Try the objects's method
try:
yield from obj.yield_hashables(self)
except AttributeError:
raise TypeError("No helper registered and no yield_hashabled method on '{}'".format(
type(obj)))
else:
yield from equator.yield_hashables(obj, self)
def hash(self, obj):
return self._hasher(*self.yield_hashables(obj))
def eq(self, obj1, obj2) -> bool: # pylint: disable=invalid-name
if not type(obj1) == type(obj2): # pylint: disable=unidiomatic-typecheck
return False
try:
equator = self.get_equator(obj1)
except TypeError:
# Fallback to python eq
return obj1 == obj2
else:
return equator.eq(obj1, obj2)
def float_to_str(self, value, sig=14): # pylint: disable=no-self-use
"""
Convert float to text string for computing hash.
Preserve up to N significant number given by sig.
:param value: the float value to convert
:param sig: choose how many digits after the comma should be output
"""
fmt = u'{{:.{}g}}'.format(sig)
return fmt.format(value)