2018-07-21 03:24:50 -07:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
import sys; assert sys.version_info[0] >= 3, "Python 3 required."
|
|
|
|
|
2018-06-05 01:36:37 -07:00
|
|
|
import argparse
|
2018-06-04 21:32:30 -07:00
|
|
|
from binascii import hexlify
|
2018-06-05 01:36:37 -07:00
|
|
|
import json
|
2018-06-04 21:32:30 -07:00
|
|
|
|
|
|
|
|
2018-06-04 21:10:43 -07:00
|
|
|
def chunk(h):
|
2018-06-04 07:31:12 -07:00
|
|
|
hstr = str(h, 'utf-8')
|
|
|
|
return '0x' + ', 0x'.join([hstr[i:i+2] for i in range(0, len(hstr), 2)])
|
2018-06-04 21:32:30 -07:00
|
|
|
|
2018-07-21 06:10:28 -07:00
|
|
|
class Some(object):
|
|
|
|
def __init__(self, thing):
|
|
|
|
self.thing = thing
|
|
|
|
|
|
|
|
def option(x):
|
|
|
|
return Some(x) if x else None
|
2018-06-05 01:36:37 -07:00
|
|
|
|
|
|
|
#
|
|
|
|
# JSON (with string comments)
|
2018-06-12 16:48:17 -07:00
|
|
|
# If bitcoin_flavoured == True, 32-byte values are reversed
|
2018-06-05 01:36:37 -07:00
|
|
|
#
|
|
|
|
|
2018-06-12 16:48:17 -07:00
|
|
|
def tv_value_json(value, bitcoin_flavoured):
|
2018-07-21 06:10:28 -07:00
|
|
|
if isinstance(value, Some):
|
|
|
|
value = value.thing
|
|
|
|
|
2018-06-05 01:36:37 -07:00
|
|
|
if type(value) == bytes:
|
2018-06-12 16:48:17 -07:00
|
|
|
if bitcoin_flavoured and len(value) == 32:
|
|
|
|
value = value[::-1]
|
2018-06-05 01:36:37 -07:00
|
|
|
value = hexlify(value).decode()
|
|
|
|
return value
|
|
|
|
|
2018-06-12 16:48:17 -07:00
|
|
|
def tv_json(filename, parts, vectors, bitcoin_flavoured):
|
2018-06-05 01:36:37 -07:00
|
|
|
if type(vectors) == type({}):
|
|
|
|
vectors = [vectors]
|
|
|
|
|
|
|
|
print('''[
|
|
|
|
["From https://github.com/zcash-hackworks/zcash-test-vectors/blob/master/%s.py"],
|
|
|
|
["%s"],''' % (
|
|
|
|
filename,
|
|
|
|
', '.join([p[0] for p in parts])
|
|
|
|
))
|
|
|
|
print(' ' + ',\n '.join([
|
2018-06-12 16:48:17 -07:00
|
|
|
json.dumps([tv_value_json(v[p[0]], bitcoin_flavoured) for p in parts]) for v in vectors
|
2018-06-05 01:36:37 -07:00
|
|
|
]))
|
|
|
|
print(']')
|
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Rust
|
|
|
|
#
|
|
|
|
|
2018-06-04 22:30:16 -07:00
|
|
|
def tv_bytes_rust(name, value, pad):
|
2018-06-04 22:06:05 -07:00
|
|
|
print('''%s%s: [
|
|
|
|
%s%s
|
|
|
|
%s],''' % (
|
|
|
|
pad,
|
|
|
|
name,
|
|
|
|
pad,
|
|
|
|
chunk(hexlify(value)),
|
|
|
|
pad,
|
|
|
|
))
|
2018-06-04 21:32:30 -07:00
|
|
|
|
2018-07-21 06:10:28 -07:00
|
|
|
def tv_option_bytes_rust(name, value, pad):
|
|
|
|
if value:
|
|
|
|
print('''%s%s: Some([
|
|
|
|
%s%s
|
|
|
|
%s]),''' % (
|
|
|
|
pad,
|
|
|
|
name,
|
|
|
|
pad,
|
|
|
|
chunk(hexlify(value.thing)),
|
|
|
|
pad,
|
|
|
|
))
|
|
|
|
else:
|
|
|
|
print('%s%s: None,' % (pad, name))
|
|
|
|
|
2018-06-04 22:30:16 -07:00
|
|
|
def tv_int_rust(name, value, pad):
|
|
|
|
print('%s%s: %d,' % (pad, name, value))
|
|
|
|
|
|
|
|
def tv_part_rust(name, value, indent=3):
|
|
|
|
pad = ' ' * indent
|
|
|
|
if type(value) == bytes:
|
|
|
|
tv_bytes_rust(name, value, pad)
|
2018-07-21 06:10:28 -07:00
|
|
|
elif isinstance(value, Some) or value is None:
|
|
|
|
tv_option_bytes_rust(name, value, pad)
|
2018-06-04 22:30:16 -07:00
|
|
|
elif type(value) == int:
|
|
|
|
tv_int_rust(name, value, pad)
|
|
|
|
else:
|
|
|
|
raise ValueError('Invalid type(%s): %s' % (name, type(value)))
|
|
|
|
|
2018-06-04 21:32:30 -07:00
|
|
|
def tv_rust(filename, parts, vectors):
|
|
|
|
print(' struct TestVector {')
|
2018-07-21 06:11:07 -07:00
|
|
|
for p in parts: print(' %s: %s,' % p)
|
2018-06-04 21:32:30 -07:00
|
|
|
print(''' };
|
|
|
|
|
|
|
|
// From https://github.com/zcash-hackworks/zcash-test-vectors/blob/master/%s.py''' % (
|
|
|
|
filename,
|
|
|
|
))
|
|
|
|
if type(vectors) == type({}):
|
|
|
|
print(' let test_vector = TestVector {')
|
2018-07-21 06:11:07 -07:00
|
|
|
for p in parts: tv_part_rust(p[0], vectors[p[0]])
|
2018-06-04 21:32:30 -07:00
|
|
|
print(' };')
|
2018-06-04 22:06:05 -07:00
|
|
|
elif type(vectors) == type([]):
|
|
|
|
print(' let test_vectors = vec![')
|
|
|
|
for vector in vectors:
|
|
|
|
print(' TestVector {')
|
2018-07-21 06:11:07 -07:00
|
|
|
for p in parts: tv_part_rust(p[0], vector[p[0]], 4)
|
2018-06-04 22:06:05 -07:00
|
|
|
print(' },')
|
|
|
|
print(' ];')
|
2018-06-04 21:32:30 -07:00
|
|
|
else:
|
|
|
|
raise ValueError('Invalid type(vectors)')
|
2018-06-05 01:36:37 -07:00
|
|
|
|
|
|
|
|
|
|
|
#
|
|
|
|
# Rendering functions
|
|
|
|
#
|
|
|
|
|
|
|
|
def render_args():
|
|
|
|
parser = argparse.ArgumentParser()
|
2018-07-21 06:12:05 -07:00
|
|
|
parser.add_argument('-t', '--target', choices=['zcash', 'json', 'rust'], default='rust')
|
2018-06-05 01:36:37 -07:00
|
|
|
return parser.parse_args()
|
|
|
|
|
|
|
|
def render_tv(args, filename, parts, vectors):
|
|
|
|
if args.target == 'rust':
|
|
|
|
tv_rust(filename, parts, vectors)
|
2018-06-12 16:48:17 -07:00
|
|
|
elif args.target == 'zcash':
|
|
|
|
tv_json(filename, parts, vectors, True)
|
2018-07-21 06:12:05 -07:00
|
|
|
elif args.target == 'json':
|
|
|
|
tv_json(filename, parts, vectors, False)
|