2019-02-26 04:28:37 +00:00
|
|
|
"""
|
|
|
|
Checks that every .c and .h file in an implementation is present as a
|
|
|
|
dependency of that scheme's Makefile.
|
|
|
|
"""
|
|
|
|
|
2019-02-26 04:42:48 +00:00
|
|
|
import datetime
|
2019-07-29 09:38:25 +01:00
|
|
|
import glob
|
|
|
|
import os
|
2019-04-18 09:00:08 +01:00
|
|
|
|
2019-07-29 09:38:25 +01:00
|
|
|
import pytest
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-07-29 09:38:25 +01:00
|
|
|
import helpers
|
|
|
|
import pqclean
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-04-18 09:00:08 +01:00
|
|
|
|
2019-07-29 09:38:25 +01:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
'implementation,test_dir,impl_path, init, destr',
|
|
|
|
[(impl,
|
|
|
|
*helpers.isolate_test_files(impl.path(), 'test_makefile_deps_'))
|
|
|
|
for impl in pqclean.Scheme.all_implementations()],
|
|
|
|
ids=[str(impl) for impl in pqclean.Scheme.all_implementations()],
|
|
|
|
)
|
|
|
|
@helpers.filtered_test
|
|
|
|
def test_makefile_dependencies(implementation, impl_path, test_dir,
|
|
|
|
init, destr):
|
|
|
|
init()
|
|
|
|
# initial build - want to have *all* files in place at beginning
|
|
|
|
helpers.make('clean', working_dir=impl_path)
|
|
|
|
helpers.make(working_dir=impl_path)
|
|
|
|
# test case for each candidate file
|
|
|
|
cfiles = glob.glob(os.path.join(impl_path, '*.c'))
|
|
|
|
hfiles = glob.glob(os.path.join(impl_path, '*.h'))
|
|
|
|
for file in (cfiles + hfiles):
|
|
|
|
check_makefile_dependencies(implementation, impl_path, file)
|
|
|
|
destr()
|
2019-02-27 14:27:41 +00:00
|
|
|
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-03-04 15:56:05 +00:00
|
|
|
def touch(time, *files):
|
2019-03-05 14:45:09 +00:00
|
|
|
for path in files:
|
|
|
|
times = (time.timestamp(), time.timestamp())
|
|
|
|
os.utime(path, times)
|
2019-03-04 15:56:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
def make_check(path, expect_error=False):
|
|
|
|
makeflag = '-q' if os.name != 'nt' else '/Q'
|
|
|
|
expected_returncode = 0
|
|
|
|
if expect_error:
|
|
|
|
expected_returncode = 1 if os.name != 'nt' else 255
|
|
|
|
helpers.make(makeflag, working_dir=path,
|
|
|
|
expected_returncode=expected_returncode)
|
|
|
|
|
|
|
|
|
2019-07-29 09:38:25 +01:00
|
|
|
def check_makefile_dependencies(implementation, impl_path, file):
|
|
|
|
cfiles = glob.glob(os.path.join(impl_path, '*.c'))
|
|
|
|
hfiles = glob.glob(os.path.join(impl_path, '*.h'))
|
|
|
|
ofiles = glob.glob(
|
|
|
|
os.path.join(impl_path,
|
|
|
|
'*.o' if os.name != 'nt' else '*.obj'))
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-07-29 09:38:25 +01:00
|
|
|
libfile = os.path.join(impl_path, implementation.libname())
|
2019-02-26 04:28:37 +00:00
|
|
|
|
|
|
|
# modification time-based calculations is tricky on a sub-second basis
|
|
|
|
# so we reset all the modification times to a known and "sensible" order
|
2019-03-04 15:56:05 +00:00
|
|
|
now = datetime.datetime.now() - datetime.timedelta(seconds=10)
|
|
|
|
ago15 = now - datetime.timedelta(minutes=15)
|
|
|
|
ago10 = now - datetime.timedelta(minutes=10)
|
|
|
|
ago5 = now - datetime.timedelta(minutes=5)
|
|
|
|
|
|
|
|
touch(ago15, *cfiles, *hfiles)
|
|
|
|
touch(ago10, *ofiles)
|
|
|
|
touch(ago5, libfile)
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-02-27 14:29:20 +00:00
|
|
|
# Sanity check: the scheme is up to date
|
2019-07-29 09:38:25 +01:00
|
|
|
make_check(impl_path)
|
2019-02-27 14:29:20 +00:00
|
|
|
|
2019-02-26 04:28:37 +00:00
|
|
|
# touch the candidate .c / .h file
|
2019-03-04 15:56:05 +00:00
|
|
|
touch(now, file)
|
2019-02-26 04:28:37 +00:00
|
|
|
|
2019-02-27 14:27:41 +00:00
|
|
|
# check if it needs to be rebuilt using make -q
|
2019-07-29 09:38:25 +01:00
|
|
|
make_check(impl_path, expect_error=True)
|
2019-02-26 04:28:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2019-07-29 09:38:25 +01:00
|
|
|
import sys
|
|
|
|
pytest.main(sys.argv)
|