mirror of
https://github.com/waku-org/nwaku.git
synced 2025-01-28 15:46:33 +00:00
205 lines
5.6 KiB
Nim
205 lines
5.6 KiB
Nim
|
import
|
||
|
std/[sequtils, hashes, os, parseopt, strutils, algorithm],
|
||
|
fuzzing_engines
|
||
|
|
||
|
const
|
||
|
Usage = """
|
||
|
|
||
|
Usage:
|
||
|
ntu COMMAND [options] <path>
|
||
|
|
||
|
Available commands:
|
||
|
|
||
|
$ ntu test [options] <path>
|
||
|
|
||
|
Run the test(s) specified at path. Will search recursively for test files
|
||
|
provided path is a directory.
|
||
|
|
||
|
Options:
|
||
|
--backends:"c cpp js objc" Run tests for specified targets
|
||
|
--include:"test1 test2" Run only listed tests (space/comma separated)
|
||
|
--exclude:"test1 test2" Skip listed tests (space/comma separated)
|
||
|
--update Rewrite failed tests with new output
|
||
|
--sort:"source,test" Sort the tests by program and/or test mtime
|
||
|
--reverse Reverse the order of tests
|
||
|
--random Shuffle the order of tests
|
||
|
--help Display this help and exit
|
||
|
|
||
|
$ ntu fuzz [options] <module>
|
||
|
|
||
|
Start a fuzzing test with a Nim module based on testutils/fuzzing.
|
||
|
|
||
|
Options:
|
||
|
--fuzzer:libFuzzer The fuzzing engine to use.
|
||
|
Possible values: libFuzzer, honggfuzz, afl
|
||
|
--corpus:<path> A directory with initial input cases
|
||
|
|
||
|
""".unindent.strip
|
||
|
|
||
|
type
|
||
|
FlagKind* = enum
|
||
|
UpdateOutputs = "--update"
|
||
|
UseThreads = "--threads:on"
|
||
|
DebugBuild = "--define:debug"
|
||
|
ReleaseBuild = "--define:release"
|
||
|
DangerBuild = "--define:danger"
|
||
|
CpuAffinity = "--affinity"
|
||
|
|
||
|
SortBy* {.pure.} = enum
|
||
|
Random = "random"
|
||
|
Source = "source"
|
||
|
Test = "test"
|
||
|
Reverse = "reverse"
|
||
|
|
||
|
Command* = enum
|
||
|
noCommand
|
||
|
test
|
||
|
fuzz
|
||
|
|
||
|
TestConfig* = object
|
||
|
case cmd*: Command
|
||
|
of test:
|
||
|
path*: string
|
||
|
includedTests*: seq[string]
|
||
|
excludedTests*: seq[string]
|
||
|
|
||
|
flags*: set[FlagKind]
|
||
|
# options
|
||
|
backendNames*: seq[string]
|
||
|
orderBy*: set[SortBy]
|
||
|
of fuzz:
|
||
|
fuzzer*: FuzzingEngine
|
||
|
corpusDir*: string
|
||
|
target*: string
|
||
|
of noCommand:
|
||
|
discard
|
||
|
|
||
|
const
|
||
|
defaultFlags = {UseThreads}
|
||
|
compilerFlags* = {DebugBuild, ReleaseBuild, DangerBuild, UseThreads}
|
||
|
# --define:testutilsBackends="cpp js"
|
||
|
testutilsBackends* {.strdefine.} = "c"
|
||
|
defaultSort = {Source, Reverse}
|
||
|
|
||
|
proc `backends=`*(config: var TestConfig; inputs: seq[string]) =
|
||
|
config.backendNames = inputs.sorted
|
||
|
|
||
|
proc `backends=`*(config: var TestConfig; input: string) =
|
||
|
config.backends = input.split(" ")
|
||
|
|
||
|
proc backends*(config: TestConfig): seq[string] =
|
||
|
result = config.backendNames
|
||
|
|
||
|
proc hash*(config: TestConfig): Hash =
|
||
|
var h: Hash = 0
|
||
|
h = h !& config.backends.hash
|
||
|
h = h !& hash(ReleaseBuild in config.flags)
|
||
|
h = h !& hash(DangerBuild in config.flags)
|
||
|
h = h !& hash(UseThreads notin config.flags)
|
||
|
result = !$h
|
||
|
|
||
|
proc compilationFlags*(config: TestConfig): string =
|
||
|
for flag in compilerFlags * config.flags:
|
||
|
result &= " " & $flag
|
||
|
|
||
|
proc cache*(config: TestConfig; backend: string): string =
|
||
|
## return the path to the nimcache for the given backend and
|
||
|
## compile-time flags
|
||
|
result = getTempDir()
|
||
|
result = result / "testutils-nimcache-$#-$#" % [ backend,
|
||
|
$getCurrentProcessId() ]
|
||
|
|
||
|
proc processArguments*(): TestConfig =
|
||
|
## consume the arguments supplied to ntu and yield a computed
|
||
|
## configuration object
|
||
|
var
|
||
|
opt = initOptParser()
|
||
|
|
||
|
func toSet[SortBy](list: seq[SortBy]): set[SortBy] =
|
||
|
for element in list.items:
|
||
|
result.incl element
|
||
|
|
||
|
for kind, key, value in opt.getOpt:
|
||
|
if result.cmd == noCommand:
|
||
|
doAssert kind == cmdArgument
|
||
|
result.cmd = parseEnum[Command](key)
|
||
|
if result.cmd == test:
|
||
|
result.flags = defaultFlags
|
||
|
result.backends = testutilsBackends
|
||
|
result.orderBy = defaultSort
|
||
|
continue
|
||
|
|
||
|
case result.cmd
|
||
|
of test:
|
||
|
case kind
|
||
|
of cmdArgument:
|
||
|
if result.path == "":
|
||
|
result.path = absolutePath(key)
|
||
|
of cmdLongOption, cmdShortOption:
|
||
|
case key.toLowerAscii
|
||
|
of "help", "h":
|
||
|
quit(Usage, QuitSuccess)
|
||
|
of "reverse", "random":
|
||
|
let
|
||
|
flag = parseEnum[SortBy](value)
|
||
|
if flag in result.orderBy:
|
||
|
result.orderBy.excl flag
|
||
|
else:
|
||
|
result.orderBy.incl flag
|
||
|
of "sort":
|
||
|
result.orderBy = toSet value.split(",").mapIt parseEnum[SortBy](it)
|
||
|
of "backend", "backends", "targets", "t":
|
||
|
result.backends = value
|
||
|
of "release", "danger":
|
||
|
result.flags.incl ReleaseBuild
|
||
|
result.flags.incl DangerBuild
|
||
|
of "nothreads":
|
||
|
result.flags.excl UseThreads
|
||
|
of "update":
|
||
|
result.flags.incl UpdateOutputs
|
||
|
of "include":
|
||
|
result.includedTests.add value.split(Whitespace + {','})
|
||
|
of "exclude":
|
||
|
result.excludedTests.add value.split(Whitespace + {','})
|
||
|
else:
|
||
|
quit(Usage)
|
||
|
of cmdEnd:
|
||
|
quit(Usage)
|
||
|
|
||
|
of fuzz:
|
||
|
case kind
|
||
|
of cmdArgument:
|
||
|
result.target = key
|
||
|
of cmdLongOption, cmdShortOption:
|
||
|
case key.toLowerAscii:
|
||
|
of "f", "fuzzer":
|
||
|
result.fuzzer = parseEnum[FuzzingEngine](value)
|
||
|
of "c", "corpus":
|
||
|
result.corpusDir = absolutePath(value)
|
||
|
else:
|
||
|
quit(Usage)
|
||
|
else:
|
||
|
echo "got kind ", kind
|
||
|
quit(Usage)
|
||
|
|
||
|
of noCommand:
|
||
|
discard
|
||
|
|
||
|
case result.cmd
|
||
|
of test:
|
||
|
if result.path == "":
|
||
|
quit(Usage)
|
||
|
of fuzz:
|
||
|
if result.target == "":
|
||
|
quit(Usage)
|
||
|
else:
|
||
|
quit(Usage)
|
||
|
|
||
|
func shouldSkip*(config: TestConfig, name: string): bool =
|
||
|
## true if the named test should be skipped
|
||
|
if name in config.excludedTests:
|
||
|
result = true
|
||
|
elif config.includedTests.len > 0:
|
||
|
if name notin config.includedTests:
|
||
|
result = true
|