Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp5482761ybl; Tue, 4 Feb 2020 15:03:44 -0800 (PST) X-Google-Smtp-Source: APXvYqyCvbkDDKf6x2RLzo4G8Y4M54f4/9S1mX7ks6MgMKfIgA6oGRKXOIpsrTJguYF5qtK55n1Z X-Received: by 2002:a9d:65cb:: with SMTP id z11mr22407670oth.348.1580857424557; Tue, 04 Feb 2020 15:03:44 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1580857424; cv=none; d=google.com; s=arc-20160816; b=U/nYLHgO4I5iSGLrZSq7+VH0lV2hByHI0MZfHMJAGWA+bST872bodc3mLbhD8USR8j yehZmrx7rritcNhA6sRvJmJK5akDpFmv3XOYw5oW/KGXaXL0Qpmtkt1KJHlP3frkEQPP crmZytoQL26klfkkc2gQZOWNnOyzR9gK1wPU7drHMFMIBwzESKHxGlVZOdqpaXhUcp5y mvggu3X3t3HrbsfVzw8b2HXWwaK67jK4avTh89xizIb97s/vd1huXDiR8pRlddVytlpd wzsF+kjAwNzdLHOEdTgjveSapmrh2QUZowDUU2MvK1XTFOOgM2FxW9HDomY3CbxRkHWr hkZA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dkim-signature; bh=zd0zXG6JPXFB9v47RJxr9eccifc+yyrlfmNJAs/p+ps=; b=zswR/mi0D6RcmAQ/L/n5JeROvZ+M0RzPgMnDmK/V06tXi4eY3EeegaTA+TLTe2hZ6x jN8vmSvFs+SCM/iIdd8oVWwI0s7nfWlJdxtkvHcJO/sc0Iw/aykIHhOgtZKzR09FtyHY VqNkbJyC8s7iYoAkgyzrFkq0v9h8NdzhvrACk7FFILqsIa7PgH7XSgW70XOVmRZlRRGk 8HsV+vB68lcKSJJI3NZ1w7Y0jZvMAAu6S//WjE0VcXNazT1FnkCoXInk95UBpjp3ZMeO yBRSf4nw+d+AbvpvFI5egZVokCG/tPaREV2XV5eTLsVqwspgC85qFSRBh6i+IV8kXXZw oG1w== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=BnjStEmM; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id h8si10907647oib.104.2020.02.04.15.03.31; Tue, 04 Feb 2020 15:03:44 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=BnjStEmM; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727664AbgBDXBn (ORCPT + 99 others); Tue, 4 Feb 2020 18:01:43 -0500 Received: from mail-yw1-f66.google.com ([209.85.161.66]:36594 "EHLO mail-yw1-f66.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727483AbgBDXBn (ORCPT ); Tue, 4 Feb 2020 18:01:43 -0500 Received: by mail-yw1-f66.google.com with SMTP id n184so583367ywc.3; Tue, 04 Feb 2020 15:01:42 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=subject:to:cc:references:from:message-id:date:user-agent :mime-version:in-reply-to:content-language:content-transfer-encoding; bh=zd0zXG6JPXFB9v47RJxr9eccifc+yyrlfmNJAs/p+ps=; b=BnjStEmMUZZjMcNWgu7L8b8UaxeO+auGj5PsxmkQF/hvjxp0DmFytmYndtfphxQQ4g uIb7eEQzVk8gJK2wIJYk9j8shtd+MX04IVCpZRE3LjVSSU0TOPts9dPPYGGuAC7LLFvv SQC68sB8dzRwqVATXoLZhz1fJXCjHZK9zgIpCuSuMfco8W9lC9Ebc77KQeDq5b9U4OOE 0aJo3v1JElZx1G6L7Wl/DiYrJ+g9Vd+o7rLa/cPT9zJfvT1KRgn3GAvQ9q6LBYobMxBm reHzRlfYxdzFll5sMP4Ltb7Uva/CjECKmtsCy8HoPnQJLNRj3duMqiZbJTs0jle09QvZ gLCg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:subject:to:cc:references:from:message-id:date :user-agent:mime-version:in-reply-to:content-language :content-transfer-encoding; bh=zd0zXG6JPXFB9v47RJxr9eccifc+yyrlfmNJAs/p+ps=; b=HNDepISTYv5iLhsu5v4Qd4nSLH/7jsIL0y2bUuaOWcYm9X204rRl4V2w1FOTDY+UL8 X99NFB4n/1RVFFd24g2iUXF6PZTnz0g5QQtA1P9Lx3I+yhkUpFIhdL51yHdbP+Sn1WiH LQq2rgGlJYqY36WyXGrj4jTz9/gkdrV8dJ5xLe2ODF7DZ4OR5CHR5U/714jZ3sHiuJRG Zglk55zwC9+IoqlqTOR0766u0Ai7qJMyvJt3Bj3w9McEphltpG5akCeTpgj2+AxlkM8a 9vNjJU0Alqq9AFjNbw90Z2JR/saW9vQOpinwXmPdgH7lds1z/57Dy67YHja+2K9/sn7x fr7w== X-Gm-Message-State: APjAAAXFnb+Vq5sR5zSPk/hSUev765rOOfMMBaOrAXOJIRYoVp/kdXhE iJMtG7JxT9hqnmBe5Sib6+o= X-Received: by 2002:a25:58c6:: with SMTP id m189mr25967018ybb.98.1580857301582; Tue, 04 Feb 2020 15:01:41 -0800 (PST) Received: from [192.168.1.46] (c-73-88-245-53.hsd1.tn.comcast.net. [73.88.245.53]) by smtp.gmail.com with ESMTPSA id k133sm10240035ywc.9.2020.02.04.15.01.40 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Tue, 04 Feb 2020 15:01:41 -0800 (PST) Subject: Re: [PATCH v2 5/7] kunit: test: add test plan to KUnit TAP format To: Brendan Higgins , jdike@addtoit.com, richard@nod.at, anton.ivanov@cambridgegreys.com, arnd@arndb.de, keescook@chromium.org, skhan@linuxfoundation.org, alan.maguire@oracle.com, yzaikin@google.com, davidgow@google.com, akpm@linux-foundation.org, rppt@linux.ibm.com Cc: gregkh@linuxfoundation.org, sboyd@kernel.org, logang@deltatee.com, mcgrof@kernel.org, knut.omang@oracle.com, linux-um@lists.infradead.org, linux-arch@vger.kernel.org, linux-kselftest@vger.kernel.org, kunit-dev@googlegroups.com, linux-kernel@vger.kernel.org, linux-doc@vger.kernel.org References: <20200130230812.142642-1-brendanhiggins@google.com> <20200130230812.142642-6-brendanhiggins@google.com> From: Frank Rowand Message-ID: <69582db9-d379-7d2a-1342-6de9a5d79141@gmail.com> Date: Tue, 4 Feb 2020 17:01:40 -0600 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:60.0) Gecko/20100101 Thunderbird/60.9.0 MIME-Version: 1.0 In-Reply-To: <20200130230812.142642-6-brendanhiggins@google.com> Content-Type: text/plain; charset=utf-8 Content-Language: en-US Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 1/30/20 5:08 PM, Brendan Higgins wrote: > TAP 14 allows an optional test plan to be emitted before the start of > the start of testing[1]; this is valuable because it makes it possible > for a test harness to detect whether the number of tests run matches the > number of tests expected to be run, ensuring that no tests silently > failed. > > Link[1]: https://github.com/isaacs/testanything.github.io/blob/tap14/tap-version-14-specification.md#the-plan > Signed-off-by: Brendan Higgins > Reviewed-by: Stephen Boyd > --- > lib/kunit/executor.c | 17 +++++ > lib/kunit/test.c | 11 --- > tools/testing/kunit/kunit_parser.py | 74 ++++++++++++++++--- > .../test_is_test_passed-all_passed.log | 1 + > .../test_data/test_is_test_passed-crash.log | 1 + > .../test_data/test_is_test_passed-failure.log | 1 + > 6 files changed, 82 insertions(+), 23 deletions(-) > > diff --git a/lib/kunit/executor.c b/lib/kunit/executor.c > index b75a46c560847..7fd16feff157e 100644 > --- a/lib/kunit/executor.c > +++ b/lib/kunit/executor.c > @@ -11,11 +11,28 @@ extern struct kunit_suite * const * const __kunit_suites_end[]; > > #if IS_BUILTIN(CONFIG_KUNIT) > > +static void kunit_print_tap_header(void) > +{ > + struct kunit_suite * const * const *suites, * const *subsuite; > + int num_of_suites = 0; > + > + for (suites = __kunit_suites_start; > + suites < __kunit_suites_end; > + suites++) > + for (subsuite = *suites; *subsuite != NULL; subsuite++) > + num_of_suites++; > + > + pr_info("TAP version 14\n"); > + pr_info("1..%d\n", num_of_suites); > +} > + Subsuites are not in Linux 5.5, as far as I can tell, so it is hard to review this. But I think this observation will still be valid even though I am not up to speed on subsuites. num_of_suites is all test suites declared (as built in). In Linux 5.5, each kunit_suite contains one or more test cases. Each test case leads to either "ok" or "not ok". So instead of counting number of tests suites, the number of test cases needs to be counted. Ahhh, I think my lack of visibility of what tree this series is against is about to lead me astray! Are the test cases in the subsuite implemented as what TAP 14 calls "subtests"? If so, then the above code that counts suites instead of test cases would be correct. I'm spinning my wheels now, since I don't know what I am reviewing against, so I'm going to stop reviewing, other than maybe a quick scan. -Frank > int kunit_run_all_tests(void) > { > struct kunit_suite * const * const *suites, * const *subsuite; > bool has_test_failed = false; > > + kunit_print_tap_header(); > + > for (suites = __kunit_suites_start; > suites < __kunit_suites_end; > suites++) { > diff --git a/lib/kunit/test.c b/lib/kunit/test.c > index 9242f932896c7..da56b94261b43 100644 > --- a/lib/kunit/test.c > +++ b/lib/kunit/test.c > @@ -18,16 +18,6 @@ static void kunit_set_failure(struct kunit *test) > WRITE_ONCE(test->success, false); > } > > -static void kunit_print_tap_version(void) > -{ > - static bool kunit_has_printed_tap_version; > - > - if (!kunit_has_printed_tap_version) { > - pr_info("TAP version 14\n"); > - kunit_has_printed_tap_version = true; > - } > -} > - > static size_t kunit_test_cases_len(struct kunit_case *test_cases) > { > struct kunit_case *test_case; > @@ -41,7 +31,6 @@ static size_t kunit_test_cases_len(struct kunit_case *test_cases) > > static void kunit_print_subtest_start(struct kunit_suite *suite) > { > - kunit_print_tap_version(); > pr_info("\t# Subtest: %s\n", suite->name); > pr_info("\t1..%zd\n", kunit_test_cases_len(suite->test_cases)); > } > diff --git a/tools/testing/kunit/kunit_parser.py b/tools/testing/kunit/kunit_parser.py > index 4ffbae0f67325..78b3bdd03b1e4 100644 > --- a/tools/testing/kunit/kunit_parser.py > +++ b/tools/testing/kunit/kunit_parser.py > @@ -45,6 +45,7 @@ class TestStatus(Enum): > FAILURE = auto() > TEST_CRASHED = auto() > NO_TESTS = auto() > + FAILURE_TO_PARSE_TESTS = auto() > > kunit_start_re = re.compile(r'^TAP version [0-9]+$') > kunit_end_re = re.compile('List of all partitions:') > @@ -106,7 +107,7 @@ OkNotOkResult = namedtuple('OkNotOkResult', ['is_ok','description', 'text']) > > OK_NOT_OK_SUBTEST = re.compile(r'^\t(ok|not ok) [0-9]+ - (.*)$') > > -OK_NOT_OK_MODULE = re.compile(r'^(ok|not ok) [0-9]+ - (.*)$') > +OK_NOT_OK_MODULE = re.compile(r'^(ok|not ok) ([0-9]+) - (.*)$') > > def parse_ok_not_ok_test_case(lines: List[str], > test_case: TestCase, > @@ -196,7 +197,9 @@ def max_status(left: TestStatus, right: TestStatus) -> TestStatus: > else: > return TestStatus.SUCCESS > > -def parse_ok_not_ok_test_suite(lines: List[str], test_suite: TestSuite) -> bool: > +def parse_ok_not_ok_test_suite(lines: List[str], > + test_suite: TestSuite, > + expected_suite_index: int) -> bool: > consume_non_diagnositic(lines) > if not lines: > test_suite.status = TestStatus.TEST_CRASHED > @@ -209,6 +212,12 @@ def parse_ok_not_ok_test_suite(lines: List[str], test_suite: TestSuite) -> bool: > test_suite.status = TestStatus.SUCCESS > else: > test_suite.status = TestStatus.FAILURE > + suite_index = int(match.group(2)) > + if suite_index != expected_suite_index: > + print_with_timestamp( > + red('[ERROR] ') + 'expected_suite_index ' + > + str(expected_suite_index) + ', but got ' + > + str(suite_index)) > return True > else: > return False > @@ -221,7 +230,7 @@ def bubble_up_test_case_errors(test_suite: TestSuite) -> TestStatus: > max_test_case_status = bubble_up_errors(lambda x: x.status, test_suite.cases) > return max_status(max_test_case_status, test_suite.status) > > -def parse_test_suite(lines: List[str]) -> TestSuite: > +def parse_test_suite(lines: List[str], expected_suite_index: int) -> TestSuite: > if not lines: > return None > consume_non_diagnositic(lines) > @@ -240,7 +249,7 @@ def parse_test_suite(lines: List[str]) -> TestSuite: > test_suite.cases.append(test_case) > test_case = parse_test_case(lines, expected_test_case_num > 0) > expected_test_case_num -= 1 > - if parse_ok_not_ok_test_suite(lines, test_suite): > + if parse_ok_not_ok_test_suite(lines, test_suite, expected_suite_index): > test_suite.status = bubble_up_test_case_errors(test_suite) > return test_suite > elif not lines: > @@ -260,6 +269,17 @@ def parse_tap_header(lines: List[str]) -> bool: > else: > return False > > +TEST_PLAN = re.compile(r'[0-9]+\.\.([0-9]+)') > + > +def parse_test_plan(lines: List[str]) -> int: > + consume_non_diagnositic(lines) > + match = TEST_PLAN.match(lines[0]) > + if match: > + lines.pop(0) > + return int(match.group(1)) > + else: > + return None > + > def bubble_up_suite_errors(test_suite_list: List[TestSuite]) -> TestStatus: > return bubble_up_errors(lambda x: x.status, test_suite_list) > > @@ -268,19 +288,34 @@ def parse_test_result(lines: List[str]) -> TestResult: > return TestResult(TestStatus.NO_TESTS, [], lines) > consume_non_diagnositic(lines) > if not parse_tap_header(lines): > - return None > + return TestResult(TestStatus.NO_TESTS, [], lines) > + expected_test_suite_num = parse_test_plan(lines) > + if not expected_test_suite_num: > + return TestResult(TestStatus.FAILURE_TO_PARSE_TESTS, [], lines) > test_suites = [] > - test_suite = parse_test_suite(lines) > - while test_suite: > - test_suites.append(test_suite) > - test_suite = parse_test_suite(lines) > - return TestResult(bubble_up_suite_errors(test_suites), test_suites, lines) > + for i in range(1, expected_test_suite_num + 1): > + test_suite = parse_test_suite(lines, i) > + if test_suite: > + test_suites.append(test_suite) > + else: > + print_with_timestamp( > + red('[ERROR] ') + ' expected ' + > + str(expected_test_suite_num) + > + ' test suites, but got ' + str(i - 2)) > + break > + test_suite = parse_test_suite(lines, -1) > + if test_suite: > + print_with_timestamp(red('[ERROR] ') + > + 'got unexpected test suite: ' + test_suite.name) > + if test_suites: > + return TestResult(bubble_up_suite_errors(test_suites), test_suites, lines) > + else: > + return TestResult(TestStatus.NO_TESTS, [], lines) > > -def parse_run_tests(kernel_output) -> TestResult: > +def print_and_count_results(test_result: TestResult) -> None: > total_tests = 0 > failed_tests = 0 > crashed_tests = 0 > - test_result = parse_test_result(list(isolate_kunit_output(kernel_output))) > for test_suite in test_result.suites: > if test_suite.status == TestStatus.SUCCESS: > print_suite_divider(green('[PASSED] ') + test_suite.name) > @@ -302,6 +337,21 @@ def parse_run_tests(kernel_output) -> TestResult: > print_with_timestamp(red('[FAILED] ') + test_case.name) > print_log(map(yellow, test_case.log)) > print_with_timestamp('') > + return total_tests, failed_tests, crashed_tests > + > +def parse_run_tests(kernel_output) -> TestResult: > + total_tests = 0 > + failed_tests = 0 > + crashed_tests = 0 > + test_result = parse_test_result(list(isolate_kunit_output(kernel_output))) > + if test_result.status == TestStatus.NO_TESTS: > + print(red('[ERROR] ') + yellow('no tests run!')) > + elif test_result.status == TestStatus.FAILURE_TO_PARSE_TESTS: > + print(red('[ERROR] ') + yellow('could not parse test results!')) > + else: > + (total_tests, > + failed_tests, > + crashed_tests) = print_and_count_results(test_result) > print_with_timestamp(DIVIDER) > fmt = green if test_result.status == TestStatus.SUCCESS else red > print_with_timestamp( > diff --git a/tools/testing/kunit/test_data/test_is_test_passed-all_passed.log b/tools/testing/kunit/test_data/test_is_test_passed-all_passed.log > index 62ebc0288355c..bc0dc8fe35b76 100644 > --- a/tools/testing/kunit/test_data/test_is_test_passed-all_passed.log > +++ b/tools/testing/kunit/test_data/test_is_test_passed-all_passed.log > @@ -1,4 +1,5 @@ > TAP version 14 > +1..2 > # Subtest: sysctl_test > 1..8 > # sysctl_test_dointvec_null_tbl_data: sysctl_test_dointvec_null_tbl_data passed > diff --git a/tools/testing/kunit/test_data/test_is_test_passed-crash.log b/tools/testing/kunit/test_data/test_is_test_passed-crash.log > index 0b249870c8be4..4d97f6708c4a5 100644 > --- a/tools/testing/kunit/test_data/test_is_test_passed-crash.log > +++ b/tools/testing/kunit/test_data/test_is_test_passed-crash.log > @@ -1,6 +1,7 @@ > printk: console [tty0] enabled > printk: console [mc-1] enabled > TAP version 14 > +1..2 > # Subtest: sysctl_test > 1..8 > # sysctl_test_dointvec_null_tbl_data: sysctl_test_dointvec_null_tbl_data passed > diff --git a/tools/testing/kunit/test_data/test_is_test_passed-failure.log b/tools/testing/kunit/test_data/test_is_test_passed-failure.log > index 9e89d32d5667a..7a416497e3bec 100644 > --- a/tools/testing/kunit/test_data/test_is_test_passed-failure.log > +++ b/tools/testing/kunit/test_data/test_is_test_passed-failure.log > @@ -1,4 +1,5 @@ > TAP version 14 > +1..2 > # Subtest: sysctl_test > 1..8 > # sysctl_test_dointvec_null_tbl_data: sysctl_test_dointvec_null_tbl_data passed >