From 60db123206c128482f67796b5b3412c08e9d2515 Mon Sep 17 00:00:00 2001 From: Simon Date: Wed, 22 Aug 2018 16:50:27 -0700 Subject: [PATCH 1/2] Implement ZIP 243 test vectors. Co-authored-by: Jack Grigg --- transaction.py | 111 +++++++++++++++++++++++++++++-- zip_0243.py | 175 +++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 282 insertions(+), 4 deletions(-) create mode 100644 zip_0243.py diff --git a/transaction.py b/transaction.py index 25aa5b2..c81acef 100644 --- a/transaction.py +++ b/transaction.py @@ -1,6 +1,9 @@ #!/usr/bin/env python3 import struct +from sapling_generators import find_group_hash, SPENDING_KEY_BASE +from sapling_jubjub import Fq +from sapling_utils import leos2ip from zc_utils import write_compact_size MAX_MONEY = 21000000 * 100000000 @@ -9,6 +12,24 @@ TX_EXPIRY_HEIGHT_THRESHOLD = 500000000 OVERWINTER_VERSION_GROUP_ID = 0x03C48270 OVERWINTER_TX_VERSION = 3 +SAPLING_VERSION_GROUP_ID = 0x892F2085 +SAPLING_TX_VERSION = 4 + +# Sapling note magic values, copied from src/zcash/Zcash.h +NOTEENCRYPTION_AUTH_BYTES = 16 +ZC_NOTEPLAINTEXT_LEADING = 1 +ZC_V_SIZE = 8 +ZC_RHO_SIZE = 32 +ZC_R_SIZE = 32 +ZC_MEMO_SIZE = 512 +ZC_DIVERSIFIER_SIZE = 11 +ZC_JUBJUB_POINT_SIZE = 32 +ZC_JUBJUB_SCALAR_SIZE = 32 +ZC_NOTEPLAINTEXT_SIZE = ZC_NOTEPLAINTEXT_LEADING + ZC_V_SIZE + ZC_RHO_SIZE + ZC_R_SIZE + ZC_MEMO_SIZE +ZC_SAPLING_ENCPLAINTEXT_SIZE = ZC_NOTEPLAINTEXT_LEADING + ZC_DIVERSIFIER_SIZE + ZC_V_SIZE + ZC_R_SIZE + ZC_MEMO_SIZE +ZC_SAPLING_OUTPLAINTEXT_SIZE = ZC_JUBJUB_POINT_SIZE + ZC_JUBJUB_SCALAR_SIZE +ZC_SAPLING_ENCCIPHERTEXT_SIZE = ZC_SAPLING_ENCPLAINTEXT_SIZE + NOTEENCRYPTION_AUTH_BYTES +ZC_SAPLING_OUTCIPHERTEXT_SIZE = ZC_SAPLING_OUTPLAINTEXT_SIZE + NOTEENCRYPTION_AUTH_BYTES # BN254 encoding of G1 elements. p[1] is big-endian. def pack_g1(p): @@ -41,9 +62,59 @@ class PHGRProof(object): pack_g1(self.g_H) ) +class GrothProof(object): + def __init__(self, rand): + self.g_A = rand.b(48) + self.g_B = rand.b(96) + self.g_C = rand.b(48) + + def __bytes__(self): + return ( + self.g_A + + self.g_B + + self.g_C + ) + +class SpendDescription(object): + def __init__(self, rand): + self.cv = find_group_hash(b'TVRandPt', rand.b(32)) + self.anchor = Fq(leos2ip(rand.b(32))) + self.nullifier = rand.b(32) + self.rk = rand.b(32) + self.proof = GrothProof(rand) + self.spendAuthSig = rand.b(64) # Invalid + + def __bytes__(self): + return ( + bytes(self.cv) + + bytes(self.anchor) + + self.nullifier + + self.rk + + bytes(self.proof) + + self.spendAuthSig + ) + +class OutputDescription(object): + def __init__(self, rand): + self.cv = find_group_hash(b'TVRandPt', rand.b(32)) + self.cmu = Fq(leos2ip(rand.b(32))) + self.ephemeralKey = find_group_hash(b'TVRandPt', rand.b(32)) + self.encCiphertext = rand.b(ZC_SAPLING_ENCCIPHERTEXT_SIZE) + self.outCipherText = rand.b(ZC_SAPLING_OUTCIPHERTEXT_SIZE) + self.proof = GrothProof(rand) + + def __bytes__(self): + return ( + bytes(self.cv) + + bytes(self.cmu) + + bytes(self.ephemeralKey) + + self.encCiphertext + + self.outCipherText + + bytes(self.proof) + ) class JoinSplit(object): - def __init__(self, rand): + def __init__(self, rand, fUseGroth = False): self.vpub_old = 0 self.vpub_new = 0 self.anchor = rand.b(32) @@ -52,7 +123,7 @@ class JoinSplit(object): self.ephemeralKey = rand.b(32) self.randomSeed = rand.b(32) self.macs = (rand.b(32), rand.b(32)) - self.proof = PHGRProof(rand) + self.proof = GrothProof(rand) if fUseGroth else PHGRProof(rand) self.ciphertexts = (rand.b(601), rand.b(601)) def __bytes__(self): @@ -132,6 +203,10 @@ class Transaction(object): self.fOverwintered = True self.nVersionGroupId = OVERWINTER_VERSION_GROUP_ID self.nVersion = OVERWINTER_TX_VERSION + elif version == SAPLING_TX_VERSION: + self.fOverwintered = True + self.nVersionGroupId = SAPLING_VERSION_GROUP_ID + self.nVersion = SAPLING_TX_VERSION else: self.fOverwintered = False self.nVersion = rand.u32() & ((1 << 31) - 1) @@ -146,15 +221,26 @@ class Transaction(object): self.nLockTime = rand.u32() self.nExpiryHeight = rand.u32() % TX_EXPIRY_HEIGHT_THRESHOLD + self.valueBalance = rand.u64() % (MAX_MONEY + 1) + + self.vShieldedSpends = [] + self.vShieldedOutputs = [] + if self.nVersion >= SAPLING_TX_VERSION: + for _ in range(rand.u8() % 5): + self.vShieldedSpends.append(SpendDescription(rand)) + for _ in range(rand.u8() % 5): + self.vShieldedOutputs.append(OutputDescription(rand)) self.vJoinSplit = [] if self.nVersion >= 2: for i in range(rand.u8() % 3): - self.vJoinSplit.append(JoinSplit(rand)) + self.vJoinSplit.append(JoinSplit(rand, self.fOverwintered and self.nVersion >= SAPLING_TX_VERSION)) if len(self.vJoinSplit) > 0: self.joinSplitPubKey = rand.b(32) # Potentially invalid self.joinSplitSig = rand.b(64) # Invalid + self.bindingSig = rand.b(64) # Invalid + def header(self): return self.nVersion | (1 << 31 if self.fOverwintered else 0) @@ -169,6 +255,11 @@ class Transaction(object): self.nVersionGroupId == OVERWINTER_VERSION_GROUP_ID and \ self.nVersion == OVERWINTER_TX_VERSION + isSaplingV4 = \ + self.fOverwintered and \ + self.nVersionGroupId == SAPLING_VERSION_GROUP_ID and \ + self.nVersion == SAPLING_TX_VERSION + ret += write_compact_size(len(self.vin)) for x in self.vin: ret += bytes(x) @@ -178,9 +269,18 @@ class Transaction(object): ret += bytes(x) ret += struct.pack('= 2: ret += write_compact_size(len(self.vJoinSplit)) for jsdesc in self.vJoinSplit: @@ -189,4 +289,7 @@ class Transaction(object): ret += self.joinSplitPubKey ret += self.joinSplitSig + if isSaplingV4 and not (len(self.vShieldedSpends) == 0 and len(self.vShieldedOutputs) == 0): + ret += self.bindingSig + return ret diff --git a/zip_0243.py b/zip_0243.py new file mode 100644 index 0000000..9ec1a31 --- /dev/null +++ b/zip_0243.py @@ -0,0 +1,175 @@ +#!/usr/bin/env python3 +from pyblake2 import blake2b +import struct + +from transaction import ( + MAX_MONEY, + SAPLING_TX_VERSION, + Script, + Transaction, +) +from tv_output import render_args, render_tv, Some +from tv_rand import Rand + +from zip_0143 import ( + getHashJoinSplits, + getHashOutputs, + getHashPrevouts, + getHashSequence, + NOT_AN_INPUT, + SIGHASH_ALL, + SIGHASH_ANYONECANPAY, + SIGHASH_NONE, + SIGHASH_SINGLE, +) + + +def getHashShieldedSpends(tx): + digest = blake2b(digest_size=32, person=b'ZcashSSpendsHash') + for desc in tx.vShieldedSpends: + # We don't pass in serialized form of desc as spendAuthSig is not part of the hash + digest.update(bytes(desc.cv)) + digest.update(bytes(desc.anchor)) + digest.update(desc.nullifier) + digest.update(desc.rk) + digest.update(bytes(desc.proof)) + return digest.digest() + +def getHashShieldedOutputs(tx): + digest = blake2b(digest_size=32, person=b'ZcashSOutputHash') + for desc in tx.vShieldedOutputs: + digest.update(bytes(desc)) + return digest.digest() + +def signature_hash(scriptCode, tx, nIn, nHashType, amount, consensusBranchId): + hashPrevouts = b'\x00'*32 + hashSequence = b'\x00'*32 + hashOutputs = b'\x00'*32 + hashJoinSplits = b'\x00'*32 + hashShieldedSpends = b'\x00'*32 + hashShieldedOutputs = b'\x00'*32 + + if not (nHashType & SIGHASH_ANYONECANPAY): + hashPrevouts = getHashPrevouts(tx) + + if (not (nHashType & SIGHASH_ANYONECANPAY)) and \ + (nHashType & 0x1f) != SIGHASH_SINGLE and \ + (nHashType & 0x1f) != SIGHASH_NONE: + hashSequence = getHashSequence(tx) + + if (nHashType & 0x1f) != SIGHASH_SINGLE and \ + (nHashType & 0x1f) != SIGHASH_NONE: + hashOutputs = getHashOutputs(tx) + elif (nHashType & 0x1f) == SIGHASH_SINGLE and \ + 0 <= nIn and nIn < len(tx.vout): + digest = blake2b(digest_size=32, person=b'ZcashOutputsHash') + digest.update(bytes(tx.vout[nIn])) + hashOutputs = digest.digest() + + if len(tx.vJoinSplit) > 0: + hashJoinSplits = getHashJoinSplits(tx) + + if len(tx.vShieldedSpends) > 0: + hashShieldedSpends = getHashShieldedSpends(tx) + + if len(tx.vShieldedOutputs) > 0: + hashShieldedOutputs = getHashShieldedOutputs(tx) + + digest = blake2b( + digest_size=32, + person=b'ZcashSigHash' + struct.pack('', 'bitcoin_flavoured': False}), + ('script_code', 'Vec'), + ('transparent_input', { + 'rust_type': 'Option', + 'rust_fmt': lambda x: None if x == -1 else Some(x), + }), + ('hash_type', 'u32'), + ('amount', 'u64'), + ('consensus_branch_id', 'u32'), + ('sighash', '[u8; 32]'), + ), + test_vectors, + ) + + +if __name__ == '__main__': + main() From a44f9f6c9032aa5cbff35311db140245d9120082 Mon Sep 17 00:00:00 2001 From: Jack Grigg Date: Thu, 23 Aug 2018 14:25:41 +0100 Subject: [PATCH 2/2] Fix chunk() output when given zero-length data --- tv_output.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tv_output.py b/tv_output.py index ce63ad8..fcb225d 100644 --- a/tv_output.py +++ b/tv_output.py @@ -8,7 +8,8 @@ import json def chunk(h): hstr = str(h, 'utf-8') - return '0x' + ', 0x'.join([hstr[i:i+2] for i in range(0, len(hstr), 2)]) + hstr = ', 0x'.join([hstr[i:i+2] for i in range(0, len(hstr), 2)]) + return '0x' + hstr if hstr else '' class Some(object): def __init__(self, thing):