forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			343 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			343 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			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/.
 | |
| 
 | |
| import argparse
 | |
| import logging
 | |
| import os
 | |
| import sys
 | |
| 
 | |
| import mozinfo
 | |
| import six
 | |
| from mach.decorators import Command, CommandArgument, CommandArgumentGroup
 | |
| from mozbuild.base import BinaryNotFoundException
 | |
| from mozbuild.base import MachCommandConditions as conditions
 | |
| 
 | |
| 
 | |
| def setup_awsy_argument_parser():
 | |
|     from marionette_harness.runtests import MarionetteArguments
 | |
|     from mozlog.structured import commandline
 | |
| 
 | |
|     parser = MarionetteArguments()
 | |
|     commandline.add_logging_group(parser)
 | |
| 
 | |
|     return parser
 | |
| 
 | |
| 
 | |
| from awsy import ITERATIONS, MAX_TABS, PER_TAB_PAUSE, SETTLE_WAIT_TIME
 | |
| 
 | |
| 
 | |
| def run_awsy(command_context, tests, binary=None, **kwargs):
 | |
|     import json
 | |
| 
 | |
|     from marionette_harness.runtests import MarionetteHarness, MarionetteTestRunner
 | |
|     from mozlog.structured import commandline
 | |
| 
 | |
|     parser = setup_awsy_argument_parser()
 | |
| 
 | |
|     awsy_source_dir = os.path.join(command_context.topsrcdir, "testing", "awsy")
 | |
|     if not tests:
 | |
|         if kwargs["base"]:
 | |
|             filename = "test_base_memory_usage.py"
 | |
|         else:
 | |
|             filename = "test_memory_usage.py"
 | |
|         tests = [os.path.join(awsy_source_dir, "awsy", filename)]
 | |
| 
 | |
|     args = argparse.Namespace(tests=tests)
 | |
| 
 | |
|     args.binary = binary
 | |
| 
 | |
|     if kwargs["quick"]:
 | |
|         kwargs["entities"] = 3
 | |
|         kwargs["iterations"] = 1
 | |
|         kwargs["perTabPause"] = 1
 | |
|         kwargs["settleWaitTime"] = 1
 | |
| 
 | |
|     runtime_testvars = {}
 | |
|     for arg in (
 | |
|         "webRootDir",
 | |
|         "pageManifest",
 | |
|         "resultsDir",
 | |
|         "entities",
 | |
|         "iterations",
 | |
|         "perTabPause",
 | |
|         "settleWaitTime",
 | |
|         "maxTabs",
 | |
|         "dmd",
 | |
|         "tp6",
 | |
|     ):
 | |
|         if arg in kwargs and kwargs[arg] is not None:
 | |
|             runtime_testvars[arg] = kwargs[arg]
 | |
| 
 | |
|     if "webRootDir" not in runtime_testvars:
 | |
|         awsy_tests_dir = os.path.join(command_context.topobjdir, "_tests", "awsy")
 | |
|         web_root_dir = os.path.join(awsy_tests_dir, "html")
 | |
|         runtime_testvars["webRootDir"] = web_root_dir
 | |
|     else:
 | |
|         web_root_dir = runtime_testvars["webRootDir"]
 | |
|         awsy_tests_dir = os.path.dirname(web_root_dir)
 | |
| 
 | |
|     if "resultsDir" not in runtime_testvars:
 | |
|         runtime_testvars["resultsDir"] = os.path.join(awsy_tests_dir, "results")
 | |
| 
 | |
|     runtime_testvars["bin"] = binary
 | |
|     runtime_testvars["run_local"] = True
 | |
| 
 | |
|     page_load_test_dir = os.path.join(web_root_dir, "page_load_test")
 | |
|     if not os.path.isdir(page_load_test_dir):
 | |
|         os.makedirs(page_load_test_dir)
 | |
| 
 | |
|     if not os.path.isdir(runtime_testvars["resultsDir"]):
 | |
|         os.makedirs(runtime_testvars["resultsDir"])
 | |
| 
 | |
|     runtime_testvars_path = os.path.join(awsy_tests_dir, "runtime-testvars.json")
 | |
|     if kwargs["testvars"]:
 | |
|         kwargs["testvars"].append(runtime_testvars_path)
 | |
|     else:
 | |
|         kwargs["testvars"] = [runtime_testvars_path]
 | |
| 
 | |
|     runtime_testvars_file = open(runtime_testvars_path, "wb" if six.PY2 else "w")
 | |
|     runtime_testvars_file.write(json.dumps(runtime_testvars, indent=2))
 | |
|     runtime_testvars_file.close()
 | |
| 
 | |
|     manifest_file = os.path.join(awsy_source_dir, "tp5n-pageset.manifest")
 | |
