diff options
author | 2022-01-15 10:21:17 -0800 | |
---|---|---|
committer | 2022-01-15 10:21:17 -0800 | |
commit | 2f70b8b0460b3368955f63a025b03b1f84324874 (patch) | |
tree | 8cc242a2de68f7546a4be0a76c3634aff56254d0 /tests | |
parent | Merge pull request #136 from python-discord/feat/tests/135/rel-path-coverage (diff) | |
parent | Fix typo in comment (diff) |
Merge #132 - fix entrypoint and support Python args
Diffstat (limited to 'tests')
-rw-r--r-- | tests/test_main.py | 90 |
1 files changed, 90 insertions, 0 deletions
diff --git a/tests/test_main.py b/tests/test_main.py new file mode 100644 index 0000000..5b5bfcb --- /dev/null +++ b/tests/test_main.py @@ -0,0 +1,90 @@ +import ast +import contextlib +import io +import logging +import unittest +from argparse import Namespace +from unittest.mock import patch + +import snekbox.__main__ as snekbox_main + + +class ArgParseTests(unittest.TestCase): + def test_parse_args(self): + subtests = ( + ( + ["", "code"], + Namespace(code="code", nsjail_args=[], py_args=["-c"]) + ), + ( + ["", "code", "--time_limit", "0"], + Namespace(code="code", nsjail_args=["--time_limit", "0"], py_args=["-c"]) + ), + ( + ["", "code", "---", "-m", "timeit"], + Namespace(code="code", nsjail_args=[], py_args=["-m", "timeit"]) + ), + ( + ["", "code", "--time_limit", "0", "---", "-m", "timeit"], + Namespace(code="code", nsjail_args=["--time_limit", "0"], py_args=["-m", "timeit"]) + ), + ( + ["", "code", "--time_limit", "0", "---"], + Namespace(code="code", nsjail_args=["--time_limit", "0"], py_args=[]) + ), + ( + ["", "code", "---"], + Namespace(code="code", nsjail_args=[], py_args=[]) + ) + ) + + for argv, expected in subtests: + with self.subTest(argv=argv, expected=expected), patch("sys.argv", argv): + args = snekbox_main.parse_args() + self.assertEqual(args, expected) + + @patch("sys.argv", [""]) + def test_parse_args_code_missing_exits(self): + with self.assertRaises(SystemExit) as cm: + with contextlib.redirect_stderr(io.StringIO()) as stderr: + snekbox_main.parse_args() + + self.assertEqual(cm.exception.code, 2) + self.assertIn("the following arguments are required: code", stderr.getvalue()) + + +class EntrypointTests(unittest.TestCase): + """Integration tests of the CLI entrypoint.""" + + def setUp(self): + logging.getLogger("snekbox.nsjail").setLevel(logging.WARNING) + + @patch("sys.argv", ["", "print('hello'); import sys; print('error', file=sys.stderr)"]) + def test_main_prints_stdout(self): + """Should print the stdout of the subprocess followed by its stderr.""" + with contextlib.redirect_stdout(io.StringIO()) as stdout: + snekbox_main.main() + + self.assertEqual(stdout.getvalue(), "hello\nerror\n\n") + + @patch("sys.argv", ["", "import sys; sys.exit(22)"]) + def test_main_exits_with_returncode(self): + """Should exit with the subprocess's returncode if it's non-zero.""" + with self.assertRaises(SystemExit) as cm: + snekbox_main.main() + + self.assertEqual(cm.exception.code, 22) + + def test_main_forwards_args(self): + """Should forward NsJail args to NsJail and Python args to the Python subprocess.""" + code = "import sys, time; print(sys.orig_argv); time.sleep(2)" + py_args = ["-R", "-dc"] + args = ["", code, "--time_limit", "1", "---", *py_args] + + with patch("sys.argv", args), self.assertRaises(SystemExit) as cm: + with contextlib.redirect_stdout(io.StringIO()) as stdout: + snekbox_main.main() + + orig_argv = ast.literal_eval(stdout.getvalue().strip()) + self.assertListEqual([*py_args, code], orig_argv[-3:]) + self.assertEqual(cm.exception.code, 137, "The time_limit NsJail arg was not respected.") |