ci_get_mr_info.py 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. #!/usr/bin/env python
  2. #
  3. # internal use only for CI
  4. # get latest MR information by source branch
  5. #
  6. # SPDX-FileCopyrightText: 2020-2022 Espressif Systems (Shanghai) CO LTD
  7. # SPDX-License-Identifier: Apache-2.0
  8. #
  9. import argparse
  10. import os
  11. import subprocess
  12. import typing as t
  13. from pathlib import Path
  14. from gitlab_api import Gitlab
  15. if t.TYPE_CHECKING:
  16. from gitlab.v4.objects import ProjectCommit, ProjectMergeRequest
  17. def _get_mr_obj(source_branch: str) -> t.Optional['ProjectMergeRequest']:
  18. gl = Gitlab(os.getenv('CI_PROJECT_ID', 'espressif/esp-idf'))
  19. if not gl.project:
  20. return None
  21. mrs = gl.project.mergerequests.list(state='opened', source_branch=source_branch)
  22. if mrs:
  23. return mrs[0] # one source branch can only have one opened MR at one moment
  24. else:
  25. return None
  26. def get_mr_iid(source_branch: str) -> str:
  27. mr = _get_mr_obj(source_branch)
  28. if not mr:
  29. return ''
  30. else:
  31. return str(mr.iid)
  32. def get_mr_changed_files(source_branch: str) -> t.List[str]:
  33. mr = _get_mr_obj(source_branch)
  34. if not mr:
  35. return []
  36. git_output = subprocess.check_output(
  37. ['git', 'diff', '--name-only', f'origin/{mr.target_branch}...origin/{source_branch}']
  38. ).decode('utf8')
  39. return [line.strip() for line in git_output.splitlines() if line.strip()]
  40. def get_mr_commits(source_branch: str) -> t.List['ProjectCommit']:
  41. mr = _get_mr_obj(source_branch)
  42. if not mr:
  43. return []
  44. return list(mr.commits())
  45. def get_mr_components(source_branch: str) -> t.List[str]:
  46. components: t.Set[str] = set()
  47. for f in get_mr_changed_files(source_branch):
  48. file = Path(f)
  49. if (
  50. file.parts[0] == 'components'
  51. and 'test_apps' not in file.parts
  52. and file.parts[-1] != '.build-test-rules.yml'
  53. ):
  54. components.add(file.parts[1])
  55. return list(components)
  56. def _print_list(_list: t.List[str], separator: str = '\n') -> None:
  57. print(separator.join(_list))
  58. if __name__ == '__main__':
  59. parser = argparse.ArgumentParser(description='Get the latest merge request info by pipeline')
  60. actions = parser.add_subparsers(dest='action', help='info type', required=True)
  61. common_args = argparse.ArgumentParser(add_help=False)
  62. common_args.add_argument('src_branch', help='source branch')
  63. actions.add_parser('id', parents=[common_args])
  64. actions.add_parser('files', parents=[common_args])
  65. actions.add_parser('commits', parents=[common_args])
  66. actions.add_parser('components', parents=[common_args])
  67. args = parser.parse_args()
  68. if args.action == 'id':
  69. print(get_mr_iid(args.src_branch))
  70. elif args.action == 'files':
  71. _print_list(get_mr_changed_files(args.src_branch))
  72. elif args.action == 'commits':
  73. _print_list([commit.id for commit in get_mr_commits(args.src_branch)])
  74. elif args.action == 'components':
  75. _print_list(get_mr_components(args.src_branch))
  76. else:
  77. raise NotImplementedError('not possible to get here')