using System;
using System.Diagnostics;
using System.IO;
using System.Threading.Tasks;
using SingularityGroup.HotReload.Editor.Localization;
using SingularityGroup.HotReload.Editor.Semver;
using Debug = UnityEngine.Debug;

namespace SingularityGroup.HotReload.Editor.Cli {
    class OsxCliController : ICliController {
        Process process;

        public string BinaryFileName => "HotReload.app.zip";
        public string PlatformName => "osx-x64";
        public bool CanOpenInBackground => true;

        /// In MacOS 13 Ventura, our app cannot launch a terminal window.
        /// We use a custom app that launches HotReload server and shows it's output (just like a terminal would). 
        //  Including MacOS 12 Monterey as well so I can dogfood it -Troy
        private static bool UseCustomConsoleApp() => MacOSVersion.Value.Major >= 12;

        // dont use static because null comparison on SemVersion is broken
        private static readonly Lazy<SemVersion> MacOSVersion = new Lazy<SemVersion>(() => {
            //UnityHelper.OperatingSystem; // in Unity 2018 it returns 10.16 on monterey (no idea why)
            //Environment.OSVersion returns unix version like 21.x
            var startinfo = new ProcessStartInfo {
                FileName = "/usr/bin/sw_vers",
                Arguments = "-productVersion",
                UseShellExecute = false,
                RedirectStandardOutput = true,
                CreateNoWindow = true,
            };
            var process = Process.Start(startinfo);

            string osVersion = process.StandardOutput.ReadToEnd().Trim();

            SemVersion macosVersion;
            if (SemVersion.TryParse(osVersion, out macosVersion)) {
                return macosVersion;
            }
            // should never happen
            Log.Warning(Translations.Errors.WarningMacOSVersionDetectionFailed);
            return SemVersion.None;
        });

        public async Task Start(StartArgs args) {
            // Unzip the .app.zip to temp folder .app
            var appExecutablePath = $"{args.executableTargetDir}/HotReload.app/Contents/MacOS/HotReload";
            var cliExecutablePath = $"{args.executableTargetDir}/HotReload.app/Contents/Resources/CodePatcherCLI";
            
            // ensure running on threadpool
            await ThreadUtility.SwitchToThreadPool();

            // executableTargetDir is versioned, so only need to extract once.
            if (!File.Exists(appExecutablePath)) {
                try {
                    // delete only the extracted app folder (must not delete downloaded zip which is in same folder)
                    Directory.Delete(args.executableTargetDir + "/HotReload.app", true);
                } catch (IOException) {
                    // ignore directory not found
                }
                Directory.CreateDirectory(args.executableTargetDir);
                UnzipMacOsPackage($"{args.executableTargetDir}/{BinaryFileName}", args.executableTargetDir + "/");
            }

            try {
                // Always stop first because rarely it has happened that the server process was still running after custom console closed.
                // Note: this will also stop Hot Reload started by other Unity projects.
                await Stop();
            } catch {
                // ignored
            }

            if (args.createNoWindow) {
                await StartExecutableOrApp(args, cliExecutablePath);
            } else if (UseCustomConsoleApp()) {
                await StartExecutableOrApp(args, appExecutablePath);
            } else {
                await StartTerminal(args, cliExecutablePath);
            }
        }

        public Task StartExecutableOrApp(StartArgs args, string executablePath) {
            process = Process.Start(new ProcessStartInfo {
                // Path to the HotReload.app
                FileName = executablePath,
                Arguments = args.cliArguments,
                UseShellExecute = false,
            });
            
            var pidFilePath = CliUtils.GetPidFilePath(args.hotreloadTempDir);
            // ReSharper disable once PossibleNullReferenceException
            File.WriteAllText(pidFilePath, process.Id.ToString());
            return Task.CompletedTask;
        }

