/**
* 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");
});
});
});