Source code for toy_crypto.types

"""
Helpful(?) type declarations and guards.

These are intended to make things easier for me, the author (jpgoldberg).
They are not carefully thought out.
This module is probably the least stable of any of these unstable modules.
"""

from typing import (
    Any,
    Callable,
    NewType,
    Optional,
    TypeGuard,
    Protocol,
    Union,
    runtime_checkable,
)

import operator

Prob = NewType("Prob", float)
"""Probability: A float between 0.0 and 1.0"""


[docs] def is_prob(val: Any) -> TypeGuard[Prob]: """true if val is a float, s.t. 0.0 <= va <= 1.0""" if not isinstance(val, float): return False return val >= 0.0 and val <= 1.0
PositiveInt = NewType("PositiveInt", int) """Positive integer."""
[docs] def is_positive_int(val: Any) -> TypeGuard[PositiveInt]: """true if val is a float, s.t. 0.0 <= val <= 1.0""" if not isinstance(val, int): return False return val >= 1
Byte = int """And int representing a single byte. Currently implemented as a type alias. As a consequence, type checking is not going to identify cases where an int out of the range of a byte is used. """
[docs] def is_byte(val: Any) -> bool: """True iff val is int s.t. 0 <= val < 256.""" if not isinstance(val, int): return False return 0 <= val and val < 256
[docs] @runtime_checkable class SupportsBool(Protocol):
[docs] def __bool__(self) -> bool: ...
[docs] class Bit: """ Because I made poor choices earlier of how to represent bits, I need an abstraction. """ def __init__(self, b: SupportsBool) -> None: self._value: bool = b is True self._as_int: Optional[int] = None self._as_bytes: Optional[bytes] = None def __bool__(self) -> bool: return self._value def as_bool(self) -> bool: return self._value def as_int(self) -> int: if not self._as_int: self._as_int = 1 if self._value else 0 return self._as_int def as_bytes(self) -> bytes: if not self._as_bytes: self._as_bytes = ( (1).to_bytes(1, "big") if self._value else (0).to_bytes(1, "big") ) return self._as_bytes def __eq__(self, other: Any) -> bool: ob = self._other_bool(other) if ob is None: return NotImplemented ob = other.__bool__() return self._value == ob @staticmethod def _other_bool(other: Any) -> Optional[bool]: if isinstance(other, bytes): ob = any([b != 0 for b in other]) elif not isinstance(other, SupportsBool): return None else: ob = other.__bool__() return ob def _logic( self, other: bool, expr: Callable[[bool, bool], bool] ) -> Union["Bit", int, bool, bytes]: """ Abstraction to manage type of :data:`other` for things like :func:`__and__` and :func:`__or__`. """ sb = self.as_bool() tvalue = expr(sb, other) if isinstance(other, Bit): return Bit(tvalue) if isinstance(other, int): return 1 if tvalue else 0 if isinstance(other, bytes): return (1).to_bytes(1, "big") if tvalue else (0).to_bytes(1, "big") return tvalue def __and__(self, other: Any) -> Union["Bit", int, bool, bytes]: ob = self._other_bool(other) if ob is None: return NotImplemented return self._logic(other=ob, expr=lambda s, o: s and o) def __xor__(self, other: Any) -> Union["Bit", int, bool, bytes]: ob = self._other_bool(other) if ob is None: return NotImplemented return self._logic(other=ob, expr=lambda s, o: operator.xor(s, o)) def __or__(self, other: Any) -> Union["Bit", int, bool, bytes]: ob = self._other_bool(other) if ob is None: return NotImplemented return self._logic(other=ob, expr=lambda s, o: s or o) def inv(self) -> "Bit": inv_b = not self.as_bool() return Bit(inv_b) def __inv__(self) -> "Bit": return self.inv()