"""Manipulating git repos natively, without much knowledge of mass-driver models"""
import logging
from pathlib import Path
from tempfile import mkdtemp
from git import Repo as GitRepo
from mass_driver.models.migration import MigrationLoaded
DEFAULT_CACHE = Path(".mass_driver/repos/")
[docs]
def clone_if_remote(
    repo_path: str, cache_folder: Path, logger: logging.Logger
) -> GitRepo:
    """Build a GitRepo; If repo_path isn't a directory, clone it"""
    if Path(repo_path).is_dir():
        logger.info("Given an existing (local) repo: no cloning")
        # Clone it into cache anyway
        return GitRepo(path=repo_path)  # TODO: Actually clone-move the repo on the way.
    # SSH clone URL e.g: git@github.com:OverkillGuy/python-template
    if ":" in repo_path:  # Presence of : is proxy for SSH clone URL
        *_junk, repo_blurb = repo_path.split(":")
        org, repo_name = repo_blurb.split("/")
    else:
        org = "local"
        repo_name = Path(repo_path).name
    clone_target = cache_folder / org / repo_name
    if clone_target.is_dir():
        logger.info("Given a URL for we cloned already: no cloning")
        return GitRepo(clone_target)
    logger.info("Given a URL, cache miss: cloning")
    cloned = GitRepo.clone_from(
        url=repo_path,
        to_path=clone_target,
        multi_options=["--depth=1"],
    )
    return cloned 
[docs]
def get_cache_folder(cache: bool, logger: logging.Logger) -> Path:
    """Create a cache folder, either locally or in temp"""
    cache_folder = DEFAULT_CACHE
    if not cache:
        cache_folder = Path(mkdtemp(suffix=".cache"))
        logger.info(
            f"Using repo cache folder: {cache_folder}/ (Won't wipe it on exit!)"
        )
    return cache_folder 
[docs]
def commit(repo: GitRepo, migration: MigrationLoaded):
    """Commit the repo's changes in branch_name, given the PatchDriver that did it"""
    assert repo.is_dirty(
        untracked_files=True
    ), "GitRepo shouldn't be clean on committing"
    branch = repo.create_head(migration.branch_name)
    branch.checkout()
    repo.git.add(A=True)
    author = None  # If stays None, git uses default commit author
    if migration.commit_author_email or migration.commit_author_name:
        name, email = migration.commit_author_name, migration.commit_author_email
        author = f"{name} <{email}>"  # Actor(name=migration.commit_author_name,
        #       email=migration.commit_author_email)
    repo.git.commit(m=migration.commit_message, author=author) 
[docs]
def push(repo: GitRepo, branch_name: str):
    """Push a branch of the repo to a remote"""
    remote = repo.remote()
    remote.push(refspec=branch_name) 
[docs]
def switch_branch_then_pull(repo: GitRepo, pull: bool, branch_name: str | None = None):
    """Switch branch then pull"""
    if branch_name is not None:
        repo.git.checkout(branch_name)
    if pull:
        repo.remote().pull() 
[docs]
def get_default_branch(r: GitRepo) -> str:
    """Get the default branch of a repository"""
    # From https://github.com/gitpython-developers/GitPython/discussions/1364#discussioncomment-1530384
    try:
        return r.remotes.origin.refs.HEAD.ref.remote_head
    except Exception:
        raise ValueError(
            "base_branch param could not be autodetected: no git remote available"
        )