        public Task StartTerminal(StartArgs args, string executablePath) {
            var pidFilePath = CliUtils.GetPidFilePath(args.hotreloadTempDir);
            // To run in a Terminal window (so you can see compiler logs), we must put the arguments into a script file
            // and run the script in Terminal. Terminal.app does not forward the arguments passed to it via `open --args`.
            // *.command files are opened with the user's default terminal app.
            var executableScriptPath = Path.Combine(Path.GetTempPath(), "Start_HotReloadServer.command");
            // You don't need to copy the cli executable on mac
            // omit hashbang line, let shell use the default interpreter (easier than detecting your default shell beforehand)
            File.WriteAllText(executableScriptPath, $"echo $$ > \"{pidFilePath}\"" +
                                                    $"\ncd \"{Environment.CurrentDirectory}\"" + // set cwd because 'open' launches script with $HOME as cwd.
                                                    $"\n\"{executablePath}\" {args.cliArguments} || read");

            CliUtils.Chmod(executableScriptPath); // make it executable
            CliUtils.Chmod(executablePath); // make it executable

            Directory.CreateDirectory(args.hotreloadTempDir);
            Directory.CreateDirectory(args.executableTargetDir);
            Directory.CreateDirectory(args.cliTempDir);
            
            process = Process.Start(new ProcessStartInfo {
                FileName = "open",
                Arguments = $"{(args.createNoWindow ? "-gj" : "")} '{executableScriptPath}'",
                UseShellExecute = true,
            });

            if (process.WaitForExit(1000)) {
                if (process.ExitCode != 0) {
                    Log.Warning(Translations.Errors.WarningFailedToRunServerCommand, process.ExitCode, executableScriptPath);
                }
            }
            else {
                process.EnableRaisingEvents = true;
                process.Exited += (_, __) => {
                    if (process.ExitCode != 0) {
                        Log.Warning(Translations.Errors.WarningFailedToRunServerCommand, process.ExitCode, executableScriptPath);
                    }
                };
            }
            return Task.CompletedTask;
        }

        public async Task Stop() {
            // kill HotReload server process (on mac it has different pid to the window which started it)
            await RequestHelper.KillServer();

            // process.CloseMainWindow throws if proc already exited.
            // We rely on the pid file for killing the trampoline script (in-case script is just starting and HotReload server not running yet)
            process = null;
            CliUtils.KillLastKnownHotReloadProcess();
        }

        static void UnzipMacOsPackage(string zipPath, string unzippedFolderPath) {
            //Log.Info("UnzipMacOsPackage called with {0}\n workingDirectory = {1}", zipPath, unzippedFolderPath);
            if (!zipPath.EndsWith(".zip")) {
                throw new ArgumentException(string.Format(Translations.Errors.ExceptionExpectedZipFile, zipPath), nameof(zipPath));
            }

            if (!File.Exists(zipPath)) {
                throw new ArgumentException(string.Format(Translations.Errors.ExceptionZipFileNotFound, zipPath), nameof(zipPath));
            }
            var processStartInfo = new ProcessStartInfo {
                FileName = "unzip",
                Arguments = $"-o \"{zipPath}\"",
                WorkingDirectory = unzippedFolderPath, // unzip extracts to working directory by default
                UseShellExecute = true,
                CreateNoWindow = true
            };

            Process process = Process.Start(processStartInfo);
            process.WaitForExit();
            if (process.ExitCode != 0) {
                throw new Exception(string.Format(Translations.Errors.ExceptionUnzipFailed, process.ExitCode));
            }
            //Log.Info($"did unzip to {unzippedFolderPath}");
            // Move the .app folder to unzippedFolderPath
            
            // find the .app directory which is now inside unzippedFolderPath directory
            var foundDirs = Directory.GetDirectories(unzippedFolderPath, "*.app", SearchOption.AllDirectories);
            var done = false;
            var destDir = unzippedFolderPath + "HotReload.app";
            foreach (var dir in foundDirs) {
                if (dir.EndsWith(".app")) {
                    done = true;
                    if (dir == destDir) {
                        // already in the right place
                        break;
                    }
                    Directory.Move(dir, destDir);
                    //Log.Info("Moved to " + destDir);
                    break;
                }
            }

            if (!done) {
                throw new Exception(string.Format(Translations.Errors.ExceptionFailedToFindAppDirectory, destDir));
            }
            //Log.Info($"did unzip to {unzippedFolderPath}");
        }
    }
}