import json
import os.path
from pathlib import Path
import re
import subprocess

import colorama

root_path = os.path.abspath(os.path.normpath(os.path.join(__file__, "..", "..")))
basepath = os.path.join(root_path, "esphome")
temp_folder = os.path.join(root_path, ".temp")
temp_header_file = os.path.join(temp_folder, "all-include.cpp")


def styled(color, msg, reset=True):
    prefix = "".join(color) if isinstance(color, tuple) else color
    suffix = colorama.Style.RESET_ALL if reset else ""
    return prefix + msg + suffix


def print_error_for_file(file, body):
    print(
        styled(colorama.Fore.GREEN, "### File ")
        + styled((colorama.Fore.GREEN, colorama.Style.BRIGHT), file)
    )
    print()
    if body is not None:
        print(body)
        print()


def build_all_include():
    # Build a cpp file that includes all header files in this repo.
    # Otherwise header-only integrations would not be tested by clang-tidy
    headers = []
    for path in walk_files(basepath):
        filetypes = (".h",)
        ext = os.path.splitext(path)[1]
        if ext in filetypes:
            path = os.path.relpath(path, root_path)
            include_p = path.replace(os.path.sep, "/")
            headers.append(f'#include "{include_p}"')
    headers.sort()
    headers.append("")
    content = "\n".join(headers)
    p = Path(temp_header_file)
    p.parent.mkdir(exist_ok=True)
    p.write_text(content, encoding="utf-8")


def walk_files(path):
    for root, _, files in os.walk(path):
        for name in files:
            yield os.path.join(root, name)


def get_output(*args):
    with subprocess.Popen(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE) as proc:
        output, _ = proc.communicate()
    return output.decode("utf-8")


def get_err(*args):
    with subprocess.Popen(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE) as proc:
        _, err = proc.communicate()
    return err.decode("utf-8")


def splitlines_no_ends(string):
    return [s.strip() for s in string.splitlines()]


def changed_files(branch="dev"):
    check_remotes = ["upstream", "origin"]
    check_remotes.extend(splitlines_no_ends(get_output("git", "remote")))
    for remote in check_remotes:
        command = ["git", "merge-base", f"refs/remotes/{remote}/{branch}", "HEAD"]
        try:
            merge_base = splitlines_no_ends(get_output(*command))[0]
            break
        # pylint: disable=bare-except
        except:  # noqa: E722
            pass
    else:
        raise ValueError("Git not configured")
    command = ["git", "diff", merge_base, "--name-only"]
    changed = splitlines_no_ends(get_output(*command))
    changed = [os.path.relpath(f, os.getcwd()) for f in changed]
    changed.sort()
    return changed


def filter_changed(files):
    changed = changed_files()
    files = [f for f in files if f in changed]
    print("Changed files:")
    if not files:
        print("    No changed files!")
    for c in files:
        print(f"    {c}")
    return files


def filter_grep(files, value):
    matched = []
    for file in files:
        with open(file, encoding="utf-8") as handle:
            contents = handle.read()
        if value in contents:
            matched.append(file)
    return matched


def git_ls_files(patterns=None):
    command = ["git", "ls-files", "-s"]
    if patterns is not None:
        command.extend(patterns)
    with subprocess.Popen(command, stdout=subprocess.PIPE) as proc:
        output, _ = proc.communicate()
    lines = [x.split() for x in output.decode("utf-8").splitlines()]
    return {s[3].strip(): int(s[0]) for s in lines}


