Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

dapp mutate #829

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 20 additions & 0 deletions overlay.nix
Original file line number Diff line number Diff line change
Expand Up @@ -159,4 +159,24 @@ in rec {
secp256k1 = super.secp256k1.overrideDerivation (_: {
dontDisableStatic = true;
});

universalmutator = with super.python3Packages; buildPythonPackage rec {
pname = "universalmutator";
version = "1.0.12";

format = "wheel";

propagatedBuildInputs = [ python-Levenshtein setuptools ];

src = fetchPypi {
inherit pname version format;
python = "py3";
sha256 = "sha256-HHJuoqlKPV3aus+fZE+pAzeoiTNPrXO+Put5TyCmAOc=";
};

meta = with lib; {
maintainers = with maintainers; [ xwvvvvwx ];
description = "regex based mutation testing tool";
};
};
}
5 changes: 3 additions & 2 deletions src/dapp/default.nix
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
{ lib, stdenv, fetchFromGitHub, makeWrapper, glibcLocales
, coreutils, git, gnused, gnumake, hevm, jshon, jq, nix
, nodejs, perl, python3, seth, shellcheck, solc, tre, dapptoolsSrc }:
, nodejs, perl, python3, seth, shellcheck, solc, tre
, dapptoolsSrc, parallel, universalmutator }:

stdenv.mkDerivation rec {
name = "dapp-${version}";
Expand All @@ -16,7 +17,7 @@ stdenv.mkDerivation rec {
postInstall =
let
path = lib.makeBinPath [
coreutils git gnused gnumake hevm jshon jq nix nodejs perl seth solc tre python3
parallel coreutils git gnused gnumake hevm jshon jq nix nodejs perl seth solc tre python3 universalmutator
];
in
''
Expand Down
17 changes: 15 additions & 2 deletions src/dapp/libexec/dapp/dapp
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@
### cache=<string> use the cache at directory
### ffi allow the use of the ffi cheatcode (WARNING: allows test authors to execute arbitrary code on your machine)
### coverage print coverage data

###
### RPC options:
### rpc fetch remote state via ETH_RPC_URL
### rpc-url=<string> fetch remote state via <url>
Expand All @@ -39,7 +39,7 @@
###
### Contract verifying options:
### async don't wait for confirmation

###
### Dapp testnet options:
### rpc-port=port change RPC port (default: 8545)
### rpc-addr=address change RPC address (default: 127.0.0.1)
Expand All @@ -49,6 +49,11 @@
### save=name after finishing, save snapshot
### load=name start from a previously saved snapshot
### dir=directory testnet directory
###
### Mutate Options:
### mutants-dir=<string> the directory in which mutants should be stored
### test-command=<string> the command to use for test execution when filtering mutants
### timeout=<number> timeout to use when filtering mutants


OPTS="dapp [<options>] <command> [<args>]
Expand Down Expand Up @@ -97,6 +102,11 @@ accounts=number create multiple accounts (default: 1)
save=name after finishing, save snapshot
load=name start from a previously saved snapshot
dir=directory testnet directory

Mutate Options:
mutants-dir=<string> the directory in which mutants should be stored
test-command=<string> the command to use for test execution when filtering mutants
timeout=<number> timeout to use when filtering mutants
"

set -e
Expand Down Expand Up @@ -177,6 +187,9 @@ while [[ $1 ]]; do
--load) shift; export DAPP_TESTNET_LOAD=$1;;
--dir) shift; export DAPP_TESTNET_gethdir=$1;;

--mutants-dir) shift; export DAPP_MUTANTS_DIR=$1;;
--test-command) shift; export DAPP_MUTANTS_TEST_COMMAND=$1;;
--timoeut) shift; export DAPP_MUTANTS_TIMEOUT=$1;;

