6.NUR/nur/index.py
Erik Oosting 20d4e9bd91
Switch from choosing the master branch to choosing the default branch
Co-authored-by: Jörg Thalheim <Mic92@users.noreply.github.com>
2021-01-13 11:42:54 +01:00

111 lines
3.8 KiB
Python

import json
import subprocess
import sys
from argparse import Namespace
from pathlib import Path
from tempfile import NamedTemporaryFile
from typing import Any, Dict
def resolve_source(pkg: Dict, repo: str, url: str) -> str:
# TODO commit hash
prefix = f"https://github.com/nix-community/nur-combined/tree/master/repos/{repo}"
position = pkg["meta"].get("position", None)
if position is not None and position.startswith("/nix/store"):
path_str, line = position.rsplit(":", 1)
path = Path(path_str)
# I've decided to just take these 2 repositories,
# update this whenever someone decided to use a recipe source other than
# NUR or nixpkgs to override packages on. right now this is about as accurate as
# `nix edit` is
# TODO find commit hash
prefixes = {
"nixpkgs": "https://github.com/nixos/nixpkgs/tree/master/",
"nur": "https://github.com/nix-community/nur-combined/tree/master/",
}
stripped = path.parts[4:]
if path.parts[3].endswith("source"):
canonical_url = url
# if we want to add the option of specifying branches, we have to update this
if "github" in url:
canonical_url += "/blob/HEAD/"
elif "gitlab" in url:
canonical_url += "/-/blob/HEAD/"
attr_path = "/".join(stripped)
location = f"{canonical_url}{attr_path}"
return f"{location}#L{line}"
elif stripped[0] not in prefixes:
print(path, file=sys.stderr)
print(
f"we could not find {stripped} , you can file an issue at https://github.com/nix-community/NUR/issues to the indexing file if you think this is a mistake",
file=sys.stderr,
)
return prefix
else:
attr_path = "/".join(stripped[1:])
location = f"{prefixes[stripped[0]]}{attr_path}"
return f"{location}#L{line}"
elif position is not None and "nur-combined" in position:
path_str, line = position.rsplit(":", 1)
stripped = path_str.partition(f"nur-combined/repos/{repo}")[2]
return f"{prefix}{stripped}#L{line}"
else:
return prefix
def index_repo(
directory: Path, repo: str, expression_file: str, url: str
) -> Dict[str, Any]:
default_nix = directory.joinpath("default.nix").resolve()
expr = """
with import <nixpkgs> {};
let
nur = import %s { nurpkgs = pkgs; inherit pkgs; };
in
callPackage (nur.repo-sources."%s" + "/%s") {}
""" % (
default_nix,
repo,
expression_file,
)
with NamedTemporaryFile(mode="w") as f:
f.write(expr)
f.flush()
query_cmd = ["nix-env", "-qa", "*", "--json", "-f", str(f.name)]
try:
out = subprocess.check_output(query_cmd)
except subprocess.CalledProcessError:
print(f"failed to evaluate {repo}", file=sys.stderr)
return {}
raw_pkgs = json.loads(out)
pkgs = {}
for name, pkg in raw_pkgs.items():
pkg["_attr"] = name
pkg["_repo"] = repo
pkg["meta"]["position"] = resolve_source(pkg, repo, url)
pkgs[f"nur.repos.{repo}.{name}"] = pkg
return pkgs
def index_command(args: Namespace) -> None:
directory = Path(args.directory)
manifest_path = directory.joinpath("repos.json")
with open(manifest_path) as f:
manifest = json.load(f)
repos = manifest.get("repos", [])
pkgs: Dict[str, Any] = {}
for (repo, data) in repos.items():
repo_pkgs = index_repo(
directory,
repo,
data.get("file", "default.nix"),
data.get("url", "https://github.com/nixos/nixpkgs"),
)
pkgs.update(repo_pkgs)
json.dump(pkgs, sys.stdout, indent=4)