diff options
Diffstat (limited to 'testing/skipfails.py')
-rw-r--r-- | testing/skipfails.py | 847 |
1 files changed, 625 insertions, 222 deletions
diff --git a/testing/skipfails.py b/testing/skipfails.py index a13bb6bd71..0d9f636632 100644 --- a/testing/skipfails.py +++ b/testing/skipfails.py @@ -51,23 +51,36 @@ MS_PER_MINUTE = 60 * 1000 # ms per minute DEBUG_THRESHOLD = 40 * MS_PER_MINUTE # 40 minutes in ms OPT_THRESHOLD = 20 * MS_PER_MINUTE # 20 minutes in ms +ANYJS = "anyjs" CC = "classification" DEF = "DEFAULT" DURATIONS = "durations" -FAILED_RUNS = "failed_runs" +FAILED_RUNS = "runs_failed" FAILURE_RATIO = 0.4 # more than this fraction of failures will disable LL = "label" -MEDIAN_DURATION = "median_duration" +MEDIAN_DURATION = "duration_median" MINIMUM_RUNS = 3 # mininum number of runs to consider success/failure MOCK_BUG_DEFAULTS = {"blocks": [], "comments": []} -MOCK_TASK_DEFAULTS = {"failure_types": {}, "results": []} +MOCK_TASK_DEFAULTS = {"extra": {}, "failure_types": {}, "results": []} MOCK_TASK_INITS = ["results"] OPT = "opt" PP = "path" +QUERY = "query" +RR = "result" RUNS = "runs" SUM_BY_LABEL = "sum_by_label" -TOTAL_DURATION = "total_duration" -TOTAL_RUNS = "total_runs" +TOTAL_DURATION = "duration_total" +TOTAL_RUNS = "runs_total" +WP = "testing/web-platform/" +WPT = "wpt" +WPT0 = WP + "tests/infrastructure" +WPT_META0 = WP + "tests/infrastructure/metadata" +WPT_META0_CLASSIC = WP + "meta/infrastructure" +WPT1 = WP + "tests" +WPT_META1 = WPT1.replace("tests", "meta") +WPT2 = WP + "mozilla/tests" +WPT_META2 = WPT2.replace("tests", "meta") +WPT_MOZILLA = "/_mozilla" class Mock(object): @@ -147,6 +160,8 @@ class Skipfails(object): self.bugzilla = os.environ["BUGZILLA"] else: self.bugzilla = Skipfails.BUGZILLA_SERVER_DEFAULT + if self.bugzilla == "disable": + self.bugzilla = None # Bug filing disabled self.component = "skip-fails" self._bzapi = None self._attach_rx = None @@ -163,10 +178,11 @@ class Skipfails(object): self.bugs = [] # preloaded bugs, currently not an updated cache def _initialize_bzapi(self): - """Lazily initializes the Bugzilla API""" - if self._bzapi is None: + """Lazily initializes the Bugzilla API (returns True on success)""" + if self._bzapi is None and self.bugzilla is not None: self._bzapi = bugzilla.Bugzilla(self.bugzilla) self._attach_rx = re.compile(ATTACHMENT_REGEX, flags=re.M) + return self._bzapi is not None def pprint(self, obj): if self.pp is None: @@ -202,6 +218,21 @@ class Skipfails(object): if self.verbose: self.info(e) + def full_path(self, filename): + """Returns full path for the relative filename""" + + return os.path.join(self.topsrcdir, os.path.normpath(filename)) + + def isdir(self, filename): + """Returns True if filename is a directory""" + + return os.path.isdir(self.full_path(filename)) + + def exists(self, filename): + """Returns True if filename exists""" + + return os.path.exists(self.full_path(filename)) + def run( self, meta_bug_id=None, @@ -215,55 +246,70 @@ class Skipfails(object): try_url = self.try_url revision, repo = self.get_revision(try_url) - if use_tasks is not None: - if os.path.exists(use_tasks): - self.vinfo(f"use tasks: {use_tasks}") - tasks = self.read_json(use_tasks) - tasks = [ - Mock(task, MOCK_TASK_DEFAULTS, MOCK_TASK_INITS) for task in tasks - ] - else: - self.error(f"uses tasks JSON file does not exist: {use_tasks}") - return False + tasks = self.read_tasks(use_tasks) + self.vinfo(f"use tasks: {use_tasks}") else: tasks = self.get_tasks(revision, repo) - if use_failures is not None: - if os.path.exists(use_failures): - self.vinfo(f"use failures: {use_failures}") - failures = self.read_json(use_failures) - else: - self.error(f"use failures JSON file does not exist: {use_failures}") - return False + failures = self.read_failures(use_failures) + self.vinfo(f"use failures: {use_failures}") else: failures = self.get_failures(tasks) if save_failures is not None: - self.vinfo(f"save failures: {save_failures}") self.write_json(save_failures, failures) - + self.vinfo(f"save failures: {save_failures}") if save_tasks is not None: - self.vinfo(f"save tasks: {save_tasks}") self.write_tasks(save_tasks, tasks) - + self.vinfo(f"save tasks: {save_tasks}") num_failures = 0 for manifest in failures: - if not manifest.endswith(".toml"): - self.warning(f"cannot process skip-fails on INI manifests: {manifest}") - else: + if manifest.endswith(".toml") or manifest.startswith(WP): + wpt = failures[manifest][WPT] for label in failures[manifest][LL]: for path in failures[manifest][LL][label][PP]: classification = failures[manifest][LL][label][PP][path][CC] if classification.startswith("disable_") or ( self.turbo and classification == Classification.SECONDARY ): + anyjs = {} # anyjs alternate basename = False for task_id in failures[manifest][LL][label][PP][path][ RUNS ]: - break # just use the first task_id + if not wpt: + break # just use the first task_id + filename = os.path.basename(path) + anyjs[filename] = False + if ( + QUERY + in failures[manifest][LL][label][PP][path][RUNS][ + task_id + ] + ): + query = failures[manifest][LL][label][PP][path][ + RUNS + ][task_id][QUERY] + anyjs[filename + query] = False + else: + query = None + if ( + ANYJS + in failures[manifest][LL][label][PP][path][RUNS][ + task_id + ] + ): + any_filename = os.path.basename( + failures[manifest][LL][label][PP][path][RUNS][ + task_id + ][ANYJS] + ) + anyjs[any_filename] = False + if query is not None: + anyjs[any_filename + query] = False self.skip_failure( manifest, path, + anyjs, label, classification, task_id, @@ -278,6 +324,14 @@ class Skipfails(object): f"max_failures={max_failures} threshold reached. stopping." ) return True + elif manifest.endswith(".ini"): + self.warning(f"cannot process skip-fails on INI manifest: {manifest}") + elif manifest.endswith(".list"): + self.warning(f"cannot process skip-fails on LIST manifest: {manifest}") + else: + self.warning( + f"cannot process skip-fails on unknown manifest: {manifest}" + ) return True def get_revision(self, url): @@ -321,12 +375,13 @@ class Skipfails(object): ff = {} manifest_paths = {} manifest_ = { + WPT: False, LL: {}, } label_ = { DURATIONS: {}, MEDIAN_DURATION: 0, - OPT: None, + OPT: False, PP: {}, SUM_BY_LABEL: { Classification.DISABLE_MANIFEST: 0, @@ -345,29 +400,77 @@ class Skipfails(object): RUNS: {}, TOTAL_RUNS: 0, } + run_ = { + RR: False, + } - for task in tasks: # add implicit failures + for task in tasks: # add explicit failures try: if len(task.results) == 0: continue # ignore aborted tasks - for mm in task.failure_types: - if mm not in manifest_paths: - manifest_paths[mm] = [] - if mm not in ff: - ff[mm] = deepcopy(manifest_) + for manifest in task.failure_types: + mm = manifest ll = task.label - if ll not in ff[mm][LL]: - ff[mm][LL][ll] = deepcopy(label_) - for path_type in task.failure_types[mm]: + wpt = False + if mm.endswith(".ini"): + self.warning( + f"cannot analyze skip-fails on INI manifests: {mm}" + ) + continue + elif mm.endswith(".list"): + self.warning( + f"cannot analyze skip-fails on LIST manifests: {mm}" + ) + continue + elif not mm.endswith(".toml"): + path, mm, _query, _anyjs = self.wpt_paths(mm) + if path is None: # not WPT + self.warning( + f"cannot analyze skip-fails on unknown manifest type: {manifest}" + ) + continue + wpt = True + if not wpt: + if mm not in manifest_paths: + manifest_paths[mm] = [] + if mm not in ff: + ff[mm] = deepcopy(manifest_) + ff[mm][WPT] = wpt + if ll not in ff[mm][LL]: + ff[mm][LL][ll] = deepcopy(label_) + for path_type in task.failure_types[manifest]: path, _type = path_type - if path == mm: + if wpt: + path, mmpath, query, anyjs = self.wpt_paths(path) + if path is None: + self.warning( + f"non existant failure path: {path_type[0]}" + ) + break + mm = os.path.dirname(mmpath) + if mm not in manifest_paths: + manifest_paths[mm] = [] + if mm not in ff: + ff[mm] = deepcopy(manifest_) + ff[mm][WPT] = wpt + if ll not in ff[mm][LL]: + ff[mm][LL][ll] = deepcopy(label_) + else: + query = None + anyjs = None + if not wpt and path == mm: path = DEF # refers to the manifest itself if path not in manifest_paths[mm]: manifest_paths[mm].append(path) if path not in ff[mm][LL][ll][PP]: ff[mm][LL][ll][PP][path] = deepcopy(path_) if task.id not in ff[mm][LL][ll][PP][path][RUNS]: - ff[mm][LL][ll][PP][path][RUNS][task.id] = False + ff[mm][LL][ll][PP][path][RUNS][task.id] = deepcopy(run_) + ff[mm][LL][ll][PP][path][RUNS][task.id][RR] = False + if query is not None: + ff[mm][LL][ll][PP][path][RUNS][task.id][QUERY] = query + if anyjs is not None: + ff[mm][LL][ll][PP][path][RUNS][task.id][ANYJS] = anyjs ff[mm][LL][ll][PP][path][TOTAL_RUNS] += 1 ff[mm][LL][ll][PP][path][FAILED_RUNS] += 1 except AttributeError as ae: @@ -378,7 +481,28 @@ class Skipfails(object): if len(task.results) == 0: continue # ignore aborted tasks for result in task.results: + wpt = False mm = result.group + if mm.endswith(".ini"): + self.warning( + f"cannot analyze skip-fails on INI manifests: {mm}" + ) + continue + elif mm.endswith(".list"): + self.warning( + f"cannot analyze skip-fails on LIST manifests: {mm}" + ) + continue + elif not mm.endswith(".toml"): + path, mm, _query, _anyjs = self.wpt_paths(mm) + if path is None: # not WPT + self.warning( + f"cannot analyze skip-fails on unknown manifest type: {result.group}" + ) + continue + wpt = True + if mm not in manifest_paths: + continue if mm not in ff: ff[mm] = deepcopy(manifest_) ll = task.label @@ -389,47 +513,56 @@ class Skipfails(object): ff[mm][LL][ll][DURATIONS][task.id] = result.duration or 0 if ff[mm][LL][ll][OPT] is None: ff[mm][LL][ll][OPT] = self.get_opt_for_task(task.id) - if mm not in manifest_paths: - continue for path in manifest_paths[mm]: # all known paths if path not in ff[mm][LL][ll][PP]: ff[mm][LL][ll][PP][path] = deepcopy(path_) if task.id not in ff[mm][LL][ll][PP][path][RUNS]: - ff[mm][LL][ll][PP][path][RUNS][task.id] = result.ok + ff[mm][LL][ll][PP][path][RUNS][task.id] = deepcopy(run_) + ff[mm][LL][ll][PP][path][RUNS][task.id][RR] = result.ok ff[mm][LL][ll][PP][path][TOTAL_RUNS] += 1 if not result.ok: ff[mm][LL][ll][PP][path][FAILED_RUNS] += 1 except AttributeError as ae: - self.warning(f"unknown attribute in task (#3): {ae}") + self.warning(f"unknown attribute in task (#2): {ae}") for mm in ff: # determine classifications + wpt = ff[mm][WPT] for label in ff[mm][LL]: - opt = ff[mm][LL][label][OPT] + ll = label + opt = ff[mm][LL][ll][OPT] durations = [] # summarize durations - for task_id in ff[mm][LL][label][DURATIONS]: - duration = ff[mm][LL][label][DURATIONS][task_id] + for task_id in ff[mm][LL][ll][DURATIONS]: + duration = ff[mm][LL][ll][DURATIONS][task_id] durations.append(duration) if len(durations) > 0: total_duration = sum(durations) median_duration = median(durations) - ff[mm][LL][label][TOTAL_DURATION] = total_duration - ff[mm][LL][label][MEDIAN_DURATION] = median_duration + ff[mm][LL][ll][TOTAL_DURATION] = total_duration + ff[mm][LL][ll][MEDIAN_DURATION] = median_duration if (opt and median_duration > OPT_THRESHOLD) or ( (not opt) and median_duration > DEBUG_THRESHOLD ): - if DEF not in ff[mm][LL][label][PP]: - ff[mm][LL][label][PP][DEF] = deepcopy(path_) - if task_id not in ff[mm][LL][label][PP][DEF][RUNS]: - ff[mm][LL][label][PP][DEF][RUNS][task_id] = False - ff[mm][LL][label][PP][DEF][TOTAL_RUNS] += 1 - ff[mm][LL][label][PP][DEF][FAILED_RUNS] += 1 - ff[mm][LL][label][PP][DEF][CC] = Classification.DISABLE_TOO_LONG + if wpt: + paths = ff[mm][LL][ll][PP].keys() + else: + paths = [DEF] + for path in paths: + if path not in ff[mm][LL][ll][PP]: + ff[mm][LL][ll][PP][path] = deepcopy(path_) + if task_id not in ff[mm][LL][ll][PP][path][RUNS]: + ff[mm][LL][ll][PP][path][RUNS][task.id] = deepcopy(run_) + ff[mm][LL][ll][PP][path][RUNS][task.id][RR] = False + ff[mm][LL][ll][PP][path][TOTAL_RUNS] += 1 + ff[mm][LL][ll][PP][path][FAILED_RUNS] += 1 + ff[mm][LL][ll][PP][path][ + CC + ] = Classification.DISABLE_TOO_LONG primary = True # we have not seen the first failure - for path in sort_paths(ff[mm][LL][label][PP]): - classification = ff[mm][LL][label][PP][path][CC] + for path in sort_paths(ff[mm][LL][ll][PP]): + classification = ff[mm][LL][ll][PP][path][CC] if classification == Classification.UNKNOWN: - failed_runs = ff[mm][LL][label][PP][path][FAILED_RUNS] - total_runs = ff[mm][LL][label][PP][path][TOTAL_RUNS] + failed_runs = ff[mm][LL][ll][PP][path][FAILED_RUNS] + total_runs = ff[mm][LL][ll][PP][path][TOTAL_RUNS] if total_runs >= MINIMUM_RUNS: if failed_runs / total_runs < FAILURE_RATIO: if failed_runs == 0: @@ -444,8 +577,8 @@ class Skipfails(object): primary = False else: classification = Classification.SECONDARY - ff[mm][LL][label][PP][path][CC] = classification - ff[mm][LL][label][SUM_BY_LABEL][classification] += 1 + ff[mm][LL][ll][PP][path][CC] = classification + ff[mm][LL][ll][SUM_BY_LABEL][classification] += 1 return ff def _get_os_version(self, os, platform): @@ -459,37 +592,36 @@ class Skipfails(object): def get_bug_by_id(self, id): """Get bug by bug id""" - self._initialize_bzapi() bug = None for b in self.bugs: if b.id == id: bug = b break - if bug is None: + if bug is None and self._initialize_bzapi(): bug = self._bzapi.getbug(id) return bug def get_bugs_by_summary(self, summary): """Get bug by bug summary""" - self._initialize_bzapi() bugs = [] for b in self.bugs: if b.summary == summary: bugs.append(b) if len(bugs) > 0: return bugs - query = self._bzapi.build_query(short_desc=summary) - query["include_fields"] = [ - "id", - "product", - "component", - "status", - "resolution", - "summary", - "blocks", - ] - bugs = self._bzapi.query(query) + if self._initialize_bzapi(): + query = self._bzapi.build_query(short_desc=summary) + query["include_fields"] = [ + "id", + "product", + "component", + "status", + "resolution", + "summary", + "blocks", + ] + bugs = self._bzapi.query(query) return bugs def create_bug( @@ -503,41 +635,45 @@ class Skipfails(object): ): """Create a bug""" - self._initialize_bzapi() - if not self._bzapi.logged_in: - self.error( - "Must create a Bugzilla API key per https://github.com/mozilla/mozci-tools/blob/main/citools/test_triage_bug_filer.py" + bug = None + if self._initialize_bzapi(): + if not self._bzapi.logged_in: + self.error( + "Must create a Bugzilla API key per https://github.com/mozilla/mozci-tools/blob/main/citools/test_triage_bug_filer.py" + ) + raise PermissionError(f"Not authenticated for Bugzilla {self.bugzilla}") + createinfo = self._bzapi.build_createbug( + product=product, + component=component, + summary=summary, + version=version, + description=description, ) - raise PermissionError(f"Not authenticated for Bugzilla {self.bugzilla}") - createinfo = self._bzapi.build_createbug( - product=product, - component=component, - summary=summary, - version=version, - description=description, - ) - createinfo["type"] = bugtype - bug = self._bzapi.createbug(createinfo) + createinfo["type"] = bugtype + bug = self._bzapi.createbug(createinfo) return bug def add_bug_comment(self, id, comment, meta_bug_id=None): """Add a comment to an existing bug""" - self._initialize_bzapi() - if not self._bzapi.logged_in: - self.error(BUGZILLA_AUTHENTICATION_HELP) - raise PermissionError("Not authenticated for Bugzilla") - if meta_bug_id is not None: - blocks_add = [meta_bug_id] - else: - blocks_add = None - updateinfo = self._bzapi.build_update(comment=comment, blocks_add=blocks_add) - self._bzapi.update_bugs([id], updateinfo) + if self._initialize_bzapi(): + if not self._bzapi.logged_in: + self.error(BUGZILLA_AUTHENTICATION_HELP) + raise PermissionError("Not authenticated for Bugzilla") + if meta_bug_id is not None: + blocks_add = [meta_bug_id] + else: + blocks_add = None + updateinfo = self._bzapi.build_update( + comment=comment, blocks_add=blocks_add + ) + self._bzapi.update_bugs([id], updateinfo) def skip_failure( self, manifest, path, + anyjs, label, classification, task_id, @@ -546,13 +682,24 @@ class Skipfails(object): repo, meta_bug_id=None, ): - """Skip a failure""" + """ + Skip a failure (for TOML and WPT manifests) + For wpt anyjs is a dictionary mapping from alternate basename to + a boolean (indicating if the basename has been handled in the manifest) + """ + if manifest.endswith(".toml"): + wpt = False + filename = DEF + else: + wpt = True + _path, manifest, _query, _anyjs = self.wpt_paths(path) + filename = os.path.basename(path) self.vinfo(f"===== Skip failure in manifest: {manifest} =====") if task_id is None: skip_if = "true" else: - skip_if = self.task_to_skip_if(task_id) + skip_if = self.task_to_skip_if(task_id, wpt) if skip_if is None: self.warning( f"Unable to calculate skip-if condition from manifest={manifest} from failure label={label}" @@ -560,17 +707,26 @@ class Skipfails(object): return bug_reference = "" if classification == Classification.DISABLE_MANIFEST: - filename = DEF comment = "Disabled entire manifest due to crash result" elif classification == Classification.DISABLE_TOO_LONG: - filename = DEF comment = "Disabled entire manifest due to excessive run time" else: - filename = self.get_filename_in_manifest(manifest, path) - comment = f'Disabled test due to failures: "{filename}"' + if not wpt: + filename = self.get_filename_in_manifest(manifest, path) + comment = f'Disabled test due to failures in test file: "{filename}"' if classification == Classification.SECONDARY: comment += " (secondary)" - bug_reference = " (secondary)" + if not wpt: + bug_reference = " (secondary)" + if wpt and len(anyjs) > 1: + comment += "\nAdditional WPT wildcard paths:" + for p in sorted(anyjs.keys()): + if p != filename: + comment += f'\n "{p}"' + platform, testname = self.label_to_platform_testname(label) + if platform is not None: + comment += "\nCommand line to reproduce (experimental):\n" + comment += f" \"mach try fuzzy -q '{platform}' {testname}\"\n" comment += f"\nTry URL = {try_url}" comment += f"\nrevision = {revision}" comment += f"\nrepo = {repo}" @@ -588,80 +744,95 @@ class Skipfails(object): line_number, line, log_url, - ) = self.get_bug_suggestions(repo, job_id, path) + ) = self.get_bug_suggestions(repo, job_id, path, anyjs) if log_url is not None: - comment += f"\n\nBug suggestions: {suggestions_url}" + comment += f"\nBug suggestions: {suggestions_url}" comment += f"\nSpecifically see at line {line_number} in the attached log: {log_url}" comment += f'\n\n "{line}"\n' - platform, testname = self.label_to_platform_testname(label) - if platform is not None: - comment += "\n\nCommand line to reproduce:\n\n" - comment += f" \"mach try fuzzy -q '{platform}' {testname}\"" bug_summary = f"MANIFEST {manifest}" attachments = {} - bugs = self.get_bugs_by_summary(bug_summary) - if len(bugs) == 0: - description = ( - f"This bug covers excluded failing tests in the MANIFEST {manifest}" - ) - description += "\n(generated by `mach manifest skip-fails`)" - product, component = self.get_file_info(path) - if self.dry_run: - self.warning( - f'Dry-run NOT creating bug: {product}::{component} "{bug_summary}"' - ) - bugid = "TBD" - else: - bug = self.create_bug(bug_summary, description, product, component) - bugid = bug.id - self.vinfo( - f'Created Bug {bugid} {product}::{component} : "{bug_summary}"' + bugid = "TBD" + if self.bugzilla is None: + self.warning("Bugzilla has been disabled: no bugs created or updated") + else: + bugs = self.get_bugs_by_summary(bug_summary) + if len(bugs) == 0: + description = ( + f"This bug covers excluded failing tests in the MANIFEST {manifest}" ) - bug_reference = f"Bug {bugid}" + bug_reference - elif len(bugs) == 1: - bugid = bugs[0].id - bug_reference = f"Bug {bugid}" + bug_reference - product = bugs[0].product - component = bugs[0].component - self.vinfo(f'Found Bug {bugid} {product}::{component} "{bug_summary}"') - if meta_bug_id is not None: - if meta_bug_id in bugs[0].blocks: - self.vinfo(f" Bug {bugid} already blocks meta bug {meta_bug_id}") - meta_bug_id = None # no need to add again - comments = bugs[0].getcomments() - for i in range(len(comments)): - text = comments[i]["text"] - m = self._attach_rx.findall(text) - if len(m) == 1: - a_task_id = m[0][1] - attachments[a_task_id] = m[0][0] - if a_task_id == task_id: + description += "\n(generated by `mach manifest skip-fails`)" + product, component = self.get_file_info(path) + if self.dry_run: + self.warning( + f'Dry-run NOT creating bug: {product}::{component} "{bug_summary}"' + ) + else: + bug = self.create_bug(bug_summary, description, product, component) + bugid = bug.id + self.vinfo( + f'Created Bug {bugid} {product}::{component} : "{bug_summary}"' + ) + elif len(bugs) == 1: + bugid = bugs[0].id + product = bugs[0].product + component = bugs[0].component + self.vinfo(f'Found Bug {bugid} {product}::{component} "{bug_summary}"') + if meta_bug_id is not None: + if meta_bug_id in bugs[0].blocks: self.vinfo( - f" Bug {bugid} already has the compressed log attached for this task" + f" Bug {bugid} already blocks meta bug {meta_bug_id}" ) - else: - self.error(f'More than one bug found for summary: "{bug_summary}"') - return + meta_bug_id = None # no need to add again + comments = bugs[0].getcomments() + for i in range(len(comments)): + text = comments[i]["text"] + m = self._attach_rx.findall(text) + if len(m) == 1: + a_task_id = m[0][1] + attachments[a_task_id] = m[0][0] + if a_task_id == task_id: + self.vinfo( + f" Bug {bugid} already has the compressed log attached for this task" + ) + else: + self.error(f'More than one bug found for summary: "{bug_summary}"') + return + bug_reference = f"Bug {bugid}" + bug_reference + comment += f"\nskip-if condition: {skip_if} # {bug_reference}" if self.dry_run: - self.warning(f"Dry-run NOT adding comment to Bug {bugid}: {comment}") + self.warning(f"Dry-run NOT adding comment to Bug {bugid}:\n{comment}") self.info(f'Dry-run NOT editing ["{filename}"] manifest: "{manifest}"') self.info(f'would add skip-if condition: "{skip_if}" # {bug_reference}') if task_id is not None and task_id not in attachments: self.info("would add compressed log for this task") return - self.add_bug_comment(bugid, comment, meta_bug_id) - self.info(f"Added comment to Bug {bugid}: {comment}") - if meta_bug_id is not None: - self.info(f" Bug {bugid} blocks meta Bug: {meta_bug_id}") - if task_id is not None and task_id not in attachments: - self.add_attachment_log_for_task(bugid, task_id) - self.info("Added compressed log for this task") - mp = ManifestParser(use_toml=True, document=True) - manifest_path = os.path.join(self.topsrcdir, os.path.normpath(manifest)) - mp.read(manifest_path) - document = mp.source_documents[manifest_path] - add_skip_if(document, filename, skip_if, bug_reference) - manifest_str = alphabetize_toml_str(document) + elif self.bugzilla is None: + self.warning(f"NOT adding comment to Bug {bugid}:\n{comment}") + else: + self.add_bug_comment(bugid, comment, meta_bug_id) + self.info(f"Added comment to Bug {bugid}:\n{comment}") + if meta_bug_id is not None: + self.info(f" Bug {bugid} blocks meta Bug: {meta_bug_id}") + if task_id is not None and task_id not in attachments: + self.add_attachment_log_for_task(bugid, task_id) + self.info("Added compressed log for this task") + manifest_path = self.full_path(manifest) + if wpt: + if os.path.exists(manifest_path): + manifest_str = io.open(manifest_path, "r", encoding="utf-8").read() + else: + manifest_str = "" + # ensure parent directories exist + os.makedirs(os.path.dirname(manifest_path), exist_ok=True) + manifest_str = self.wpt_add_skip_if( + manifest_str, anyjs, skip_if, bug_reference + ) + else: + mp = ManifestParser(use_toml=True, document=True) + mp.read(manifest_path) + document = mp.source_documents[manifest_path] + add_skip_if(document, filename, skip_if, bug_reference) + manifest_str = alphabetize_toml_str(document) fp = io.open(manifest_path, "w", encoding="utf-8", newline="\n") fp.write(manifest_str) fp.close() @@ -673,9 +844,7 @@ class Skipfails(object): if len(self.variants) == 0: variants_file = "taskcluster/ci/test/variants.yml" - variants_path = os.path.join( - self.topsrcdir, os.path.normpath(variants_file) - ) + variants_path = self.full_path(variants_file) fp = io.open(variants_path, "r", encoding="utf-8") raw_variants = load(fp, Loader=Loader) fp.close() @@ -708,13 +877,14 @@ class Skipfails(object): else: self.get_variants() task = self.get_task_details(task_id) or {} - os = None - os_version = None arch = None bits = None + build = None + build_types = [] display = None + os = None + os_version = None runtimes = [] - build_types = [] test_setting = task.get("extra", {}).get("test-setting", {}) platform = test_setting.get("platform", {}) platform_os = platform.get("os", {}) @@ -730,12 +900,17 @@ class Skipfails(object): os_version = platform_os["version"] if len(os_version) == 4: os_version = os_version[0:2] + "." + os_version[2:4] + if "build" in platform_os: + build = platform_os["build"] if "arch" in platform: arch = platform["arch"] if arch == "x86" or arch.find("32") >= 0: bits = "32" - if arch == "64" or arch.find("64") >= 0: + arch = "x86" + else: bits = "64" + if arch != "aarch64" and arch != "ppc": + arch = "x86_64" if "display" in platform: display = platform["display"] if "runtime" in test_setting: @@ -755,15 +930,16 @@ class Skipfails(object): build_types.append(k) unknown = None extra = { - "os": os or unknown, - "os_version": os_version or unknown, "arch": arch or unknown, "bits": bits or unknown, + "build": build or unknown, + "build_types": build_types, + "debug": debug, "display": display or unknown, - "runtimes": runtimes, "opt": opt, - "debug": debug, - "build_types": build_types, + "os": os or unknown, + "os_version": os_version or unknown, + "runtimes": runtimes, } self.extras[task_id] = extra return extra @@ -772,28 +948,51 @@ class Skipfails(object): extra = self.get_extra(task_id) return extra["opt"] - def task_to_skip_if(self, task_id): + def task_to_skip_if(self, task_id, wpt=False): """Calculate the skip-if condition for failing task task_id""" + if wpt: + qq = '"' + aa = " and " + nn = "not " + else: + qq = "'" + aa = " && " + nn = "!" + eq = " == " + arch = "processor" + version = "os_version" extra = self.get_extra(task_id) skip_if = None if extra["os"] is not None: - skip_if = "os == '" + extra["os"] + "'" if extra["os_version"] is not None: - skip_if += " && " - skip_if += "os_version == '" + extra["os_version"] + "'" - if extra["bits"] is not None: - skip_if += " && " - skip_if += "bits == '" + extra["bits"] + "'" + skip_if = "os" + eq + qq + extra["os"] + qq + if ( + extra["build"] is not None + and extra["os"] == "win" + and extra["os_version"] == "11" + and extra["build"] == "2009" + ): + skip_if = "win11_2009" # mozinfo.py:137 + else: + skip_if += aa + version + eq + qq + extra["os_version"] + qq + if extra["arch"] is not None: + skip_if += aa + arch + eq + qq + extra["arch"] + qq + # since we always give arch/processor, bits are not required + # if extra["bits"] is not None: + # skip_if += aa + "bits" + eq + extra["bits"] + if extra["debug"]: + skip_if += aa + "debug" + else: + skip_if += aa + nn + "debug" if extra["display"] is not None: - skip_if += " && " - skip_if += "display == '" + extra["display"] + "'" + skip_if += aa + "display" + eq + qq + extra["display"] + qq for runtime in extra["runtimes"]: - skip_if += " && " - skip_if += runtime + skip_if += aa + runtime for build_type in extra["build_types"]: - skip_if += " && " - skip_if += build_type + # note: lite will not evaluate on non-android platforms + if build_type not in ["debug", "lite", "opt", "shippable"]: + skip_if += aa + build_type return skip_if def get_file_info(self, path, product="Testing", component="General"): @@ -859,8 +1058,9 @@ class Skipfails(object): """Return the job_id for push_id, task_id (or None)""" self.vinfo(f"Retrieving job_id for push_id: {push_id}, task_id: {task_id} ...") - if push_id in self.job_ids: # if cached - job_id = self.job_ids[push_id] + k = f"{push_id}:{task_id}" + if k in self.job_ids: # if cached + job_id = self.job_ids[k] else: job_id = None params = {} @@ -878,10 +1078,10 @@ class Skipfails(object): if result[14] == task_id: job_id = result[1] break - self.job_ids[push_id] = job_id + self.job_ids[k] = job_id return job_id - def get_bug_suggestions(self, repo, job_id, path): + def get_bug_suggestions(self, repo, job_id, path, anyjs=None): """ Return the (suggestions_url, line_number, line, log_url) for the given repo and job_id @@ -897,24 +1097,65 @@ class Skipfails(object): if r.status_code != 200: self.warning(f"FAILED to query Treeherder = {r} for {r.url}") else: + if anyjs is not None: + pathdir = os.path.dirname(path) + "/" + paths = [pathdir + f for f in anyjs.keys()] + else: + paths = [path] response = r.json() if len(response) > 0: for sugg in response: - if sugg["path_end"] == path: - line_number = sugg["line_number"] + 1 - line = sugg["search"] - log_url = f"https://treeherder.mozilla.org/logviewer?repo={repo}&job_id={job_id}&lineNumber={line_number}" - break + for p in paths: + path_end = sugg.get("path_end", None) + # handles WPT short paths + if path_end is not None and p.endswith(path_end): + line_number = sugg["line_number"] + 1 + line = sugg["search"] + log_url = f"https://treeherder.mozilla.org/logviewer?repo={repo}&job_id={job_id}&lineNumber={line_number}" + break rv = (suggestions_url, line_number, line, log_url) return rv def read_json(self, filename): """read data as JSON from filename""" + fp = io.open(filename, "r", encoding="utf-8") data = json.load(fp) fp.close() return data + def read_tasks(self, filename): + """read tasks as JSON from filename""" + + if not os.path.exists(filename): + msg = f"use-tasks JSON file does not exist: {filename}" + raise OSError(2, msg, filename) + tasks = self.read_json(filename) + tasks = [Mock(task, MOCK_TASK_DEFAULTS, MOCK_TASK_INITS) for task in tasks] + for task in tasks: + if len(task.extra) > 0: # pre-warm cache for extra information + self.extras[task.id] = task.extra + return tasks + + def read_failures(self, filename): + """read failures as JSON from filename""" + + if not os.path.exists(filename): + msg = f"use-failures JSON file does not exist: {filename}" + raise OSError(2, msg, filename) + failures = self.read_json(filename) + return failures + + def read_bugs(self, filename): + """read bugs as JSON from filename""" + + if not os.path.exists(filename): + msg = f"bugs JSON file does not exist: {filename}" + raise OSError(2, msg, filename) + bugs = self.read_json(filename) + bugs = [Mock(bug, MOCK_BUG_DEFAULTS) for bug in bugs] + return bugs + def write_json(self, filename, data): """saves data as JSON to filename""" fp = io.open(filename, "w", encoding="utf-8") @@ -987,20 +1228,182 @@ class Skipfails(object): fp = gzip.open(attach_fp, "wb") fp.write(r.text.encode("utf-8")) fp.close() - self._initialize_bzapi() - description = ATTACHMENT_DESCRIPTION + task_id - file_name = TASK_LOG + ".gz" - comment = "Added compressed log" - content_type = "application/gzip" - try: - self._bzapi.attachfile( - [bugid], - attach_fp.name, - description, - file_name=file_name, - comment=comment, - content_type=content_type, - is_private=False, - ) - except Fault: - pass # Fault expected: Failed to fetch key 9372091 from network storage: The specified key does not exist. + if self._initialize_bzapi(): + description = ATTACHMENT_DESCRIPTION + task_id + file_name = TASK_LOG + ".gz" + comment = "Added compressed log" + content_type = "application/gzip" + try: + self._bzapi.attachfile( + [bugid], + attach_fp.name, + description, + file_name=file_name, + comment=comment, + content_type=content_type, + is_private=False, + ) + except Fault: + pass # Fault expected: Failed to fetch key 9372091 from network storage: The specified key does not exist. + + def get_wpt_path_meta(self, shortpath): + if shortpath.startswith(WPT0): + path = shortpath + meta = shortpath.replace(WPT0, WPT_META0, 1) + elif shortpath.startswith(WPT1): + path = shortpath + meta = shortpath.replace(WPT1, WPT_META1, 1) + elif shortpath.startswith(WPT2): + path = shortpath + meta = shortpath.replace(WPT2, WPT_META2, 1) + elif shortpath.startswith(WPT_MOZILLA): + shortpath = shortpath[len(WPT_MOZILLA) :] + path = WPT2 + shortpath + meta = WPT_META2 + shortpath + else: + path = WPT1 + shortpath + meta = WPT_META1 + shortpath + return (path, meta) + + def wpt_paths(self, shortpath): + """ + Analyzes the WPT short path for a test and returns + (path, manifest, query, anyjs) where + path is the relative path to the test file + manifest is the relative path to the file metadata + query is the test file query paramters (or None) + anyjs is the html test file as reported by mozci (or None) + """ + query = None + anyjs = None + i = shortpath.find("?") + if i > 0: + query = shortpath[i:] + shortpath = shortpath[0:i] + path, manifest = self.get_wpt_path_meta(shortpath) + failure_type = not self.isdir(path) + if failure_type: + i = path.find(".any.") + if i > 0: + anyjs = path # orig path + manifest = manifest.replace(path[i:], ".any.js") + path = path[0:i] + ".any.js" + else: + i = path.find(".window.") + if i > 0: + anyjs = path # orig path + manifest = manifest.replace(path[i:], ".window.js") + path = path[0:i] + ".window.js" + else: + i = path.find(".worker.") + if i > 0: + anyjs = path # orig path + manifest = manifest.replace(path[i:], ".worker.js") + path = path[0:i] + ".worker.js" + manifest += ".ini" + manifest_classic = "" + if manifest.startswith(WPT_META0): + manifest_classic = manifest.replace(WPT_META0, WPT_META0_CLASSIC, 1) + if self.exists(manifest_classic): + if self.exists(manifest): + self.warning( + f"Both classic {manifest_classic} and metadata {manifest} manifests exist" + ) + else: + self.warning( + f"Using the classic {manifest_classic} manifest as the metadata manifest {manifest} does not exist" + ) + manifest = manifest_classic + if not self.exists(path): + return (None, None, None, None) + return (path, manifest, query, anyjs) + + def wpt_add_skip_if(self, manifest_str, anyjs, skip_if, bug_reference): + """ + Edits a WPT manifest string to add disabled condition + anyjs is a dictionary mapping from filename and any alternate basenames to + a boolean (indicating if the file has been handled in the manifest) + """ + + disabled_key = False + disabled = " disabled:" + condition_start = " if " + condition = condition_start + skip_if + ": " + bug_reference + lines = manifest_str.splitlines() + section = None # name of the section + i = 0 + n = len(lines) + while i < n: + line = lines[i] + if line.startswith("["): + if section is not None and not anyjs[section]: # not yet handled + if not disabled_key: + lines.insert(i, disabled) + i += 1 + lines.insert(i, condition) + lines.insert(i + 1, "") # blank line after condition + i += 2 + n += 2 + anyjs[section] = True + section = line[1:-1] + if section in anyjs and not anyjs[section]: + disabled_key = False + else: + section = None # ignore section we are not interested in + elif section is not None: + if line == disabled: + disabled_key = True + elif line.startswith(" ["): + if i > 0 and i - 1 < n and lines[i - 1] == "": + del lines[i - 1] + i -= 1 + n -= 1 + if not disabled_key: + lines.insert(i, disabled) + i += 1 + n += 1 + lines.insert(i, condition) + lines.insert(i + 1, "") # blank line after condition + i += 2 + n += 2 + anyjs[section] = True + section = None + elif line.startswith(" ") and not line.startswith(" "): + if disabled_key: # insert condition above new key + lines.insert(i, condition) + i += 1 + n += 1 + anyjs[section] = True + section = None + disabled_key = False + elif line.startswith(" "): + if disabled_key and line == condition: + anyjs[section] = True # condition already present + section = None + i += 1 + if section is not None and not anyjs[section]: # not yet handled + if i > 0 and i - 1 < n and lines[i - 1] == "": + del lines[i - 1] + if not disabled_key: + lines.append(disabled) + i += 1 + n += 1 + lines.append(condition) + lines.append("") # blank line after condition + i += 2 + n += 2 + anyjs[section] = True + for section in anyjs: + if not anyjs[section]: + if i > 0 and i - 1 < n and lines[i - 1] != "": + lines.append("") # blank line before condition + i += 1 + n += 1 + lines.append("[" + section + "]") + lines.append(disabled) + lines.append(condition) + lines.append("") # blank line after condition + i += 4 + n += 4 + manifest_str = "\n".join(lines) + "\n" + return manifest_str |