2021-05-14 02:21:43 -07:00
|
|
|
import struct
|
|
|
|
|
2021-05-14 04:26:00 -07:00
|
|
|
from orchard_commitments import note_commit
|
2021-05-14 02:21:43 -07:00
|
|
|
from orchard_key_components import diversify_hash, prf_expand, derive_nullifier, FullViewingKey, SpendingKey
|
|
|
|
from orchard_pallas import Point, Scalar
|
|
|
|
from orchard_utils import to_base, to_scalar
|
|
|
|
|
|
|
|
from utils import leos2bsp
|
|
|
|
|
|
|
|
class OrchardNote(object):
|
2021-05-25 23:08:08 -07:00
|
|
|
def __init__(self, d, pk_d, v, rho, rseed):
|
2021-05-28 08:37:49 -07:00
|
|
|
assert isinstance(v, int)
|
2021-05-14 02:21:43 -07:00
|
|
|
self.d = d
|
|
|
|
self.pk_d = pk_d
|
|
|
|
self.v = v
|
|
|
|
self.rho = rho
|
|
|
|
self.rseed = rseed
|
2021-05-26 21:48:44 -07:00
|
|
|
self.rcm = self.rcm()
|
|
|
|
self.psi = self.psi()
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-25 23:08:08 -07:00
|
|
|
def __eq__(self, other):
|
|
|
|
if other is None:
|
|
|
|
return False
|
2021-05-14 04:26:00 -07:00
|
|
|
return (
|
2021-05-25 23:08:08 -07:00
|
|
|
self.d == other.d and
|
|
|
|
self.pk_d == other.pk_d and
|
|
|
|
self.v == other.v and
|
|
|
|
self.rho == other.rho and
|
|
|
|
self.rcm == other.rcm and
|
|
|
|
self.psi == other.psi
|
2021-05-14 04:26:00 -07:00
|
|
|
)
|
|
|
|
|
2021-05-26 21:48:44 -07:00
|
|
|
def rcm(self):
|
|
|
|
return to_scalar(prf_expand(self.rseed, b'\x05' + bytes(self.rho)))
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-26 21:48:44 -07:00
|
|
|
def psi(self):
|
|
|
|
return to_base(prf_expand(self.rseed, b'\x09' + bytes(self.rho)))
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-14 04:26:00 -07:00
|
|
|
def note_commitment(self):
|
|
|
|
g_d = diversify_hash(self.d)
|
2021-05-25 23:08:08 -07:00
|
|
|
return note_commit(self.rcm, leos2bsp(bytes(g_d)), leos2bsp(bytes(self.pk_d)), self.v, self.rho, self.psi)
|
2021-05-14 04:26:00 -07:00
|
|
|
|
2021-05-14 02:21:43 -07:00
|
|
|
def note_plaintext(self, memo):
|
|
|
|
return OrchardNotePlaintext(self.d, self.v, self.rseed, memo)
|
|
|
|
|
|
|
|
# https://zips.z.cash/protocol/nu5.pdf#notept
|
|
|
|
class OrchardNotePlaintext(object):
|
|
|
|
def __init__(self, d, v, rseed, memo):
|
2021-05-14 04:26:00 -07:00
|
|
|
self.leadbyte = bytes.fromhex('02')
|
2021-05-14 02:21:43 -07:00
|
|
|
self.d = d
|
|
|
|
self.v = v
|
|
|
|
self.rseed = rseed
|
|
|
|
self.memo = memo
|
|
|
|
|
|
|
|
def __bytes__(self):
|
|
|
|
return (
|
|
|
|
self.leadbyte +
|
|
|
|
self.d +
|
2021-05-25 23:08:08 -07:00
|
|
|
struct.pack('<Q', self.v) +
|
2021-05-25 23:32:17 -07:00
|
|
|
self.rseed +
|
2021-05-14 02:21:43 -07:00
|
|
|
self.memo
|
|
|
|
)
|
|
|
|
|
|
|
|
def dummy_nullifier(self, rand):
|
|
|
|
sk = SpendingKey(rand.b(32))
|
|
|
|
fvk = FullViewingKey(sk)
|
|
|
|
pk_d = fvk.default_pkd()
|
2021-05-25 23:08:08 -07:00
|
|
|
d = fvk.default_d()
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-25 23:08:08 -07:00
|
|
|
v = 0
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-25 23:08:08 -07:00
|
|
|
rseed = rand.b(32)
|
|
|
|
rho = Point.rand(rand).extract()
|
2021-05-14 02:21:43 -07:00
|
|
|
|
2021-05-25 23:08:08 -07:00
|
|
|
note = OrchardNote(d, pk_d, v, rho, rseed)
|
|
|
|
cm = note.note_commitment()
|
2021-05-26 21:48:44 -07:00
|
|
|
return derive_nullifier(fvk.nk, rho, note.psi, cm)
|