2024-10-30 12:42:27 -07:00
|
|
|
import networkx as nx
|
|
|
|
from sgqlc.endpoint.http import HTTPEndpoint
|
|
|
|
from sgqlc.operation import Operation
|
|
|
|
|
2024-11-11 15:39:59 -08:00
|
|
|
from helpers.repos import CORE_REPOS, TFL_REPOS, WALLET_REPOS, ZF_REPOS, ZF_FROST_REPOS
|
2024-10-30 12:42:27 -07:00
|
|
|
from zenhub_schema import zenhub_schema
|
|
|
|
|
|
|
|
WORKSPACE_SETS = {
|
|
|
|
# ecc-core
|
2024-11-01 15:51:10 -07:00
|
|
|
'5dc1fd615862290001229f21': list(CORE_REPOS.keys()) + list(TFL_REPOS.keys()),
|
2024-10-30 12:42:27 -07:00
|
|
|
# ecc-wallet
|
|
|
|
'5db8aa0244512d0001e0968e': WALLET_REPOS.keys(),
|
|
|
|
# zf
|
|
|
|
'5fb24d9264a3e8000e666a9e': ZF_REPOS.keys(),
|
|
|
|
# zf-frost
|
|
|
|
'607d75e0169bd50011d5410f': ZF_FROST_REPOS.keys(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def api(token):
|
|
|
|
return HTTPEndpoint(
|
|
|
|
'https://api.zenhub.com/public/graphql',
|
|
|
|
{'Authorization': 'Bearer %s' % token},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def fetch_workspace_graph(op, workspace_id, repos, cursor):
|
|
|
|
dependencies = op.workspace(id=workspace_id).issue_dependencies(
|
|
|
|
# TODO: This causes a 500 Internal Server Error. We need the ZenHub repo IDs here,
|
|
|
|
# not the GitHub repo IDs (which the previous REST API used).
|
|
|
|
# repository_ids=repos,
|
|
|
|
first=100,
|
|
|
|
after=cursor,
|
|
|
|
)
|
|
|
|
dependencies.nodes.id()
|
|
|
|
dependencies.nodes.blocked_issue.number()
|
|
|
|
dependencies.nodes.blocked_issue.repository.gh_id()
|
|
|
|
dependencies.nodes.blocking_issue.number()
|
|
|
|
dependencies.nodes.blocking_issue.repository.gh_id()
|
|
|
|
dependencies.page_info.has_next_page()
|
|
|
|
dependencies.page_info.end_cursor()
|
|
|
|
|
|
|
|
|
2024-11-11 16:57:11 -08:00
|
|
|
# Fetches the dependency graph involving the given `repos` from the given `workspace_id`.
|
|
|
|
#
|
|
|
|
# `repos` is a list of GitHub repo IDs.
|
|
|
|
#
|
|
|
|
# Returns a list of `(blocking, blocked)` tuples corresponding to DAG edges.
|
|
|
|
# `blocking` and `blocked` are both `(repo_github_id, issue_number)` tuples.
|
2024-10-30 12:42:27 -07:00
|
|
|
def get_dependency_graph(endpoint, workspace_id, repos):
|
|
|
|
edges = []
|
|
|
|
cursor = None
|
|
|
|
|
|
|
|
while True:
|
|
|
|
op = Operation(zenhub_schema.Query)
|
|
|
|
fetch_workspace_graph(op, workspace_id, repos, cursor)
|
|
|
|
|
|
|
|
d = endpoint(op)
|
|
|
|
data = op + d
|
|
|
|
|
2024-10-30 21:39:46 -07:00
|
|
|
if hasattr(data.workspace, 'issue_dependencies'):
|
|
|
|
dependencies = data.workspace.issue_dependencies
|
|
|
|
edges += [
|
|
|
|
(
|
|
|
|
(node.blocking_issue.repository.gh_id, node.blocking_issue.number),
|
|
|
|
(node.blocked_issue.repository.gh_id, node.blocked_issue.number),
|
|
|
|
)
|
|
|
|
for node in dependencies.nodes
|
|
|
|
]
|
2024-10-30 12:42:27 -07:00
|
|
|
|
2024-10-30 21:39:46 -07:00
|
|
|
if dependencies.page_info.has_next_page:
|
|
|
|
cursor = dependencies.page_info.end_cursor
|
|
|
|
print('.', end='', flush=True)
|
|
|
|
else:
|
|
|
|
print()
|
|
|
|
break
|
2024-10-30 12:42:27 -07:00
|
|
|
else:
|
|
|
|
print()
|
|
|
|
break
|
|
|
|
|
|
|
|
return nx.DiGraph(edges)
|
|
|
|
|
|
|
|
|
|
|
|
def fetch_epics(op, workspace_id, repos, cursor):
|
|
|
|
epics = op.workspace(id=workspace_id).epics(
|
|
|
|
# TODO: This causes a 500 Internal Server Error. We need the ZenHub repo IDs here,
|
|
|
|
# not the GitHub repo IDs (which the previous REST API used).
|
|
|
|
# repository_ids=repos,
|
|
|
|
first=100,
|
|
|
|
after=cursor,
|
|
|
|
)
|
|
|
|
epics.nodes.id()
|
|
|
|
epics.nodes.issue.number()
|
|
|
|
epics.nodes.issue.repository.gh_id()
|
|
|
|
epics.page_info.has_next_page()
|
|
|
|
epics.page_info.end_cursor()
|
|
|
|
|
|
|
|
|
|
|
|
def get_epics(endpoint, workspace_id, repos):
|
|
|
|
epics = []
|
|
|
|
cursor = None
|
|
|
|
|
|
|
|
while True:
|
|
|
|
op = Operation(zenhub_schema.Query)
|
|
|
|
fetch_epics(op, workspace_id, repos, cursor)
|
|
|
|
|
|
|
|
d = endpoint(op)
|
|
|
|
data = op + d
|
|
|
|
|
2024-11-01 16:50:30 -07:00
|
|
|
if hasattr(data.workspace, 'epics'):
|
|
|
|
epics_page = data.workspace.epics
|
|
|
|
epics += [
|
|
|
|
(node.id, (node.issue.repository.gh_id, node.issue.number))
|
|
|
|
for node in epics_page.nodes
|
|
|
|
]
|
|
|
|
|
|
|
|
if epics_page.page_info.has_next_page:
|
|
|
|
cursor = epics_page.page_info.end_cursor
|
|
|
|
print('.', end='', flush=True)
|
|
|
|
else:
|
|
|
|
print()
|
|
|
|
break
|
2024-10-30 12:42:27 -07:00
|
|
|
else:
|
|
|
|
print()
|
|
|
|
break
|
|
|
|
|
|
|
|
return epics
|
|
|
|
|
|
|
|
|
|
|
|
def fetch_epic_issues(op, workspace_id, epic_id, cursor):
|
|
|
|
epic = op.workspace(id=workspace_id).epics(ids=[epic_id])
|
|
|
|
child_issues = epic.nodes.child_issues(
|
|
|
|
first=100,
|
|
|
|
after=cursor,
|
|
|
|
)
|
|
|
|
child_issues.nodes.number()
|
|
|
|
child_issues.nodes.repository.gh_id()
|
|
|
|
child_issues.page_info.has_next_page()
|
|
|
|
child_issues.page_info.end_cursor()
|
|
|
|
|
|
|
|
|
|
|
|
def get_epic_issues(endpoint, workspace_id, epic_id):
|
|
|
|
epic_issues = []
|
|
|
|
cursor = None
|
|
|
|
|
|
|
|
while True:
|
|
|
|
op = Operation(zenhub_schema.Query)
|
|
|
|
fetch_epic_issues(op, workspace_id, epic_id, cursor)
|
|
|
|
|
|
|
|
d = endpoint(op)
|
|
|
|
data = op + d
|
|
|
|
|
|
|
|
epic = data.workspace.epics.nodes[0]
|
|
|
|
epic_issues += [
|
|
|
|
(node.repository.gh_id, node.number) for node in epic.child_issues.nodes
|
|
|
|
]
|
|
|
|
|
|
|
|
if epic.child_issues.page_info.has_next_page:
|
|
|
|
cursor = epic.child_issues.page_info.end_cursor
|
|
|
|
print('.', end='', flush=True)
|
|
|
|
else:
|
|
|
|
print()
|
|
|
|
break
|
|
|
|
|
|
|
|
return epic_issues
|