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')
|
2018-08-23 06:25:41 -07:00
|
|
|
hstr = ', 0x'.join([hstr[i:i+2] for i in range(0, len(hstr), 2)])
|
|
|
|
return '0x' + hstr if hstr else ''
|
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
|
|
|
|
|
2022-01-05 13:03:56 -08:00
|
|
|
def bitcoinify(value):
|
2022-02-07 12:33:14 -08:00
|
|
|
if type(value) == list:
|
|
|
|
return [bitcoinify(v) for v in value]
|
|
|
|
|
2022-09-19 12:59:00 -07:00
|
|
|
if type(value) == str:
|
|
|
|
return value
|
|
|
|
|
2022-01-05 13:03:56 -08:00
|
|
|
if type(value) == bytes:
|
|
|
|
if bitcoin_flavoured and len(value) == 32:
|
|
|
|
value = value[::-1]
|
|
|
|
value = hexlify(value).decode()
|
|
|
|
return value
|
|
|
|
|
2022-02-07 12:33:14 -08:00
|
|
|
return bitcoinify(value)
|
2018-06-05 01:36:37 -07:00
|
|
|
|
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-15 16:00:11 -07:00
|
|
|
json.dumps([tv_value_json(v[p[0]], p[1].get('bitcoin_flavoured', 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-08-21 11:44:02 -07:00
|
|
|
def tv_vec_bytes_rust(name, value, pad):
|
|
|
|
print('''%s%s: vec![
|
|
|
|
%s%s
|
|
|
|
%s],''' % (
|
|
|
|
pad,
|
|
|
|
name,
|
|
|
|
pad,
|
|
|
|
chunk(hexlify(value)),
|
|
|
|
pad,
|
|
|
|
))
|
|
|
|
|
2021-04-26 20:41:19 -07:00
|
|
|
def tv_vec_bool_rust(name, value, pad):
|
|
|
|
print('''%s%s: vec![
|
|
|
|
%s%s
|
|
|
|
%s],''' % (
|
|
|
|
pad,
|
|
|
|
name,
|
|
|
|
pad,
|
|
|
|
', '.join(['true' if x else 'false' for x in value]),
|
|
|
|
pad,
|
|
|
|
))
|
|
|
|
|
2022-09-19 12:59:00 -07:00
|
|
|
def tv_str_rust(name, value, pad):
|
|
|
|
print('''%s%s: "%s",''' % (
|
|
|
|
pad,
|
|
|
|
name,
|
|
|
|
value,
|
|
|
|
))
|
|
|
|
|
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-08-21 12:13:11 -07:00
|
|
|
def tv_option_vec_bytes_rust(name, value, pad):
|
|
|
|
if value:
|
|
|
|
print('''%s%s: Some(vec![
|
|
|
|
%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))
|
|
|
|
|
2018-08-21 12:13:11 -07:00
|
|
|
def tv_option_int_rust(name, value, pad):
|
|
|
|
if value:
|
|
|
|
print('%s%s: Some(%d),' % (pad, name, value.thing))
|
|
|
|
else:
|
|
|
|
print('%s%s: None,' % (pad, name))
|
|
|
|
|
|
|
|
def tv_part_rust(name, value, config, indent=3):
|
|
|
|
if 'rust_fmt' in config:
|
|
|
|
value = config['rust_fmt'](value)
|
2022-02-12 12:46:02 -08:00
|
|
|
elif config['rust_type'].startswith('Option<') and not (value is None or isinstance(value, Some)):
|
|
|
|
value = Some(value)
|
2018-08-21 12:13:11 -07:00
|
|
|
|
2018-06-04 22:30:16 -07:00
|
|
|
pad = ' ' * indent
|
2018-08-21 12:13:11 -07:00
|
|
|
if config['rust_type'] == 'Option<Vec<u8>>':
|
|
|
|
tv_option_vec_bytes_rust(name, value, pad)
|
|
|
|
elif config['rust_type'] == 'Vec<u8>':
|
|
|
|
tv_vec_bytes_rust(name, value, pad)
|
2021-04-26 20:41:19 -07:00
|
|
|
elif config['rust_type'] == 'Vec<bool>':
|
|
|
|
tv_vec_bool_rust(name, value, pad)
|
2022-09-19 12:59:00 -07:00
|
|
|
elif config['rust_type'] == '&\'static str':
|
|
|
|
tv_str_rust(name, value, pad)
|
2018-08-21 12:13:11 -07:00
|
|
|
elif config['rust_type'].startswith('Option<['):
|
2018-07-21 06:10:28 -07:00
|
|
|
tv_option_bytes_rust(name, value, pad)
|
2018-08-21 12:13:11 -07:00
|
|
|
elif type(value) == bytes:
|
|
|
|
tv_bytes_rust(name, value, pad)
|
|
|
|
elif config['rust_type'].startswith('Option<'):
|
|
|
|
tv_option_int_rust(name, value, pad)
|
2018-06-04 22:30:16 -07:00
|
|
|
elif type(value) == int:
|
|
|
|
tv_int_rust(name, value, pad)
|
2021-05-06 21:27:08 -07:00
|
|
|
elif type(value) == list:
|
2022-01-25 13:57:29 -08:00
|
|
|
print('''%s%s: %s[''' % (
|
2021-05-06 21:27:08 -07:00
|
|
|
pad,
|
|
|
|
name,
|
2022-01-25 13:57:29 -08:00
|
|
|
'vec!' if config['rust_type'].startswith('Vec<') else '',
|
2021-05-06 21:27:08 -07:00
|
|
|
))
|
|
|
|
for item in value:
|
2022-01-05 13:03:56 -08:00
|
|
|
if 'Vec<u8>' in config['rust_type']:
|
|
|
|
print('''%svec![
|
|
|
|
%s%s
|
|
|
|
%s],''' % (
|
|
|
|
' ' * (indent + 1),
|
|
|
|
' ' * (indent + 1),
|
|
|
|
chunk(hexlify(item)),
|
|
|
|
' ' * (indent + 1),
|
|
|
|
))
|
|
|
|
elif type(item) == bytes:
|
2021-05-06 21:27:08 -07:00
|
|
|
print('''%s[%s],''' % (
|
|
|
|
' ' * (indent + 1),
|
|
|
|
chunk(hexlify(item)),
|
|
|
|
))
|
2022-01-05 13:03:56 -08:00
|
|
|
elif type(item) == int:
|
|
|
|
print('%s%d,' % (' ' * (indent + 1), item))
|
2021-09-01 08:01:54 -07:00
|
|
|
elif type(item) == list:
|
|
|
|
print('''%s[''' % (
|
|
|
|
' ' * (indent + 1)
|
|
|
|
))
|
|
|
|
for subitem in item:
|
|
|
|
if type(subitem) == bytes:
|
|
|
|
print('''%s[%s],''' % (
|
|
|
|
' ' * (indent + 2),
|
|
|
|
chunk(hexlify(subitem)),
|
|
|
|
))
|
|
|
|
else:
|
|
|
|
raise ValueError('Invalid sublist type(%s): %s' % (name, type(subitem)))
|
|
|
|
print('''%s],''' % (
|
|
|
|
' ' * (indent + 1)
|
|
|
|
))
|
|
|
|
else:
|
|
|
|
raise ValueError('Invalid list type(%s): %s' % (name, type(item)))
|
2021-05-06 21:27:08 -07:00
|
|
|
print('''%s],''' % (
|
|
|
|
pad,
|
|
|
|
))
|
2018-06-04 22:30:16 -07:00
|
|
|
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-08-21 12:13:11 -07:00
|
|
|
for p in parts: print(' %s: %s,' % (p[0], p[1]['rust_type']))
|
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-08-21 12:13:11 -07:00
|
|
|
for p in parts: tv_part_rust(p[0], vectors[p[0]], p[1])
|
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-08-21 12:13:11 -07:00
|
|
|
for p in parts: tv_part_rust(p[0], vector[p[0]], p[1], 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):
|
2018-06-15 16:00:11 -07:00
|
|
|
# Convert older format
|
2018-08-21 12:13:11 -07:00
|
|
|
parts = [(p[0], p[1] if type(p[1]) == type({}) else {'rust_type': p[1]}) for p in parts]
|
2018-06-15 16:00:11 -07:00
|
|
|
|
2018-06-05 01:36:37 -07:00
|
|
|
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)
|