|
| 1 | +from os import error |
| 2 | +from pathlib import Path |
| 3 | +from jinja2 import Template |
| 4 | +import datetime |
| 5 | +import logging |
| 6 | +import requests |
| 7 | +import os |
| 8 | +import sys |
| 9 | +import yaml |
| 10 | + |
| 11 | +logger = logging.getLogger(__name__) |
| 12 | + |
| 13 | +headers = {} |
| 14 | +env = dict(os.environ) |
| 15 | +if "GITHUB_TOKEN" in env: |
| 16 | + headers["Authorization"] = " ".join(["Bearer", os.environ["GITHUB_TOKEN"]]) |
| 17 | +else: |
| 18 | + logger.warning("No GITHUB_TOKEN found in env") |
| 19 | + |
| 20 | +def unslugify(s: str) -> str: |
| 21 | + return s.replace("-", " ").replace("_", " ").title() |
| 22 | + |
| 23 | +def get_repo(response) -> dict: |
| 24 | + data = { |
| 25 | + "title": unslugify(response["name"]), |
| 26 | + "url": response["html_url"], |
| 27 | + "description": response["description"] |
| 28 | + } |
| 29 | + if response["archived"]: |
| 30 | + data["description"] = "**DEPRECATED.** " + data["description"] |
| 31 | + return data |
| 32 | + |
| 33 | +def get_repos() -> list: |
| 34 | + logger.info(f"Reading list of repositories") |
| 35 | + |
| 36 | + with requests.get("https://api.github.com/users/fiboa/repos?per_page=1000", headers=headers) as site: |
| 37 | + repos = site.json() |
| 38 | + return repos |
| 39 | + |
| 40 | +def filter_repos(repos, type) -> list: |
| 41 | + data = [] |
| 42 | + try: |
| 43 | + for repo in repos: |
| 44 | + if not isinstance(repo, dict): |
| 45 | + logger.error(f"response invalid") |
| 46 | + continue |
| 47 | + if repo["is_template"] or repo["visibility"] != "public" or type not in repo["topics"]: |
| 48 | + continue |
| 49 | + data.append(get_repo(repo)) |
| 50 | + except error as e: |
| 51 | + logger.error(f"fiboa org not available: {e}") |
| 52 | + return data |
| 53 | + |
| 54 | +def get_externals(base) -> list: |
| 55 | + data = [] |
| 56 | + config = yaml.safe_load((base / "external.yaml").read_text()) |
| 57 | + for r in config["github"]: |
| 58 | + try: |
| 59 | + logger.info(f"Reading community GitHub repos individually") |
| 60 | + with requests.get(f"https://api.github.com/repos/{r['org']}/{r['repo']}", headers=headers) as repo: |
| 61 | + data.append(get_repo(repo.json())) |
| 62 | + except error as e: |
| 63 | + logger.error(f"community repo not available: {e}") |
| 64 | + |
| 65 | + for r in config["external"]: |
| 66 | + data.append(r) |
| 67 | + |
| 68 | + return data |
| 69 | + |
| 70 | +def create_overview(repos, folder, label): |
| 71 | + print(f"Generating {folder}") |
| 72 | + base = Path(f"./{folder}/") |
| 73 | + data = filter_repos(repos, label) |
| 74 | + data.extend(get_externals(base)) |
| 75 | + data.sort(key = lambda x: x["title"]) |
| 76 | + count = len(data) |
| 77 | + now = datetime.datetime.now(datetime.timezone.utc).strftime("%b %d %Y, %H:%M %Z") |
| 78 | + template = Template((base / "index.md.jinja").read_text()) |
| 79 | + target = base / "index.md" |
| 80 | + content = template.render(data=data, updated=now, count=count) |
| 81 | + |
| 82 | + with open(target, "w", encoding="utf-8") as f: |
| 83 | + f.write(content) |
| 84 | + |
| 85 | +def main(): |
| 86 | + repos = get_repos() |
| 87 | + create_overview(repos, "extensions", "extension") |
| 88 | + create_overview(repos, "software", "software") |
| 89 | + sys.exit(0) |
| 90 | + |
| 91 | + |
| 92 | +if __name__ == "__main__": |
| 93 | + main() |
0 commit comments