from pathlib import Path
import shutil
import subprocess
import tempfile
from typing import List, Tuple
from logging import Logger

def convertPatchToGitDiff(repoPath: Path, changes: List[Tuple[str, str, int, str]], langs:str, logger: Logger) -> Tuple[str, str]:
    """
    Args:
        repoPath: Path to git repository
        changes: List of tuples (filePath, symbolName, startLine, newBody)
    Returns:
        Generated git patch as string, error if git diff fails
    """
    with tempfile.TemporaryDirectory() as backupDir:
        backupFiles: List[Tuple[Path, Path]] = []
        for filePath, _, _, _ in changes:
            fullPath = (repoPath / filePath).resolve()
            backupPath = (Path(backupDir) / filePath).resolve()
            backupPath.parent.mkdir(parents=True, exist_ok=True)
            shutil.copy2(fullPath, backupPath)
            backupFiles.append((fullPath, backupPath))

        try:
            applyChanges(repoPath, changes, langs, logger)

            result = subprocess.run(
                ['git', 'diff'], 
                cwd=repoPath,
                capture_output=True,
                text=True,
                encoding='utf-8',
            )
            patch = result.stdout
            error = result.stderr
        except Exception as e:
            patch = ""
            error = str(e)
        finally:
            for origPath, backupPath in backupFiles:
                shutil.copy2(backupPath, origPath)

    return patch, error

def getFunctionBounds(filePath: str, functionName: str, langs:str) -> List[Tuple[int, int]]:
    """
    Get function start and end lines using ctags and readtags.
    Returns a list of all candidates.
    """
    with tempfile.NamedTemporaryFile(suffix='.tags') as temp_tags:
        subprocess.run(
            ['ctags', '--fields=+ne', f'--languages={langs}', 
             '-f', temp_tags.name, filePath],
        )

        result = subprocess.run(
            ['readtags', '-t', temp_tags.name, '-F', '(list $line "," $end #t)', 
             '-n', functionName],
            capture_output=True,
            text=True
        )

        if result.stdout:
            candidates = []
            for line in result.stdout.splitlines():
                start, end = map(int, line.strip().split(','))
                candidates.append((start, end))
            if not candidates:
                raise ValueError(f"Function {functionName} not found")
            return candidates

        raise ValueError(f"Function {functionName} not found")

def applyChanges(repoPath: Path, changes: List[Tuple[str, str, int, str]], langs:str, logger: Logger) -> None:
    """
    Apply the specified changes to files in the repository.

    Args:
        repoPath: Path to git repository
        changes: List of tuples (filePath, symbolName, startLine, newBody)
    """
    # Group changes by file
    changesByFile = {}
    for change in changes:
        filePath = change[0]
        if filePath not in changesByFile:
            changesByFile[filePath] = []
        changesByFile[filePath].append(change)

    for filePath, fileChanges in changesByFile.items():
        fullPath = (repoPath / filePath).resolve()

        # Read file content
        with open(fullPath, 'r') as f:
            lines = f.readlines()

        # Resolve bounds for all changes and adjust startLine based on closest bounds
        resolvedChanges = []
        for filePath, functionName, startLine, newBody in fileChanges:
            try:
                candidates = getFunctionBounds(str(fullPath), functionName, langs)
                closest = min(candidates, key=lambda x: (abs(x[0] - startLine), x[0]))  # Stable minimum
                resolvedChanges.append((filePath, functionName, closest[0], closest[1], newBody))
            except Exception as e:
                logger.error(f"Failed to resolve bounds for {functionName} in {filePath}: {e}")
                logger.error("Adding the change at the end of the file")
                # Add it at the end of the file if we can't resolve bounds
                resolvedChanges.append((filePath, functionName, len(lines)+1, len(lines)+1, newBody))

        # Sort resolved changes by start line in descending order (stable sort)
        resolvedChanges.sort(key=lambda x: x[2], reverse=True)

        # Apply changes while tracking line offsets
        for _, functionName, start, end, newBody in resolvedChanges:
            # Update lines with the new function body
            lines = lines[:start - 1] + [newBody + "\n"] + lines[end:]

        # Write updated content back to the file
        with open(fullPath, 'w') as f:
            f.writelines(lines)
