// SPDX-License-Identifier: Apache-2.0

import {inject, injectable} from 'tsyringe-neo';
import {ShellRunner} from './shell-runner.js';
import {InjectTokens} from './dependency-injection/inject-tokens.js';
import {BrewPackageManager} from './package-managers/brew-package-manager.js';
import {OsPackageManager} from './package-managers/os-package-manager.js';
import {patchInject} from './dependency-injection/container-helper.js';
import {PodmanMode, SoloListrTask, type SoloListrTaskWrapper} from '../types/index.js';
import {InitContext} from '../commands/init/init-context.js';
import {AptGetPackageManager} from './package-managers/apt-get-package-manager.js';
import {SoloError} from './errors/solo-error.js';
import * as constants from './constants.js';
import {getTemporaryDirectory} from './helpers.js';
import fs from 'node:fs';
import * as yaml from 'yaml';
import {type AnyObject} from '../types/aliases.js';
import path from 'node:path';
import {KindClient} from '../integration/kind/kind-client.js';
import {ClusterCreateResponse} from '../integration/kind/model/create-cluster/cluster-create-response.js';
import {type ClusterCreateOptions} from '../integration/kind/model/create-cluster/cluster-create-options.js';
import {ClusterCreateOptionsBuilder} from '../integration/kind/model/create-cluster/create-cluster-options-builder.js';
import {type DefaultKindClientBuilder} from '../integration/kind/impl/default-kind-client-builder.js';
import {type DependencyManager, KindDependencyManager, PodmanDependencyManager} from './dependency-managers/index.js';
import {K8} from '../integration/kube/k8.js';
import {MissingActiveContextError} from '../integration/kube/errors/missing-active-context-error.js';
import {MissingActiveClusterError} from '../integration/kube/errors/missing-active-cluster-error.js';
import {type K8Factory} from '../integration/kube/k8-factory.js';
import {type GitClient} from '../integration/git/git-client.js';
import {ImageCacheHandler} from '../integration/cache/impl/image-cache-handler.js';
import {KindNodeImageTargetProvider} from '../integration/cache/target-providers/kind-image-target-provider.js';
import {ImageCacheHandlerBuilder} from '../integration/cache/impl/image-cache-handler-builder.js';
import {type ContainerEngineClient} from '../integration/container-engine/container-engine-client.js';

@injectable()
export class ClusterTaskManager extends ShellRunner {
  public constructor(
    @inject(InjectTokens.BrewPackageManager) protected readonly brewPackageManager: BrewPackageManager,
    @inject(InjectTokens.OsPackageManager) protected readonly osPackageManager: OsPackageManager,
    @inject(InjectTokens.KindBuilder) protected readonly kindBuilder: DefaultKindClientBuilder,
    @inject(InjectTokens.PodmanDependencyManager) protected readonly podmanDependencyManager: PodmanDependencyManager,
    @inject(InjectTokens.KindDependencyManager) protected readonly kindDependencyManager: KindDependencyManager,
    @inject(InjectTokens.PodmanInstallationDirectory) protected readonly podmanInstallationDirectory: string,
    @inject(InjectTokens.K8Factory) protected readonly k8Factory: K8Factory,
    @inject(InjectTokens.DependencyManager) protected readonly depManager: DependencyManager,
    @inject(InjectTokens.KindInstallationDirectory) protected readonly kindInstallationDirectory: string,
    @inject(InjectTokens.GitClient) protected readonly gitClient: GitClient,
    @inject(InjectTokens.ContainerEngineClient) protected readonly containerEngineClient: ContainerEngineClient,
  ) {
    super();

    this.brewPackageManager = patchInject(brewPackageManager, InjectTokens.BrewPackageManager, ClusterTaskManager.name);
    this.osPackageManager = patchInject(osPackageManager, InjectTokens.OsPackageManager, ClusterTaskManager.name);
    this.kindBuilder = patchInject(kindBuilder, InjectTokens.KindBuilder, ClusterTaskManager.name);
    this.podmanDependencyManager = patchInject(
      podmanDependencyManager,
      InjectTokens.KindBuilder,
      ClusterTaskManager.name,
    );
    this.kindDependencyManager = patchInject(kindDependencyManager, InjectTokens.KindBuilder, ClusterTaskManager.name);
    this.podmanInstallationDirectory = patchInject(
      podmanInstallationDirectory,
      InjectTokens.PodmanInstallationDirectory,
      ClusterTaskManager.name,
    );
    this.k8Factory = patchInject(k8Factory, InjectTokens.K8Factory, ClusterTaskManager.name);
    this.depManager = patchInject(depManager, InjectTokens.DependencyManager, ClusterTaskManager.name);
    this.kindInstallationDirectory = patchInject(
      kindInstallationDirectory,
      InjectTokens.KindInstallationDirectory,
      ClusterTaskManager.name,
    );
    this.gitClient = patchInject(gitClient, InjectTokens.GitClient, ClusterTaskManager.name);
    this.containerEngineClient = patchInject(
      containerEngineClient,
      InjectTokens.ContainerEngineClient,
      ClusterTaskManager.name,
    );
  }