*) printf "${0##*/}: internal error: %q\\n" "$1"; exit 1
esac; shift
Expand Down
95 changes: 95 additions & 0 deletions src/dapp/libexec/dapp/dapp-mutate
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
#!/usr/bin/env bash
### dapp mutate -- mutation testing framework
### Usage: dapp mutate <cmd> [<options>]
###
### dapp mutate provides utilities for mutation testing. Four subcommands are available:
###
### gen: apply mutations and store them in --mutants-dir
### filter: run the project test suite against each mutant, keep the ones where the test suite passes
### show-diffs: display the mutatations that were not detected by the test suite
### status: display the current status of the mutation campaign
###
### To begin a mutation testing campaign, run dapp mutate gen, this will
### generate mutated versions of all non test source files in your project and
### place them in ./mutants.
### Once you have some mutants you can run dapp mutate filter to check if the
### mutations are detected by the test suite. You can run dapp mutate show-diffs
### to display the undetected mutations and dapp mutate status to display the current
### progress of the mutation campgaign.
###
### Options:
### -m, --match <string> only operate on files matching the given regex
### --timeout <number> maximum time given for the test command when filtering
### --test-command <string> the command to use for test execution when filtering mutants
### --mutants-dir <string> the directory in which mutants should be stored

set -euo pipefail

export DAPP_TEST_MATCH=${DAPP_TEST_MATCH:-".+[^\.][^t]\.sol$"}
export DAPP_MUTANTS_DIR=${DAPP_MUTANTS_DIR:-mutants}
export DAPP_MUTANTS_TEST_COMMAND=${DAPP_MUTANTS_TEST_COMMAND:-"dapp test"}
export DAPP_MUTANTS_TIMEOUT=${DAPP_MUTANTS_TIMEOUT:-60}
export DAPP_SOLC=${DAPP_SOLC:-"solc"}

mkdir -p "./$DAPP_MUTANTS_DIR"

[[ -z $1 ]] && dapp help mutate && exit 0

export root
root=$(pwd)

case "$1" in
"gen")
# shellcheck disable=SC2153
files=$(find "$root/$DAPP_SRC" -type f -regex "$DAPP_TEST_MATCH")

# we run in a tmpdir since universalmutator spams tmp files in the cwd...
tmp=$(mktemp -d)
cd "$tmp" || exit
# shellcheck disable=SC2064
trap "rm -rf $tmp; exit 1" EXIT

for f in $files; do
mutate \
--mutantDir "$root/$DAPP_MUTANTS_DIR" \
--cmd "eval $DAPP_SOLC $(dapp remappings) MUTANT --asm --optimize --allow-paths ." \
"$f"
done
;;

"filter")
files=$(find "$root/$DAPP_SRC" -type f -regex "$DAPP_TEST_MATCH")

for f in $files; do
analyze_mutants "$f" "$DAPP_MUTANTS_TEST_COMMAND" \
--mutantDir "./$DAPP_MUTANTS_DIR" \
--timeout "$DAPP_MUTANTS_TIMEOUT" \
--prefix mutants \
--show \
--resume
done
;;

"status")
total=$(find "$DAPP_MUTANTS_DIR" -type f | wc -l)
killed=$(wc -l < mutants.killed.txt)
notkilled=$(wc -l < mutants.notkilled.txt)
tested=$(bc <<< "$killed + $notkilled")

echo "$total mutants generated, $tested were tested"
echo "$killed mutations were detected by the test suite"
echo "$notkilled mutations were undetected by the test suite"
;;

"show-diffs")
mapfile -t notkilled < <(cat mutants.notkilled.txt)
for f in "${notkilled[@]}"; do
echo
src="$DAPP_SRC/$(echo "$f" | sed -e 's|\.mutant\..*\.sol$|\.sol|g')"
echo "$f -> $src"
diff --color "$DAPP_MUTANTS_DIR/$f" "$src" || :
done
;;

*) printf "${0##*/}: unrecognised subcommand: %q\\n" "$1"; exit 1
esac