/**
 * store-paths.test.ts - Comprehensive unit tests for Windows path support
 *
 * Tests all path-related utility functions for cross-platform compatibility:
 * - isAbsolutePath() - Unix, Windows (C:\, C:/), and Git Bash (/c/) paths
 * - normalizePathSeparators() - backslash to forward slash conversion
 * - getRelativePathFromPrefix() - relative path extraction
 * - resolve() - path resolution with Unix and Windows paths
 *
 * Run with: bun test store-paths.test.ts
 */

import { describe, test, expect, beforeEach, afterEach } from "vitest";
import {
  isAbsolutePath,
  normalizePathSeparators,
  getRelativePathFromPrefix,
  resolve,
} from "../src/store.js";

// =============================================================================
// Test Utilities
// =============================================================================

let originalPWD: string | undefined;
let originalProcessCwd: () => string;

beforeEach(() => {
  // Save original environment
  originalPWD = process.env.PWD;
  originalProcessCwd = process.cwd;
});

afterEach(() => {
  // Restore original environment
  if (originalPWD !== undefined) {
    process.env.PWD = originalPWD;
  } else {
    delete process.env.PWD;
  }
  process.cwd = originalProcessCwd;
});

/**
 * Mock the current working directory for testing.
 * Sets both process.env.PWD and process.cwd() to simulate different environments.
 */
function mockPWD(path: string): void {
  process.env.PWD = path;
  process.cwd = () => path;
}

// =============================================================================
// Path Utilities - Cross-platform Support
// =============================================================================

