1*27162e4eSAndroid Build Coastguard Worker#!/usr/bin/env python3 2*27162e4eSAndroid Build Coastguard Worker"""Test LZ4 interoperability between versions""" 3*27162e4eSAndroid Build Coastguard Worker 4*27162e4eSAndroid Build Coastguard Worker# 5*27162e4eSAndroid Build Coastguard Worker# Copyright (C) 2011-present, Takayuki Matsuoka 6*27162e4eSAndroid Build Coastguard Worker# All rights reserved. 7*27162e4eSAndroid Build Coastguard Worker# GPL v2 License 8*27162e4eSAndroid Build Coastguard Worker# 9*27162e4eSAndroid Build Coastguard Worker 10*27162e4eSAndroid Build Coastguard Workerimport argparse 11*27162e4eSAndroid Build Coastguard Workerimport glob 12*27162e4eSAndroid Build Coastguard Workerimport subprocess 13*27162e4eSAndroid Build Coastguard Workerimport filecmp 14*27162e4eSAndroid Build Coastguard Workerimport os 15*27162e4eSAndroid Build Coastguard Workerimport shutil 16*27162e4eSAndroid Build Coastguard Workerimport sys 17*27162e4eSAndroid Build Coastguard Workerimport hashlib 18*27162e4eSAndroid Build Coastguard Worker 19*27162e4eSAndroid Build Coastguard Workerrepo_url = 'https://github.com/lz4/lz4.git' 20*27162e4eSAndroid Build Coastguard Workertmp_dir_name = 'tests/versionsTest' 21*27162e4eSAndroid Build Coastguard Workermake_cmd = 'make' 22*27162e4eSAndroid Build Coastguard Workergit_cmd = 'git' 23*27162e4eSAndroid Build Coastguard Workertest_dat_src = 'README.md' 24*27162e4eSAndroid Build Coastguard Workertest_dat = 'test_dat' 25*27162e4eSAndroid Build Coastguard Workerhead = 'v999' 26*27162e4eSAndroid Build Coastguard Worker 27*27162e4eSAndroid Build Coastguard Workerparser = argparse.ArgumentParser() 28*27162e4eSAndroid Build Coastguard Workerparser.add_argument("--verbose", action="store_true", help="increase output verbosity") 29*27162e4eSAndroid Build Coastguard Workerargs = parser.parse_args() 30*27162e4eSAndroid Build Coastguard Worker 31*27162e4eSAndroid Build Coastguard Workerdef debug_message(msg): 32*27162e4eSAndroid Build Coastguard Worker if args.verbose: 33*27162e4eSAndroid Build Coastguard Worker print(msg) 34*27162e4eSAndroid Build Coastguard Worker 35*27162e4eSAndroid Build Coastguard Workerdef env_or_empty(env, key): 36*27162e4eSAndroid Build Coastguard Worker if key in env: 37*27162e4eSAndroid Build Coastguard Worker return " " + env[key] 38*27162e4eSAndroid Build Coastguard Worker return "" 39*27162e4eSAndroid Build Coastguard Worker 40*27162e4eSAndroid Build Coastguard Workerdef proc(cmd_args, pipe=True, env=False): 41*27162e4eSAndroid Build Coastguard Worker if env == False: 42*27162e4eSAndroid Build Coastguard Worker env = os.environ.copy() 43*27162e4eSAndroid Build Coastguard Worker debug_message("Executing command {} with env {}".format(cmd_args, env)) 44*27162e4eSAndroid Build Coastguard Worker if pipe: 45*27162e4eSAndroid Build Coastguard Worker s = subprocess.Popen(cmd_args, 46*27162e4eSAndroid Build Coastguard Worker stdout=subprocess.PIPE, 47*27162e4eSAndroid Build Coastguard Worker stderr=subprocess.PIPE, 48*27162e4eSAndroid Build Coastguard Worker env = env) 49*27162e4eSAndroid Build Coastguard Worker else: 50*27162e4eSAndroid Build Coastguard Worker s = subprocess.Popen(cmd_args, env = env) 51*27162e4eSAndroid Build Coastguard Worker stdout_data, stderr_data = s.communicate() 52*27162e4eSAndroid Build Coastguard Worker if s.poll() != 0: 53*27162e4eSAndroid Build Coastguard Worker print('Error Code:', s.poll()) 54*27162e4eSAndroid Build Coastguard Worker print('Standard Error:', stderr_data.decode()) 55*27162e4eSAndroid Build Coastguard Worker sys.exit(1) 56*27162e4eSAndroid Build Coastguard Worker return stdout_data, stderr_data 57*27162e4eSAndroid Build Coastguard Worker 58*27162e4eSAndroid Build Coastguard Workerdef make(args, pipe=True, env=False): 59*27162e4eSAndroid Build Coastguard Worker if env == False: 60*27162e4eSAndroid Build Coastguard Worker env = os.environ.copy() 61*27162e4eSAndroid Build Coastguard Worker # favor compilation speed for faster total test time, actual runtime is very short 62*27162e4eSAndroid Build Coastguard Worker env["CFLAGS"] = env_or_empty(env, 'CFLAGS') + " -O0" 63*27162e4eSAndroid Build Coastguard Worker # old versions of lz4 may require MOREFLAGS 64*27162e4eSAndroid Build Coastguard Worker env["MOREFLAGS"] = env_or_empty(env, 'MOREFLAGS') + env_or_empty(env, 'CFLAGS') + env_or_empty(env, 'CPPFLAGS') + env_or_empty(env, 'LDFLAGS') 65*27162e4eSAndroid Build Coastguard Worker return proc([make_cmd] + args, pipe, env) 66*27162e4eSAndroid Build Coastguard Worker 67*27162e4eSAndroid Build Coastguard Workerdef git(args, pipe=True): 68*27162e4eSAndroid Build Coastguard Worker return proc([git_cmd] + args, pipe) 69*27162e4eSAndroid Build Coastguard Worker 70*27162e4eSAndroid Build Coastguard Workerdef get_git_tags(): 71*27162e4eSAndroid Build Coastguard Worker stdout, stderr = git(['tag', '-l', 'r[0-9][0-9][0-9]']) 72*27162e4eSAndroid Build Coastguard Worker tags = stdout.decode('utf-8').split() 73*27162e4eSAndroid Build Coastguard Worker stdout, stderr = git(['tag', '-l', 'v[1-9].[0-9].[0-9]']) 74*27162e4eSAndroid Build Coastguard Worker tags += stdout.decode('utf-8').split() 75*27162e4eSAndroid Build Coastguard Worker return tags 76*27162e4eSAndroid Build Coastguard Worker 77*27162e4eSAndroid Build Coastguard Worker# https://stackoverflow.com/a/19711609/2132223 78*27162e4eSAndroid Build Coastguard Workerdef sha1_of_file(filepath): 79*27162e4eSAndroid Build Coastguard Worker with open(filepath, 'rb') as f: 80*27162e4eSAndroid Build Coastguard Worker return hashlib.sha1(f.read()).hexdigest() 81*27162e4eSAndroid Build Coastguard Worker 82*27162e4eSAndroid Build Coastguard Workerif __name__ == '__main__': 83*27162e4eSAndroid Build Coastguard Worker error_code = 0 84*27162e4eSAndroid Build Coastguard Worker base_dir = os.getcwd() + '/..' # /path/to/lz4 85*27162e4eSAndroid Build Coastguard Worker tmp_dir = base_dir + '/' + tmp_dir_name # /path/to/lz4/tests/versionsTest 86*27162e4eSAndroid Build Coastguard Worker clone_dir = tmp_dir + '/' + 'lz4' # /path/to/lz4/tests/versionsTest/lz4 87*27162e4eSAndroid Build Coastguard Worker programs_dir = base_dir + '/programs' # /path/to/lz4/programs 88*27162e4eSAndroid Build Coastguard Worker os.makedirs(tmp_dir, exist_ok=True) 89*27162e4eSAndroid Build Coastguard Worker 90*27162e4eSAndroid Build Coastguard Worker # since Travis clones limited depth, we should clone full repository 91*27162e4eSAndroid Build Coastguard Worker if not os.path.isdir(clone_dir): 92*27162e4eSAndroid Build Coastguard Worker git(['clone', repo_url, clone_dir]) 93*27162e4eSAndroid Build Coastguard Worker 94*27162e4eSAndroid Build Coastguard Worker shutil.copy2(base_dir + '/' + test_dat_src, tmp_dir + '/' + test_dat) 95*27162e4eSAndroid Build Coastguard Worker 96*27162e4eSAndroid Build Coastguard Worker # Retrieve all release tags 97*27162e4eSAndroid Build Coastguard Worker print('Retrieve all release tags :') 98*27162e4eSAndroid Build Coastguard Worker os.chdir(clone_dir) 99*27162e4eSAndroid Build Coastguard Worker tags = [head] + get_git_tags() 100*27162e4eSAndroid Build Coastguard Worker print(tags); 101*27162e4eSAndroid Build Coastguard Worker 102*27162e4eSAndroid Build Coastguard Worker # Build all release lz4c and lz4c32 103*27162e4eSAndroid Build Coastguard Worker for tag in tags: 104*27162e4eSAndroid Build Coastguard Worker print("processing tag " + tag) 105*27162e4eSAndroid Build Coastguard Worker os.chdir(base_dir) 106*27162e4eSAndroid Build Coastguard Worker dst_lz4c = '{}/lz4c.{}' .format(tmp_dir, tag) # /path/to/lz4/test/lz4test/lz4c.<TAG> 107*27162e4eSAndroid Build Coastguard Worker dst_lz4c32 = '{}/lz4c32.{}'.format(tmp_dir, tag) # /path/to/lz4/test/lz4test/lz4c32.<TAG> 108*27162e4eSAndroid Build Coastguard Worker if not os.path.isfile(dst_lz4c) or not os.path.isfile(dst_lz4c32) or tag == head: 109*27162e4eSAndroid Build Coastguard Worker if tag != head: 110*27162e4eSAndroid Build Coastguard Worker r_dir = '{}/{}'.format(tmp_dir, tag) # /path/to/lz4/test/lz4test/<TAG> 111*27162e4eSAndroid Build Coastguard Worker os.makedirs(r_dir, exist_ok=True) 112*27162e4eSAndroid Build Coastguard Worker os.chdir(clone_dir) 113*27162e4eSAndroid Build Coastguard Worker git(['--work-tree=' + r_dir, 'checkout', tag, '--', '.'], False) 114*27162e4eSAndroid Build Coastguard Worker os.chdir(r_dir + '/programs') # /path/to/lz4/lz4test/<TAG>/programs 115*27162e4eSAndroid Build Coastguard Worker else: 116*27162e4eSAndroid Build Coastguard Worker os.chdir(programs_dir) 117*27162e4eSAndroid Build Coastguard Worker make(['clean'], False) 118*27162e4eSAndroid Build Coastguard Worker make(['lz4c'], False) 119*27162e4eSAndroid Build Coastguard Worker shutil.copy2('lz4c', dst_lz4c) 120*27162e4eSAndroid Build Coastguard Worker make(['clean'], False) 121*27162e4eSAndroid Build Coastguard Worker make(['lz4c32'], False) 122*27162e4eSAndroid Build Coastguard Worker shutil.copy2('lz4c32', dst_lz4c32) 123*27162e4eSAndroid Build Coastguard Worker 124*27162e4eSAndroid Build Coastguard Worker # Compress test.dat by all released lz4c and lz4c32 125*27162e4eSAndroid Build Coastguard Worker print('Compress test.dat by all released lz4c and lz4c32') 126*27162e4eSAndroid Build Coastguard Worker os.chdir(tmp_dir) 127*27162e4eSAndroid Build Coastguard Worker for lz4 in glob.glob("*.lz4"): 128*27162e4eSAndroid Build Coastguard Worker os.remove(lz4) 129*27162e4eSAndroid Build Coastguard Worker for tag in tags: 130*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c.' + tag, '-1fz', test_dat, test_dat + '_1_64_' + tag + '.lz4']) 131*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c.' + tag, '-9fz', test_dat, test_dat + '_9_64_' + tag + '.lz4']) 132*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c32.' + tag, '-1fz', test_dat, test_dat + '_1_32_' + tag + '.lz4']) 133*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c32.' + tag, '-9fz', test_dat, test_dat + '_9_32_' + tag + '.lz4']) 134*27162e4eSAndroid Build Coastguard Worker 135*27162e4eSAndroid Build Coastguard Worker print('Full list of compressed files') 136*27162e4eSAndroid Build Coastguard Worker lz4s = sorted(glob.glob('*.lz4')) 137*27162e4eSAndroid Build Coastguard Worker for lz4 in lz4s: 138*27162e4eSAndroid Build Coastguard Worker print(lz4 + ' : ' + repr(os.path.getsize(lz4))) 139*27162e4eSAndroid Build Coastguard Worker 140*27162e4eSAndroid Build Coastguard Worker # Remove duplicated .lz4 files 141*27162e4eSAndroid Build Coastguard Worker print('') 142*27162e4eSAndroid Build Coastguard Worker print('Duplicated files') 143*27162e4eSAndroid Build Coastguard Worker lz4s = sorted(glob.glob('*.lz4')) 144*27162e4eSAndroid Build Coastguard Worker for i, lz4 in enumerate(lz4s): 145*27162e4eSAndroid Build Coastguard Worker if not os.path.isfile(lz4): 146*27162e4eSAndroid Build Coastguard Worker continue 147*27162e4eSAndroid Build Coastguard Worker for j in range(i+1, len(lz4s)): 148*27162e4eSAndroid Build Coastguard Worker lz4t = lz4s[j] 149*27162e4eSAndroid Build Coastguard Worker if not os.path.isfile(lz4t): 150*27162e4eSAndroid Build Coastguard Worker continue 151*27162e4eSAndroid Build Coastguard Worker if filecmp.cmp(lz4, lz4t): 152*27162e4eSAndroid Build Coastguard Worker os.remove(lz4t) 153*27162e4eSAndroid Build Coastguard Worker print('{} == {}'.format(lz4, lz4t)) 154*27162e4eSAndroid Build Coastguard Worker 155*27162e4eSAndroid Build Coastguard Worker print('Enumerate only different compressed files') 156*27162e4eSAndroid Build Coastguard Worker lz4s = sorted(glob.glob('*.lz4')) 157*27162e4eSAndroid Build Coastguard Worker for lz4 in lz4s: 158*27162e4eSAndroid Build Coastguard Worker print(lz4 + ' : ' + repr(os.path.getsize(lz4)) + ', ' + sha1_of_file(lz4)) 159*27162e4eSAndroid Build Coastguard Worker 160*27162e4eSAndroid Build Coastguard Worker # Decompress remained .lz4 files by all released lz4c and lz4c32 161*27162e4eSAndroid Build Coastguard Worker print('Decompression tests and verifications') 162*27162e4eSAndroid Build Coastguard Worker lz4s = sorted(glob.glob('*.lz4')) 163*27162e4eSAndroid Build Coastguard Worker for dec in glob.glob("*.dec"): 164*27162e4eSAndroid Build Coastguard Worker os.remove(dec) 165*27162e4eSAndroid Build Coastguard Worker for lz4 in lz4s: 166*27162e4eSAndroid Build Coastguard Worker print(lz4, end=" ") 167*27162e4eSAndroid Build Coastguard Worker for tag in tags: 168*27162e4eSAndroid Build Coastguard Worker print(tag, end=" ") 169*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c.' + tag, '-df', lz4, lz4 + '_d64_' + tag + '.dec']) 170*27162e4eSAndroid Build Coastguard Worker proc(['./lz4c32.' + tag, '-df', lz4, lz4 + '_d32_' + tag + '.dec']) 171*27162e4eSAndroid Build Coastguard Worker print(' OK') # well, here, decompression has worked; but file is not yet verified 172*27162e4eSAndroid Build Coastguard Worker 173*27162e4eSAndroid Build Coastguard Worker # Compare all '.dec' files with test_dat 174*27162e4eSAndroid Build Coastguard Worker decs = glob.glob('*.dec') 175*27162e4eSAndroid Build Coastguard Worker for dec in decs: 176*27162e4eSAndroid Build Coastguard Worker if not filecmp.cmp(dec, test_dat): 177*27162e4eSAndroid Build Coastguard Worker print('ERR : ' + dec) 178*27162e4eSAndroid Build Coastguard Worker error_code = 1 179*27162e4eSAndroid Build Coastguard Worker else: 180*27162e4eSAndroid Build Coastguard Worker print('OK : ' + dec) 181*27162e4eSAndroid Build Coastguard Worker os.remove(dec) 182*27162e4eSAndroid Build Coastguard Worker 183*27162e4eSAndroid Build Coastguard Worker if error_code != 0: 184*27162e4eSAndroid Build Coastguard Worker print('ERROR') 185*27162e4eSAndroid Build Coastguard Worker 186*27162e4eSAndroid Build Coastguard Worker sys.exit(error_code) 187