|     tooltool_args = {
 | |
|         "args": [
 | |
|             sys.executable,
 | |
|             os.path.join(command_context.topsrcdir, "mach"),
 | |
|             "artifact",
 | |
|             "toolchain",
 | |
|             "-v",
 | |
|             "--tooltool-manifest=%s" % manifest_file,
 | |
|             "--cache-dir=%s"
 | |
|             % os.path.join(command_context._mach_context.state_dir, "tooltool-cache"),
 | |
|         ]
 | |
|     }
 | |
|     command_context.run_process(cwd=page_load_test_dir, **tooltool_args)
 | |
|     tp5nzip = os.path.join(page_load_test_dir, "tp5n.zip")
 | |
|     tp5nmanifest = os.path.join(page_load_test_dir, "tp5n", "tp5n.manifest")
 | |
|     if not os.path.exists(tp5nmanifest):
 | |
|         unzip_args = {"args": ["unzip", "-q", "-o", tp5nzip, "-d", page_load_test_dir]}
 | |
|         try:
 | |
|             command_context.run_process(**unzip_args)
 | |
|         except Exception as exc:
 | |
|             troubleshoot = ""
 | |
|             if mozinfo.os == "win":
 | |
|                 troubleshoot = (
 | |
|                     " Try using --web-root to specify a "
 | |
|                     "directory closer to the drive root."
 | |
|                 )
 | |
| 
 | |
|             command_context.log(
 | |
|                 logging.ERROR,
 | |
|                 "awsy",
 | |
|                 {"directory": page_load_test_dir, "exception": exc},
 | |
|                 "Failed to unzip `tp5n.zip` into "
 | |
|                 "`{directory}` with `{exception}`." + troubleshoot,
 | |
|             )
 | |
|             raise exc
 | |
| 
 | |
|     # If '--preferences' was not specified supply our default set.
 | |
|     if not kwargs["prefs_files"]:
 | |
|         kwargs["prefs_files"] = [os.path.join(awsy_source_dir, "conf", "prefs.json")]
 | |
| 
 | |
|     # Setup DMD env vars if necessary.
 | |
|     if kwargs["dmd"]:
 | |
|         bin_dir = os.path.dirname(binary)
 | |
| 
 | |
|         if "DMD" not in os.environ:
 | |
|             os.environ["DMD"] = "1"
 | |
| 
 | |
|         # Work around a startup crash with DMD on windows
 | |
|         if mozinfo.os == "win":
 | |
|             kwargs["pref"] = "security.sandbox.content.level:0"
 | |
|             command_context.log(
 | |
|                 logging.WARNING,
 | |
|                 "awsy",
 | |
|                 {},
 | |
|                 "Forcing 'security.sandbox.content.level' = 0 because DMD is enabled.",
 | |
|             )
 | |
|         elif mozinfo.os == "mac":
 | |
|             # On mac binary is in MacOS and dmd.py is in Resources, ie:
 | |
|             #   Name.app/Contents/MacOS/libdmd.dylib
 | |
|             #   Name.app/Contents/Resources/dmd.py
 | |
|             bin_dir = os.path.join(bin_dir, "../Resources/")
 | |
| 
 | |
|         # Also add the bin dir to the python path so we can use dmd.py
 | |
|         if bin_dir not in sys.path:
 | |
|             sys.path.append(bin_dir)
 | |
| 
 | |
|     for k, v in six.iteritems(kwargs):
 | |
|         setattr(args, k, v)
 | |
| 
 | |
|     parser.verify_usage(args)
 | |
| 
 | |
|     args.logger = commandline.setup_logging(
 | |
|         "Are We Slim Yet Tests", args, {"mach": sys.stdout}
 | |
|     )
 | |
|     failed = MarionetteHarness(MarionetteTestRunner, args=vars(args)).run()
 | |
|     if failed > 0:
 | |
|         return 1
 | |
|     else:
 | |
|         return 0
 | |
| 
 | |
| 
 | |
| @Command(
 | |
|     "awsy-test",
 | |
|     category="testing",
 | |
|     description="Run Are We Slim Yet (AWSY) memory usage testing using marionette.",
 | |
|     parser=setup_awsy_argument_parser,
 | |
| )
 | |
| @CommandArgumentGroup("AWSY")
 | |
