2018-06-15 14:29:42 +00:00
|
|
|
#!/usr/bin/env python2
|
2016-09-30 11:09:45 +00:00
|
|
|
#
|
2017-07-10 21:52:47 +00:00
|
|
|
# This script reads C++ or RST source files and writes all
|
2016-09-30 11:09:45 +00:00
|
|
|
# multi-line strings into individual files.
|
|
|
|
# This can be used to extract the Solidity test cases
|
2016-10-10 20:04:11 +00:00
|
|
|
# into files for e.g. fuzz testing as
|
2016-12-06 22:21:38 +00:00
|
|
|
# scripts/isolate_tests.py test/libsolidity/*
|
2016-09-30 11:09:45 +00:00
|
|
|
|
|
|
|
import sys
|
2017-03-15 11:07:59 +00:00
|
|
|
import re
|
2017-03-22 19:19:20 +00:00
|
|
|
import os
|
|
|
|
import hashlib
|
|
|
|
from os.path import join
|
2016-12-06 22:21:38 +00:00
|
|
|
|
2017-07-10 21:52:47 +00:00
|
|
|
def extract_test_cases(path):
|
2017-03-22 19:19:20 +00:00
|
|
|
lines = open(path, 'rb').read().splitlines()
|
2016-12-06 22:21:38 +00:00
|
|
|
|
|
|
|
inside = False
|
2017-03-15 11:07:59 +00:00
|
|
|
delimiter = ''
|
2016-12-06 22:21:38 +00:00
|
|
|
tests = []
|
|
|
|
|
|
|
|
for l in lines:
|
|
|
|
if inside:
|
2017-03-15 11:07:59 +00:00
|
|
|
if l.strip().endswith(')' + delimiter + '";'):
|
2016-12-06 22:21:38 +00:00
|
|
|
inside = False
|
|
|
|
else:
|
|
|
|
tests[-1] += l + '\n'
|
|
|
|
else:
|
2017-03-15 11:07:59 +00:00
|
|
|
m = re.search(r'R"([^(]*)\($', l.strip())
|
|
|
|
if m:
|
2016-12-06 22:21:38 +00:00
|
|
|
inside = True
|
2017-03-15 11:07:59 +00:00
|
|
|
delimiter = m.group(1)
|
2016-12-06 22:21:38 +00:00
|
|
|
tests += ['']
|
|
|
|
|
|
|
|
return tests
|
|
|
|
|
2017-07-10 21:52:47 +00:00
|
|
|
# Contract sources are indented by 4 spaces.
|
|
|
|
# Look for `pragma solidity` and abort a line not indented properly.
|
|
|
|
# If the comment `// This will not compile` is above the pragma,
|
|
|
|
# the code is skipped.
|
|
|
|
def extract_docs_cases(path):
|
|
|
|
# Note: this code works, because splitlines() removes empty new lines
|
|
|
|
# and thus even if the empty new lines are missing indentation
|
|
|
|
lines = open(path, 'rb').read().splitlines()
|
|
|
|
|
|
|
|
ignore = False
|
|
|
|
inside = False
|
|
|
|
tests = []
|
|
|
|
|
|
|
|
for l in lines:
|
|
|
|
if inside:
|
|
|
|
# Abort if indentation is missing
|
|
|
|
m = re.search(r'^[^ ]+', l)
|
|
|
|
if m:
|
|
|
|
inside = False
|
|
|
|
else:
|
|
|
|
tests[-1] += l + '\n'
|
|
|
|
else:
|
|
|
|
m = re.search(r'^ // This will not compile', l)
|
|
|
|
if m:
|
|
|
|
ignore = True
|
|
|
|
|
|
|
|
if ignore:
|
|
|
|
# Abort if indentation is missing
|
|
|
|
m = re.search(r'^[^ ]+', l)
|
|
|
|
if m:
|
|
|
|
ignore = False
|
|
|
|
else:
|
|
|
|
m = re.search(r'^ pragma solidity .*[0-9]+\.[0-9]+\.[0-9]+;$', l)
|
|
|
|
if m:
|
|
|
|
inside = True
|
|
|
|
tests += [l]
|
|
|
|
|
|
|
|
return tests
|
2016-12-06 22:21:38 +00:00
|
|
|
|
2017-03-22 19:19:20 +00:00
|
|
|
def write_cases(tests):
|
|
|
|
for test in tests:
|
|
|
|
open('test_%s.sol' % hashlib.sha256(test).hexdigest(), 'wb').write(test)
|
2016-12-06 22:21:38 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2017-03-22 19:19:20 +00:00
|
|
|
path = sys.argv[1]
|
2017-07-10 21:52:47 +00:00
|
|
|
docs = False
|
|
|
|
if len(sys.argv) > 2 and sys.argv[2] == 'docs':
|
|
|
|
docs = True
|
2016-12-06 22:21:38 +00:00
|
|
|
|
2017-07-12 17:06:04 +00:00
|
|
|
for root, subdirs, files in os.walk(path):
|
|
|
|
if '_build' in subdirs:
|
|
|
|
subdirs.remove('_build')
|
2018-03-06 19:45:34 +00:00
|
|
|
if 'compilationTests' in subdirs:
|
|
|
|
subdirs.remove('compilationTests')
|
2017-03-22 19:19:20 +00:00
|
|
|
for f in files:
|
2017-07-10 21:52:47 +00:00
|
|
|
path = join(root, f)
|
|
|
|
if docs:
|
|
|
|
cases = extract_docs_cases(path)
|
|
|
|
else:
|
2018-03-06 19:45:34 +00:00
|
|
|
if f.endswith(".sol"):
|
|
|
|
cases = [open(path, "r").read()]
|
|
|
|
else:
|
|
|
|
cases = extract_test_cases(path)
|
2017-03-22 19:19:20 +00:00
|
|
|
write_cases(cases)
|