mirror of
https://github.com/denoland/std.git
synced 2024-11-21 20:50:22 +00:00
1ac89853f5
* BREAKING(fs): remove `GlobToRegExpOptions.os` * fix
496 lines
17 KiB
TypeScript
496 lines
17 KiB
TypeScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||
import { assert, assertEquals } from "@std/assert";
|
||
import { type GlobOptions, globToRegExp } from "./glob_to_regexp.ts";
|
||
import { globToRegExp as posixGlobToRegExp } from "./posix/glob_to_regexp.ts";
|
||
import { globToRegExp as windowsGlobToRegExp } from "./windows/glob_to_regexp.ts";
|
||
|
||
function posixMatch(glob: string, path: string, options?: GlobOptions) {
|
||
const match = path.match(posixGlobToRegExp(glob, options));
|
||
if (match) {
|
||
assertEquals(match.length, 1);
|
||
}
|
||
return !!match;
|
||
}
|
||
|
||
function windowsMatch(glob: string, path: string, options?: GlobOptions) {
|
||
const match = path.match(windowsGlobToRegExp(glob, options));
|
||
if (match) {
|
||
assertEquals(match.length, 1);
|
||
}
|
||
return !!match;
|
||
}
|
||
|
||
function match(glob: string, path: string, options?: GlobOptions) {
|
||
return posixMatch(glob, path, options) && windowsMatch(glob, path, options);
|
||
}
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() returns RegExp from a glob",
|
||
fn() {
|
||
assertEquals(posixGlobToRegExp("*.js"), /^[^/]*\.js\/*$/);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() returns RegExp from an empty glob",
|
||
fn() {
|
||
assertEquals(globToRegExp(""), /(?!)/);
|
||
assertEquals(posixGlobToRegExp("*.js"), /^[^/]*\.js\/*$/);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: `globToRegExp() checks "*" (wildcard)`,
|
||
fn() {
|
||
assert(match("*", "foo", { extended: false, globstar: false }));
|
||
assert(match("*", "foo", { extended: false, globstar: false }));
|
||
assert(match("f*", "foo", { extended: false, globstar: false }));
|
||
assert(match("f*", "foo", { extended: false, globstar: false }));
|
||
assert(match("*o", "foo", { extended: false, globstar: false }));
|
||
assert(match("*o", "foo", { extended: false, globstar: false }));
|
||
assert(match("u*orn", "unicorn", { extended: false, globstar: false }));
|
||
assert(match("u*orn", "unicorn", { extended: false, globstar: false }));
|
||
assert(!match("ico", "unicorn", { extended: false, globstar: false }));
|
||
assert(match("u*nicorn", "unicorn", { extended: false, globstar: false }));
|
||
assert(match("u*nicorn", "unicorn", { extended: false, globstar: false }));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: `globToRegExp() checks "?" (match one character)`,
|
||
fn() {
|
||
assert(match("f?o", "foo", { extended: false, globstar: false }));
|
||
assert(match("f?o?", "fooo", { extended: false, globstar: false }));
|
||
assert(!match("f?oo", "foo", { extended: false, globstar: false }));
|
||
assert(!match("?fo", "fooo", { extended: false, globstar: false }));
|
||
assert(!match("f?oo", "foo", { extended: false, globstar: false }));
|
||
assert(!match("foo?", "foo", { extended: false, globstar: false }));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks [seq] (character range)",
|
||
fn() {
|
||
assert(match("fo[oz]", "foo", { extended: false, globstar: false }));
|
||
assert(match("fo[oz]", "foz", { extended: false, globstar: false }));
|
||
assert(!match("fo[oz]", "fog", { extended: false, globstar: false }));
|
||
assert(match("fo[a-z]", "fob", { extended: false, globstar: false }));
|
||
assert(!match("fo[a-d]", "fot", { extended: false, globstar: false }));
|
||
assert(!match("fo[!tz]", "fot", { extended: false, globstar: false }));
|
||
assert(match("fo[!tz]", "fob", { extended: false, globstar: false }));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks [[:alnum:]] (character class in range)",
|
||
fn() {
|
||
assert(
|
||
match(
|
||
"[[:alnum:]]/bar.txt",
|
||
"a/bar.txt",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
match(
|
||
"[[:alnum:]abc]/bar.txt",
|
||
"1/bar.txt",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
!match(
|
||
"[[:alnum:]]/bar.txt",
|
||
"!/bar.txt",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
for (const c of "09AGZagz") {
|
||
assert(match("[[:alnum:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "AGZagz") {
|
||
assert(match("[[:alpha:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\x00\x20\x7F") {
|
||
assert(match("[[:ascii:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\t ") {
|
||
assert(match("[[:blank:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\x00\x1F\x7F") {
|
||
assert(match("[[:cntrl:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "09") {
|
||
assert(match("[[:digit:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\x21\x7E") {
|
||
assert(match("[[:graph:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "az") {
|
||
assert(match("[[:lower:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\x20\x7E") {
|
||
assert(match("[[:print:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "!\"#$%&'()*+,-./:;<=>?@[\\]^_‘{|}~") {
|
||
assert(match("[[:punct:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "\t\n\v\f\r ") {
|
||
assert(match("[[:space:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "AZ") {
|
||
assert(match("[[:upper:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "09AZaz_") {
|
||
assert(match("[[:word:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
for (const c of "09AFaf") {
|
||
assert(match("[[:xdigit:]]", c, { extended: false, globstar: false }), c);
|
||
}
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: `globToRegExp() checks {} (brace expansion)`,
|
||
fn() {
|
||
assert(
|
||
match("foo{bar,baaz}", "foobaaz", { extended: false, globstar: false }),
|
||
);
|
||
assert(
|
||
match("foo{bar,baaz}", "foobar", { extended: false, globstar: false }),
|
||
);
|
||
assert(
|
||
!match("foo{bar,baaz}", "foobuzz", { extended: false, globstar: false }),
|
||
);
|
||
assert(
|
||
match("foo{bar,b*z}", "foobuzz", { extended: false, globstar: false }),
|
||
);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks complex matches",
|
||
fn() {
|
||
assert(
|
||
match(
|
||
"http://?o[oz].b*z.com/{*.js,*.html}",
|
||
"http://foo.baaz.com/jquery.min.js",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
match(
|
||
"http://?o[oz].b*z.com/{*.js,*.html}",
|
||
"http://moz.buzz.com/index.html",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
!match(
|
||
"http://?o[oz].b*z.com/{*.js,*.html}",
|
||
"http://moz.buzz.com/index.htm",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
!match(
|
||
"http://?o[oz].b*z.com/{*.js,*.html}",
|
||
"http://moz.bar.com/index.html",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
assert(
|
||
!match(
|
||
"http://?o[oz].b*z.com/{*.js,*.html}",
|
||
"http://flozz.buzz.com/index.html",
|
||
{ extended: false, globstar: false },
|
||
),
|
||
);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: `globToRegExp() checks "**" (globstar)`,
|
||
fn() {
|
||
assert(match("/foo/**", "/foo/bar.txt"));
|
||
assert(match("/foo/**", "/foo/bar/baz.txt"));
|
||
assert(!match("/foo/**", "/foo/bar/baz.txt", { globstar: false }));
|
||
assert(match("/foo/**", "/foo/bar", { globstar: false }));
|
||
assert(match("/foo/**/*.txt", "/foo/bar/baz.txt"));
|
||
assert(match("/foo/**/*.txt", "/foo/bar/baz/qux.txt"));
|
||
assert(match("/foo/**/bar.txt", "/foo/bar.txt"));
|
||
assert(match("/foo/**/**/bar.txt", "/foo/bar.txt"));
|
||
assert(match("/foo/**/*/baz.txt", "/foo/bar/baz.txt"));
|
||
assert(match("/foo/**/*.txt", "/foo/bar.txt"));
|
||
assert(match("/foo/**/**/*.txt", "/foo/bar.txt"));
|
||
assert(match("/foo/**/*/*.txt", "/foo/bar/baz.txt"));
|
||
assert(match("**/*.txt", "/foo/bar/baz/qux.txt"));
|
||
assert(match("**/foo.txt", "foo.txt"));
|
||
assert(match("**/*.txt", "foo.txt"));
|
||
assert(!match("/foo/**.txt", "/foo/bar/baz/qux.txt"));
|
||
assert(
|
||
!match("/foo/bar**/*.txt", "/foo/bar/baz/qux.txt"),
|
||
);
|
||
assert(!match("/foo/bar**", "/foo/bar/baz.txt"));
|
||
assert(!match("**/.txt", "/foo/bar/baz/qux.txt"));
|
||
assert(
|
||
!match(
|
||
"http://foo.com/*",
|
||
"http://foo.com/bar/baz/jquery.min.js",
|
||
),
|
||
);
|
||
assert(
|
||
!match("http://foo.com/*", "http://foo.com/bar/baz/jquery.min.js"),
|
||
);
|
||
assert(
|
||
match("http://foo.com/**", "http://foo.com/bar/baz/jquery.min.js"),
|
||
);
|
||
assert(
|
||
match(
|
||
"http://foo.com/**/jquery.min.js",
|
||
"http://foo.com/bar/baz/jquery.min.js",
|
||
),
|
||
);
|
||
assert(
|
||
!match(
|
||
"http://foo.com/*/jquery.min.js",
|
||
"http://foo.com/bar/baz/jquery.min.js",
|
||
),
|
||
);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name:
|
||
`globToRegExp() checks "?" (pattern-list) (extended: match zero or one)`,
|
||
fn() {
|
||
assert(match("?(foo).txt", "foo.txt"));
|
||
assert(!match("?(foo).txt", "foo.txt", { extended: false }));
|
||
assert(match("?(foo).txt", "a(foo).txt", { extended: false }));
|
||
assert(match("?(foo).txt", ".txt"));
|
||
assert(match("?(foo|bar)baz.txt", "foobaz.txt"));
|
||
assert(match("?(ba[zr]|qux)baz.txt", "bazbaz.txt"));
|
||
assert(match("?(ba[zr]|qux)baz.txt", "barbaz.txt"));
|
||
assert(match("?(ba[zr]|qux)baz.txt", "quxbaz.txt"));
|
||
assert(match("?(ba[!zr]|qux)baz.txt", "batbaz.txt"));
|
||
assert(match("?(ba*|qux)baz.txt", "batbaz.txt"));
|
||
assert(match("?(ba*|qux)baz.txt", "batttbaz.txt"));
|
||
assert(match("?(ba*|qux)baz.txt", "quxbaz.txt"));
|
||
assert(match("?(ba?(z|r)|qux)baz.txt", "bazbaz.txt"));
|
||
assert(match("?(ba?(z|?(r))|qux)baz.txt", "bazbaz.txt"));
|
||
assert(!match("?(foo|bar)baz.txt", "foobarbaz.txt"));
|
||
assert(!match("?(ba[zr]|qux)baz.txt", "bazquxbaz.txt"));
|
||
assert(!match("?(ba[!zr]|qux)baz.txt", "bazbaz.txt"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks *(pattern-list) (extended: match zero or more)",
|
||
fn() {
|
||
assert(match("*(foo).txt", "foo.txt"));
|
||
assert(!match("*(foo).txt", "foo.txt", { extended: false }));
|
||
assert(match("*(foo).txt", "bar(foo).txt", { extended: false }));
|
||
assert(match("*(foo).txt", "foofoo.txt"));
|
||
assert(match("*(foo).txt", ".txt"));
|
||
assert(match("*(fooo).txt", ".txt"));
|
||
assert(!match("*(fooo).txt", "foo.txt"));
|
||
assert(match("*(foo|bar).txt", "foobar.txt"));
|
||
assert(match("*(foo|bar).txt", "barbar.txt"));
|
||
assert(match("*(foo|bar).txt", "barfoobar.txt"));
|
||
assert(match("*(foo|bar).txt", ".txt"));
|
||
assert(match("*(foo|ba[rt]).txt", "bat.txt"));
|
||
assert(match("*(foo|b*[rt]).txt", "blat.txt"));
|
||
assert(!match("*(foo|b*[rt]).txt", "tlat.txt"));
|
||
assert(match("*(*).txt", "whatever.txt"));
|
||
assert(match("*(foo|bar)/**/*.txt", "foo/hello/world/bar.txt"));
|
||
assert(match("*(foo|bar)/**/*.txt", "foo/world/bar.txt"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks +(pattern-list) (extended: match 1 or more)",
|
||
fn() {
|
||
assert(match("+(foo).txt", "foo.txt"));
|
||
assert(!match("+(foo).txt", "foo.txt", { extended: false }));
|
||
assert(match("+(foo).txt", "+(foo).txt", { extended: false }));
|
||
assert(!match("+(foo).txt", ".txt"));
|
||
assert(match("+(foo|bar).txt", "foobar.txt"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks @(pattern-list) (extended: match one)",
|
||
fn() {
|
||
assert(match("@(foo).txt", "foo.txt"));
|
||
assert(!match("@(foo).txt", "foo.txt", { extended: false }));
|
||
assert(match("@(foo).txt", "@(foo).txt", { extended: false }));
|
||
assert(match("@(foo|baz)bar.txt", "foobar.txt"));
|
||
assert(!match("@(foo|baz)bar.txt", "foobazbar.txt"));
|
||
assert(!match("@(foo|baz)bar.txt", "foofoobar.txt"));
|
||
assert(!match("@(foo|baz)bar.txt", "toofoobar.txt"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks !(pattern-list) (extended: match any except)",
|
||
fn() {
|
||
assert(match("!(boo).txt", "foo.txt"));
|
||
assert(!match("!(boo).txt", "foo.txt", { extended: false }));
|
||
assert(match("!(boo).txt", "!(boo).txt", { extended: false }));
|
||
assert(match("!(foo|baz)bar.txt", "buzbar.txt"));
|
||
assert(match("!({foo,bar})baz.txt", "notbaz.txt"));
|
||
assert(!match("!({foo,bar})baz.txt", "foobaz.txt"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() matches special extended characters with themselves",
|
||
fn() {
|
||
const glob = "\\/$^+.()=!|,.*";
|
||
assert(match(glob, glob));
|
||
assert(match(glob, glob, { extended: false }));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() matches special extended characters in range",
|
||
fn() {
|
||
assertEquals(posixGlobToRegExp("[?*+@!|]"), /^[?*+@!|]\/*$/);
|
||
assertEquals(posixGlobToRegExp("[!?*+@!|]"), /^[^?*+@!|]\/*$/);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() matches special RegExp characters in range",
|
||
fn() {
|
||
// Excluding characters checked in the previous test.
|
||
assertEquals(posixGlobToRegExp("[\\\\$^.=]"), /^[\\$^.=]\/*$/);
|
||
assertEquals(posixGlobToRegExp("[!\\\\$^.=]"), /^[^\\$^.=]\/*$/);
|
||
assertEquals(posixGlobToRegExp("[^^]"), /^[\^^]\/*$/);
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks repeating separators",
|
||
fn() {
|
||
assert(match("foo/bar", "foo//bar"));
|
||
assert(match("foo//bar", "foo/bar"));
|
||
assert(match("foo//bar", "foo//bar"));
|
||
assert(match("**/bar", "foo//bar"));
|
||
assert(match("**//bar", "foo/bar"));
|
||
assert(match("**//bar", "foo//bar"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks trailing separators",
|
||
fn() {
|
||
assert(match("foo", "foo/"));
|
||
assert(match("foo/", "foo"));
|
||
assert(match("foo/", "foo/"));
|
||
assert(match("**", "foo/"));
|
||
assert(match("**/", "foo"));
|
||
assert(match("**/", "foo/"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks backslashes on Windows",
|
||
fn() {
|
||
assert(windowsMatch("foo/bar", "foo\\bar"));
|
||
assert(windowsMatch("foo\\bar", "foo/bar"));
|
||
assert(windowsMatch("foo\\bar", "foo\\bar"));
|
||
assert(windowsMatch("**/bar", "foo\\bar"));
|
||
assert(windowsMatch("**\\bar", "foo/bar"));
|
||
assert(windowsMatch("**\\bar", "foo\\bar"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks unclosed groups",
|
||
fn() {
|
||
assert(match("{foo,bar}/[ab", "foo/[ab"));
|
||
assert(match("{foo,bar}/{foo,bar", "foo/{foo,bar"));
|
||
assert(match("{foo,bar}/?(foo|bar", "foo/?(foo|bar"));
|
||
assert(match("{foo,bar}/@(foo|bar", "foo/@(foo|bar"));
|
||
assert(match("{foo,bar}/*(foo|bar", "foo/*(foo|bar"));
|
||
assert(match("{foo,bar}/+(foo|bar", "foo/+(foo|bar"));
|
||
assert(match("{foo,bar}/!(foo|bar", "foo/!(foo|bar"));
|
||
assert(match("{foo,bar}/?({)}", "foo/?({)}"));
|
||
assert(match("{foo,bar}/{?(})", "foo/{?(})"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() escapes glob characters",
|
||
fn() {
|
||
assert(posixMatch("\\[ab]", "[ab]"));
|
||
assert(windowsMatch("`[ab]", "[ab]"));
|
||
assert(posixMatch("\\{foo,bar}", "{foo,bar}"));
|
||
assert(windowsMatch("`{foo,bar}", "{foo,bar}"));
|
||
assert(posixMatch("\\?(foo|bar)", "?(foo|bar)"));
|
||
assert(windowsMatch("`?(foo|bar)", "?(foo|bar)"));
|
||
assert(posixMatch("\\@(foo|bar)", "@(foo|bar)"));
|
||
assert(windowsMatch("`@(foo|bar)", "@(foo|bar)"));
|
||
assert(posixMatch("\\*(foo|bar)", "*(foo|bar)"));
|
||
assert(windowsMatch("`*(foo|bar)", "*(foo|bar)"));
|
||
assert(posixMatch("\\+(foo|bar)", "+(foo|bar)"));
|
||
assert(windowsMatch("`+(foo|bar)", "+(foo|bar)"));
|
||
assert(posixMatch("\\!(foo|bar)", "!(foo|bar)"));
|
||
assert(windowsMatch("`!(foo|bar)", "!(foo|bar)"));
|
||
assert(posixMatch("@\\(foo|bar)", "@(foo|bar)"));
|
||
assert(windowsMatch("@`(foo|bar)", "@(foo|bar)"));
|
||
assert(posixMatch("{foo,bar}/[ab]\\", "foo/[ab]\\"));
|
||
assert(windowsMatch("{foo,bar}/[ab]`", "foo/[ab]`"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks dangling escape prefix",
|
||
fn() {
|
||
assert(posixMatch("{foo,bar}/[ab]\\", "foo/[ab]\\"));
|
||
assert(windowsMatch("{foo,bar}/[ab]`", "foo/[ab]`"));
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks options.extended",
|
||
fn() {
|
||
const pattern1 = globToRegExp("?(foo|bar)");
|
||
assertEquals("foo".match(pattern1)?.[0], "foo");
|
||
assertEquals("bar".match(pattern1)?.[0], "bar");
|
||
|
||
const pattern2 = globToRegExp("?(foo|bar)", { extended: false });
|
||
assertEquals("foo".match(pattern2)?.[0], undefined);
|
||
assertEquals("bar".match(pattern2)?.[0], undefined);
|
||
assertEquals("?(foo|bar)".match(pattern2)?.[0], "?(foo|bar)");
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks options.globstar",
|
||
fn() {
|
||
const pattern1 = globToRegExp("**/foo");
|
||
assertEquals("foo".match(pattern1)?.[0], "foo");
|
||
assertEquals("path/to/foo".match(pattern1)?.[0], "path/to/foo");
|
||
|
||
const pattern2 = globToRegExp("**/foo", { globstar: false });
|
||
assertEquals("foo".match(pattern2)?.[0], undefined);
|
||
assertEquals("path/to/foo".match(pattern2)?.[0], undefined);
|
||
assertEquals("path-to/foo".match(pattern2)?.[0], "path-to/foo");
|
||
},
|
||
});
|
||
|
||
Deno.test({
|
||
name: "globToRegExp() checks options.caseInsensitive",
|
||
fn() {
|
||
const pattern1 = globToRegExp("foo/bar", { caseInsensitive: false });
|
||
assertEquals("foo/bar".match(pattern1)?.[0], "foo/bar");
|
||
assertEquals("Foo/Bar".match(pattern1)?.[0], undefined);
|
||
|
||
const pattern2 = globToRegExp("foo/bar", { caseInsensitive: true });
|
||
assertEquals("foo/bar".match(pattern2)?.[0], "foo/bar");
|
||
assertEquals("Foo/Bar".match(pattern2)?.[0], "Foo/Bar");
|
||
},
|
||
});
|