import backoff
+
+class GitCacheError(Exception):
+ pass
+
+
Path = str # eg: "/home/user/.cache/git-cache/v1"
Repo = str # eg: "https://github.com/NixOS/nixpkgs.git"
Ref = str # eg: "master" or "v1.0.0"
'merge-base',
'--is-ancestor',
ancestor,
- descendant],
+ descendant.removeprefix('tag ')],
check=False)
return process.returncode == 0
ancestor: RefOrRev,
force: bool = False) -> None:
if not force and not is_ancestor(repo, descendant, ancestor):
- raise Exception(f'{ancestor} is not an ancestor of {descendant}')
+ raise GitCacheError(f'{ancestor} is not an ancestor of {descendant}')
def _read_fetch_log(repo: Repo) -> Iterator[_LogEntry]:
repo: Repo,
ref: Ref,
force: bool = False) -> None:
+ refargs = (['tag', ref.removeprefix('tag ')]
+ if ref.startswith('tag ')
+ else [f'{ref}:{ref}'])
subprocess.run(['git', '-C', cachedir, 'fetch'] +
(['--force'] if force else []) +
- [repo, f'{ref}:{ref}'], check=True)
+ [repo] + refargs, check=True)
def fetch(repo: Repo, ref: Ref, force: bool = False) -> Tuple[Path, Rev]:
logging.debug('Fetching ref "%s" from %s', ref, repo)
_git_fetch(cachedir, repo, ref, force=force)
- with open(os.path.join(cachedir, 'refs', 'heads', ref), encoding='utf-8') as rev_file:
+ rev_path = (['tags', ref.removeprefix('tag ')]
+ if ref.startswith('tag ')
+ else ['heads', ref])
+ with open(os.path.join(cachedir, 'refs', *rev_path), encoding='utf-8') as rev_file:
rev = Rev(rev_file.read(999).strip())
verify_ancestry(repo, ref, rev, force=force)
_log_fetch(repo, ref, rev, force=force)