import { Command, Option } from "@commander-js/extra-typings";
import chalk from "chalk";
import {
  ensureHasConvexDependency,
  logAndHandleAxiosError,
  deploymentClient,
  waitUntilCalled,
} from "./lib/utils.js";
import { version } from "./version.js";
import {
  logFailure,
  oneoffContext,
  Context,
  showSpinner,
  logFinishedStep,
  logError,
  stopSpinner,
  changeSpinner,
} from "../bundler/context.js";
import {
  fetchDeploymentCredentialsProvisionProd,
  deploymentSelectionFromOptions,
} from "./lib/api.js";
import { subscribe } from "./lib/run.js";
import { AxiosResponse } from "axios";
import { nodeFs } from "../bundler/fs.js";
import path from "path";
import { deploymentDashboardUrlPage } from "./dashboard.js";
import { actionDescription } from "./lib/command.js";

export const convexExport = new Command("export")
  .summary("Export data from your deployment to a ZIP file")
  .description(
    "Export data, and optionally file storage, from your Convex deployment to a ZIP file.\n" +
      "By default, this exports from your dev deployment.",
  )
  .requiredOption(
    "--path <zipFilePath>",
    "Exports data into a ZIP file at this path, which may be a directory or unoccupied .zip path",
  )
  .addOption(
    new Option(
      "--include-file-storage",
      "Includes stored files (https://dashboard.convex.dev/deployment/files) in a _storage folder within the ZIP file",
    ),
  )
  .addDeploymentSelectionOptions(actionDescription("Export data from"))
  .showHelpAfterError()
  .action(async (options) => {
    const ctx = oneoffContext;

    const deploymentSelection = deploymentSelectionFromOptions(options);

    const {
      adminKey,
      url: deploymentUrl,
      deploymentName,
    } = await fetchDeploymentCredentialsProvisionProd(ctx, deploymentSelection);

    const inputPath = options.path;
    const includeStorage = !!options.includeFileStorage;

    await ensureHasConvexDependency(ctx, "export");

    const deploymentNotice = options.prod
      ? ` in your ${chalk.bold("prod")} deployment`
      : "";
    showSpinner(ctx, `Creating snapshot export${deploymentNotice}`);

    const client = deploymentClient(deploymentUrl);
    const headers = {
      Authorization: `Convex ${adminKey}`,
      "Convex-Client": `npm-cli-${version}`,
    };
    try {
      await client.post(
        `/api/export/request/zip?includeStorage=${includeStorage}`,
        null,
        { headers },
      );
    } catch (e) {
      return await logAndHandleAxiosError(ctx, e);
    }

    const snapshotExportState = await waitForStableExportState(
      ctx,
      deploymentUrl,
      adminKey,
    );
    switch (snapshotExportState.state) {
      case "completed":
        stopSpinner(ctx);
        logFinishedStep(
          ctx,
          `Created snapshot export at timestamp ${snapshotExportState.start_ts}`,
        );
        logFinishedStep(
          ctx,
          `Export is available at ${await deploymentDashboardUrlPage(
            deploymentName ?? null,
            "/settings/snapshot-export",
          )}`,
        );
        break;
      case "requested":
      case "in_progress": {
        logFailure(ctx, `WARNING: Export is continuing to run on the server.`);
        return await ctx.crash(1);
      }
      default: {
        const _: never = snapshotExportState;
        logFailure(
          ctx,
          `unknown error: unexpected state ${snapshotExportState as any}`,
        );
        return await ctx.crash(1);
      }
    }

    showSpinner(ctx, `Downloading snapshot export to ${chalk.bold(inputPath)}`);
    const exportUrl = `/api/export/zip/${snapshotExportState.start_ts.toString()}?adminKey=${encodeURIComponent(
      adminKey,
    )}`;
    let response: AxiosResponse;
    try {
      response = await client.get(exportUrl, {
        headers,
        responseType: "stream",
      });
    } catch (e) {
      return await logAndHandleAxiosError(ctx, e);
    }

    let filePath;
    if (ctx.fs.exists(inputPath)) {
      const st = ctx.fs.stat(inputPath);
      if (st.isDirectory()) {
        const contentDisposition =
          response.headers["content-disposition"] ?? "";
        let filename = `snapshot_${snapshotExportState.start_ts.toString()}.zip`;
        if (contentDisposition.startsWith("attachment; filename=")) {
          filename = contentDisposition.slice("attachment; filename=".length);
        }
        filePath = path.join(inputPath, filename);
      } else {
        logFailure(ctx, `Error: Path ${chalk.bold(inputPath)} already exists.`);
        return await ctx.crash(1, "invalid filesystem data");
      }
    } else {
      filePath = inputPath;
    }
    changeSpinner(
      ctx,
      `Downloading snapshot export to ${chalk.bold(filePath)}`,
    );

    try {
      await nodeFs.writeFileStream(filePath, response.data);
    } catch (e) {
      logFailure(ctx, `Exporting data failed`);
      logError(ctx, chalk.red(e));
      return await ctx.crash(1);
    }
    stopSpinner(ctx);
    logFinishedStep(
      ctx,
      `Downloaded snapshot export to ${chalk.bold(filePath)}`,
    );
  });

type SnapshotExportState =
  | { state: "requested" }
  | { state: "in_progress" }
  | {
      state: "completed";
      complete_ts: bigint;
      start_ts: bigint;
      zip_object_key: string;
    };

async function waitForStableExportState(
  ctx: Context,
  deploymentUrl: string,
  adminKey: string,
): Promise<SnapshotExportState> {
  const [donePromise, onDone] = waitUntilCalled();
  let snapshotExportState: SnapshotExportState;
  await subscribe(
    ctx,
    deploymentUrl,
    adminKey,
    "_system/cli/exports:getLatest",
    {},
    donePromise,
    {
      onChange: (value: any) => {
        // NOTE: `value` would only be `null` if there has never been an export
        // requested.
        snapshotExportState = value;
        switch (snapshotExportState.state) {
          case "requested":
          case "in_progress":
            // Not a stable state.
            break;
          case "completed":
            onDone();
            break;
          default: {
            const _: never = snapshotExportState;
            onDone();
          }
        }
      },
    },
  );
  return snapshotExportState!;
}
