Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: sync metadata from catalog-info.yaml with repo_checks (WIP) #428

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
83 changes: 73 additions & 10 deletions edx_repo_tools/repo_checks/repo_checks.py
Original file line number Diff line number Diff line change
Expand Up @@ -106,6 +106,7 @@ def __init__(self, api, org, repo):
self.api = api
self.org_name = org
self.repo_name = repo
self._repo_cache: dict[tuple[str, str], object] = {}

def is_relevant(self) -> bool:
"""
Expand Down Expand Up @@ -149,6 +150,21 @@ def dry_run(self):
"""
raise NotImplementedError

@cache
@property
def repo(self) -> object:
"""
Metadata for the checked repo, loaded from the GitHub API.

Cached (...per check instance! It would be nice to globally cache this, but check fixes
are likely to alter repo metadata, so a global repo metadata cache would fall
out of date during the lifetime of this script.)
"""
key = (self.org_name, self.repo_name)
if key not in self._repo_cache:
self._repo_cache[key] = self.api.repos.get(self.org_name, self.repo_name)
return self._repo_cache[key]


class EnsureWorkflowTemplates(Check):
"""
Expand Down Expand Up @@ -188,8 +204,7 @@ def check(self):
as the default templates in the `.github` repo.
"""
# Get the current default branch.
repo = self.api.repos.get(self.org_name, self.repo_name)
default_branch = repo.default_branch
default_branch = self.repo.default_branch

files_that_differ, files_that_are_missing = self._check_branch(default_branch)
# Return False and save the list of files that need to be updated.
Expand Down Expand Up @@ -288,8 +303,7 @@ def fix(self, dry_run=False):
raise # For any other unexpected errors.

# Get the hash of the default branch.
repo = self.api.repos.get(self.org_name, self.repo_name)
default_branch = repo.default_branch
default_branch = self.repo.default_branch
default_branch_sha = self.api.git.get_ref(
self.org_name,
self.repo_name,
Expand Down Expand Up @@ -668,8 +682,7 @@ def _check_cla_is_required_check(self) -> tuple[bool, str]:
"""
Is the CLA required on the repo? If not, what's wrong?
"""
repo = self.api.repos.get(self.org_name, self.repo_name)
default_branch = repo.default_branch
default_branch = self.repo.default_branch
# Branch protection rule might not exist.
try:
branch_protection = self.api.repos.get_branch_protection(
Expand Down Expand Up @@ -723,8 +736,7 @@ def _fix_branch_protection(self, dry_run=False):
if self.required_checks_has_cla_required:
return steps

repo = self.api.repos.get(self.org_name, self.repo_name)
default_branch = repo.default_branch
default_branch = self.repo.default_branch

# While the API docs claim that "contexts" is a required part
# of the put body, it is only required if "checks" is not supplied.
Expand Down Expand Up @@ -834,8 +846,7 @@ def _get_update_params_from_get_branch_protection(self):
"""

# TODO: Could use Glom here in the future, but didn't need it.
repo = self.api.repos.get(self.org_name, self.repo_name)
default_branch = repo.default_branch
default_branch = self.repo.default_branch
protection = self.api.repos.get_branch_protection(
self.org_name, self.repo_name, default_branch
)
Expand Down Expand Up @@ -880,11 +891,63 @@ def _get_update_params_from_get_branch_protection(self):
return params


class ApplyCatalogInfo:
"""
If catalog-info.yml exists, apply the description and first link as the repo's description and link.
"""

def __init__(self):
self.catalog_description: str | None = None
self.catalog_link: str | None = None

@cache
@property
def catalog_info(self) -> dict | None:
"""
Returns a YAML dict if catalog-info.yml exists, or None if it doesn't or can't be parsed.
"""
try:
get_github_file_contents(self.org_name, self.repo_name, self.repo.default_branch, "catalog-info.yml")
except:
# TODO more targeted error handling
return None

@property
def catalog_description(self) -> str | None:

def is_relevant(self) -> bool:
return bool(self.catalog_info)

def check(self):
"""
TODO
"""
self.catalog_description = (self.catalog_info or {}).get("metadata", {}).get("description")
links = (self.catalog_info or {}).get("metadata", {}).get("links")
if links:
self.catalog_link = links[0]
repo_description = ... # TODO
repo_link = ... # TODO
# TODO compare the two

def fix(self):
"""
TODO
"""

def dry_run(self):
"""
TODO
"""



CHECKS = [
RequiredCLACheck,
RequireTriageTeamAccess,
EnsureLabels,
EnsureWorkflowTemplates,
ApplyCatalogInfoDescription,
]
CHECKS_BY_NAME = {check_cls.__name__: check_cls for check_cls in CHECKS}
CHECKS_BY_NAME_LOWER = {check_cls.__name__.lower(): check_cls for check_cls in CHECKS}
Expand Down