| #!/usr/bin/env python |
| # -*- coding: utf-8 -*- |
| |
| # This script computes the new "current" toolstate for the toolstate repo (not to be |
| # confused with publishing the test results, which happens in `src/bootstrap/toolstate.rs`). |
| # It gets called from `src/ci/publish_toolstate.sh` at the end of an `auto` build. |
| |
| from __future__ import print_function |
| |
| import sys |
| import re |
| import os |
| import json |
| import datetime |
| import collections |
| import textwrap |
| |
| try: |
| import urllib2 |
| from urllib2 import HTTPError |
| except ImportError: |
| import urllib.request as urllib2 |
| from urllib.error import HTTPError |
| try: |
| import typing # noqa: F401 FIXME: py2 |
| except ImportError: |
| pass |
| |
| # List of people to ping when the status of a tool or a book changed. |
| # These should be collaborators of the rust-lang/rust repository (with at least |
| # read privileges on it). CI will fail otherwise. |
| MAINTAINERS = { |
| "book": {"ehuss", "chriskrycho", "carols10cents"}, |
| "nomicon": {"ehuss", "JohnTitor"}, |
| "reference": {"ehuss"}, |
| "rust-by-example": {"ehuss", "marioidival"}, |
| "embedded-book": { |
| "ehuss", |
| "adamgreig", |
| "andre-richter", |
| "jamesmunns", |
| "therealprof", |
| }, |
| "edition-guide": {"ehuss"}, |
| } |
| |
| LABELS = { |
| "book": ["C-bug"], |
| "nomicon": ["C-bug"], |
| "reference": ["C-bug"], |
| "rust-by-example": ["C-bug"], |
| "embedded-book": ["C-bug"], |
| "edition-guide": ["C-bug"], |
| } |
| |
| REPOS = { |
| "book": "https://github.com/rust-lang/book", |
| "nomicon": "https://github.com/rust-lang/nomicon", |
| "reference": "https://github.com/rust-lang/reference", |
| "rust-by-example": "https://github.com/rust-lang/rust-by-example", |
| "embedded-book": "https://github.com/rust-embedded/book", |
| "edition-guide": "https://github.com/rust-lang/edition-guide", |
| } |
| |
| |
| def load_json_from_response(resp): |
| # type: (typing.Any) -> typing.Any |
| content = resp.read() |
| if isinstance(content, bytes): |
| content_str = content.decode("utf-8") |
| else: |
| print("Refusing to decode " + str(type(content)) + " to str") |
| return json.loads(content_str) |
| |
| |
| def read_current_status(current_commit, path): |
| # type: (str, str) -> typing.Mapping[str, typing.Any] |
| """Reads build status of `current_commit` from content of `history/*.tsv`""" |
| with open(path, "r") as f: |
| for line in f: |
| (commit, status) = line.split("\t", 1) |
| if commit == current_commit: |
| return json.loads(status) |
| return {} |
| |
| |
| def gh_url(): |
| # type: () -> str |
| return os.environ["TOOLSTATE_ISSUES_API_URL"] |
| |
| |
| def maybe_remove_mention(message): |
| # type: (str) -> str |
| if os.environ.get("TOOLSTATE_SKIP_MENTIONS") is not None: |
| return message.replace("@", "") |
| return message |
| |
| |
| def issue( |
| tool, |
| status, |
| assignees, |
| relevant_pr_number, |
| relevant_pr_user, |
| labels, |
| github_token, |
| ): |
| # type: (str, str, typing.Iterable[str], str, str, typing.List[str], str) -> None |
| """Open an issue about the toolstate failure.""" |
| if status == "test-fail": |
| status_description = "has failing tests" |
| else: |
| status_description = "no longer builds" |
| request = json.dumps( |
| { |
| "body": maybe_remove_mention( |
| textwrap.dedent("""\ |
| Hello, this is your friendly neighborhood mergebot. |
| After merging PR {}, I observed that the tool {} {}. |
| A follow-up PR to the repository {} is needed to fix the fallout. |
| |
| cc @{}, do you think you would have time to do the follow-up work? |
| If so, that would be great! |
| """).format( |
| relevant_pr_number, |
| tool, |
| status_description, |
| REPOS.get(tool), |
| relevant_pr_user, |
| ) |
| ), |
| "title": "`{}` no longer builds after {}".format(tool, relevant_pr_number), |
| "assignees": list(assignees), |
| "labels": labels, |
| } |
| ) |
| print("Creating issue:\n{}".format(request)) |
| response = urllib2.urlopen( |
| urllib2.Request( |
| gh_url(), |
| request.encode(), |
| { |
| "Authorization": "token " + github_token, |
| "Content-Type": "application/json", |
| }, |
| ) |
| ) |
| response.read() |
| |
| |
| def update_latest( |
| current_commit, |
| relevant_pr_number, |
| relevant_pr_url, |
| relevant_pr_user, |
| pr_reviewer, |
| current_datetime, |
| github_token, |
| ): |
| # type: (str, str, str, str, str, str, str) -> str |
| """Updates `_data/latest.json` to match build result of the given commit.""" |
| with open("_data/latest.json", "r+") as f: |
| latest = json.load(f, object_pairs_hook=collections.OrderedDict) |
| |
| current_status = { |
| os_: read_current_status(current_commit, "history/" + os_ + ".tsv") |
| for os_ in ["windows", "linux"] |
| } |
| |
| slug = "rust-lang/rust" |
| message = textwrap.dedent("""\ |
| 📣 Toolstate changed by {}! |
| |
| Tested on commit {}@{}. |
| Direct link to PR: <{}> |
| |
| """).format(relevant_pr_number, slug, current_commit, relevant_pr_url) |
| anything_changed = False |
| for status in latest: |
| tool = status["tool"] |
| changed = False |
| create_issue_for_status = None # set to the status that caused the issue |
| |
| for os_, s in current_status.items(): |
| old = status[os_] |
| new = s.get(tool, old) |
| status[os_] = new |
| maintainers = " ".join("@" + name for name in MAINTAINERS.get(tool, ())) |
| # comparing the strings, but they are ordered appropriately: |
| # "test-pass" > "test-fail" > "build-fail" |
| if new > old: |
| # things got fixed or at least the status quo improved |
| changed = True |
| message += "🎉 {} on {}: {} → {} (cc {}).\n".format( |
| tool, os_, old, new, maintainers |
| ) |
| elif new < old: |
| # tests or builds are failing and were not failing before |
| changed = True |
| title = "💔 {} on {}: {} → {}".format(tool, os_, old, new) |
| message += "{} (cc {}).\n".format(title, maintainers) |
| # See if we need to create an issue. |
| # Create issue if things no longer build. |
| # (No issue for mere test failures to avoid spurious issues.) |
| if new == "build-fail": |
| create_issue_for_status = new |
| |
| if create_issue_for_status is not None: |
| try: |
| issue( |
| tool, |
| create_issue_for_status, |
| MAINTAINERS.get(tool, ()), |
| relevant_pr_number, |
| relevant_pr_user, |
| LABELS.get(tool, []), |
| github_token, |
| ) |
| except HTTPError as e: |
| # network errors will simply end up not creating an issue, but that's better |
| # than failing the entire build job |
| print( |
| "HTTPError when creating issue for status regression: {0}\n{1!r}".format( |
| e, e.read() |
| ) |
| ) |
| except IOError as e: |
| print( |
| "I/O error when creating issue for status regression: {0}".format( |
| e |
| ) |
| ) |
| except: |
| print( |
| "Unexpected error when creating issue for status regression: {0}".format( |
| sys.exc_info()[0] |
| ) |
| ) |
| raise |
| |
| if changed: |
| status["commit"] = current_commit |
| status["datetime"] = current_datetime |
| anything_changed = True |
| |
| if not anything_changed: |
| return "" |
| |
| f.seek(0) |
| f.truncate(0) |
| json.dump(latest, f, indent=4, separators=(",", ": ")) |
| return message |
| |
| |
| # Warning: Do not try to add a function containing the body of this try block. |
| # There are variables declared within that are implicitly global; it is unknown |
| # which ones precisely but at least this is true for `github_token`. |
| try: |
| if __name__ != "__main__": |
| exit(0) |
| |
| cur_commit = sys.argv[1] |
| cur_datetime = datetime.datetime.now(datetime.timezone.utc).strftime( |
| "%Y-%m-%dT%H:%M:%SZ" |
| ) |
| cur_commit_msg = sys.argv[2] |
| save_message_to_path = sys.argv[3] |
| github_token = sys.argv[4] |
| |
| # assume that PR authors are also owners of the repo where the branch lives |
| relevant_pr_match = re.search( |
| r"Auto merge of #([0-9]+) - ([^:]+):[^,]+, r=(\S+)", |
| cur_commit_msg, |
| ) |
| if relevant_pr_match: |
| number = relevant_pr_match.group(1) |
| relevant_pr_user = relevant_pr_match.group(2) |
| relevant_pr_number = "rust-lang/rust#" + number |
| relevant_pr_url = "https://github.com/rust-lang/rust/pull/" + number |
| pr_reviewer = relevant_pr_match.group(3) |
| else: |
| number = "-1" |
| relevant_pr_user = "ghost" |
| relevant_pr_number = "<unknown PR>" |
| relevant_pr_url = "<unknown>" |
| pr_reviewer = "ghost" |
| |
| message = update_latest( |
| cur_commit, |
| relevant_pr_number, |
| relevant_pr_url, |
| relevant_pr_user, |
| pr_reviewer, |
| cur_datetime, |
| github_token, |
| ) |
| if not message: |
| print("<Nothing changed>") |
| sys.exit(0) |
| |
| print(message) |
| |
| if not github_token: |
| print("Dry run only, not committing anything") |
| sys.exit(0) |
| |
| with open(save_message_to_path, "w") as f: |
| f.write(message) |
| |
| # Write the toolstate comment on the PR as well. |
| issue_url = gh_url() + "/{}/comments".format(number) |
| response = urllib2.urlopen( |
| urllib2.Request( |
| issue_url, |
| json.dumps({"body": maybe_remove_mention(message)}).encode(), |
| { |
| "Authorization": "token " + github_token, |
| "Content-Type": "application/json", |
| }, |
| ) |
| ) |
| response.read() |
| except HTTPError as e: |
| print("HTTPError: %s\n%r" % (e, e.read())) |
| raise |