  private sudoCallbacks(task: SoloListrTaskWrapper<InitContext>): {
    onSudoRequested: (message: string) => void;
    onSudoGranted: (message: string) => void;
  } {
    const originalTitle: string = task.title;
    const onSudoRequested: (message: string) => void = (message: string): void => {
      task.title = message;
    };
    const onSudoGranted: (message: string) => void = (message: string): void => {
      void message;
      task.title = originalTitle;
    };
    return {onSudoGranted, onSudoRequested};
  }

  public rootfullInstallTasks(parentTask: SoloListrTaskWrapper<InitContext>): SoloListrTask<InitContext>[] {
    return [
      {
        title: 'Install git, iptables...',
        task: async (): Promise<void> => {
          try {
            await this.gitClient.version();
          } catch {
            this.logger.info('Git not found, installing git...');
            const {onSudoGranted, onSudoRequested} = this.sudoCallbacks(parentTask);
            const osPackageManager: AptGetPackageManager =
              this.osPackageManager.getPackageManager() as AptGetPackageManager;
            osPackageManager.setOnSudoGranted(onSudoGranted);
            osPackageManager.setOnSudoRequested(onSudoRequested);
            await osPackageManager.update();
            await osPackageManager.installPackages(['git', 'iptables']);
          }
        },
      },
      {
        title: 'Install brew...',
        task: async (): Promise<void> => {
          const brewInstalled: boolean = await this.brewPackageManager.isAvailable();
          if (!brewInstalled) {
            this.logger.info('Homebrew not found, installing Homebrew...');
            if (!(await this.brewPackageManager.install())) {
              throw new SoloError('Failed to install Homebrew');
            }
          }
        },
      },
      {
        title: 'Install podman...',
        task: async (): Promise<void> => {
          try {
            const podmanVersion: string[] = await this.run('podman --version');
            this.logger.info(`Podman already installed: ${podmanVersion}`);
          } catch {
            this.logger.info('Podman not found, installing Podman...');
            await this.brewPackageManager.installPackages(['podman']);
            const brewBin: string[] = await this.run('which podman');
            process.env.PATH = `${process.env.PATH}:${brewBin.join('').replace('/podman', '')}`;
          }
        },
      } as SoloListrTask<InitContext>,
      {
        title: 'Creating local cluster...',
        task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<void> => {
          void _context;
          const whichPodman: string[] = await this.run('which podman');
          const podmanPath: string = whichPodman.join('').replace('/podman', '');
          const sudoRunOptions: [string[], boolean?, boolean?, Record<string, string>?] = [
            [],
            undefined,
            undefined,
            {
              PATH:
                `${this.podmanInstallationDirectory}${path.delimiter}` +
                `${this.kindInstallationDirectory}${path.delimiter}${process.env.PATH}`,
            },
          ];
          const {onSudoGranted, onSudoRequested} = this.sudoCallbacks(task);
          await this.sudoRun(
            onSudoRequested,
            onSudoGranted,
            `KIND_EXPERIMENTAL_PROVIDER=podman PATH="$PATH:${podmanPath}" kind create cluster --image "${constants.KIND_NODE_IMAGE}" --config "${constants.KIND_CLUSTER_CONFIG_FILE}"`,
            ...sudoRunOptions,
          );

          // Merge kubeconfig data from root user into normal user's kubeconfig
          const user: string[] = await this.run('whoami');
          const temporaryDirectory: string = getTemporaryDirectory();

          await this.sudoRun(
            onSudoRequested,
            onSudoGranted,
            `cp /root/.kube/config ${temporaryDirectory}/kube-config-root`,
            ...sudoRunOptions,
          );
          await this.sudoRun(
            onSudoRequested,
            onSudoGranted,
            `chown ${user} ${temporaryDirectory}/kube-config-root`,
            ...sudoRunOptions,
          );
          await this.sudoRun(
            onSudoRequested,
            onSudoGranted,
            `chmod 755 ${temporaryDirectory}/kube-config-root`,
            ...sudoRunOptions,
          );

          const rootYamlData: string = fs.readFileSync(`${temporaryDirectory}/kube-config-root`, 'utf8');
          const rootConfig: Record<string, AnyObject> = yaml.parse(rootYamlData) as Record<string, AnyObject>;

          let userConfig: Record<string, AnyObject>;
          const clusterName: string = 'kind-kind';

          try {
            const userYamlData: string = fs.readFileSync(`/home/${user}/.kube/config`, 'utf8');
            userConfig = yaml.parse(userYamlData) as Record<string, AnyObject>;

            if (!userConfig.clusters) {
              userConfig.clusters = [];
            }
            userConfig.clusters.push(rootConfig.clusters.find((c: AnyObject): boolean => c.name === clusterName));

            if (!userConfig.contexts) {
              userConfig.contexts = [];
            }
            userConfig.contexts.push(rootConfig.contexts.find((c: AnyObject): boolean => c.name === clusterName));

            if (!userConfig.users) {
              userConfig.users = [];
            }
            userConfig.users.push(rootConfig.users.find((c: AnyObject): boolean => c.name === clusterName));

            userConfig['current-context'] = rootConfig['current-context'];
          } catch (error) {
            if (error.code === 'ENOENT') {
              const kubeConfigDirectory: string = `/home/${user}/.kube/`;
              if (!fs.existsSync(kubeConfigDirectory)) {
                fs.mkdirSync(kubeConfigDirectory, {recursive: true});
              }
              userConfig = rootConfig;
              userConfig.clusters = userConfig.clusters.filter((c: AnyObject): boolean => c.name === clusterName);
              userConfig.contexts = userConfig.contexts.filter((c: AnyObject): boolean => c.name === clusterName);
              userConfig.users = userConfig.users.filter((c: AnyObject): boolean => c.name === clusterName);
            } else {
              throw error;
            }
          }

          fs.writeFileSync(`/home/${user}/.kube/config`, yaml.stringify(userConfig), 'utf8');
          fs.rmSync(`${temporaryDirectory}/kube-config-root`);
        },
      } as SoloListrTask<InitContext>,
    ];
  }