| @CommandArgument(
 | |
|     "--web-root",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=str,
 | |
|     dest="webRootDir",
 | |
|     help="Path to web server root directory. If not specified, "
 | |
|     "defaults to topobjdir/_tests/awsy/html.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--page-manifest",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=str,
 | |
|     dest="pageManifest",
 | |
|     help="Path to page manifest text file containing a list "
 | |
|     "of urls to test. The urls must be served from localhost. If not "
 | |
|     "specified, defaults to page_load_test/tp5n/tp5n.manifest under "
 | |
|     "the web root.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--results",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=str,
 | |
|     dest="resultsDir",
 | |
|     help="Path to results directory. If not specified, defaults "
 | |
|     "to the parent directory of the web root.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--quick",
 | |
|     group="AWSY",
 | |
|     action="store_true",
 | |
|     dest="quick",
 | |
|     default=False,
 | |
|     help="Set --entities=3, --iterations=1, --per-tab-pause=1, "
 | |
|     "--settle-wait-time=1 for a quick test. Overrides any explicit "
 | |
|     "argument settings.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--entities",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=int,
 | |
|     dest="entities",
 | |
|     help="Number of urls to load. Defaults to the total number of urls.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--max-tabs",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=int,
 | |
|     dest="maxTabs",
 | |
|     help="Maximum number of tabs to open. Defaults to %s." % MAX_TABS,
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--iterations",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=int,
 | |
|     dest="iterations",
 | |
|     help="Number of times to run through the test suite. "
 | |
|     "Defaults to %s." % ITERATIONS,
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--per-tab-pause",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=int,
 | |
|     dest="perTabPause",
 | |
|     help="Seconds to wait in between opening tabs. Defaults to %s." % PER_TAB_PAUSE,
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--settle-wait-time",
 | |
|     group="AWSY",
 | |
|     action="store",
 | |
|     type=int,
 | |
|     dest="settleWaitTime",
 | |
|     help="Seconds to wait for things to settled down. "
 | |
|     "Defaults to %s." % SETTLE_WAIT_TIME,
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--dmd",
 | |
|     group="AWSY",
 | |
|     action="store_true",
 | |
|     dest="dmd",
 | |
|     default=False,
 | |
|     help="Enable DMD during testing. Requires a DMD-enabled build.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--tp6",
 | |
|     group="AWSY",
 | |
|     action="store_true",
 | |
|     dest="tp6",
 | |
|     default=False,
 | |
|     help="Use the tp6 pageset during testing.",
 | |
| )
 | |
| @CommandArgument(
 | |
|     "--base",
 | |
|     group="AWSY",
 | |
|     action="store_true",
 | |
|     dest="base",
 | |
|     default=False,
 | |
|     help="Run base memory usage tests.",
 | |
| )
 | |
| def run_awsy_test(command_context, tests, **kwargs):
 | |
|     """mach awsy-test runs the in-tree version of the Are We Slim Yet
 | |
|     (AWSY) tests.
 | |
| 
 | |
|     awsy-test is implemented as a marionette test and marionette
 | |
|     test arguments also apply although they are not necessary
 | |
|     since reasonable defaults will be chosen.
 | |
| 
 | |
|     The AWSY specific arguments can be found in the Command
 | |
|     Arguments for AWSY section below.
 | |
| 
 | |
|     awsy-test will automatically download the tp5n.zip talos
 | |
|     pageset from tooltool and install it under
 | |
|     topobjdir/_tests/awsy/html. You can specify your own page set
 | |
|     by specifying --web-root and --page-manifest.
 | |
| 
 | |
|     The results of the test will be placed in the results
 | |
|     directory specified by the --results argument.
 | |
| 
 | |
|     On Windows, you may experience problems due to path length
 | |
|     errors when extracting the tp5n.zip file containing the
 | |
|     test pages or when attempting to write checkpoints to the
 | |
|     results directory. In that case, you should specify both
 | |
|     the --web-root and --results arguments pointing to a location
 | |
|     with a short path. For example:
 | |
| 
 | |
|     --web-root=c:\\\\tmp\\\\html --results=c:\\\\tmp\\\\results
 | |
| 
 | |
|     Note that the double backslashes are required.
 | |
|     """
 | |
|     kwargs["logger_name"] = "Awsy Tests"
 | |
|     if "test_objects" in kwargs:
 | |
|         tests = []
 | |
|         for obj in kwargs["test_objects"]:
 | |
|             tests.append(obj["file_relpath"])
 | |
|         del kwargs["test_objects"]
 | |
| 
 | |
|     if not kwargs.get("binary") and conditions.is_firefox(command_context):
 | |
|         try:
 | |
|             kwargs["binary"] = command_context.get_binary_path("app")
 | |
|         except BinaryNotFoundException as e:
 | |
|             command_context.log(
 | |
|                 logging.ERROR, "awsy", {"error": str(e)}, "ERROR: {error}"
 | |
|             )
 | |
|             command_context.log(logging.INFO, "awsy", {"help": e.help()}, "{help}")
 | |
|             return 1
 | |
|     return run_awsy(command_context, tests, **kwargs)
 | 
