選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 
 
 

127 行
3.5 KiB

  1. import json
  2. import os
  3. import re
  4. import shlex
  5. import subprocess
  6. import sys
  7. import urllib.request
  8. from functools import lru_cache
  9. @lru_cache(maxsize=None)
  10. def fetch_pr_data(pr_number, repo, endpoint=""):
  11. api_url = f"https://api.github.com/repos/{repo}/pulls/{pr_number}"
  12. if endpoint:
  13. api_url += f"/{endpoint}"
  14. req = urllib.request.Request(api_url)
  15. res = urllib.request.urlopen(req)
  16. return json.loads(res.read().decode("utf8"))
  17. def get_files_list(pr_number, repo="frappe/frappe"):
  18. return [change["filename"] for change in fetch_pr_data(pr_number, repo, "files")]
  19. def get_output(command, shell=True):
  20. print(command)
  21. command = shlex.split(command)
  22. return subprocess.check_output(command, shell=shell, encoding="utf8").strip()
  23. def has_skip_ci_label(pr_number, repo="frappe/frappe"):
  24. return has_label(pr_number, "Skip CI", repo)
  25. def has_run_server_tests_label(pr_number, repo="frappe/frappe"):
  26. return has_label(pr_number, "Run Server Tests", repo)
  27. def has_run_ui_tests_label(pr_number, repo="frappe/frappe"):
  28. return has_label(pr_number, "Run UI Tests", repo)
  29. def has_label(pr_number, label, repo="frappe/frappe"):
  30. return any(
  31. [
  32. fetched_label["name"]
  33. for fetched_label in fetch_pr_data(pr_number, repo)["labels"]
  34. if fetched_label["name"] == label
  35. ]
  36. )
  37. def is_py(file):
  38. return file.endswith("py")
  39. def is_ci(file):
  40. return ".github" in file
  41. def is_frontend_code(file):
  42. return file.lower().endswith(
  43. (".css", ".scss", ".less", ".sass", ".styl", ".js", ".ts", ".vue", ".html")
  44. )
  45. def is_docs(file):
  46. regex = re.compile(r"\.(md|png|jpg|jpeg|csv|svg)$|^.github|LICENSE")
  47. return bool(regex.search(file))
  48. if __name__ == "__main__":
  49. files_list = sys.argv[1:]
  50. build_type = os.environ.get("TYPE")
  51. pr_number = os.environ.get("PR_NUMBER")
  52. repo = os.environ.get("REPO_NAME")
  53. # this is a push build, run all builds
  54. if not pr_number:
  55. os.system('echo "::set-output name=build::strawberry"')
  56. os.system('echo "::set-output name=build-server::strawberry"')
  57. sys.exit(0)
  58. files_list = files_list or get_files_list(pr_number=pr_number, repo=repo)
  59. if not files_list:
  60. print("No files' changes detected. Build is shutting")
  61. sys.exit(0)
  62. ci_files_changed = any(f for f in files_list if is_ci(f))
  63. only_docs_changed = len(list(filter(is_docs, files_list))) == len(files_list)
  64. only_frontend_code_changed = len(list(filter(is_frontend_code, files_list))) == len(files_list)
  65. updated_py_file_count = len(list(filter(is_py, files_list)))
  66. only_py_changed = updated_py_file_count == len(files_list)
  67. if has_skip_ci_label(pr_number, repo):
  68. if build_type == "ui" and has_run_ui_tests_label(pr_number, repo):
  69. print("Running UI tests only.")
  70. elif build_type == "server" and has_run_server_tests_label(pr_number, repo):
  71. print("Running server tests only.")
  72. else:
  73. print("Found `Skip CI` label on pr, stopping build process.")
  74. sys.exit(0)
  75. elif ci_files_changed:
  76. print("CI related files were updated, running all build processes.")
  77. elif only_docs_changed:
  78. print("Only docs were updated, stopping build process.")
  79. sys.exit(0)
  80. elif (
  81. only_frontend_code_changed
  82. and build_type == "server"
  83. and not has_run_server_tests_label(pr_number, repo)
  84. ):
  85. print("Only Frontend code was updated; Stopping Python build process.")
  86. sys.exit(0)
  87. elif build_type == "ui" and only_py_changed and not has_run_ui_tests_label(pr_number, repo):
  88. print("Only Python code was updated, stopping Cypress build process.")
  89. sys.exit(0)
  90. os.system('echo "::set-output name=build::strawberry"')