2018-11-18 22:54:00 -05:00
|
|
|
import re
|
2018-08-16 00:56:46 -04:00
|
|
|
from typing import TypeVar, Sequence, Optional
|
2020-05-01 09:28:51 -04:00
|
|
|
from lbry.constants import COIN
|
2018-11-18 22:54:00 -05:00
|
|
|
|
|
|
|
|
|
|
|
def coins_to_satoshis(coins):
|
|
|
|
if not isinstance(coins, str):
|
|
|
|
raise ValueError("{coins} must be a string")
|
|
|
|
result = re.search(r'^(\d{1,10})\.(\d{1,8})$', coins)
|
|
|
|
if result is not None:
|
|
|
|
whole, fractional = result.groups()
|
|
|
|
return int(whole+fractional.ljust(8, "0"))
|
|
|
|
raise ValueError("'{lbc}' is not a valid coin decimal")
|
|
|
|
|
|
|
|
|
|
|
|
def satoshis_to_coins(satoshis):
|
|
|
|
coins = '{:.8f}'.format(satoshis / COIN).rstrip('0')
|
|
|
|
if coins.endswith('.'):
|
|
|
|
return coins+'0'
|
|
|
|
else:
|
|
|
|
return coins
|
2018-05-25 02:03:25 -04:00
|
|
|
|
|
|
|
|
2018-06-26 17:22:05 -04:00
|
|
|
T = TypeVar('T')
|
|
|
|
|
|
|
|
|
2018-07-28 20:52:54 -04:00
|
|
|
class ReadOnlyList(Sequence[T]):
|
2018-05-25 02:03:25 -04:00
|
|
|
|
|
|
|
def __init__(self, lst):
|
|
|
|
self.lst = lst
|
|
|
|
|
2018-07-28 20:52:54 -04:00
|
|
|
def __getitem__(self, key):
|
2018-05-25 02:03:25 -04:00
|
|
|
return self.lst[key]
|
|
|
|
|
2018-07-28 20:52:54 -04:00
|
|
|
def __len__(self) -> int:
|
2018-05-25 02:03:25 -04:00
|
|
|
return len(self.lst)
|
|
|
|
|
|
|
|
|
|
|
|
def subclass_tuple(name, base):
|
|
|
|
return type(name, (base,), {'__slots__': ()})
|
|
|
|
|
|
|
|
|
2018-08-16 00:43:38 -04:00
|
|
|
class ArithUint256:
|
|
|
|
# https://github.com/bitcoin/bitcoin/blob/master/src/arith_uint256.cpp
|
|
|
|
|
|
|
|
__slots__ = '_value', '_compact'
|
|
|
|
|
|
|
|
def __init__(self, value: int) -> None:
|
|
|
|
self._value = value
|
2018-08-16 00:56:46 -04:00
|
|
|
self._compact: Optional[int] = None
|
2018-08-16 00:43:38 -04:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def from_compact(cls, compact) -> 'ArithUint256':
|
|
|
|
size = compact >> 24
|
|
|
|
word = compact & 0x007fffff
|
|
|
|
if size <= 3:
|
|
|
|
return cls(word >> 8 * (3 - size))
|
|
|
|
else:
|
|
|
|
return cls(word << 8 * (size - 3))
|
|
|
|
|
|
|
|
@property
|
|
|
|
def value(self) -> int:
|
|
|
|
return self._value
|
|
|
|
|
|
|
|
@property
|
|
|
|
def compact(self) -> int:
|
|
|
|
if self._compact is None:
|
|
|
|
self._compact = self._calculate_compact()
|
|
|
|
return self._compact
|
|
|
|
|
|
|
|
@property
|
|
|
|
def negative(self) -> int:
|
|
|
|
return self._calculate_compact(negative=True)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def bits(self) -> int:
|
|
|
|
""" Returns the position of the highest bit set plus one. """
|
2018-08-16 00:56:46 -04:00
|
|
|
bits = bin(self._value)[2:]
|
|
|
|
for i, d in enumerate(bits):
|
2018-08-16 00:43:38 -04:00
|
|
|
if d:
|
2018-08-16 00:56:46 -04:00
|
|
|
return (len(bits) - i) + 1
|
2018-08-16 00:43:38 -04:00
|
|
|
return 0
|
|
|
|
|
|
|
|
@property
|
|
|
|
def low64(self) -> int:
|
|
|
|
return self._value & 0xffffffffffffffff
|
|
|
|
|
|
|
|
def _calculate_compact(self, negative=False) -> int:
|
|
|
|
size = (self.bits + 7) // 8
|
|
|
|
if size <= 3:
|
|
|
|
compact = self.low64 << 8 * (3 - size)
|
|
|
|
else:
|
|
|
|
compact = ArithUint256(self._value >> 8 * (size - 3)).low64
|
|
|
|
# The 0x00800000 bit denotes the sign.
|
|
|
|
# Thus, if it is already set, divide the mantissa by 256 and increase the exponent.
|
|
|
|
if compact & 0x00800000:
|
|
|
|
compact >>= 8
|
|
|
|
size += 1
|
|
|
|
assert (compact & ~0x007fffff) == 0
|
|
|
|
assert size < 256
|
|
|
|
compact |= size << 24
|
|
|
|
if negative and compact & 0x007fffff:
|
|
|
|
compact |= 0x00800000
|
|
|
|
return compact
|
|
|
|
|
|
|
|
def __mul__(self, x):
|
|
|
|
# Take the mod because we are limited to an unsigned 256 bit number
|
|
|
|
return ArithUint256((self._value * x) % 2 ** 256)
|
|
|
|
|
|
|
|
def __truediv__(self, x):
|
|
|
|
return ArithUint256(int(self._value / x))
|
|
|
|
|
|
|
|
def __gt__(self, other):
|
|
|
|
return self._value > other
|
|
|
|
|
|
|
|
def __lt__(self, other):
|
|
|
|
return self._value < other
|