def load_idedata(environment):
    platformio_ini = Path(root_path) / "platformio.ini"
    temp_idedata = Path(temp_folder) / f"idedata-{environment}.json"
    changed = False
    if not platformio_ini.is_file() or not temp_idedata.is_file():
        changed = True
    elif platformio_ini.stat().st_mtime >= temp_idedata.stat().st_mtime:
        changed = True

    if "idf" in environment:
        # remove full sdkconfig when the defaults have changed so that it is regenerated
        default_sdkconfig = Path(root_path) / "sdkconfig.defaults"
        temp_sdkconfig = Path(temp_folder) / f"sdkconfig-{environment}"

        if not temp_sdkconfig.is_file():
            changed = True
        elif default_sdkconfig.stat().st_mtime >= temp_sdkconfig.stat().st_mtime:
            temp_sdkconfig.unlink()
            changed = True

    if not changed:
        return json.loads(temp_idedata.read_text())

    # ensure temp directory exists before running pio, as it writes sdkconfig to it
    Path(temp_folder).mkdir(exist_ok=True)

    if "nrf" in environment:
        build_environment = environment.replace("-tidy", "")
        build_dir = Path(temp_folder) / f"build-{build_environment}"
        Path(build_dir).mkdir(exist_ok=True)
        Path(build_dir / "platformio.ini").write_text(
            Path(platformio_ini).read_text(encoding="utf-8"), encoding="utf-8"
        )
        esphome_dir = Path(build_dir / "esphome")
        esphome_dir.mkdir(exist_ok=True)
        Path(esphome_dir / "main.cpp").write_text(
            """
#include <zephyr/kernel.h>
int main() { return 0;}
""",
            encoding="utf-8",
        )
        zephyr_dir = Path(build_dir / "zephyr")
        zephyr_dir.mkdir(exist_ok=True)
        Path(zephyr_dir / "prj.conf").write_text(
            """
CONFIG_NEWLIB_LIBC=y
""",
            encoding="utf-8",
        )
        subprocess.run(
            ["pio", "run", "-e", build_environment, "-d", build_dir], check=True
        )

        def extract_include_paths(command):
            include_paths = []
            include_pattern = re.compile(
                r'("-I\s*[^"]+)|(-isystem\s*[^\s]+)|(-I\s*[^\s]+)'
            )
            for match in include_pattern.findall(command):
                split_strings = re.split(
                    r"\s*-\s*(?:I|isystem)", list(filter(lambda x: x, match))[0]
                )
                include_paths.append(split_strings[1])
            return include_paths

        def extract_defines(command):
            defines = []
            define_pattern = re.compile(r"-D\s*([^\s]+)")
            for match in define_pattern.findall(command):
                if match not in ("_ASMLANGUAGE"):
                    defines.append(match)
            return defines

        def find_cxx_path(commands):
            for entry in commands:
                command = entry["command"]
                cxx_path = command.split()[0]
                if not cxx_path.endswith("++"):
                    continue
                return cxx_path

        def get_builtin_include_paths(compiler):
            result = subprocess.run(
                [compiler, "-E", "-x", "c++", "-", "-v"],
                input="",
                text=True,
                stderr=subprocess.PIPE,
                stdout=subprocess.DEVNULL,
                check=True,
            )
            include_paths = []
            start_collecting = False
            for line in result.stderr.splitlines():
                if start_collecting:
                    if line.startswith(" "):
                        include_paths.append(line.strip())
                    else:
                        break
                if "#include <...> search starts here:" in line:
                    start_collecting = True
            return include_paths

        def extract_cxx_flags(command):
            flags = []
            # Extracts CXXFLAGS from the command string, excluding includes and defines.
            flag_pattern = re.compile(
                r"(-O[0-3s]|-g|-std=[^\s]+|-Wall|-Wextra|-Werror|--[^\s]+|-f[^\s]+|-m[^\s]+|-imacros\s*[^\s]+)"
            )
            for match in flag_pattern.findall(command):
                flags.append(match.replace("-imacros ", "-imacros"))
            return flags

        def transform_to_idedata_format(compile_commands):
            cxx_path = find_cxx_path(compile_commands)
            idedata = {
                "includes": {
                    "toolchain": get_builtin_include_paths(cxx_path),
                    "build": set(),
                },
                "defines": set(),
                "cxx_path": cxx_path,
                "cxx_flags": set(),
            }

            for entry in compile_commands:
                command = entry["command"]
                exec = command.split()[0]
                if exec != cxx_path:
                    continue

                idedata["includes"]["build"].update(extract_include_paths(command))
                idedata["defines"].update(extract_defines(command))
                idedata["cxx_flags"].update(extract_cxx_flags(command))

            idedata["defines"].update(
                [
                    "pthread_attr_t=pthread_attr",
                    "pthread_mutexattr_t=pthread_mutexattr",
                    "pthread_condattr_t=pthread_condattr",
                ]
            )

            # Convert sets to lists for JSON serialization
            idedata["includes"]["build"] = list(idedata["includes"]["build"])
            idedata["defines"] = list(idedata["defines"])
            idedata["cxx_flags"] = list(idedata["cxx_flags"])

            return idedata

        compile_commands = json.loads(
            Path(
                build_dir
                / ".pio"
                / "build"
                / build_environment
                / "compile_commands.json"
            ).read_text(encoding="utf-8")
        )
        data = transform_to_idedata_format(compile_commands)
    else:
        stdout = subprocess.check_output(
            ["pio", "run", "-t", "idedata", "-e", environment]
        )
        match = re.search(r'{\s*".*}', stdout.decode("utf-8"))
        data = json.loads(match.group())
    temp_idedata.write_text(json.dumps(data, indent=2) + "\n")
    return data


def get_binary(name: str, version: str) -> str:
    binary_file = f"{name}-{version}"
    try:
        result = subprocess.check_output([binary_file, "-version"])
        return binary_file
    except FileNotFoundError:
        pass
    binary_file = name
    try:
        result = subprocess.run(
            [binary_file, "-version"], text=True, capture_output=True, check=False
        )
        if result.returncode == 0 and (f"version {version}") in result.stdout:
            return binary_file
        raise FileNotFoundError(f"{name} not found")

    except FileNotFoundError:
        print(
            f"""
            Oops. It looks like {name} is not installed. It should be available under venv/bin
            and in PATH after running in turn:
              script/setup
              source venv/bin/activate.

            Please confirm you can run "{name} -version" or "{name}-{version} -version"
            in your terminal and install
            {name} (v{version}) if necessary.

            Note you can also upload your code as a pull request on GitHub and see the CI check
            output to apply {name}
            """
        )
        raise