165 lines
5.9 KiB
TypeScript
165 lines
5.9 KiB
TypeScript
import { describe, test, expect } from "bun:test"
|
|
import path from "path"
|
|
import fs from "fs/promises"
|
|
import { Glob } from "../../src/util/glob"
|
|
import { tmpdir } from "../fixture/fixture"
|
|
|
|
describe("Glob", () => {
|
|
describe("scan()", () => {
|
|
test("finds files matching pattern", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.writeFile(path.join(tmp.path, "a.txt"), "", "utf-8")
|
|
await fs.writeFile(path.join(tmp.path, "b.txt"), "", "utf-8")
|
|
await fs.writeFile(path.join(tmp.path, "c.md"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*.txt", { cwd: tmp.path })
|
|
|
|
expect(results.sort()).toEqual(["a.txt", "b.txt"])
|
|
})
|
|
|
|
test("returns absolute paths when absolute option is true", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*.txt", { cwd: tmp.path, absolute: true })
|
|
|
|
expect(results[0]).toBe(path.join(tmp.path, "file.txt"))
|
|
})
|
|
|
|
test("excludes directories by default", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*", { cwd: tmp.path })
|
|
|
|
expect(results).toEqual(["file.txt"])
|
|
})
|
|
|
|
test("excludes directories when include is 'file'", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*", { cwd: tmp.path, include: "file" })
|
|
|
|
expect(results).toEqual(["file.txt"])
|
|
})
|
|
|
|
test("includes directories when include is 'all'", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*", { cwd: tmp.path, include: "all" })
|
|
|
|
expect(results.sort()).toEqual(["file.txt", "subdir"])
|
|
})
|
|
|
|
test("handles nested patterns", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "nested"), { recursive: true })
|
|
await fs.writeFile(path.join(tmp.path, "nested", "deep.txt"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("**/*.txt", { cwd: tmp.path })
|
|
|
|
expect(results).toEqual([path.join("nested", "deep.txt")])
|
|
})
|
|
|
|
test("returns empty array for no matches", async () => {
|
|
await using tmp = await tmpdir()
|
|
|
|
const results = await Glob.scan("*.nonexistent", { cwd: tmp.path })
|
|
|
|
expect(results).toEqual([])
|
|
})
|
|
|
|
test("does not follow symlinks by default", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "realdir"))
|
|
await fs.writeFile(path.join(tmp.path, "realdir", "file.txt"), "", "utf-8")
|
|
await fs.symlink(path.join(tmp.path, "realdir"), path.join(tmp.path, "linkdir"))
|
|
|
|
const results = await Glob.scan("**/*.txt", { cwd: tmp.path })
|
|
|
|
expect(results).toEqual([path.join("realdir", "file.txt")])
|
|
})
|
|
|
|
test("follows symlinks when symlink option is true", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "realdir"))
|
|
await fs.writeFile(path.join(tmp.path, "realdir", "file.txt"), "", "utf-8")
|
|
await fs.symlink(path.join(tmp.path, "realdir"), path.join(tmp.path, "linkdir"))
|
|
|
|
const results = await Glob.scan("**/*.txt", { cwd: tmp.path, symlink: true })
|
|
|
|
expect(results.sort()).toEqual([path.join("linkdir", "file.txt"), path.join("realdir", "file.txt")])
|
|
})
|
|
|
|
test("includes dotfiles when dot option is true", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.writeFile(path.join(tmp.path, ".hidden"), "", "utf-8")
|
|
await fs.writeFile(path.join(tmp.path, "visible"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*", { cwd: tmp.path, dot: true })
|
|
|
|
expect(results.sort()).toEqual([".hidden", "visible"])
|
|
})
|
|
|
|
test("excludes dotfiles when dot option is false", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.writeFile(path.join(tmp.path, ".hidden"), "", "utf-8")
|
|
await fs.writeFile(path.join(tmp.path, "visible"), "", "utf-8")
|
|
|
|
const results = await Glob.scan("*", { cwd: tmp.path, dot: false })
|
|
|
|
expect(results).toEqual(["visible"])
|
|
})
|
|
})
|
|
|
|
describe("scanSync()", () => {
|
|
test("finds files matching pattern synchronously", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.writeFile(path.join(tmp.path, "a.txt"), "", "utf-8")
|
|
await fs.writeFile(path.join(tmp.path, "b.txt"), "", "utf-8")
|
|
|
|
const results = Glob.scanSync("*.txt", { cwd: tmp.path })
|
|
|
|
expect(results.sort()).toEqual(["a.txt", "b.txt"])
|
|
})
|
|
|
|
test("respects options", async () => {
|
|
await using tmp = await tmpdir()
|
|
await fs.mkdir(path.join(tmp.path, "subdir"))
|
|
await fs.writeFile(path.join(tmp.path, "file.txt"), "", "utf-8")
|
|
|
|
const results = Glob.scanSync("*", { cwd: tmp.path, include: "all" })
|
|
|
|
expect(results.sort()).toEqual(["file.txt", "subdir"])
|
|
})
|
|
})
|
|
|
|
describe("match()", () => {
|
|
test("matches simple patterns", () => {
|
|
expect(Glob.match("*.txt", "file.txt")).toBe(true)
|
|
expect(Glob.match("*.txt", "file.js")).toBe(false)
|
|
})
|
|
|
|
test("matches directory patterns", () => {
|
|
expect(Glob.match("**/*.js", "src/index.js")).toBe(true)
|
|
expect(Glob.match("**/*.js", "src/index.ts")).toBe(false)
|
|
})
|
|
|
|
test("matches dot files", () => {
|
|
expect(Glob.match(".*", ".gitignore")).toBe(true)
|
|
expect(Glob.match("**/*.md", ".github/README.md")).toBe(true)
|
|
})
|
|
|
|
test("matches brace expansion", () => {
|
|
expect(Glob.match("*.{js,ts}", "file.js")).toBe(true)
|
|
expect(Glob.match("*.{js,ts}", "file.ts")).toBe(true)
|
|
expect(Glob.match("*.{js,ts}", "file.py")).toBe(false)
|
|
})
|
|
})
|
|
})
|