added time limit CLI parameter

This commit is contained in:
dogeystamp 2023-11-02 15:23:09 -04:00
parent 4139709250
commit 91bb8a8e45
Signed by: dogeystamp
GPG Key ID: 7225FE3592EFFA38
3 changed files with 32 additions and 7 deletions

View File

@ -1,6 +1,7 @@
import asyncio import asyncio
from pathlib import Path from pathlib import Path
from testr.file_data import DirectorySuite, ExecutableRunner from testr.file_data import DirectorySuite, ExecutableRunner
from testr.runner import TestOptions
import argparse import argparse
parser = argparse.ArgumentParser() parser = argparse.ArgumentParser()
@ -11,14 +12,23 @@ test_data_group = parser.add_mutually_exclusive_group(required=True)
test_data_group.add_argument("--testdir", help="Directory for test cases") test_data_group.add_argument("--testdir", help="Directory for test cases")
test_runner_group.add_argument("--exec", help="Executable to run test cases against") test_runner_group.add_argument("--exec", help="Executable to run test cases against")
parser.add_argument("--timelim", help="Time limit in seconds", type=float, default=5.0)
args = parser.parse_args() args = parser.parse_args()
async def main(): async def main():
test_suite = DirectorySuite(Path(args.testdir)) test_suite = DirectorySuite(Path(args.testdir))
test_runner = ExecutableRunner(Path(args.exec)) test_runner = ExecutableRunner(Path(args.exec))
async for test_case in test_runner.run_test_suite(test_suite): async for test_case in test_runner.run_test_suite(
test_suite,
TestOptions(
time_limit=args.timelim
)
):
print(test_case.code) print(test_case.code)
if __name__ == "__main__": if __name__ == "__main__":
asyncio.run(main()) asyncio.run(main())

View File

@ -1,5 +1,5 @@
import asyncio import asyncio
from testr.runner import TestData, TestRunner, TestStatus, TestSuite, StatusCode from testr.runner import TestData, TestOptions, TestRunner, TestStatus, TestSuite, StatusCode
from pathlib import Path from pathlib import Path
@ -30,7 +30,7 @@ class ExecutableRunner(TestRunner):
raise ValueError(f"executable must be a file, got '{executable}'") raise ValueError(f"executable must be a file, got '{executable}'")
self.executable = executable self.executable = executable
async def run_test(self, data: TestData) -> TestStatus: async def run_test(self, data: TestData, opts: TestOptions) -> TestStatus:
proc = await asyncio.create_subprocess_shell( proc = await asyncio.create_subprocess_shell(
str(self.executable), str(self.executable),
stdout=asyncio.subprocess.PIPE, stdout=asyncio.subprocess.PIPE,
@ -42,7 +42,7 @@ class ExecutableRunner(TestRunner):
try: try:
out_stream, err_stream = await asyncio.wait_for( out_stream, err_stream = await asyncio.wait_for(
proc.communicate(input=input_data.encode()), timeout=5.0) proc.communicate(input=input_data.encode()), timeout=opts.time_limit)
except TimeoutError: except TimeoutError:
proc.kill() proc.kill()
return TestStatus(code=StatusCode.TLE, stderr="", stdout="", stdin=input_data) return TestStatus(code=StatusCode.TLE, stderr="", stdout="", stdin=input_data)

View File

@ -53,6 +53,21 @@ class TestValidator(ABC):
async def validate_output(self, output: str) -> bool: pass async def validate_output(self, output: str) -> bool: pass
@dataclass
class TestOptions:
"""
Options for running a test suite.
Attributes
----------
time_limit
Time limit in seconds for each test case.
"""
time_limit: float
class TestData(TestInput, TestValidator): class TestData(TestInput, TestValidator):
"""Combined input/output for single test case""" """Combined input/output for single test case"""
@ -70,8 +85,8 @@ class TestRunner(ABC):
"""Runner for test cases.""" """Runner for test cases."""
@abstractmethod @abstractmethod
async def run_test(self, data: TestData) -> TestStatus: pass async def run_test(self, data: TestData, opts: TestOptions) -> TestStatus: pass
async def run_test_suite(self, data: TestSuite) -> AsyncIterator[TestStatus]: async def run_test_suite(self, data: TestSuite, opts: TestOptions) -> AsyncIterator[TestStatus]:
for test_case in data: for test_case in data:
yield await self.run_test(test_case) yield await self.run_test(test_case, opts)