From 30e537fad2af0e56f2ddfed0a01dc98e7b57912d Mon Sep 17 00:00:00 2001 From: "Matthias J. Kannwischer" Date: Mon, 23 Dec 2024 15:39:40 +0800 Subject: [PATCH 1/4] Eliminate click dependency Our test scripts rely on click to argument parsing. This commit removes that dependency and uses the native argparse instead. Signed-off-by: Matthias J. Kannwischer --- requirements.txt | 2 - scripts/tests | 470 ++++++++++++++++++++--------------------------- 2 files changed, 195 insertions(+), 277 deletions(-) delete mode 100644 requirements.txt diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index db7919bb3..000000000 --- a/requirements.txt +++ /dev/null @@ -1,2 +0,0 @@ -# SPDX-License-Identifier: Apache-2.0 -click==8.1.7 diff --git a/scripts/tests b/scripts/tests index e559663e4..41e3ed9cb 100755 --- a/scripts/tests +++ b/scripts/tests @@ -5,313 +5,233 @@ import platform import os import sys -import click +import argparse from functools import reduce sys.path.append(f"{os.path.join(os.path.dirname(__file__), 'lib')}") from mlkem_test import * from util import path -""" -Click interface configuration -""" +def cli(): + common_parser = argparse.ArgumentParser(add_help=False) -def __callback(n): - def callback(ctx, param, value): - state = ctx.ensure_object(Options) - state.__dict__[n] = value - return value - - return callback - - -def add_options(options): - """Shortcurt for adding multiple options for Click command""" - return lambda func: reduce(lambda f, o: o(f), reversed(options), func) - + # Common arguments for all sub-commands + common_parser.add_argument( + "-v", "--verbose", help="Show verbose output or not", action="store_true" + ) + common_parser.add_argument( + "-cp", "--cross-prefix", help="Cross prefix for compilation", default="" + ) + common_parser.add_argument( + "--cflags", help="Extra cflags to passed in (e.g. '-mcpu=cortex-a72')" + ) + common_parser.add_argument( + "--arch-flags", help="Extra arch flags to passed in (e.g. '-march=armv8')" + ) -_shared_options = [ - click.option( - "-v", - "--verbose", - expose_value=False, - is_flag=True, - show_default=True, - default=False, - type=bool, - help="Show verbose output or not", - callback=__callback("verbose"), - ), - click.option( - "-cp", - "--cross-prefix", - expose_value=False, - default="", - show_default=True, - nargs=1, - help="Cross prefix for compilation", - callback=__callback("cross_prefix"), - ), - click.option( - "--cflags", - expose_value=False, - nargs=1, - help="Extra cflags to passed in (e.g. '-mcpu=cortex-a72')", - callback=__callback("cflags"), - ), - click.option( - "--arch-flags", - expose_value=False, - nargs=1, - help="Extra arch flags to passed in (e.g. '-march=armv8')", - callback=__callback("arch_flags"), - ), - click.option( - "--auto/--no-auto", - expose_value=False, - is_flag=True, - show_default=True, - default=True, + # --auto / --no-auto + auto_group = common_parser.add_mutually_exclusive_group() + auto_group.add_argument( + "--auto", + action="store_true", + dest="auto", help="Allow makefile to auto configure system specific preprocessor", - callback=__callback("auto"), - ), - click.option( + default=True, + ) + auto_group.add_argument( + "--no-auto", + action="store_false", + dest="auto", + help="Disallow makefile to auto configure system specific preprocessor", + ) + + common_parser.add_argument( "--opt", - expose_value=False, - nargs=1, - type=click.Choice(["ALL", "OPT", "NO_OPT"], case_sensitive=False), - show_default=True, - default="ALL", help="Determine whether to compile/run the opt/no_opt binary or both", - callback=__callback("opt"), - ), - click.option( - "--compile/--no-compile", - expose_value=False, - is_flag=True, - show_default=True, - default=True, - help="Determine to compile the binary or not", - callback=__callback("compile"), - ), - click.option( - "--run/--no-run", - expose_value=False, - is_flag=True, - show_default=True, + choices=["ALL", "OPT", "NO_OPT"], + type=str.upper, + default="ALL", + ) + + # --compile / --no-compile + compile_group = common_parser.add_mutually_exclusive_group() + compile_group.add_argument( + "--compile", + action="store_true", + dest="compile", + help="Compile the binaries", default=True, - help="Determine to run the compiled binary or not", - callback=__callback("run"), - ), - click.option( - "-w", - "--exec-wrapper", - expose_value=False, - show_default=True, - default="", - help="Run the benchmark binary with the user-customized wrapper.", - callback=__callback("exec_wrapper"), - ), - click.option( - "-r", - "--run-as-root", - expose_value=False, - is_flag=True, - show_default=True, - default=False, - type=bool, - help="Benchmarking binary is run with sudo.", - callback=__callback("run_as_root"), - ), -] + ) + compile_group.add_argument( + "--no-compile", + action="store_false", + dest="compile", + help="Do not compile the binaries", + ) -_bench_options = [ - click.option( - "-c", - "--cycles", - nargs=1, - type=click.Choice(["NO", "PMU", "PERF", "M1"]), - show_default=True, - default="NO", - help="Method for counting clock cycles. PMU requires (user-space) access to the Arm Performance Monitor Unit (PMU). PERF requires a kernel with perf support. M1 only works on Apple silicon.", - ), - click.option( - "-o", - "--output", - nargs=1, - help="Path to output file in json format", - ), - click.option( - "-t", - "--mac-taskpolicy", - nargs=1, - type=click.Choice(["utility", "background", "maintenance"]), - hidden=platform.system() != "Darwin", - show_default=True, - default=None, - help="Run the program using the specified QoS clamp. Applies to MacOS only. Setting this flag to 'background' guarantees running on E-cores. This is an abbreviation of --exec-wrapper 'taskpolicy -c {mac_taskpolicy}'.", - ), - click.option( - "--components", - is_flag=True, - type=bool, - show_default=True, - default=False, - help="Benchmark low-level components", - ), -] + # --run / --no-run + run_group = common_parser.add_mutually_exclusive_group() + run_group.add_argument( + "--run", action="store_true", dest="run", help="Run the binaries", default=True + ) + run_group.add_argument( + "--no-run", action="store_false", dest="run", help="Do not run the binaries" + ) + common_parser.add_argument( + "-w", "--exec-wrapper", help="Run the binary with the user-customized wrapper" + ) + common_parser.add_argument("-r", "--run-as-root", help="Run the binary as root") -@click.group(invoke_without_command=True) -def cli(): - pass + main_parser = argparse.ArgumentParser() + + cmd_subparsers = main_parser.add_subparsers(title="Commands", dest="cmd") + + # all arguments + all_parser = cmd_subparsers.add_parser( + "all", help="Run all tests (except benchmark for now)", parents=[common_parser] + ) + func_group = all_parser.add_mutually_exclusive_group() + func_group.add_argument( + "--func", action="store_true", dest="func", help="Run func test", default=True + ) + func_group.add_argument( + "--no-func", action="store_false", dest="func", help="Do not run func test" + ) -@cli.command( - short_help="Run the functional tests for all parameter sets", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@click.make_pass_decorator(Options, ensure=True) -def func(opts: Options): - Tests(opts).func() + kat_group = all_parser.add_mutually_exclusive_group() + kat_group.add_argument( + "--kat", action="store_true", dest="kat", help="Run kat test", default=True + ) + kat_group.add_argument( + "--no-kat", action="store_false", dest="kat", help="Do not run kat test" + ) + nistkat_group = all_parser.add_mutually_exclusive_group() + nistkat_group.add_argument( + "--nistkat", + action="store_true", + dest="nistkat", + help="Run nistkat test", + default=True, + ) + nistkat_group.add_argument( + "--no-nistkatkat", + action="store_false", + dest="nistkat", + help="Do not run nistkat test", + ) -@cli.command( - short_help="Run the nistkat tests for all parameter sets", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@click.make_pass_decorator(Options, ensure=True) -def nistkat(opts: Options): - Tests(opts).nistkat() + acvp_group = all_parser.add_mutually_exclusive_group() + acvp_group.add_argument( + "--acvp", action="store_true", dest="acvp", help="Run acvp test", default=True + ) + acvp_group.add_argument( + "--no-acvp", action="store_false", dest="acvp", help="Do not run acvp test" + ) + # acvp arguments + acvp_parser = cmd_subparsers.add_parser( + "acvp", help="Run ACVP client", parents=[common_parser] + ) -@cli.command( - short_help="Run the kat tests for all parameter sets", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@click.make_pass_decorator(Options, ensure=True) -def kat(opts: Options): - Tests(opts).kat() + acvp_parser.add_argument( + "-d", + "--acvp_dir", + dest="acvp_dir", + default=path("test/acvp_data"), + help="Path to acvp directory", + ) + # bench arguments + bench_parser = cmd_subparsers.add_parser( + "bench", + help="Run the benchmarks for all parameter sets", + parents=[common_parser], + ) -@cli.command( - short_help="Run ACVP client", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@add_options( - [ - click.option( - "-d", - "--acvp_dir", - nargs=1, - show_default=True, - type=click.Path(), - default=path("test/acvp_data"), - help="Path to acvp directory", - ), - ] -) -@click.make_pass_decorator(Options, ensure=True) -def acvp(opts: Options, acvp_dir: str): - Tests(opts).acvp(acvp_dir) + bench_parser.add_argument( + "-c", + "--cycles", + help="Method for counting clock cycles. PMU requires (user-space) access to the Arm Performance Monitor Unit (PMU). PERF requires a kernel with perf support. M1 only works on Apple silicon.", + choices=["NO", "PMU", "PERF", "M1"], + type=str.upper, + default="NO", + ) + bench_parser.add_argument( + "-o", "--output", help="Path to output file in json format" + ) + if platform.system() == "Darwin": + bench_parser.add_argument( + "-t", + "--mac-taskpolicy", + help="Run the program using the specified QoS clamp. Applies to MacOS only. Setting this flag to 'background' guarantees running on E-cores. This is an abbreviation of --exec-wrapper 'taskpolicy -c {mac_taskpolicy}'.", + choices=["utility", "background", "maintenance"], + type=str.lower, + ) + bench_parser.add_argument( + "--components", + help="Benchmark low-level components", + action="store_true", + default=False, + ) + # cbmc arguments + cbmc_parser = cmd_subparsers.add_parser( + "cbmc", + help="Run the CBMC proofs for all parameter sets", + parents=[common_parser], + ) -@cli.command( - short_help="Run the benchmarks for all parameter sets", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@add_options(_bench_options) -@click.make_pass_decorator(Options, ensure=True) -def bench( - opts: Options, - cycles: str, - output, - mac_taskpolicy, - components, -): - Tests(opts).bench( - cycles, - output, - mac_taskpolicy, - components, + cbmc_parser.add_argument( + "--k", + help="MLKEM parameter set (MLKEM_K)", + choices=["2", "3", "4", "ALL"], + type=str.upper, + default="ALL", ) + # func arguments + func_parser = cmd_subparsers.add_parser( + "func", + help="Run the functional tests for all parameter sets", + parents=[common_parser], + ) -@cli.command( - short_help="Run all tests (except benchmark for now)", - context_settings={"show_default": True}, -) -@add_options(_shared_options) -@add_options( - [ - click.option( - "--func/--no-func", - is_flag=True, - show_default=True, - default=True, - help="Determine whether to run func test or not", - ), - click.option( - "--kat/--no-kat", - is_flag=True, - show_default=True, - default=True, - help="Determine whether to run kat test or not", - ), - click.option( - "--nistkat/--no-nistkat", - is_flag=True, - show_default=True, - default=True, - help="Determine whether to run nistkat test or not", - ), - click.option( - "--acvp/--no-acvp", - is_flag=True, - show_default=True, - default=True, - help="Determine whether to run acvp test or not", - ), - ] -) -@click.make_pass_decorator(Options, ensure=True) -def all( - opts: Options, - func: bool, - kat: bool, - nistkat: bool, - acvp: bool, -): - Tests(opts).all(func, kat, nistkat, acvp) + # kat arguments + kat_parser = cmd_subparsers.add_parser( + "kat", help="Run the kat tests for all parameter sets", parents=[common_parser] + ) + # nistkat arguments + nistkat_parser = cmd_subparsers.add_parser( + "nistkat", + help="Run the nistkat tests for all parameter sets", + parents=[common_parser], + ) -@cli.command( - short_help="Run the CBMC proofs for all parameter sets", - context_settings={"show_default": True}, -) -@click.make_pass_decorator(Options, ensure=True) -@add_options( - [ - click.option( - "--k", - expose_value=False, - nargs=1, - type=click.Choice(["2", "3", "4", "ALL"]), - show_default=True, - default="ALL", - help="MLKEM parameter set (MLKEM_K).", - callback=__callback("k"), + args = main_parser.parse_args() + + if args.cmd == "all": + Tests(args).all(args.func, args.kat, args.nistkat, args.acvp) + elif args.cmd == "acvp": + Tests(args).acvp(args.acvp_dir) + elif args.cmd == "bench": + if not hasattr(args, "mac_taskpolicy"): + args.mac_taskpolicy = None + Tests(args).bench( + args.cycles, args.output, args.mac_taskpolicy, args.components ) - ] -) -def cbmc(opts: Options): - Tests(opts).cbmc(opts.k) + elif args.cmd == "cbmc": + Tests(args).cbmc(args.k) + elif args.cmd == "func": + Tests(args).func() + elif args.cmd == "kat": + Tests(args).kat() + elif args.cmd == "nistkat": + Tests(args).nistkat() if __name__ == "__main__": From f72376071f5bc0b63a9c2c66ef8a6da07d485edd Mon Sep 17 00:00:00 2001 From: "Matthias J. Kannwischer" Date: Mon, 23 Dec 2024 15:43:23 +0800 Subject: [PATCH 2/4] Doc: pip no longer needed Signed-off-by: Matthias J. Kannwischer --- BUILDING.md | 9 +-------- README.md | 7 +------ 2 files changed, 2 insertions(+), 14 deletions(-) diff --git a/BUILDING.md b/BUILDING.md index df01753c9..e8d7f0791 100644 --- a/BUILDING.md +++ b/BUILDING.md @@ -4,14 +4,7 @@ ### Prerequisites -To build **mlkem-native**, you need `make` and a C90 compiler. To use the test scripts, you need Python3 with -dependencies as specified in [requirements.txt](requirements.txt). We recommend using a virtual environment, e.g.: - -```bash -python3 -m venv venv -./venv/bin/python3 -m pip install -r requirements.txt -source venv/bin/activate -``` +To build **mlkem-native**, you need `make` and a C90 compiler. To use the test scripts, you need Python3 (>= 3.7). ### Using `make` diff --git a/README.md b/README.md index 85ee50eb6..242ff105a 100644 --- a/README.md +++ b/README.md @@ -28,12 +28,7 @@ cd mlkem-native # Install base packages sudo apt-get update -sudo apt-get install python3-venv python3-pip make - -# Setup Python environment -python3 -m venv venv -source venv/bin/activate -python3 -m pip install -r requirements.txt +sudo apt-get install make gcc python3 # Build and run base tests make quickcheck From 83e587f439453ab0fc3df74f12caf7dfaa250841 Mon Sep 17 00:00:00 2001 From: "Matthias J. Kannwischer" Date: Mon, 23 Dec 2024 15:43:53 +0800 Subject: [PATCH 3/4] Dependabot: remove pip Signed-off-by: Matthias J. Kannwischer --- .github/dependabot.yml | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/.github/dependabot.yml b/.github/dependabot.yml index fc4c82dbe..10eea6c61 100644 --- a/.github/dependabot.yml +++ b/.github/dependabot.yml @@ -4,8 +4,4 @@ updates: directory: / schedule: interval: monthly - - - package-ecosystem: pip - directory: / - schedule: - interval: monthly + From 19990c51c435f241c41973d8741136de4641f3f0 Mon Sep 17 00:00:00 2001 From: "Matthias J. Kannwischer" Date: Mon, 23 Dec 2024 15:45:36 +0800 Subject: [PATCH 4/4] CI: remove pip dependencies Signed-off-by: Matthias J. Kannwischer --- .github/actions/setup-apt/action.yml | 10 +--------- .github/actions/setup-yum/action.yml | 10 +--------- 2 files changed, 2 insertions(+), 18 deletions(-) diff --git a/.github/actions/setup-apt/action.yml b/.github/actions/setup-apt/action.yml index a4348deba..c676aee50 100644 --- a/.github/actions/setup-apt/action.yml +++ b/.github/actions/setup-apt/action.yml @@ -22,17 +22,9 @@ runs: - name: Install base packages shell: bash run: | - ${{ inputs.sudo }} apt-get install python3-venv python3-pip make -y + ${{ inputs.sudo }} apt-get install make gcc python3 -y - name: Install additional packages if: ${{ inputs.packages != ''}} shell: bash run: | ${{ inputs.sudo }} apt-get install ${{ inputs.packages }} -y - - name: Setup Python venv - shell: bash - run: | - python3 -m venv venv - source venv/bin/activate - python3 -m pip install -r requirements.txt - deactivate - echo "$(pwd)/venv/bin/" >> "$GITHUB_PATH" diff --git a/.github/actions/setup-yum/action.yml b/.github/actions/setup-yum/action.yml index 16f9c3eda..72f9ccbaf 100644 --- a/.github/actions/setup-yum/action.yml +++ b/.github/actions/setup-yum/action.yml @@ -18,17 +18,9 @@ runs: - name: Install base packages shell: bash run: | - ${{ inputs.sudo }} yum install make git python3-pip -y - ${{ inputs.sudo }} pip3 install virtualenv + ${{ inputs.sudo }} yum install make gcc python3 git -y - name: Install additional packages if: ${{ inputs.packages != ''}} shell: bash run: | ${{ inputs.sudo }} yum install ${{ inputs.packages }} -y - - name: Setup Python venv - shell: bash - run: | - virtualenv venv - source venv/bin/activate - python3 -m pip install -r requirements.txt - echo "$(pwd)/venv/bin/" >> "$GITHUB_PATH"