  public async installationTasks(parentTask: SoloListrTaskWrapper<InitContext>): Promise<SoloListrTask<InitContext>[]> {
    const skipPodmanTasks: boolean = !(await this.podmanDependencyManager.shouldInstall());
    if (this.podmanDependencyManager.mode === PodmanMode.ROOTFUL) {
      {
        return skipPodmanTasks ? [this.defaultCreateClusterTask(parentTask)] : this.rootfullInstallTasks(parentTask);
      }
    } else if (this.podmanDependencyManager.mode === PodmanMode.VIRTUAL_MACHINE) {
      {
        return [
          {
            title: 'Create Podman machine...',
            task: async (): Promise<void> => {
              const podmanRunOptions: [string[], boolean?, boolean?, Record<string, string>?] = [
                [],
                undefined,
                undefined,
                {
                  PATH: `${this.podmanInstallationDirectory}${path.delimiter}${process.env.PATH}`,
                },
              ];
              await this.podmanDependencyManager.setupConfig();
              const podmanExecutable: string = await this.podmanDependencyManager.getExecutable();
              try {
                await this.run(
                  `${podmanExecutable} machine inspect ${constants.PODMAN_MACHINE_NAME}`,
                  ...podmanRunOptions,
                );
              } catch (error) {
                if (error.message.includes('VM does not exist')) {
                  await this.run(
                    `${podmanExecutable} machine init ${constants.PODMAN_MACHINE_NAME} --memory=16384`, // 16GB
                    ...podmanRunOptions,
                  );
                  await this.run(
                    `${podmanExecutable} machine start ${constants.PODMAN_MACHINE_NAME}`,
                    ...podmanRunOptions,
                  );
                } else {
                  throw new SoloError(`Failed to inspect Podman machine: ${error.message}`);
                }
              }
            },
            skip: (): boolean => skipPodmanTasks,
          } as SoloListrTask<InitContext>,
          {
            title: 'Configure kind to use podman...',
            task: async (): Promise<void> => {
              process.env.KIND_EXPERIMENTAL_PROVIDER = 'podman';
            },
            skip: (): boolean => skipPodmanTasks,
          } as SoloListrTask<InitContext>,
          this.defaultCreateClusterTask(parentTask),
        ];
      }
    }

    return [];
  }

