aboutsummaryrefslogtreecommitdiffstats
path: root/tests/test-runner/bin
diff options
context:
space:
mode:
Diffstat (limited to 'tests/test-runner/bin')
-rw-r--r--tests/test-runner/bin/Makefile.am3
-rwxr-xr-xtests/test-runner/bin/zts-report.py369
2 files changed, 371 insertions, 1 deletions
diff --git a/tests/test-runner/bin/Makefile.am b/tests/test-runner/bin/Makefile.am
index 223622dda..e843e4e09 100644
--- a/tests/test-runner/bin/Makefile.am
+++ b/tests/test-runner/bin/Makefile.am
@@ -1,3 +1,4 @@
pkgdatadir = $(datadir)/@PACKAGE@/test-runner/bin
dist_pkgdata_SCRIPTS = \
- test-runner.py
+ test-runner.py \
+ zts-report.py
diff --git a/tests/test-runner/bin/zts-report.py b/tests/test-runner/bin/zts-report.py
new file mode 100755
index 000000000..976ca1f8a
--- /dev/null
+++ b/tests/test-runner/bin/zts-report.py
@@ -0,0 +1,369 @@
+#!/usr/bin/python
+
+#
+# This file and its contents are supplied under the terms of the
+# Common Development and Distribution License ("CDDL"), version 1.0.
+# You may only use this file in accordance with the terms of version
+# 1.0 of the CDDL.
+#
+# A full copy of the text of the CDDL should have accompanied this
+# source. A copy of the CDDL is also available via the Internet at
+# http://www.illumos.org/license/CDDL.
+#
+
+#
+# Copyright (c) 2017 by Delphix. All rights reserved.
+# Copyright (c) 2018 by Lawrence Livermore National Security, LLC.
+#
+
+import os
+import re
+import sys
+
+#
+# This script parses the stdout of zfstest, which has this format:
+#
+# Test: /path/to/testa (run as root) [00:00] [PASS]
+# Test: /path/to/testb (run as jkennedy) [00:00] [PASS]
+# Test: /path/to/testc (run as root) [00:00] [FAIL]
+# [...many more results...]
+#
+# Results Summary
+# FAIL 22
+# SKIP 32
+# PASS 1156
+#
+# Running Time: 02:50:31
+# Percent passed: 95.5%
+# Log directory: /var/tmp/test_results/20180615T205926
+#
+
+#
+# Common generic reasons for a test or test group to be skipped.
+#
+# Some test cases are known to fail in ways which are not harmful or dangerous.
+# In these cases simply mark the test as a known failure until it can be
+# updated and the issue resolved. Note that it's preferable to open a unique
+# issue on the GitHub issue tracker for each test case failure.
+#
+known_reason = 'Known issue'
+
+#
+# Some tests require that a test user be able to execute the zfs utilities.
+# This may not be possible when testing in-tree due to the default permissions
+# on the user's home directory. When testing this can be resolved by granting
+# group read access.
+#
+# chmod 0750 $HOME
+#
+exec_reason = 'Test user execute permissions required for utilities'
+
+#
+# Some tests require that the DISKS provided can be partitioned. This is
+# normally not an issue because loop back devices are used for DISKS and they
+# can be partition. There is one notable exception, the CentOS 6.x kernel is
+# old enough that it does not support partitioning loop back devices.
+#
+disk_reason = 'Partitionable DISKS required'
+
+#
+# Some tests require a minimum python version of 3.5 and will be skipped when
+# the default system version is too old. There may also be tests which require
+# additional python modules be installed, for example python-cffi is required
+# by the pyzfs tests.
+#
+python_reason = 'Python v3.5 or newer required'
+python_deps_reason = 'Python modules missing: python-cffi'
+
+#
+# Some tests require the O_TMPFILE flag which was first introduced in the
+# 3.11 kernel.
+#
+tmpfile_reason = 'Kernel O_TMPFILE support required'
+
+#
+# Some tests require that the NFS client and server utilities be installed.
+#
+share_reason = 'NFS client and server utilities required'
+
+#
+# Some tests require that the lsattr utility support the project id feature.
+#
+project_id_reason = 'lsattr with set/show project ID required'
+
+#
+# Some tests require that the kernel support user namespaces.
+#
+user_ns_reason = 'Kernel user namespace support required'
+
+#
+# Some rewind tests can fail since nothing guarantees that old MOS blocks
+# are not overwritten. Snapshots protect datasets and data files but not
+# the MOS. Reasonable efforts are made in the test case to increase the
+# odds that some txgs will have their MOS data left untouched, but it is
+# never a sure thing.
+#
+rewind_reason = 'Arbitrary pool rewind is not guaranteed'
+
+#
+# Some tests are not applicable to Linux or need to be updated to operate
+# in the manor required by Linux. Any tests which are skipped for this
+# reason will be suppressed in the final analysis output.
+#
+na_reason = "N/A on Linux"
+
+summary = {
+ 'total': float(0),
+ 'passed': float(0),
+ 'logfile': "Could not determine logfile location."
+}
+
+#
+# These tests are known to fail, thus we use this list to prevent these
+# failures from failing the job as a whole; only unexpected failures
+# bubble up to cause this script to exit with a non-zero exit status.
+#
+# Format: { 'test-name': ['expected result', 'issue-number | reason'] }
+#
+# For each known failure it is recommended to link to a GitHub issue by
+# setting the reason to the issue number. Alternately, one of the generic
+# reasons listed above can be used.
+#
+known = {
+ 'acl/posix/posix_001_pos': ['FAIL', known_reason],
+ 'acl/posix/posix_002_pos': ['FAIL', known_reason],
+ 'casenorm/sensitive_none_lookup': ['FAIL', '7633'],
+ 'casenorm/sensitive_none_delete': ['FAIL', '7633'],
+ 'casenorm/sensitive_formd_lookup': ['FAIL', '7633'],
+ 'casenorm/sensitive_formd_delete': ['FAIL', '7633'],
+ 'casenorm/insensitive_none_lookup': ['FAIL', '7633'],
+ 'casenorm/insensitive_none_delete': ['FAIL', '7633'],
+ 'casenorm/insensitive_formd_lookup': ['FAIL', '7633'],
+ 'casenorm/insensitive_formd_delete': ['FAIL', '7633'],
+ 'casenorm/mixed_none_lookup': ['FAIL', '7633'],
+ 'casenorm/mixed_none_lookup_ci': ['FAIL', '7633'],
+ 'casenorm/mixed_none_delete': ['FAIL', '7633'],
+ 'casenorm/mixed_formd_lookup': ['FAIL', '7633'],
+ 'casenorm/mixed_formd_lookup_ci': ['FAIL', '7633'],
+ 'casenorm/mixed_formd_delete': ['FAIL', '7633'],
+ 'cli_root/zfs_mount/zfs_mount_006_pos': ['SKIP', '4990'],
+ 'cli_root/zfs_receive/zfs_receive_004_neg': ['FAIL', known_reason],
+ 'cli_root/zfs_unshare/zfs_unshare_002_pos': ['SKIP', na_reason],
+ 'cli_root/zfs_unshare/zfs_unshare_006_pos': ['SKIP', na_reason],
+ 'cli_root/zpool_create/zpool_create_016_pos': ['SKIP', na_reason],
+ 'cli_root/zpool_expand/zpool_expand_001_pos': ['SKIP', '5771'],
+ 'cli_root/zpool_expand/zpool_expand_003_neg': ['SKIP', '5771'],
+ 'cli_user/misc/zfs_share_001_neg': ['SKIP', na_reason],
+ 'cli_user/misc/zfs_unshare_001_neg': ['SKIP', na_reason],
+ 'inuse/inuse_001_pos': ['SKIP', na_reason],
+ 'inuse/inuse_003_pos': ['SKIP', na_reason],
+ 'inuse/inuse_006_pos': ['SKIP', na_reason],
+ 'inuse/inuse_007_pos': ['SKIP', na_reason],
+ 'privilege/setup': ['SKIP', na_reason],
+ 'refreserv/refreserv_004_pos': ['FAIL', known_reason],
+ 'removal/removal_condense_export': ['SKIP', known_reason],
+ 'removal/removal_with_zdb': ['SKIP', known_reason],
+ 'rootpool/setup': ['SKIP', na_reason],
+ 'rsend/rsend_008_pos': ['SKIP', '6066'],
+ 'vdev_zaps/vdev_zaps_007_pos': ['FAIL', known_reason],
+ 'xattr/xattr_008_pos': ['SKIP', na_reason],
+ 'xattr/xattr_009_neg': ['SKIP', na_reason],
+ 'xattr/xattr_010_neg': ['SKIP', na_reason],
+ 'zvol/zvol_misc/zvol_misc_001_neg': ['SKIP', na_reason],
+ 'zvol/zvol_misc/zvol_misc_003_neg': ['SKIP', na_reason],
+ 'zvol/zvol_misc/zvol_misc_004_pos': ['SKIP', na_reason],
+ 'zvol/zvol_misc/zvol_misc_005_neg': ['SKIP', na_reason],
+ 'zvol/zvol_misc/zvol_misc_006_pos': ['SKIP', na_reason],
+ 'zvol/zvol_swap/zvol_swap_003_pos': ['SKIP', na_reason],
+ 'zvol/zvol_swap/zvol_swap_005_pos': ['SKIP', na_reason],
+ 'zvol/zvol_swap/zvol_swap_006_pos': ['SKIP', na_reason],
+}
+
+#
+# These tests may occasionally fail or be skipped. We want there failures
+# to be reported but only unexpected failures should bubble up to cause
+# this script to exit with a non-zero exit status.
+#
+# Format: { 'test-name': ['expected result', 'issue-number | reason'] }
+#
+# For each known failure it is recommended to link to a GitHub issue by
+# setting the reason to the issue number. Alternately, one of the generic
+# reasons listed above can be used.
+#
+maybe = {
+ 'cache/setup': ['SKIP', disk_reason],
+ 'cache/cache_010_neg': ['FAIL', known_reason],
+ 'chattr/setup': ['SKIP', exec_reason],
+ 'clean_mirror/setup': ['SKIP', disk_reason],
+ 'cli_root/zdb/zdb_006_pos': ['FAIL', known_reason],
+ 'cli_root/zfs_get/zfs_get_004_pos': ['FAIL', known_reason],
+ 'cli_root/zfs_get/zfs_get_009_pos': ['SKIP', '5479'],
+ 'cli_root/zfs_receive/receive-o-x_props_override':
+ ['FAIL', known_reason],
+ 'cli_root/zfs_rename/zfs_rename_006_pos': ['FAIL', '5647'],
+ 'cli_root/zfs_rename/zfs_rename_009_neg': ['FAIL', '5648'],
+ 'cli_root/zfs_rollback/zfs_rollback_001_pos': ['FAIL', '6415'],
+ 'cli_root/zfs_rollback/zfs_rollback_002_pos': ['FAIL', '6416'],
+ 'cli_root/zfs_share/setup': ['SKIP', share_reason],
+ 'cli_root/zfs_snapshot/zfs_snapshot_002_neg': ['FAIL', known_reason],
+ 'cli_root/zfs_unshare/setup': ['SKIP', share_reason],
+ 'cli_root/zpool_add/setup': ['SKIP', disk_reason],
+ 'cli_root/zpool_add/zpool_add_004_pos': ['FAIL', known_reason],
+ 'cli_root/zpool_create/setup': ['SKIP', disk_reason],
+ 'cli_root/zpool_create/zpool_create_008_pos': ['FAIL', known_reason],
+ 'cli_root/zpool_destroy/zpool_destroy_001_pos': ['SKIP', '6145'],
+ 'cli_root/zpool_export/setup': ['SKIP', disk_reason],
+ 'cli_root/zpool_import/setup': ['SKIP', disk_reason],
+ 'cli_root/zpool_import/import_rewind_device_replaced':
+ ['FAIL', rewind_reason],
+ 'cli_root/zpool_import/import_rewind_config_changed':
+ ['FAIL', rewind_reason],
+ 'cli_root/zpool_import/zpool_import_missing_003_pos': ['SKIP', '6839'],
+ 'cli_root/zpool_remove/setup': ['SKIP', disk_reason],
+ 'cli_root/zpool_upgrade/zpool_upgrade_004_pos': ['FAIL', '6141'],
+ 'cli_user/misc/arc_summary3_001_pos': ['SKIP', python_reason],
+ 'delegate/setup': ['SKIP', exec_reason],
+ 'fault/auto_online_001_pos': ['SKIP', disk_reason],
+ 'fault/auto_replace_001_pos': ['SKIP', disk_reason],
+ 'history/history_004_pos': ['FAIL', '7026'],
+ 'history/history_005_neg': ['FAIL', '6680'],
+ 'history/history_006_neg': ['FAIL', '5657'],
+ 'history/history_008_pos': ['FAIL', known_reason],
+ 'history/history_010_pos': ['SKIP', exec_reason],
+ 'inuse/inuse_005_pos': ['SKIP', disk_reason],
+ 'inuse/inuse_008_pos': ['SKIP', disk_reason],
+ 'inuse/inuse_009_pos': ['SKIP', disk_reason],
+ 'largest_pool/largest_pool_001_pos': ['FAIL', known_reason],
+ 'pyzfs/pyzfs_unittest': ['SKIP', python_deps_reason],
+ 'no_space/setup': ['SKIP', disk_reason],
+ 'no_space/enospc_002_pos': ['FAIL', known_reason],
+ 'projectquota/setup': ['SKIP', exec_reason],
+ 'reservation/reservation_018_pos': ['FAIL', '5642'],
+ 'rsend/rsend_019_pos': ['FAIL', '6086'],
+ 'rsend/rsend_020_pos': ['FAIL', '6446'],
+ 'rsend/rsend_021_pos': ['FAIL', '6446'],
+ 'rsend/rsend_024_pos': ['FAIL', '5665'],
+ 'rsend/send-c_volume': ['FAIL', '6087'],
+ 'scrub_mirror/setup': ['SKIP', disk_reason],
+ 'snapshot/clone_001_pos': ['FAIL', known_reason],
+ 'snapused/snapused_004_pos': ['FAIL', '5513'],
+ 'tmpfile/setup': ['SKIP', tmpfile_reason],
+ 'threadsappend/threadsappend_001_pos': ['FAIL', '6136'],
+ 'upgrade/upgrade_projectquota_001_pos': ['SKIP', project_id_reason],
+ 'user_namespace/setup': ['SKIP', user_ns_reason],
+ 'userquota/setup': ['SKIP', exec_reason],
+ 'vdev_zaps/vdev_zaps_004_pos': ['FAIL', '6935'],
+ 'write_dirs/setup': ['SKIP', disk_reason],
+ 'zvol/zvol_ENOSPC/zvol_ENOSPC_001_pos': ['FAIL', '5848'],
+}
+
+
+def usage(s):
+ print s
+ sys.exit(1)
+
+
+def process_results(pathname):
+ try:
+ f = open(pathname)
+ except IOError, e:
+ print 'Error opening file: %s' % e
+ sys.exit(1)
+
+ prefix = '/zfs-tests/tests/functional/'
+ pattern = '^Test:\s*\S*%s(\S+)\s*\(run as (\S+)\)\s*\[(\S+)\]\s*\[(\S+)\]'\
+ % prefix
+ pattern_log = '^\s*Log directory:\s*(\S*)'
+
+ d = {}
+ for l in f.readlines():
+ m = re.match(pattern, l)
+ if m and len(m.groups()) == 4:
+ summary['total'] += 1
+ if m.group(4) == "PASS":
+ summary['passed'] += 1
+ d[m.group(1)] = m.group(4)
+ continue
+
+ m = re.match(pattern_log, l)
+ if m:
+ summary['logfile'] = m.group(1)
+
+ return d
+
+
+if __name__ == "__main__":
+ if len(sys.argv) is not 2:
+ usage('usage: %s <pathname>' % sys.argv[0])
+ results = process_results(sys.argv[1])
+
+ if summary['total'] == 0:
+ print "\n\nNo test results were found."
+ print "Log directory: %s" % summary['logfile']
+ sys.exit(0)
+
+ expected = []
+ unexpected = []
+
+ for test in results.keys():
+ if results[test] == "PASS":
+ continue
+
+ setup = test.replace(os.path.basename(test), "setup")
+ if results[test] == "SKIP" and test != setup:
+ if setup in known and known[setup][0] == "SKIP":
+ continue
+ if setup in maybe and maybe[setup][0] == "SKIP":
+ continue
+
+ if ((test not in known or results[test] not in known[test][0]) and
+ (test not in maybe or results[test] not in maybe[test][0])):
+ unexpected.append(test)
+ else:
+ expected.append(test)
+
+ print "\nTests with results other than PASS that are expected:"
+ for test in sorted(expected):
+ issue_url = 'https://github.com/zfsonlinux/zfs/issues/'
+
+ # Include the reason why the result is expected, given the following:
+ # 1. Suppress test results which set the "N/A on Linux" reason.
+ # 2. Numerical reasons are assumed to be GitHub issue numbers.
+ # 3. When an entire test group is skipped only report the setup reason.
+ if test in known:
+ if known[test][1] == na_reason:
+ continue
+ elif known[test][1].isdigit():
+ expect = issue_url + known[test][1]
+ else:
+ expect = known[test][1]
+ elif test in maybe:
+ if maybe[test][1].isdigit():
+ expect = issue_url + maybe[test][1]
+ else:
+ expect = maybe[test][1]
+ elif setup in known and known[setup][0] == "SKIP" and setup != test:
+ continue
+ elif setup in maybe and maybe[setup][0] == "SKIP" and setup != test:
+ continue
+ else:
+ expect = "UNKNOWN REASON"
+ print " %s %s (%s)" % (results[test], test, expect)
+
+ print "\nTests with result of PASS that are unexpected:"
+ for test in sorted(known.keys()):
+ # We probably should not be silently ignoring the case
+ # where "test" is not in "results".
+ if test not in results or results[test] != "PASS":
+ continue
+ print " %s %s (expected %s)" % (results[test], test, known[test][0])
+
+ print "\nTests with results other than PASS that are unexpected:"
+ for test in sorted(unexpected):
+ expect = "PASS" if test not in known else known[test][0]
+ print " %s %s (expected %s)" % (results[test], test, expect)
+
+ if len(unexpected) == 0:
+ sys.exit(0)
+ else:
+ sys.exit(1)