summaryrefslogtreecommitdiffstats
path: root/js/src/tests/jstests.py
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/tests/jstests.py')
-rwxr-xr-xjs/src/tests/jstests.py877
1 files changed, 877 insertions, 0 deletions
diff --git a/js/src/tests/jstests.py b/js/src/tests/jstests.py
new file mode 100755
index 0000000000..f00897ddb0
--- /dev/null
+++ b/js/src/tests/jstests.py
@@ -0,0 +1,877 @@
+#!/usr/bin/env python
+# This Source Code Form is subject to the terms of the Mozilla Public
+# License, v. 2.0. If a copy of the MPL was not distributed with this
+# file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+"""
+The JS Shell Test Harness.
+
+See the adjacent README.txt for more details.
+"""
+
+import math
+import os
+import platform
+import posixpath
+import re
+import shlex
+import sys
+import tempfile
+from contextlib import contextmanager
+from copy import copy
+from datetime import datetime
+from itertools import chain
+from os.path import abspath, dirname, isfile, realpath
+from subprocess import call, list2cmdline
+
+from lib.adaptor import xdr_annotate
+from lib.progressbar import ProgressBar
+from lib.results import ResultsSink, TestOutput
+from lib.tempfile import TemporaryDirectory
+from lib.tests import (
+ RefTestCase,
+ change_env,
+ get_cpu_count,
+ get_environment_overlay,
+ get_jitflags,
+)
+
+if sys.platform.startswith("linux") or sys.platform.startswith("darwin"):
+ from lib.tasks_unix import run_all_tests
+else:
+ from lib.tasks_win import run_all_tests
+
+here = dirname(abspath(__file__))
+
+
+@contextmanager
+def changedir(dirname):
+ pwd = os.getcwd()
+ os.chdir(dirname)
+ try:
+ yield
+ finally:
+ os.chdir(pwd)
+
+
+class PathOptions(object):
+ def __init__(self, location, requested_paths, excluded_paths):
+ self.requested_paths = requested_paths
+ self.excluded_files, self.excluded_dirs = PathOptions._split_files_and_dirs(
+ location, excluded_paths
+ )
+
+ @staticmethod
+ def _split_files_and_dirs(location, paths):
+ """Split up a set of paths into files and directories"""
+ files, dirs = set(), set()
+ for path in paths:
+ fullpath = os.path.join(location, path)
+ if path.endswith("/"):
+ dirs.add(path[:-1])
+ elif os.path.isdir(fullpath):
+ dirs.add(path)
+ elif os.path.exists(fullpath):
+ files.add(path)
+
+ return files, dirs
+
+ def should_run(self, filename):
+ # If any tests are requested by name, skip tests that do not match.
+ if self.requested_paths and not any(
+ req in filename for req in self.requested_paths
+ ):
+ return False
+
+ # Skip excluded tests.
+ if filename in self.excluded_files:
+ return False
+
+ for dir in self.excluded_dirs:
+ if filename.startswith(dir + "/"):
+ return False
+
+ return True
+
+
+def parse_args():
+ """
+ Parse command line arguments.
+ Returns a tuple of: (options, js_shell, requested_paths, excluded_paths)
+ options :object: The raw OptionParser output.
+ js_shell :str: The absolute location of the shell to test with.
+ requested_paths :set<str>: Test paths specially requested on the CLI.
+ excluded_paths :set<str>: Test paths specifically excluded by the CLI.
+ """
+ from argparse import ArgumentParser
+
+ op = ArgumentParser(
+ description="Run jstests JS shell tests",
+ epilog="Shell output format: [ pass | fail | timeout | skip ] progress | time",
+ )
+ op.add_argument(
+ "--xul-info",
+ dest="xul_info_src",
+ help="config data for xulRuntime" " (avoids search for config/autoconf.mk)",
+ )
+
+ harness_og = op.add_argument_group("Harness Controls", "Control how tests are run.")
+ harness_og.add_argument(
+ "-j",
+ "--worker-count",
+ type=int,
+ default=max(1, get_cpu_count()),
+ help="Number of tests to run in parallel" " (default %(default)s)",
+ )
+ harness_og.add_argument(
+ "-t",
+ "--timeout",
+ type=float,
+ default=150.0,
+ help="Set maximum time a test is allows to run" " (in seconds).",
+ )
+ harness_og.add_argument(
+ "--show-slow",
+ action="store_true",
+ help="Show tests taking longer than a minimum time" " (in seconds).",
+ )
+ harness_og.add_argument(
+ "--slow-test-threshold",
+ type=float,
+ default=5.0,
+ help="Time in seconds a test can take until it is"
+ "considered slow (default %(default)s).",
+ )
+ harness_og.add_argument(
+ "-a",
+ "--args",
+ dest="shell_args",
+ default="",
+ help="Extra args to pass to the JS shell.",
+ )
+ harness_og.add_argument(
+ "--feature-args",
+ dest="feature_args",
+ default="",
+ help="Extra args to pass to the JS shell even when feature-testing.",
+ )
+ harness_og.add_argument(
+ "--jitflags",
+ dest="jitflags",
+ default="none",
+ type=str,
+ help="IonMonkey option combinations. One of all,"
+ " debug, ion, and none (default %(default)s).",
+ )
+ harness_og.add_argument(
+ "--tbpl",
+ action="store_true",
+ help="Runs each test in all configurations tbpl" " tests.",
+ )
+ harness_og.add_argument(
+ "--tbpl-debug",
+ action="store_true",
+ help="Runs each test in some faster configurations" " tbpl tests.",
+ )
+ harness_og.add_argument(
+ "-g", "--debug", action="store_true", help="Run a test in debugger."
+ )
+ harness_og.add_argument(
+ "--debugger", default="gdb -q --args", help="Debugger command."
+ )
+ harness_og.add_argument(
+ "-J", "--jorendb", action="store_true", help="Run under JS debugger."
+ )
+ harness_og.add_argument(
+ "--passthrough",
+ action="store_true",
+ help="Run tests with stdin/stdout attached to" " caller.",
+ )
+ harness_og.add_argument(
+ "--test-reflect-stringify",
+ dest="test_reflect_stringify",
+ help="instead of running tests, use them to test the "
+ "Reflect.stringify code in specified file",
+ )
+ harness_og.add_argument(
+ "--valgrind", action="store_true", help="Run tests in valgrind."
+ )
+ harness_og.add_argument(
+ "--valgrind-args", default="", help="Extra args to pass to valgrind."
+ )
+ harness_og.add_argument(
+ "--rr",
+ action="store_true",
+ help="Run tests under RR record-and-replay debugger.",
+ )
+ harness_og.add_argument(
+ "-C",
+ "--check-output",
+ action="store_true",
+ help="Run tests to check output for different jit-flags",
+ )
+ harness_og.add_argument(
+ "--remote", action="store_true", help="Run tests on a remote device"
+ )
+ harness_og.add_argument(
+ "--deviceIP",
+ action="store",
+ type=str,
+ dest="device_ip",
+ help="IP address of remote device to test",
+ )
+ harness_og.add_argument(
+ "--devicePort",
+ action="store",
+ type=int,
+ dest="device_port",
+ default=20701,
+ help="port of remote device to test",
+ )
+ harness_og.add_argument(
+ "--deviceSerial",
+ action="store",
+ type=str,
+ dest="device_serial",
+ default=None,
+ help="ADB device serial number of remote device to test",
+ )
+ harness_og.add_argument(
+ "--remoteTestRoot",
+ dest="remote_test_root",
+ action="store",
+ type=str,
+ default="/data/local/tmp/test_root",
+ help="The remote directory to use as test root" " (e.g. %(default)s)",
+ )
+ harness_og.add_argument(
+ "--localLib",
+ dest="local_lib",
+ action="store",
+ type=str,
+ help="The location of libraries to push -- preferably" " stripped",
+ )
+ harness_og.add_argument(
+ "--no-xdr",
+ dest="use_xdr",
+ action="store_false",
+ help="Whether to disable caching of self-hosted parsed content in XDR format.",
+ )
+
+ input_og = op.add_argument_group("Inputs", "Change what tests are run.")
+ input_og.add_argument(
+ "-f",
+ "--file",
+ dest="test_file",
+ action="append",
+ help="Get tests from the given file.",
+ )
+ input_og.add_argument(
+ "-x",
+ "--exclude-file",
+ action="append",
+ help="Exclude tests from the given file.",
+ )
+ input_og.add_argument(
+ "--wpt",
+ dest="wpt",
+ choices=["enabled", "disabled", "if-running-everything"],
+ default="if-running-everything",
+ help="Enable or disable shell web-platform-tests "
+ "(default: enable if no test paths are specified).",
+ )
+ input_og.add_argument(
+ "--include",
+ action="append",
+ dest="requested_paths",
+ default=[],
+ help="Include the given test file or directory.",
+ )
+ input_og.add_argument(
+ "--exclude",
+ action="append",
+ dest="excluded_paths",
+ default=[],
+ help="Exclude the given test file or directory.",
+ )
+ input_og.add_argument(
+ "-d",
+ "--exclude-random",
+ dest="random",
+ action="store_false",
+ help='Exclude tests marked as "random."',
+ )
+ input_og.add_argument(
+ "--run-skipped", action="store_true", help='Run tests marked as "skip."'
+ )
+ input_og.add_argument(
+ "--run-only-skipped",
+ action="store_true",
+ help='Run only tests marked as "skip."',
+ )
+ input_og.add_argument(
+ "--run-slow-tests",
+ action="store_true",
+ help='Do not skip tests marked as "slow."',
+ )
+ input_og.add_argument(
+ "--no-extensions",
+ action="store_true",
+ help="Run only tests conforming to the ECMAScript 5" " standard.",
+ )
+ input_og.add_argument(
+ "--repeat", type=int, default=1, help="Repeat tests the given number of times."
+ )
+
+ output_og = op.add_argument_group("Output", "Modify the harness and tests output.")
+ output_og.add_argument(
+ "-s",
+ "--show-cmd",
+ action="store_true",
+ help="Show exact commandline used to run each test.",
+ )
+ output_og.add_argument(
+ "-o",
+ "--show-output",
+ action="store_true",
+ help="Print each test's output to the file given by" " --output-file.",
+ )
+ output_og.add_argument(
+ "-F",
+ "--failed-only",
+ action="store_true",
+ help="If a --show-* option is given, only print" " output for failed tests.",
+ )
+ output_og.add_argument(
+ "--no-show-failed",
+ action="store_true",
+ help="Don't print output for failed tests" " (no-op with --show-output).",
+ )
+ output_og.add_argument(
+ "-O",
+ "--output-file",
+ help="Write all output to the given file" " (default: stdout).",
+ )
+ output_og.add_argument(
+ "--failure-file", help="Write all not-passed tests to the given file."
+ )
+ output_og.add_argument(
+ "--no-progress",
+ dest="hide_progress",
+ action="store_true",
+ help="Do not show the progress bar.",
+ )
+ output_og.add_argument(
+ "--tinderbox",
+ dest="format",
+ action="store_const",
+ const="automation",
+ help="Use automation-parseable output format.",
+ )
+ output_og.add_argument(
+ "--format",
+ dest="format",
+ default="none",
+ choices=["automation", "none"],
+ help="Output format. Either automation or none" " (default %(default)s).",
+ )
+ output_og.add_argument(
+ "--log-wptreport",
+ dest="wptreport",
+ action="store",
+ help="Path to write a Web Platform Tests report (wptreport)",
+ )
+ output_og.add_argument(
+ "--this-chunk", type=int, default=1, help="The test chunk to run."
+ )
+ output_og.add_argument(
+ "--total-chunks", type=int, default=1, help="The total number of test chunks."
+ )
+
+ special_og = op.add_argument_group(
+ "Special", "Special modes that do not run tests."
+ )
+ special_og.add_argument(
+ "--make-manifests",
+ metavar="BASE_TEST_PATH",
+ help="Generate reftest manifest files.",
+ )
+
+ op.add_argument("--js-shell", metavar="JS_SHELL", help="JS shell to run tests with")
+ op.add_argument(
+ "-z", "--gc-zeal", help="GC zeal mode to use when running the shell"
+ )
+
+ options, args = op.parse_known_args()
+
+ # Need a shell unless in a special mode.
+ if not options.make_manifests:
+ if not args:
+ op.error("missing JS_SHELL argument")
+ options.js_shell = os.path.abspath(args.pop(0))
+
+ requested_paths = set(args)
+
+ # Valgrind, gdb, and rr are mutually exclusive.
+ if sum(map(bool, (options.valgrind, options.debug, options.rr))) > 1:
+ op.error("--valgrind, --debug, and --rr are mutually exclusive.")
+
+ # Fill the debugger field, as needed.
+ if options.debug:
+ if options.debugger == "lldb":
+ debugger_prefix = ["lldb", "--"]
+ else:
+ debugger_prefix = options.debugger.split()
+ else:
+ debugger_prefix = []
+
+ if options.valgrind:
+ debugger_prefix = ["valgrind"] + options.valgrind_args.split()
+ if os.uname()[0] == "Darwin":
+ debugger_prefix.append("--dsymutil=yes")
+ options.show_output = True
+ if options.rr:
+ debugger_prefix = ["rr", "record"]
+
+ js_cmd_args = shlex.split(options.shell_args) + shlex.split(options.feature_args)
+ if options.jorendb:
+ options.passthrough = True
+ options.hide_progress = True
+ options.worker_count = 1
+ debugger_path = realpath(
+ os.path.join(
+ abspath(dirname(abspath(__file__))),
+ "..",
+ "..",
+ "examples",
+ "jorendb.js",
+ )
+ )
+ js_cmd_args.extend(["-d", "-f", debugger_path, "--"])
+ prefix = RefTestCase.build_js_cmd_prefix(
+ options.js_shell, js_cmd_args, debugger_prefix
+ )
+
+ # If files with lists of tests to run were specified, add them to the
+ # requested tests set.
+ if options.test_file:
+ for test_file in options.test_file:
+ requested_paths |= set(
+ [line.strip() for line in open(test_file).readlines()]
+ )
+
+ excluded_paths = set(options.excluded_paths)
+
+ # If files with lists of tests to exclude were specified, add them to the
+ # excluded tests set.
+ if options.exclude_file:
+ for filename in options.exclude_file:
+ with open(filename, "r") as fp:
+ for line in fp:
+ if line.startswith("#"):
+ continue
+ line = line.strip()
+ if not line:
+ continue
+ excluded_paths.add(line)
+
+ # Handle output redirection, if requested and relevant.
+ options.output_fp = sys.stdout
+ if options.output_file:
+ if not options.show_cmd:
+ options.show_output = True
+ try:
+ options.output_fp = open(options.output_file, "w")
+ except IOError as ex:
+ raise SystemExit("Failed to open output file: " + str(ex))
+
+ # Hide the progress bar if it will get in the way of other output.
+ options.hide_progress = (
+ options.format == "automation"
+ or not ProgressBar.conservative_isatty()
+ or options.hide_progress
+ )
+
+ return (options, prefix, requested_paths, excluded_paths)
+
+
+def load_wpt_tests(xul_tester, requested_paths, excluded_paths, update_manifest=True):
+ """Return a list of `RefTestCase` objects for the jsshell testharness.js
+ tests filtered by the given paths and debug-ness."""
+ repo_root = abspath(os.path.join(here, "..", "..", ".."))
+ wp = os.path.join(repo_root, "testing", "web-platform")
+ wpt = os.path.join(wp, "tests")
+
+ sys_paths = [
+ "python/mozterm",
+ "python/mozboot",
+ "testing/mozbase/mozcrash",
+ "testing/mozbase/mozdevice",
+ "testing/mozbase/mozfile",
+ "testing/mozbase/mozinfo",
+ "testing/mozbase/mozleak",
+ "testing/mozbase/mozlog",
+ "testing/mozbase/mozprocess",
+ "testing/mozbase/mozprofile",
+ "testing/mozbase/mozrunner",
+ "testing/mozbase/mozversion",
+ "testing/web-platform/",
+ "testing/web-platform/tests/tools",
+ "testing/web-platform/tests/tools/third_party/html5lib",
+ "testing/web-platform/tests/tools/third_party/webencodings",
+ "testing/web-platform/tests/tools/wptrunner",
+ "testing/web-platform/tests/tools/wptserve",
+ "third_party/python/requests",
+ ]
+ abs_sys_paths = [os.path.join(repo_root, path) for path in sys_paths]
+
+ failed = False
+ for path in abs_sys_paths:
+ if not os.path.isdir(path):
+ failed = True
+ print("Could not add '%s' to the path")
+ if failed:
+ return []
+
+ sys.path[0:0] = abs_sys_paths
+
+ import manifestupdate
+ from wptrunner import products, testloader, wptcommandline, wptlogging, wpttest
+
+ manifest_root = tempfile.gettempdir()
+ (maybe_dist, maybe_bin) = os.path.split(os.path.dirname(xul_tester.js_bin))
+ if maybe_bin == "bin":
+ (maybe_root, maybe_dist) = os.path.split(maybe_dist)
+ if maybe_dist == "dist":
+ if os.path.exists(os.path.join(maybe_root, "_tests")):
+ # Assume this is a gecko objdir.
+ manifest_root = maybe_root
+
+ logger = wptlogging.setup({}, {})
+
+ test_manifests = manifestupdate.run(
+ repo_root, manifest_root, logger, update=update_manifest
+ )
+
+ kwargs = vars(wptcommandline.create_parser().parse_args([]))
+ kwargs.update(
+ {
+ "config": os.path.join(
+ manifest_root, "_tests", "web-platform", "wptrunner.local.ini"
+ ),
+ "gecko_e10s": False,
+ "product": "firefox",
+ "verify": False,
+ "wasm": xul_tester.test("wasmIsSupported()"),
+ }
+ )
+ wptcommandline.set_from_config(kwargs)
+
+ def filter_jsshell_tests(it):
+ for item_type, path, tests in it:
+ tests = set(item for item in tests if item.jsshell)
+ if tests:
+ yield item_type, path, tests
+
+ run_info_extras = products.Product(kwargs["config"], "firefox").run_info_extras(
+ logger, **kwargs
+ )
+ run_info = wpttest.get_run_info(
+ kwargs["run_info"],
+ "firefox",
+ debug=xul_tester.test("isDebugBuild"),
+ extras=run_info_extras,
+ )
+ release_or_beta = xul_tester.test("getBuildConfiguration('release_or_beta')")
+ run_info["release_or_beta"] = release_or_beta
+ run_info["nightly_build"] = not release_or_beta
+ early_beta_or_earlier = xul_tester.test(
+ "getBuildConfiguration('early_beta_or_earlier')"
+ )
+ run_info["early_beta_or_earlier"] = early_beta_or_earlier
+
+ path_filter = testloader.TestFilter(
+ test_manifests, include=requested_paths, exclude=excluded_paths
+ )
+ subsuites = testloader.load_subsuites(logger, run_info, None, set())
+ loader = testloader.TestLoader(
+ test_manifests,
+ ["testharness"],
+ run_info,
+ subsuites=subsuites,
+ manifest_filters=[path_filter, filter_jsshell_tests],
+ )
+
+ extra_helper_paths = [
+ os.path.join(here, "web-platform-test-shims.js"),
+ os.path.join(wpt, "resources", "testharness.js"),
+ os.path.join(here, "testharnessreport.js"),
+ ]
+
+ def resolve(test_path, script):
+ if script.startswith("/"):
+ return os.path.join(wpt, script[1:])
+
+ return os.path.join(wpt, os.path.dirname(test_path), script)
+
+ tests = []
+ for test in loader.tests[""]["testharness"]:
+ test_path = os.path.relpath(test.path, wpt)
+ scripts = [resolve(test_path, s) for s in test.scripts]
+ extra_helper_paths_for_test = extra_helper_paths + scripts
+
+ # We must create at least one test with the default options, along with
+ # one test for each option given in a test-also annotation.
+ options = [None]
+ for m in test.itermeta():
+ if m.has_key("test-also"): # NOQA: W601
+ options += m.get("test-also").split()
+ for option in options:
+ test_case = RefTestCase(
+ wpt,
+ test_path,
+ extra_helper_paths=extra_helper_paths_for_test[:],
+ wpt=test,
+ )
+ if option:
+ test_case.options.append(option)
+ tests.append(test_case)
+ return tests
+
+
+def load_tests(options, requested_paths, excluded_paths):
+ """
+ Returns a tuple: (test_count, test_gen)
+ test_count: [int] Number of tests that will be in test_gen
+ test_gen: [iterable<Test>] Tests found that should be run.
+ """
+ import lib.manifest as manifest
+
+ if options.js_shell is None:
+ xul_tester = manifest.NullXULInfoTester()
+ else:
+ if options.xul_info_src is None:
+ xul_info = manifest.XULInfo.create(options.js_shell)
+ else:
+ xul_abi, xul_os, xul_debug = options.xul_info_src.split(r":")
+ xul_debug = xul_debug.lower() == "true"
+ xul_info = manifest.XULInfo(xul_abi, xul_os, xul_debug)
+ feature_args = shlex.split(options.feature_args)
+ xul_tester = manifest.XULInfoTester(xul_info, options, feature_args)
+
+ test_dir = dirname(abspath(__file__))
+ path_options = PathOptions(test_dir, requested_paths, excluded_paths)
+ test_count = manifest.count_tests(test_dir, path_options)
+ test_gen = manifest.load_reftests(test_dir, path_options, xul_tester)
+
+ # WPT tests are already run in the browser in their own harness.
+ wpt_enabled = options.wpt == "enabled" or (
+ options.wpt == "if-running-everything"
+ and len(requested_paths) == 0
+ and not options.make_manifests
+ )
+ if wpt_enabled:
+ wpt_tests = load_wpt_tests(xul_tester, requested_paths, excluded_paths)
+ test_count += len(wpt_tests)
+ test_gen = chain(test_gen, wpt_tests)
+
+ if options.test_reflect_stringify is not None:
+
+ def trs_gen(tests):
+ for test in tests:
+ test.test_reflect_stringify = options.test_reflect_stringify
+ # Even if the test is not normally expected to pass, we still
+ # expect reflect-stringify to be able to handle it.
+ test.expect = True
+ test.random = False
+ test.slow = False
+ yield test
+
+ test_gen = trs_gen(test_gen)
+
+ if options.make_manifests:
+ manifest.make_manifests(options.make_manifests, test_gen)
+ sys.exit()
+
+ # Create a new test list. Apply each TBPL configuration to every test.
+ flags_list = None
+ if options.tbpl:
+ flags_list = get_jitflags("all")
+ elif options.tbpl_debug:
+ flags_list = get_jitflags("debug")
+ else:
+ flags_list = get_jitflags(options.jitflags, none=None)
+
+ if flags_list:
+
+ def flag_gen(tests):
+ for test in tests:
+ for jitflags in flags_list:
+ tmp_test = copy(test)
+ tmp_test.jitflags = copy(test.jitflags)
+ tmp_test.jitflags.extend(jitflags)
+ yield tmp_test
+
+ test_count = test_count * len(flags_list)
+ test_gen = flag_gen(test_gen)
+
+ if options.test_file:
+ paths = set()
+ for test_file in options.test_file:
+ paths |= set([line.strip() for line in open(test_file).readlines()])
+ test_gen = (_ for _ in test_gen if _.path in paths)
+
+ if options.no_extensions:
+ pattern = os.sep + "extensions" + os.sep
+ test_gen = (_ for _ in test_gen if pattern not in _.path)
+
+ if not options.random:
+ test_gen = (_ for _ in test_gen if not _.random)
+
+ if options.run_only_skipped:
+ options.run_skipped = True
+ test_gen = (_ for _ in test_gen if not _.enable)
+
+ if not options.run_slow_tests:
+ test_gen = (_ for _ in test_gen if not _.slow)
+
+ if options.repeat:
+ test_gen = (test for test in test_gen for i in range(options.repeat))
+ test_count *= options.repeat
+
+ return test_count, test_gen
+
+
+def main():
+ options, prefix, requested_paths, excluded_paths = parse_args()
+ if options.js_shell is not None and not (
+ isfile(options.js_shell) and os.access(options.js_shell, os.X_OK)
+ ):
+ if (
+ platform.system() != "Windows"
+ or isfile(options.js_shell)
+ or not isfile(options.js_shell + ".exe")
+ or not os.access(options.js_shell + ".exe", os.X_OK)
+ ):
+ print("Could not find executable shell: " + options.js_shell)
+ return 1
+
+ test_count, test_gen = load_tests(options, requested_paths, excluded_paths)
+ test_environment = get_environment_overlay(options.js_shell, options.gc_zeal)
+
+ if test_count == 0:
+ print("no tests selected")
+ return 1
+
+ test_dir = dirname(abspath(__file__))
+
+ if options.debug:
+ if test_count > 1:
+ print(
+ "Multiple tests match command line arguments,"
+ " debugger can only run one"
+ )
+ for tc in test_gen:
+ print(" {}".format(tc.path))
+ return 2
+
+ with changedir(test_dir), change_env(
+ test_environment
+ ), TemporaryDirectory() as tempdir:
+ cmd = next(test_gen).get_command(prefix, tempdir)
+ if options.show_cmd:
+ print(list2cmdline(cmd))
+ call(cmd)
+ return 0
+
+ # The test_gen generator is converted into a list in
+ # run_all_tests. Go ahead and do it here so we can apply
+ # chunking.
+ #
+ # If chunking is enabled, determine which tests are part of this chunk.
+ # This code was adapted from testing/mochitest/runtestsremote.py.
+ if options.total_chunks > 1:
+ tests_per_chunk = math.ceil(test_count / float(options.total_chunks))
+ start = int(round((options.this_chunk - 1) * tests_per_chunk))
+ end = int(round(options.this_chunk * tests_per_chunk))
+ test_gen = list(test_gen)[start:end]
+
+ if options.remote:
+ results = ResultsSink("jstests", options, test_count)
+ try:
+ from lib.remote import init_device, init_remote_dir
+
+ device = init_device(options)
+ tempdir = posixpath.join(options.remote_test_root, "tmp")
+ jtd_tests = posixpath.join(options.remote_test_root, "tests", "tests")
+ init_remote_dir(device, jtd_tests)
+ device.push(test_dir, jtd_tests, timeout=600)
+ device.chmod(jtd_tests, recursive=True)
+ prefix[0] = options.js_shell
+ if options.use_xdr:
+ test_gen = xdr_annotate(test_gen, options)
+ for test in test_gen:
+ out = run_test_remote(test, device, prefix, tempdir, options)
+ results.push(out)
+ results.finish(True)
+ except KeyboardInterrupt:
+ results.finish(False)
+
+ return 0 if results.all_passed() else 1
+
+ with changedir(test_dir), change_env(
+ test_environment
+ ), TemporaryDirectory() as tempdir:
+ results = ResultsSink("jstests", options, test_count)
+ try:
+ for out in run_all_tests(test_gen, prefix, tempdir, results.pb, options):
+ results.push(out)
+ results.finish(True)
+ except KeyboardInterrupt:
+ results.finish(False)
+
+ return 0 if results.all_passed() else 1
+
+ return 0
+
+
+def run_test_remote(test, device, prefix, tempdir, options):
+ from mozdevice import ADBDevice, ADBProcessError
+
+ cmd = test.get_command(prefix, tempdir)
+ test_root_parent = os.path.dirname(test.root)
+ jtd_tests = posixpath.join(options.remote_test_root, "tests")
+ cmd = [_.replace(test_root_parent, jtd_tests) for _ in cmd]
+
+ env = {"TZ": "PST8PDT", "LD_LIBRARY_PATH": os.path.dirname(prefix[0])}
+
+ adb_cmd = ADBDevice._escape_command_line(cmd)
+ start = datetime.now()
+ try:
+ # Allow ADBError or ADBTimeoutError to terminate the test run,
+ # but handle ADBProcessError in order to support the use of
+ # non-zero exit codes in the JavaScript shell tests.
+ out = device.shell_output(
+ adb_cmd, env=env, cwd=options.remote_test_root, timeout=int(options.timeout)
+ )
+ returncode = 0
+ except ADBProcessError as e:
+ # Treat ignorable intermittent adb communication errors as
+ # skipped tests.
+ out = str(e.adb_process.stdout)
+ returncode = e.adb_process.exitcode
+ re_ignore = re.compile(r"error: (closed|device .* not found)")
+ if returncode == 1 and re_ignore.search(out):
+ print("Skipping {} due to ignorable adb error {}".format(test.path, out))
+ test.skip_if_cond = "true"
+ returncode = test.SKIPPED_EXIT_STATUS
+
+ elapsed = (datetime.now() - start).total_seconds()
+
+ # We can't distinguish between stdout and stderr so we pass
+ # the same buffer to both.
+ return TestOutput(test, cmd, out, out, returncode, elapsed, False)
+
+
+if __name__ == "__main__":
+ sys.exit(main())