  private defaultCreateClusterTask(parentTask: SoloListrTaskWrapper<InitContext>): SoloListrTask<InitContext> {
    return {
      title: 'Creating local cluster...',
      task: async (): Promise<void> => {
        const kindExecutable: string = await this.kindDependencyManager.getExecutable();
        const kindClient: KindClient = await this.kindBuilder.executable(kindExecutable).build();

        if (constants.CONFIG.ENABLE_IMAGE_CACHE) {
          const kindImageCacheHandler: ImageCacheHandler = new ImageCacheHandlerBuilder()
            .provider(new KindNodeImageTargetProvider())
            .engine(this.containerEngineClient)
            .build();

          await kindImageCacheHandler.pullKindNodeImageIfMissing();
          await kindImageCacheHandler.loadKindNodeImageIntoEngine();
        }

        const clusterCreateOptions: ClusterCreateOptions = ClusterCreateOptionsBuilder.builder()
          .image(constants.KIND_NODE_IMAGE)
          .config(constants.KIND_CLUSTER_CONFIG_FILE)
          .build();

        const clusterResponse: ClusterCreateResponse = await kindClient.createCluster(
          constants.DEFAULT_CLUSTER,
          clusterCreateOptions,
        );

        parentTask.title = `Created local cluster '${clusterResponse.name}'; connect with context '${clusterResponse.context}'`;
      },
    } as SoloListrTask<InitContext>;
  }

  public setupLocalClusterTasks(): SoloListrTask<InitContext>[] {
    return [
      {
        title: 'Install Kind',
        task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<unknown> => {
          void _context;
          const podmanDependency: PodmanDependencyManager = this.podmanDependencyManager;
          const shouldInstallPodman: boolean = await podmanDependency.shouldInstall();

          const podmanDependencies: string[] =
            shouldInstallPodman && podmanDependency.mode === PodmanMode.VIRTUAL_MACHINE
              ? [constants.PODMAN, constants.VFKIT, constants.GVPROXY]
              : [];

          const deps: string[] = [...podmanDependencies, constants.KIND];

          const subTasks: SoloListrTask<InitContext>[] = this.depManager.taskCheckDependencies<InitContext>(deps);

          // set up the sub-tasks
          return task.newListr(subTasks, {
            concurrent: true,
            rendererOptions: {
              collapseSubtasks: false,
            },
          });
        },
        skip: this.skipKindSetup.bind(this),
      },
      {
        title: 'Create default cluster',
        task: async (_context: InitContext, task: SoloListrTaskWrapper<InitContext>): Promise<unknown> => {
          void _context;
          const subTasks: SoloListrTask<InitContext>[] = await this.installationTasks(task);
          return task.newListr(subTasks, {
            concurrent: false, // should not use concurrent as cluster creation may be called before dependencies are finished installing
            rendererOptions: {
              collapseSubtasks: false,
            },
          });
        },
        skip: this.skipKindSetup.bind(this),
      },
    ];
  }

  private async skipKindSetup(): Promise<boolean> {
    try {
      const k8: K8 = this.k8Factory.default();
      const contextName: string = k8.contexts().readCurrent();
      if (!contextName) {
        return false;
      }

      // Try to verify the cluster is actually accessible by making a simple API call
      try {
        await k8.namespaces().list();
        return true;
      } catch {
        // If we can't connect to the cluster, don't skip cluster creation
        // This handles cases where contexts exist but clusters are not running
        return false;
      }
    } catch (error) {
      return !(error instanceof MissingActiveContextError || error instanceof MissingActiveClusterError);
    }
  }
}