describe("Path utilities - Cross-platform support", () => {
  
  // ===========================================================================
  // isAbsolutePath
  // ===========================================================================
  
  describe("isAbsolutePath", () => {
    test("Unix absolute paths", () => {
      expect(isAbsolutePath("/path/to/file")).toBe(true);
      expect(isAbsolutePath("/")).toBe(true);
      expect(isAbsolutePath("/home/user/documents")).toBe(true);
      expect(isAbsolutePath("/usr/local/bin")).toBe(true);
    });

    test("Unix relative paths", () => {
      expect(isAbsolutePath("path/to/file")).toBe(false);
      expect(isAbsolutePath("./path/to/file")).toBe(false);
      expect(isAbsolutePath("../path/to/file")).toBe(false);
      expect(isAbsolutePath("./file")).toBe(false);
      expect(isAbsolutePath("../file")).toBe(false);
      expect(isAbsolutePath("file.txt")).toBe(false);
    });

    test("Windows absolute paths (native) - forward slash", () => {
      expect(isAbsolutePath("C:/path/to/file")).toBe(true);
      expect(isAbsolutePath("C:/")).toBe(true);
      expect(isAbsolutePath("D:/Users/Documents")).toBe(true);
      expect(isAbsolutePath("Z:/")).toBe(true);
      expect(isAbsolutePath("c:/lowercase")).toBe(true);
    });

    test("Windows absolute paths (native) - backslash", () => {
      expect(isAbsolutePath("C:\\path\\to\\file")).toBe(true);
      expect(isAbsolutePath("C:\\")).toBe(true);
      expect(isAbsolutePath("D:\\Users\\Documents")).toBe(true);
      expect(isAbsolutePath("Z:\\")).toBe(true);
      expect(isAbsolutePath("c:\\lowercase")).toBe(true);
    });

    test("Windows relative paths", () => {
      expect(isAbsolutePath("path\\to\\file")).toBe(false);
      expect(isAbsolutePath(".\\path\\to\\file")).toBe(false);
      expect(isAbsolutePath("..\\path\\to\\file")).toBe(false);
      expect(isAbsolutePath(".\\file")).toBe(false);
      expect(isAbsolutePath("..\\file")).toBe(false);
      expect(isAbsolutePath("file.txt")).toBe(false);
    });

    test("Git Bash style paths", () => {
      expect(isAbsolutePath("/c/Users/name/file")).toBe(true);
      expect(isAbsolutePath("/C/Users/name/file")).toBe(true);
      expect(isAbsolutePath("/d/Projects")).toBe(true);
      expect(isAbsolutePath("/D/Projects")).toBe(true);
      expect(isAbsolutePath("/z/")).toBe(true);
    });

    test("Edge cases", () => {
      expect(isAbsolutePath("")).toBe(false);
      expect(isAbsolutePath("C:")).toBe(true); // Drive letter only
      expect(isAbsolutePath("C")).toBe(false); // Just a letter
      expect(isAbsolutePath(":")).toBe(false);
      expect(isAbsolutePath("/a")).toBe(true); // Short Unix path
      expect(isAbsolutePath("/1/")).toBe(true); // Number after slash (not Git Bash)
    });
  });

  // ===========================================================================
  // normalizePathSeparators
  // ===========================================================================
  
  describe("normalizePathSeparators", () => {
    test("Windows paths with backslashes", () => {
      expect(normalizePathSeparators("C:\\Users\\name\\file.txt"))
        .toBe("C:/Users/name/file.txt");
      expect(normalizePathSeparators("D:\\Projects\\qmd\\src"))
        .toBe("D:/Projects/qmd/src");
      expect(normalizePathSeparators("\\path\\to\\file"))
        .toBe("/path/to/file");
    });

    test("Mixed separators", () => {
      expect(normalizePathSeparators("C:\\Users/name\\file.txt"))
        .toBe("C:/Users/name/file.txt");
      expect(normalizePathSeparators("path\\to/file/here"))
        .toBe("path/to/file/here");
    });

    test("Unix paths (should remain unchanged)", () => {
      expect(normalizePathSeparators("/path/to/file"))
        .toBe("/path/to/file");
      expect(normalizePathSeparators("/usr/local/bin"))
        .toBe("/usr/local/bin");
      expect(normalizePathSeparators("relative/path"))
        .toBe("relative/path");
    });

    test("Multiple consecutive backslashes", () => {
      expect(normalizePathSeparators("path\\\\to\\\\file"))
        .toBe("path//to//file");
      expect(normalizePathSeparators("C:\\\\Users\\\\name"))
        .toBe("C://Users//name");
    });

    test("Edge cases", () => {
      expect(normalizePathSeparators("")).toBe("");
      expect(normalizePathSeparators("\\")).toBe("/");
      expect(normalizePathSeparators("\\\\")).toBe("//");
      expect(normalizePathSeparators("file.txt")).toBe("file.txt");
    });
  });

  // ===========================================================================
  // getRelativePathFromPrefix
  // ===========================================================================
  
  describe("getRelativePathFromPrefix", () => {
    test("Exact match (path equals prefix)", () => {
      expect(getRelativePathFromPrefix("/home/user", "/home/user")).toBe("");
      expect(getRelativePathFromPrefix("C:/Users/name", "C:/Users/name")).toBe("");
      expect(getRelativePathFromPrefix("/path", "/path")).toBe("");
    });

    test("Path under prefix", () => {
      expect(getRelativePathFromPrefix("/home/user/documents", "/home/user"))
        .toBe("documents");
      expect(getRelativePathFromPrefix("/home/user/documents/file.txt", "/home/user"))
        .toBe("documents/file.txt");
      expect(getRelativePathFromPrefix("C:/Users/name/Documents/file.txt", "C:/Users/name"))
        .toBe("Documents/file.txt");
    });

    test("Path not under prefix", () => {
      expect(getRelativePathFromPrefix("/home/other", "/home/user")).toBeNull();
      expect(getRelativePathFromPrefix("/usr/local", "/home/user")).toBeNull();
      expect(getRelativePathFromPrefix("C:/Users/other", "D:/Users")).toBeNull();
    });

    test("Windows paths with normalized separators", () => {
      // Backslashes should be normalized
      expect(getRelativePathFromPrefix("C:\\Users\\name\\Documents", "C:\\Users\\name"))
        .toBe("Documents");
      expect(getRelativePathFromPrefix("C:\\Users\\name\\Documents\\file.txt", "C:/Users/name"))
        .toBe("Documents/file.txt");
    });

    test("Prefix with trailing slash", () => {
      expect(getRelativePathFromPrefix("/home/user/documents", "/home/user/"))
        .toBe("documents");
      expect(getRelativePathFromPrefix("C:/Users/name/Documents", "C:/Users/name/"))
        .toBe("Documents");
    });

    test("Prefix without trailing slash", () => {
      expect(getRelativePathFromPrefix("/home/user/documents", "/home/user"))
        .toBe("documents");
      expect(getRelativePathFromPrefix("C:/Users/name/Documents", "C:/Users/name"))
        .toBe("Documents");
    });

    test("Edge cases", () => {
      // Empty prefix
      expect(getRelativePathFromPrefix("/path/to/file", "")).toBeNull();
      
      // Path is prefix substring but not in hierarchy
      expect(getRelativePathFromPrefix("/home/username", "/home/user")).toBeNull();
      
      // Root prefix
      expect(getRelativePathFromPrefix("/home/user", "/")).toBe("home/user");
    });
  });

  // ===========================================================================
  // resolve - Unix environment
  // ===========================================================================
  
  describe("resolve - Unix environment", () => {
    beforeEach(() => {
      mockPWD("/home/user");
    });

    test("Unix relative paths", () => {
      expect(resolve("/base", "relative")).toBe("/base/relative");
      expect(resolve("/base", "a/b/c")).toBe("/base/a/b/c");
      expect(resolve("/home", "user/documents")).toBe("/home/user/documents");
    });

    test("Unix absolute paths", () => {
      expect(resolve("/base", "/absolute")).toBe("/absolute");
      expect(resolve("/home/user", "/usr/local")).toBe("/usr/local");
      expect(resolve("/any", "/")).toBe("/");
    });

    test("Path with .. and .", () => {
      expect(resolve("/base", "../other")).toBe("/other");
      expect(resolve("/base/sub", "..")).toBe("/base");
      expect(resolve("/base", "./file")).toBe("/base/file");
      expect(resolve("/base/a/b", "../../c")).toBe("/base/c");
    });

    test("Multiple path segments", () => {
      expect(resolve("/a", "b", "c")).toBe("/a/b/c");
      expect(resolve("/a", "b", "../c")).toBe("/a/c");
      expect(resolve("/a", "b", "/c")).toBe("/c");
    });

    test("Relative path without base (uses PWD)", () => {
      expect(resolve("relative")).toBe("/home/user/relative");
      expect(resolve("a/b/c")).toBe("/home/user/a/b/c");
      expect(resolve("./file")).toBe("/home/user/file");
    });

    test("Absolute path alone", () => {
      expect(resolve("/absolute/path")).toBe("/absolute/path");
      expect(resolve("/")).toBe("/");
    });
  });

  // ===========================================================================
  // resolve - Windows environment
  // ===========================================================================
  
  describe("resolve - Windows environment", () => {
    beforeEach(() => {
      mockPWD("C:/Users/name");
    });

    test("Windows relative paths", () => {
      expect(resolve("C:/base", "relative")).toBe("C:/base/relative");
      expect(resolve("C:/base", "a/b/c")).toBe("C:/base/a/b/c");
      expect(resolve("D:/Projects", "qmd/src")).toBe("D:/Projects/qmd/src");
    });

    test("Windows absolute paths", () => {
      expect(resolve("C:/base", "D:/other")).toBe("D:/other");
      expect(resolve("C:/Users", "C:/Program Files")).toBe("C:/Program Files");
      expect(resolve("D:/any", "E:/other")).toBe("E:/other");
    });

    test("Windows with backslashes", () => {
      expect(resolve("C:\\base", "relative")).toBe("C:/base/relative");
      expect(resolve("C:\\Users\\name", "Documents")).toBe("C:/Users/name/Documents");
      expect(resolve("C:\\base", "a\\b\\c")).toBe("C:/base/a/b/c");
    });

    test("Path with .. and .", () => {
      expect(resolve("C:/base", "../other")).toBe("C:/other");
      expect(resolve("C:/base/sub", "..")).toBe("C:/base");
      expect(resolve("C:/base", "./file")).toBe("C:/base/file");
      expect(resolve("C:/base/a/b", "../../c")).toBe("C:/base/c");
    });

    test("Multiple path segments", () => {
      expect(resolve("C:/a", "b", "c")).toBe("C:/a/b/c");
      expect(resolve("C:/a", "b", "../c")).toBe("C:/a/c");
      expect(resolve("C:/a", "b", "D:/c")).toBe("D:/c");
    });

    test("Relative path without base (uses PWD)", () => {
      expect(resolve("relative")).toBe("C:/Users/name/relative");
      expect(resolve("a/b/c")).toBe("C:/Users/name/a/b/c");
      expect(resolve(".\\file")).toBe("C:/Users/name/file");
    });

    test("Drive letter only", () => {
      expect(resolve("C:")).toBe("C:/");
      expect(resolve("D:")).toBe("D:/");
    });
  });

  // ===========================================================================
  // resolve - Git Bash style paths
  // ===========================================================================
  
  describe("resolve - Git Bash style paths", () => {
    test("Git Bash to Windows conversion", () => {
      expect(resolve("/c/Users/name")).toBe("C:/Users/name");
      expect(resolve("/C/Users/name")).toBe("C:/Users/name");
      expect(resolve("/d/Projects")).toBe("D:/Projects");
      expect(resolve("/D/Projects")).toBe("D:/Projects");
    });

    test("Git Bash with relative paths", () => {
      expect(resolve("/c/base", "relative")).toBe("C:/base/relative");
      expect(resolve("/d/Projects", "qmd/src")).toBe("D:/Projects/qmd/src");
    });

    test("Git Bash with .. and .", () => {
      expect(resolve("/c/base", "../other")).toBe("C:/other");
      expect(resolve("/c/base/sub", "..")).toBe("C:/base");
      expect(resolve("/c/base", "./file")).toBe("C:/base/file");
    });

    test("Multiple Git Bash segments", () => {
      expect(resolve("/c/a", "b", "c")).toBe("C:/a/b/c");
      expect(resolve("/c/a", "b", "/d/c")).toBe("D:/c");
    });
  });

  // ===========================================================================
  // resolve - Edge cases and mixed scenarios
  // ===========================================================================
  
  describe("resolve - Edge cases", () => {
    test("Empty path segments are filtered", () => {
      expect(resolve("/base", "", "file")).toBe("/base/file");
      expect(resolve("C:/base", "", "file")).toBe("C:/base/file");
    });

    test("Multiple consecutive slashes", () => {
      expect(resolve("/base//path///file")).toBe("/base/path/file");
      expect(resolve("C:/base//path///file")).toBe("C:/base/path/file");
    });

    test("Trailing slashes", () => {
      expect(resolve("/base/", "file")).toBe("/base/file");
      expect(resolve("C:/base/", "file")).toBe("C:/base/file");
    });

    test("Complex .. navigation", () => {
      expect(resolve("/a/b/c/d", "../../../e")).toBe("/a/e");
      expect(resolve("C:/a/b/c/d", "../../../e")).toBe("C:/a/e");
    });

    test("Too many .. (should not go above root)", () => {
      expect(resolve("/base", "../../../../other")).toBe("/other");
      expect(resolve("C:/base", "../../../../other")).toBe("C:/other");
    });

    test("Mixed Unix and Windows (normalized)", () => {
      mockPWD("C:/Users/name");
      expect(resolve("/unix/path")).toBe("/unix/path");
      expect(resolve("relative")).toBe("C:/Users/name/relative");
    });

    test("Error on no arguments", () => {
      expect(() => resolve()).toThrow("resolve: at least one path segment is required");
    });
  });
});