mirror of
https://github.com/httpie/cli.git
synced 2025-01-25 14:58:43 +01:00
a65771e271
* Add a script that lists all contributors to a release We will keep a contributors database (simple JSON file) where each entry is a contributor (either a committer, either an issue reporter, either both) with some nicknames (GitHub, and Twitter). The file will be used to craft credits on our release blog posts and to ping them on Twitter. * Add templates * Missing docstring * Clean-up * Tweak
281 lines
8.7 KiB
Python
281 lines
8.7 KiB
Python
"""
|
|
Generate the contributors database.
|
|
|
|
FIXME: replace `requests` calls with the HTTPie API, when available.
|
|
"""
|
|
import json
|
|
import os
|
|
import re
|
|
import sys
|
|
from copy import deepcopy
|
|
from datetime import datetime
|
|
from pathlib import Path
|
|
from subprocess import check_output
|
|
from time import sleep
|
|
from typing import Any, Dict, Optional, Set
|
|
|
|
import requests
|
|
|
|
FullNames = Set[str]
|
|
GitHubLogins = Set[str]
|
|
Person = Dict[str, str]
|
|
People = Dict[str, Person]
|
|
UserInfo = Dict[str, Any]
|
|
|
|
CO_AUTHORS = re.compile(r'Co-authored-by: ([^<]+) <').finditer
|
|
API_URL = 'https://api.github.com'
|
|
REPO = OWNER = 'httpie'
|
|
REPO_URL = f'{API_URL}/repos/{REPO}/{OWNER}'
|
|
|
|
HERE = Path(__file__).parent
|
|
DB_FILE = HERE / 'people.json'
|
|
|
|
DEFAULT_PERSON: Person = {'committed': [], 'reported': [], 'github': '', 'twitter': ''}
|
|
SKIPPED_LABELS = {'invalid'}
|
|
|
|
GITHUB_TOKEN = os.getenv('GITHUB_TOKEN')
|
|
assert GITHUB_TOKEN, 'GITHUB_TOKEN envar is missing'
|
|
|
|
|
|
class FinishedForNow(Exception):
|
|
"""Raised when remaining GitHub rate limit is zero."""
|
|
|
|
|
|
def main(previous_release: str, current_release: str) -> int:
|
|
since = release_date(previous_release)
|
|
until = release_date(current_release)
|
|
|
|
contributors = load_awesome_people()
|
|
try:
|
|
committers = find_committers(since, until)
|
|
reporters = find_reporters(since, until)
|
|
except Exception as exc:
|
|
# We want to save what we fetched so far. So pass.
|
|
print(' !! ', exc)
|
|
|
|
try:
|
|
merge_all_the_people(current_release, contributors, committers, reporters)
|
|
fetch_missing_users_details(contributors)
|
|
except FinishedForNow:
|
|
# We want to save what we fetched so far. So pass.
|
|
print(' !! Committers:', committers)
|
|
print(' !! Reporters:', reporters)
|
|
exit_status = 1
|
|
else:
|
|
exit_status = 0
|
|
|
|
save_awesome_people(contributors)
|
|
return exit_status
|
|
|
|
|
|
def find_committers(since: str, until: str) -> FullNames:
|
|
url = f'{REPO_URL}/commits'
|
|
page = 1
|
|
per_page = 100
|
|
params = {
|
|
'since': since,
|
|
'until': until,
|
|
'per_page': per_page,
|
|
}
|
|
committers: FullNames = set()
|
|
|
|
while 'there are commits':
|
|
params['page'] = page
|
|
data = fetch(url, params=params)
|
|
|
|
for item in data:
|
|
commit = item['commit']
|
|
committers.add(commit['author']['name'])
|
|
debug(' >>> Commit', item['html_url'])
|
|
for co_author in CO_AUTHORS(commit['message']):
|
|
name = co_author.group(1)
|
|
committers.add(name)
|
|
|
|
if len(data) < per_page:
|
|
break
|
|
page += 1
|
|
|
|
return committers
|
|
|
|
|
|
def find_reporters(since: str, until: str) -> GitHubLogins:
|
|
url = f'{API_URL}/search/issues'
|
|
page = 1
|
|
per_page = 100
|
|
params = {
|
|
'q': f'repo:{REPO}/{OWNER} is:issue closed:{since}..{until}',
|
|
'per_page': per_page,
|
|
}
|
|
reporters: GitHubLogins = set()
|
|
|
|
while 'there are issues':
|
|
params['page'] = page
|
|
data = fetch(url, params=params)
|
|
|
|
for item in data['items']:
|
|
# Filter out unwanted labels.
|
|
if any(label['name'] in SKIPPED_LABELS for label in item['labels']):
|
|
continue
|
|
debug(' >>> Issue', item['html_url'])
|
|
reporters.add(item['user']['login'])
|
|
|
|
if len(data['items']) < per_page:
|
|
break
|
|
page += 1
|
|
|
|
return reporters
|
|
|
|
|
|
def merge_all_the_people(release: str, contributors: People, committers: FullNames, reporters: GitHubLogins) -> None:
|
|
"""
|
|
>>> contributors = {'Alice': new_person(github='alice', twitter='alice')}
|
|
>>> merge_all_the_people('2.6.0', contributors, {}, {})
|
|
>>> contributors
|
|
{'Alice': {'committed': [], 'reported': [], 'github': 'alice', 'twitter': 'alice'}}
|
|
|
|
>>> contributors = {'Bob': new_person(github='bob', twitter='bob')}
|
|
>>> merge_all_the_people('2.6.0', contributors, {'Bob'}, {'bob'})
|
|
>>> contributors
|
|
{'Bob': {'committed': ['2.6.0'], 'reported': ['2.6.0'], 'github': 'bob', 'twitter': 'bob'}}
|
|
|
|
>>> contributors = {'Charlotte': new_person(github='charlotte', twitter='charlotte', committed=['2.5.0'], reported=['2.5.0'])}
|
|
>>> merge_all_the_people('2.6.0', contributors, {'Charlotte'}, {'charlotte'})
|
|
>>> contributors
|
|
{'Charlotte': {'committed': ['2.5.0', '2.6.0'], 'reported': ['2.5.0', '2.6.0'], 'github': 'charlotte', 'twitter': 'charlotte'}}
|
|
|
|
"""
|
|
# Update known contributors.
|
|
for name, details in contributors.items():
|
|
if name in committers:
|
|
if release not in details['committed']:
|
|
details['committed'].append(release)
|
|
committers.remove(name)
|
|
if details['github'] in reporters:
|
|
if release not in details['reported']:
|
|
details['reported'].append(release)
|
|
reporters.remove(details['github'])
|
|
|
|
# Add new committers.
|
|
for name in committers:
|
|
user_info = user(fullname=name)
|
|
contributors[name] = new_person(
|
|
github=user_info['login'],
|
|
twitter=user_info['twitter_username'],
|
|
committed=[release],
|
|
)
|
|
if user_info['login'] in reporters:
|
|
contributors[name]['reported'].append(release)
|
|
reporters.remove(user_info['login'])
|
|
|
|
# Add new reporters.
|
|
for github_username in reporters:
|
|
user_info = user(github_username=github_username)
|
|
contributors[user_info['name'] or user_info['login']] = new_person(
|
|
github=github_username,
|
|
twitter=user_info['twitter_username'],
|
|
reported=[release],
|
|
)
|
|
|
|
|
|
def release_date(release: str) -> str:
|
|
date = check_output(['git', 'log', '-1', '--format=%ai', release], text=True).strip()
|
|
return datetime.strptime(date, '%Y-%m-%d %H:%M:%S %z').isoformat()
|
|
|
|
|
|
def load_awesome_people() -> People:
|
|
try:
|
|
with DB_FILE.open(encoding='utf-8') as fh:
|
|
return json.load(fh)
|
|
except (FileNotFoundError, ValueError):
|
|
return {}
|
|
|
|
|
|
def fetch(url: str, params: Optional[Dict[str, str]] = None) -> UserInfo:
|
|
headers = {
|
|
'Accept': 'application/vnd.github.v3+json',
|
|
'Authentication': f'token {GITHUB_TOKEN}'
|
|
}
|
|
for retry in range(1, 6):
|
|
debug(f'[{retry}/5]', f'{url = }', f'{params = }')
|
|
with requests.get(url, params=params, headers=headers) as req:
|
|
try:
|
|
req.raise_for_status()
|
|
except requests.exceptions.HTTPError as exc:
|
|
if exc.response.status_code == 403:
|
|
# 403 Client Error: rate limit exceeded for url: ...
|
|
now = int(datetime.utcnow().timestamp())
|
|
xrate_limit_reset = int(exc.response.headers['X-RateLimit-Reset'])
|
|
wait = xrate_limit_reset - now
|
|
if wait > 20:
|
|
raise FinishedForNow()
|
|
debug(' !', 'Waiting', wait, 'seconds before another try ...')
|
|
sleep(wait)
|
|
continue
|
|
return req.json()
|
|
assert ValueError('Rate limit exceeded')
|
|
|
|
|
|
def new_person(**kwargs: str) -> Person:
|
|
data = deepcopy(DEFAULT_PERSON)
|
|
data.update(**kwargs)
|
|
return data
|
|
|
|
|
|
def user(fullname: Optional[str] = '', github_username: Optional[str] = '') -> UserInfo:
|
|
if github_username:
|
|
url = f'{API_URL}/users/{github_username}'
|
|
return fetch(url)
|
|
|
|
url = f'{API_URL}/search/users'
|
|
for query in (f'fullname:{fullname}', f'user:{fullname}'):
|
|
params = {
|
|
'q': f'repo:{REPO}/{OWNER} {query}',
|
|
'per_page': 1,
|
|
}
|
|
user_info = fetch(url, params=params)
|
|
if user_info['items']:
|
|
user_url = user_info['items'][0]['url']
|
|
return fetch(user_url)
|
|
|
|
|
|
def fetch_missing_users_details(people: People) -> None:
|
|
for name, details in people.items():
|
|
if details['github'] and details['twitter']:
|
|
continue
|
|
user_info = user(github_username=details['github'], fullname=name)
|
|
if not details['github']:
|
|
details['github'] = user_info['login']
|
|
if not details['twitter']:
|
|
details['twitter'] = user_info['twitter_username']
|
|
|
|
|
|
def save_awesome_people(people: People) -> None:
|
|
with DB_FILE.open(mode='w', encoding='utf-8') as fh:
|
|
json.dump(people, fh, indent=4, sort_keys=True)
|
|
|
|
|
|
def debug(*args: Any) -> None:
|
|
if os.getenv('DEBUG') == '1':
|
|
print(*args)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
ret = 1
|
|
try:
|
|
ret = main(*sys.argv[1:])
|
|
except TypeError:
|
|
ret = 2
|
|
print(f'''
|
|
Fetch contributors to a release.
|
|
|
|
Usage:
|
|
python {sys.argv[0]} {sys.argv[0]} <RELEASE N-1> <RELEASE N>
|
|
Example:
|
|
python {sys.argv[0]} 2.4.0 2.5.0
|
|
|
|
Define the DEBUG=1 environment variable to enable verbose output.
|
|
''')
|
|
except KeyboardInterrupt:
|
|
ret = 255
|
|
sys.exit(ret)
|