#!/bin/bash # Tested on Linux, MSys2 REFDIR='test/ref' if uname -a | grep -q MSYS; then SUDO='' MSYS2=1; else SUDO='sudo' MSYS2='' fi RED="\e[31;1m" GREEN="\e[32;1m" YELLOW="\e[33;1m" RESET="\e[0m" trap 'echo -e "${GREEN}Exiting at user request$RESET"; exit' INT umask 0022 export MMGEN_TEST_SUITE=1 export MMGEN_NO_LICENSE=1 export PYTHONPATH=. test_py='test/test.py -n' objtest_py='test/objtest.py' unit_tests_py='test/unit_tests.py --names --quiet' tooltest_py='test/tooltest.py' tooltest2_py='test/tooltest2.py --names' gentest_py='test/gentest.py' scrambletest_py='test/scrambletest.py' mmgen_tool='cmds/mmgen-tool' mmgen_keygen='cmds/mmgen-keygen' python='python3' rounds=100 rounds_min=20 rounds_mid=250 rounds_max=500 monero_addrs='3,99,2,22-24,101-104' dfl_tests='obj unit hash alts monero eth autosign btc btc_tn btc_rt bch bch_rt ltc ltc_tn ltc_rt tool tool2 gen' add_tests='autosign_minimal autosign_live' PROGNAME=$(basename $0) while getopts hbCfFiIlOpRtvV OPT do case "$OPT" in h) printf " %-16s Test MMGen release\n" "${PROGNAME}:" echo " USAGE: $PROGNAME [options] [tests]" echo " OPTIONS: '-h' Print this help message" echo " '-b' Buffer keypresses for all invocations of 'test/test.py'" echo " '-C' Run tests in coverage mode" echo " '-f' Speed up the tests by using fewer rounds" echo " '-F' Reduce rounds even further" echo " '-i' Create and install Python package, then run tests. A branch" echo " must be supplied as the first argument" echo " '-I' Install the package only; don't run tests" echo " '-l' List the test name symbols" echo " '-O' Use pexpect.spawn rather than popen_spawn for applicable tests" echo " '-p' Pause between tests" echo " '-R' Don't remove temporary files after program has exited" echo " '-t' Print the tests without running them" echo " '-v' Run test/test.py with '--exact-output' and other commands with" echo " '--verbose' switch" echo " '-V' Run test/test.py and other commands with '--verbose' switch" echo " AVAILABLE TESTS:" echo " obj - data objects" echo " unit - unit tests" echo " hash - internal hash function implementations" echo " alts - operations for all supported gen-only altcoins" echo " monero - operations for Monero" echo " eth - operations for Ethereum" echo " autosign - autosign" echo " btc - bitcoin" echo " btc_tn - bitcoin testnet" echo " btc_rt - bitcoin regtest" echo " bch - bitcoin cash (BCH)" echo " bch_rt - bitcoin cash (BCH) regtest" echo " ltc - litecoin" echo " ltc_tn - litecoin testnet" echo " ltc_rt - litecoin regtest" echo " tool - tooltest (all supported coins)" echo " tool2 - tooltest2 (all supported coins)" echo " gen - gentest (all supported coins)" echo " By default, all tests are run" exit ;; b) test_py+=" --buf-keypress" ;; C) mkdir -p 'test/trace' touch 'test/trace.acc' test_py+=" --coverage" tooltest_py+=" --coverage" tooltest2_py+=" --fork --coverage" scrambletest_py+=" --coverage" python="python3 -m trace --count --file=test/trace.acc --coverdir=test/trace" unit_tests_py="$python $unit_tests_py" objtest_py="$python $objtest_py" gentest_py="$python $gentest_py" mmgen_tool="$python $mmgen_tool" mmgen_keygen="$python $mmgen_keygen" ;& f) FAST=1 rounds=10 rounds_min=3 rounds_mid=25 rounds_max=50 monero_addrs='3,23' unit_tests_py+=" --fast" ;; F) FAST=1 rounds=2 rounds_min=1 rounds_mid=3 rounds_max=5 monero_addrs='3,23' unit_tests_py+=" --fast" ;; i) INSTALL=1 ;; I) INSTALL_ONLY=1 ;; l) echo -e "Default tests:\n $dfl_tests" echo -e "Additional tests:\n $add_tests" exit ;; O) test_py+=" --pexpect-spawn" ;; p) PAUSE=1 ;; R) NO_TMPFILE_REMOVAL=1 ;; t) TESTING=1 ;; v) EXACT_OUTPUT=1 test_py+=" --exact-output" ;& V) VERBOSE=1 [ "$EXACT_OUTPUT" ] || test_py+=" --verbose" tooltest_py+=" --verbose" tooltest2_py+=" --verbose" gentest_py+=" --verbose" mmgen_tool+=" --verbose" unit_tests_py="${unit_tests_py/--quiet/--verbose}" scrambletest_py+=" --verbose" ;; *) exit ;; esac done [ "$MSYS2" -a ! "$FAST" ] && tooltest2_py+=' --fork' [ "$EXACT_OUTPUT" -o "$VERBOSE" ] || objtest_py+=" -S" shift $((OPTIND-1)) [ "$INSTALL" ] && { BRANCH=$1; shift BRANCHES=$(git branch) FOUND_BRANCH=$(for b in ${BRANCHES/\*}; do [ "$b" == "$BRANCH" ] && echo ok; done) [ "$FOUND_BRANCH" ] || { echo "Branch '$BRANCH' not found!"; exit; } } set -e check() { [ "$BRANCH" ] || { echo 'No branch specified. Exiting'; exit; } [ "$(git diff $BRANCH)" == "" ] || { echo "Unmerged changes from branch '$BRANCH'. Exiting" exit } git diff $BRANCH >/dev/null 2>&1 || exit } uninstall() { set +e eval "$SUDO ./scripts/uninstall-mmgen.py" [ "$?" -ne 0 ] && { echo 'Uninstall failed, but proceeding anyway'; sleep 1; } set -e } install() { set -x eval "$SUDO rm -rf .test-release" git clone --branch $BRANCH --single-branch . .test-release ( cd .test-release ./setup.py sdist mkdir pydist && cd pydist if [ "$MSYS2" ]; then unzip ../dist/mmgen-*.zip; else tar zxvf ../dist/mmgen-*gz; fi cd mmgen-* eval "$SUDO ./setup.py clean --all" [ "$MSYS2" ] && ./setup.py build --compiler=mingw32 eval "$SUDO ./setup.py install --force" ) set +x } do_test() { set +x tests=$(eval echo \"'$'"t_$1"\") skips=$(eval echo \"'$'"t_$1_skip"\") declare -a tests_arr n=0 while read test; do tests_arr[n]="$test" let n+=1 done <<-EOF $tests EOF n=0 for test in "${tests_arr[@]}"; do [ -z "$test" -o "${test:0:1}" == '#' ] && continue let n+=1 echo $skips | grep -q "\<$n\>" && continue echo -e "${GREEN}Running:$RESET $YELLOW$test$RESET" # continue [ "$TESTING" ] || eval "$test" || { echo -e $RED"Test '$CUR_TEST' failed at command '$test'"$RESET exit } done } i_obj='Data object' s_obj='Testing data objects' t_obj=" $objtest_py --coin=btc $objtest_py --coin=btc --testnet=1 $objtest_py --coin=ltc $objtest_py --coin=ltc --testnet=1 " f_obj='Data object test complete' i_unit='Unit' s_unit='Running unit' t_unit="$unit_tests_py" f_unit='Unit tests run complete' i_hash='Internal hash function implementations' s_hash='Testing internal hash function implementations' t_hash=" $python test/hashfunc.py sha256 $rounds_max $python test/hashfunc.py sha512 $rounds_max $python test/hashfunc.py keccak $rounds_max " f_hash='Hash function tests complete' [ "$MSYS2" ] && t_hash_skip='2' # gmp issues i_alts='Gen-only altcoin' s_alts='The following tests will test generation operations for all supported altcoins' t_alts=" $scrambletest_py $test_py ref_altcoin # generated addrfiles verified against checksums $gentest_py --all 2:keyconv $rounds_mid # speed tests, no verification $gentest_py --coin=btc 2 $rounds $gentest_py --coin=btc --type=compressed 2 $rounds $gentest_py --coin=btc --type=segwit 2 $rounds $gentest_py --coin=btc --type=bech32 2 $rounds $gentest_py --coin=ltc 2 $rounds $gentest_py --coin=ltc --type=compressed 2 $rounds $gentest_py --coin=ltc --type=segwit 2 $rounds $gentest_py --coin=ltc --type=bech32 2 $rounds $gentest_py --coin=etc 2 $rounds $gentest_py --coin=etc --use-internal-keccak-module 2 $rounds_min $gentest_py --coin=eth 2 $rounds $gentest_py --coin=eth --use-internal-keccak-module 2 $rounds_min $gentest_py --coin=xmr 2 $rounds $gentest_py --coin=xmr --use-internal-keccak-module 2 $rounds_min $gentest_py --coin=zec 2 $rounds $gentest_py --coin=zec --type=zcash_z 2 $rounds_mid " # disabled, pycoin generates old-style LTC Segwit addrs: # $gentest_py --coin=ltc --type=segwit 2:ext $rounds # disabled, no pyethereum: # $gentest_py --coin=etc 2:ext $rounds # $gentest_py --coin=eth 2:ext $rounds # $gentest_py --all 2:pyethereum $rounds [ "$MSYS2" ] || { # no pycoin, zcash-mini t_alts="$t_alts $gentest_py --coin=zec --type=zcash_z 2:ext $rounds_mid $gentest_py --all 2:zcash_mini $rounds_mid $gentest_py --all 2:pycoin $rounds $gentest_py --coin=btc 2:ext $rounds $gentest_py --coin=btc --type=compressed 2:ext $rounds $gentest_py --coin=btc --type=segwit 2:ext $rounds $gentest_py --coin=ltc 2:ext $rounds $gentest_py --coin=ltc --type=compressed 2:ext $rounds $gentest_py --coin=zec 2:ext $rounds " } f_alts='Gen-only altcoin tests completed' [ "$NO_TMPFILE_REMOVAL" ] || rm -rf /tmp/mmgen-test-release* if [ "$MSYS2" ]; then TMPDIR='/tmp/mmgen-test-release' else TMPDIR='/tmp/mmgen-test-release-'$(cat /dev/urandom | base32 - | head -n1 | cut -b 1-16) fi mkdir -p $TMPDIR i_monero='Monero' s_monero='Testing key-address file generation and wallet creation and sync operations for Monero' s_monero='The monerod (mainnet) daemon must be running for the following tests' t_monero=" mmgen-walletgen -q -r0 -p1 -Llabel --outdir $TMPDIR -o words $mmgen_keygen -q --accept-defaults --use-internal-keccak-module --outdir $TMPDIR --coin=xmr $TMPDIR/*.mmwords $monero_addrs cs1=\$(mmgen-tool -q --accept-defaults --coin=xmr keyaddrfile_chksum $TMPDIR/*-XMR*.akeys) $mmgen_keygen -q --use-old-ed25519 --accept-defaults --outdir $TMPDIR --coin=xmr $TMPDIR/*.mmwords $monero_addrs cs2=\$(mmgen-tool -q --accept-defaults --coin=xmr keyaddrfile_chksum $TMPDIR/*-XMR*.akeys) [ \"\$cs1\" == \"\$cs2\" ] " f_monero='Monero tests completed' [ "$MSYS2" ] || { # password file descriptor issues, cannot use popen_spawn() t_monero="$t_monero $mmgen_tool -q --accept-defaults --outdir $TMPDIR keyaddrlist2monerowallets $TMPDIR/*-XMR*.akeys addrs=23 $mmgen_tool -q --accept-defaults --outdir $TMPDIR keyaddrlist2monerowallets $TMPDIR/*-XMR*.akeys addrs=103-200 rm $TMPDIR/*-MoneroWallet* $mmgen_tool -q --accept-defaults --outdir $TMPDIR keyaddrlist2monerowallets $TMPDIR/*-XMR*.akeys $mmgen_tool -q --accept-defaults --outdir $TMPDIR syncmonerowallets $TMPDIR/*-XMR*.akeys addrs=3 $mmgen_tool -q --accept-defaults --outdir $TMPDIR syncmonerowallets $TMPDIR/*-XMR*.akeys addrs=23-29 $mmgen_tool -q --accept-defaults --outdir $TMPDIR syncmonerowallets $TMPDIR/*-XMR*.akeys " } [ "$monero_addrs" == '3,23' ] && t_monero_skip='4 8 13' i_eth='Ethereum' s_eth='Testing transaction and tracking wallet operations for Ethereum and Ethereum Classic' t_eth=" $test_py --coin=eth ethdev $test_py --coin=etc ethdev " f_eth='Ethereum tests completed' i_autosign='Autosign' s_autosign='The bitcoin, bitcoin-abc and litecoin mainnet and testnet daemons must be running for the following test' t_autosign="$test_py autosign" f_autosign='Autosign test complete' i_autosign_minimal='Autosign Minimal' s_autosign_minimal='The bitcoin mainnet and testnet daemons must be running for the following test' t_autosign_minimal="$test_py autosign_minimal" f_autosign_minimal='Autosign Minimal test complete' i_autosign_live='Autosign Live' s_autosign_live="The bitcoin mainnet and testnet daemons must be running for the following test\n" s_autosign_live+="${YELLOW}Mountpoint, '/etc/fstab' and removable device must be configured " s_autosign_live+="as described in 'mmgen-autosign --help'${RESET}" t_autosign_live="$test_py autosign_live" f_autosign_live='Autosign Live test complete' i_btc='Bitcoin mainnet' s_btc='The bitcoin (mainnet) daemon must both be running for the following tests' t_btc=" $test_py --exclude regtest $test_py --segwit $test_py --segwit-random $test_py --bech32 $python scripts/compute-file-chksum.py $REFDIR/*testnet.rawtx >/dev/null 2>&1 " f_btc='You may stop the bitcoin (mainnet) daemon if you wish' i_btc_tn='Bitcoin testnet' s_btc_tn='The bitcoin testnet daemon must both be running for the following tests' t_btc_tn=" $test_py --testnet=1 $test_py --testnet=1 --segwit $test_py --testnet=1 --segwit-random $test_py --testnet=1 --bech32 " f_btc_tn='You may stop the bitcoin testnet daemon if you wish' i_btc_rt='Bitcoin regtest' s_btc_rt="The following tests will test MMGen's regtest (Bob and Alice) mode" t_btc_rt="$test_py regtest" f_btc_rt='Regtest (Bob and Alice) mode tests for BTC completed' i_bch='Bitcoin cash (BCH)' s_bch='The bitcoin cash daemon (Bitcoin ABC) must both be running for the following tests' t_bch="$test_py --coin=bch --exclude regtest" f_bch='You may stop the Bitcoin ABC daemon if you wish' i_bch_rt='Bitcoin cash (BCH) regtest' s_bch_rt="The following tests will test MMGen's regtest (Bob and Alice) mode" t_bch_rt="$test_py --coin=bch regtest" f_bch_rt='Regtest (Bob and Alice) mode tests for BCH completed' i_ltc='Litecoin' s_ltc='The litecoin daemon must both be running for the following tests' t_ltc=" $test_py --coin=ltc --exclude regtest $test_py --coin=ltc --segwit $test_py --coin=ltc --segwit-random $test_py --coin=ltc --bech32 " f_ltc='You may stop the litecoin daemon if you wish' i_ltc_tn='Litecoin testnet' s_ltc_tn='The litecoin testnet daemon must both be running for the following tests' t_ltc_tn=" $test_py --coin=ltc --testnet=1 --exclude regtest $test_py --coin=ltc --testnet=1 --segwit $test_py --coin=ltc --testnet=1 --segwit-random $test_py --coin=ltc --testnet=1 --bech32 " f_ltc_tn='You may stop the litecoin testnet daemon if you wish' i_ltc_rt='Litecoin regtest' s_ltc_rt="The following tests will test MMGen's regtest (Bob and Alice) mode" t_ltc_rt="$test_py --coin=ltc regtest" f_ltc_rt='Regtest (Bob and Alice) mode tests for LTC completed' i_tool2='Tooltest2' s_tool2="The following tests will run '$tooltest2_py' for all supported coins" t_tool2=" $tooltest2_py --quiet $tooltest2_py --quiet --coin=btc $tooltest2_py --quiet --coin=btc --testnet=1 $tooltest2_py --quiet --coin=ltc $tooltest2_py --quiet --coin=ltc --testnet=1 $tooltest2_py --quiet --coin=bch $tooltest2_py --quiet --coin=bch --testnet=1 $tooltest2_py --quiet --coin=zec $tooltest2_py --quiet --coin=zec --type=zcash_z $tooltest2_py --quiet --coin=xmr $tooltest2_py --quiet --coin=dash $tooltest2_py --quiet --coin=eth $tooltest2_py --quiet --coin=eth --testnet=1 $tooltest2_py --quiet --coin=eth --token=mm1 $tooltest2_py --quiet --coin=eth --token=mm1 --testnet=1 $tooltest2_py --quiet --coin=etc " f_tool2='tooltest2 tests completed' i_tool='Tooltest' s_tool="The following tests will run '$tooltest_py' for all supported coins" t_tool=" $tooltest_py --coin=btc cryptocoin $tooltest_py --coin=btc mnemonic $tooltest_py --coin=ltc cryptocoin $tooltest_py --coin=eth cryptocoin $tooltest_py --coin=etc cryptocoin $tooltest_py --coin=dash cryptocoin $tooltest_py --coin=doge cryptocoin $tooltest_py --coin=emc cryptocoin $tooltest_py --coin=zec cryptocoin $tooltest_py --coin=zec --type=zcash_z cryptocoin " [ "$MSYS2" ] && t_tool_skip='10' f_tool='tooltest tests completed' i_gen='Gentest' s_gen="The following tests will run '$gentest_py' on mainnet and testnet for all supported coins" t_gen=" $gentest_py -q 2 $REFDIR/btcwallet.dump $gentest_py -q --type=segwit 2 $REFDIR/btcwallet-segwit.dump $gentest_py -q --type=bech32 2 $REFDIR/btcwallet-bech32.dump $gentest_py -q 1:2 $rounds $gentest_py -q --type=segwit 1:2 $rounds $gentest_py -q --type=bech32 1:2 $rounds $gentest_py -q --testnet=1 2 $REFDIR/btcwallet-testnet.dump $gentest_py -q --testnet=1 1:2 $rounds $gentest_py -q --testnet=1 --type=segwit 1:2 $rounds $gentest_py -q --coin=ltc 2 $REFDIR/litecoin/ltcwallet.dump $gentest_py -q --coin=ltc --type=segwit 2 $REFDIR/litecoin/ltcwallet-segwit.dump $gentest_py -q --coin=ltc --type=bech32 2 $REFDIR/litecoin/ltcwallet-bech32.dump $gentest_py -q --coin=ltc 1:2 $rounds $gentest_py -q --coin=ltc --type=segwit 1:2 $rounds $gentest_py -q --coin=ltc --testnet=1 2 $REFDIR/litecoin/ltcwallet-testnet.dump $gentest_py -q --coin=ltc --testnet=1 1:2 $rounds $gentest_py -q --coin=ltc --testnet=1 --type=segwit 1:2 $rounds " f_gen='gentest tests completed' [ -d .git -a -n "$INSTALL" -a -z "$TESTING" ] && { check uninstall install cd .test-release/pydist/mmgen-* } [ "$INSTALL_ONLY" ] && exit prompt_skip() { echo -n "Enter 's' to skip, or ENTER to continue: "; read -n1; echo [ "$REPLY" == 's' ] && return 0 return 1 } run_tests() { for t in $1; do eval echo -e "'\n'"\${GREEN}'###' Running $(echo \$i_$t) tests\$RESET eval echo -e $(echo \$s_$t) [ "$PAUSE" ] && prompt_skip && continue CUR_TEST=$t do_test $t eval echo -e \$GREEN$(echo \$f_$t)\$RESET done } check_args() { for i in $tests; do echo "$dfl_tests $add_tests" | grep -q "\<$i\>" || { echo "$i: unrecognized argument"; exit; } done } tests=$dfl_tests [ "$*" ] && tests="$*" check_args echo "Running tests: $tests" START=$(date +%s) run_tests "$tests" TIME=$(($(date +%s)-START)) MS=$(printf %02d:%02d $((TIME/60)) $((TIME%60))) [ "$NO_TMPFILE_REMOVAL" ] || rm -rf /tmp/mmgen-test-release-* echo -e "${GREEN}All OK. Total elapsed time: $MS$RESET"