forked from mozilla/github-org-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
get_org_info.py
executable file
·242 lines (219 loc) · 8.09 KB
/
get_org_info.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
#!/usr/bin/env python3
# PYTHON_ARGCOMPLETE_OK
"""Report Basic info about orgs."""
# additional help text
_epilog = """
This script outputs some information about organizations, mostly immutable
identifiers and contact info.
Current owner list is available via the '--owners' option.
"""
# hwine believes keeping the doc above together is more important than PEP-8
import argparse # NOQA
import base64
import logging # NOQA
import json
import sys
import time
import typing
from collections import defaultdict # NOQA
import argcomplete
import github3 # NOQA
from client import get_github3_client # NOQA
logger = logging.getLogger(__name__)
DEBUG = False
gh: typing.Any = None
def jsonl_out(d):
print(json.dumps(d))
def show_info(
gh,
org_name,
show_owners=False,
show_emails=False,
show_json=False,
owners_only=False,
):
def miss():
return "<hidden>"
try:
org = gh.organization(org_name)
orgd = defaultdict(miss, org.as_dict())
if show_json and not show_owners:
jsonl_out(orgd)
if not owners_only:
v4decoded = "{:03}:{}{}".format(
len(orgd["type"]), orgd["type"], str(org.id)
)
v4encoded = base64.b64encode(bytes(v4decoded, "utf-8"))
print(
"{:>15}: {!s} ({})".format("Name", org.name or org_name, orgd["login"])
)
print("{:>15}: {!s}".format("API v3 id", org.id))
print("{:>15}: {!s}".format("API v4 id", f"{v4encoded} ({v4decoded})"))
print("{:>15}: {!s}".format("contact", org.email))
print("{:>15}: {!s}".format("billing", orgd["billing_email"]))
print(
"{:>15}: {!s}".format(
"2FA required", orgd["two_factor_requirement_enabled"]
)
)
print("{:>15}: {!s}".format("private repos", orgd["owned_private_repos"]))
# Nested dictionaries need special handling
plan = orgd["plan"]
if not isinstance(plan, dict):
plan = defaultdict(miss)
print("{:>15}: {!s}".format("plan", plan["name"]))
print("{:>15}: {!s}".format("seats", plan["filled_seats"]))
if show_owners:
if not owners_only:
print("{:>15}:".format("Org Owners"))
owner_info = defaultdict(miss)
owner_info["org"] = org.login
owner_info["org_v3_id"] = org.id
owner_info["org_v4_id"] = org.as_dict()["node_id"]
for owner in org.members(role="admin"):
owner = owner.refresh() # get name
owner_info["name"] = owner.name or "<hidden>"
owner_info["login"] = owner.login or "<hidden>"
owner_info["id_v3"] = owner.id or "<hidden>"
owner_info["id_v4"] = owner.node_id or "<hidden>"
if show_emails:
email = " " + (owner.email or "<email hidden>")
else:
email = ""
owner_info["email"] = email
if show_json:
jsonl_out(owner_info)
else:
print(
f' {owner_info["name"]} ({owner_info["login"]}{email})'
)
except Exception as e:
logger.error("Error %s obtaining data for org '%s'", str(e), str(org))
finally:
if DEBUG:
from pprint import pprint
pprint(orgd, stream=sys.stderr)
def parse_args():
parser = argparse.ArgumentParser(description=__doc__, epilog=_epilog)
parser.add_argument(
"--debug", action="store_true", help="include dump of all data returned"
)
parser.add_argument("--owners", action="store_true", help="Also show owners")
parser.add_argument("--owners-only", action="store_true", help="Only show owners")
parser.add_argument("--email", action="store_true", help="include owner email")
parser.add_argument("--json", action="store_true", help="output as json lines")
parser.add_argument(
"--all-my-orgs",
action="store_true",
help="act on all orgs for which you're an owner",
)
parser.add_argument(
"--names-only",
action="store_true",
help="Only output your org names for which you're an owner",
)
parser.add_argument(
"orgs",
nargs="*",
help="github organizations to check (defaults to " "mozilla)",
)
argcomplete.autocomplete(parser)
args = parser.parse_args()
if args.names_only:
args.all_my_orgs = True
if args.owners or args.email:
parser.error("Can't specify owners or emails with --all-my-orgs")
if not args.all_my_orgs and len(args.orgs) == 0:
args.orgs = ["mozilla"]
if args.owners_only and args.json and not args.email:
# implies we want email
args.email = True
if args.email and not args.owners:
# option implies owners
args.owners = True
if args.debug:
global DEBUG
DEBUG = args.debug
return args
# api.github.com/user/orgs endpoint not natively supported
# belongs in authenticated user
class MyOrganizationsIterator(github3.structs.GitHubIterator):
def __init__(self, me):
super().__init__(
count=-1, # get all
url=me.session.base_url + "/user/orgs",
cls=github3.orgs.ShortOrganization,
session=me.session,
)
def print_limits(e=None, verbose=False):
if e:
# display("API limit reached, try again in 5 minutes.\\n")
print(str(e), file=sys.stderr)
reset_max = reset_min = 0
global gh
limits = gh.rate_limit()
resources = limits["resources"]
# print("{:3d} keys: ".format(len(resources.keys())), resources.keys())
# print(resources)
for reset in resources.keys():
reset_at = resources[reset]["reset"]
reset_max = max(reset_at, reset_max)
if not resources[reset]["remaining"]:
reset_min = min(reset_at, reset_min if reset_min else reset_at)
if verbose:
print(
"EXPIRED for {} {}".format(reset, resources[reset]["remaining"]),
file=sys.stderr,
)
else:
if verbose:
print(
"remaining for {} {}".format(
reset, resources[reset]["remaining"], file=sys.stderr
)
)
if not reset_min:
print("No limits reached currently.", file=sys.stderr)
else:
print(
"Minimum reset at {} UTC ({})".format(
time.asctime(time.gmtime(reset_min)),
time.asctime(time.localtime(reset_min)),
file=sys.stderr,
)
)
print(
"All reset at {} UTC".format(
time.asctime(time.gmtime(reset_max)),
time.asctime(time.localtime(reset_max)),
file=sys.stderr,
)
)
def main():
args = parse_args()
if args.orgs or args.all_my_orgs:
global gh
gh = get_github3_client()
try:
if args.all_my_orgs:
authorized_user = gh.me()
me = gh.user(authorized_user.login)
my_orgs = MyOrganizationsIterator(me)
for org in my_orgs:
owner_logins = [u.login for u in org.members(role="admin")]
if me.login in owner_logins:
args.orgs.append(org.login)
if args.names_only:
print("\n".join(sorted(args.orgs)))
return
newline = ""
for org in args.orgs:
if len(args.orgs) > 1 and not args.json:
print(f"{newline}Processing org {org}")
newline = "\n"
show_info(gh, org, args.owners, args.email, args.json, args.owners_only)
except github3.exceptions.ForbiddenError as e:
print_limits(e)
if __name__ == "__main__":
logging.basicConfig(level=logging.WARN, format="%(asctime)s %(message)s")
main()