Fix #182
This commit is contained in:
parent
0389d798d7
commit
b7f045fd5e
|
@ -61,9 +61,10 @@ mixin template StandardEditorConfigFields()
|
||||||
|
|
||||||
void merge(ref const typeof(this) other, const string fileName)
|
void merge(ref const typeof(this) other, const string fileName)
|
||||||
{
|
{
|
||||||
import std.path : globMatch;
|
import dfmt.globmatch_editorconfig : globMatchEditorConfig;
|
||||||
|
import std.array : front, popFront, empty, save;
|
||||||
|
|
||||||
if (other.pattern is null || !fileName.globMatch(other.pattern))
|
if (other.pattern is null || !ecMatch(fileName, other.pattern))
|
||||||
return;
|
return;
|
||||||
foreach (N; FieldNameTuple!(typeof(this)))
|
foreach (N; FieldNameTuple!(typeof(this)))
|
||||||
{
|
{
|
||||||
|
@ -82,6 +83,33 @@ mixin template StandardEditorConfigFields()
|
||||||
static assert(false);
|
static assert(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bool ecMatch(string fileName, string patt)
|
||||||
|
{
|
||||||
|
import std.algorithm : canFind;
|
||||||
|
import std.path : baseName;
|
||||||
|
import dfmt.globmatch_editorconfig : globMatchEditorConfig;
|
||||||
|
|
||||||
|
if (!pattern.canFind("/"))
|
||||||
|
fileName = fileName.baseName;
|
||||||
|
return fileName.globMatchEditorConfig(patt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest
|
||||||
|
{
|
||||||
|
struct Config
|
||||||
|
{
|
||||||
|
mixin StandardEditorConfigFields;
|
||||||
|
}
|
||||||
|
|
||||||
|
Config config1;
|
||||||
|
Config config2;
|
||||||
|
config2.pattern = "test.d";
|
||||||
|
config2.end_of_line = EOL.crlf;
|
||||||
|
assert(config1.end_of_line != config2.end_of_line);
|
||||||
|
config1.merge(config2, "a/b/test.d");
|
||||||
|
assert(config1.end_of_line == config2.end_of_line);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -0,0 +1,240 @@
|
||||||
|
module dfmt.globmatch_editorconfig;
|
||||||
|
|
||||||
|
import std.path : CaseSensitive;
|
||||||
|
import std.range : isForwardRange, ElementEncodingType;
|
||||||
|
import std.string : isSomeChar, isSomeString, empty, save, front, popFront;
|
||||||
|
import std.typecons : Unqual;
|
||||||
|
import std.conv : to;
|
||||||
|
import std.path : filenameCharCmp, isDirSeparator;
|
||||||
|
|
||||||
|
// From std.path with changes:
|
||||||
|
// * changes meaning to match all characters except '/'
|
||||||
|
// ** added to take over the old meaning of *
|
||||||
|
bool globMatchEditorConfig(CaseSensitive cs = CaseSensitive.osDefault, C, Range)
|
||||||
|
(Range path, const(C)[] pattern)
|
||||||
|
@safe pure nothrow
|
||||||
|
if (isForwardRange!Range && isSomeChar!(ElementEncodingType!Range) &&
|
||||||
|
isSomeChar!C && is(Unqual!C == Unqual!(ElementEncodingType!Range)))
|
||||||
|
in
|
||||||
|
{
|
||||||
|
// Verify that pattern[] is valid
|
||||||
|
import std.algorithm : balancedParens;
|
||||||
|
assert(balancedParens(pattern, '[', ']', 0));
|
||||||
|
assert(balancedParens(pattern, '{', '}', 0));
|
||||||
|
}
|
||||||
|
body
|
||||||
|
{
|
||||||
|
alias RC = Unqual!(ElementEncodingType!Range);
|
||||||
|
|
||||||
|
static if (RC.sizeof == 1 && isSomeString!Range)
|
||||||
|
{
|
||||||
|
import std.utf : byChar;
|
||||||
|
return globMatchEditorConfig!cs(path.byChar, pattern);
|
||||||
|
}
|
||||||
|
else static if (RC.sizeof == 2 && isSomeString!Range)
|
||||||
|
{
|
||||||
|
import std.utf : byWchar;
|
||||||
|
return globMatchEditorConfig!cs(path.byWchar, pattern);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
C[] pattmp;
|
||||||
|
foreach (ref pi; 0 .. pattern.length)
|
||||||
|
{
|
||||||
|
const pc = pattern[pi];
|
||||||
|
switch (pc)
|
||||||
|
{
|
||||||
|
case '*':
|
||||||
|
if (pi < pattern.length-1 && pattern[pi+1] == '*')
|
||||||
|
{
|
||||||
|
if (pi + 2 == pattern.length)
|
||||||
|
return true;
|
||||||
|
for (; !path.empty; path.popFront())
|
||||||
|
{
|
||||||
|
auto p = path.save;
|
||||||
|
if (globMatchEditorConfig!(cs, C)(p,
|
||||||
|
pattern[pi + 2 .. pattern.length]))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (pi + 1 == pattern.length)
|
||||||
|
return true;
|
||||||
|
for (; !path.empty; path.popFront())
|
||||||
|
{
|
||||||
|
auto p = path.save;
|
||||||
|
//if (p[0].to!dchar.isDirSeparator() && !pattern[pi+1].isDirSeparator())
|
||||||
|
// return false;
|
||||||
|
if (globMatchEditorConfig!(cs, C)(p,
|
||||||
|
pattern[pi + 1 .. pattern.length]))
|
||||||
|
return true;
|
||||||
|
if (p[0].to!dchar.isDirSeparator())
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
case '?':
|
||||||
|
if (path.empty)
|
||||||
|
return false;
|
||||||
|
path.popFront();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '[':
|
||||||
|
if (path.empty)
|
||||||
|
return false;
|
||||||
|
auto nc = path.front;
|
||||||
|
path.popFront();
|
||||||
|
auto not = false;
|
||||||
|
++pi;
|
||||||
|
if (pattern[pi] == '!')
|
||||||
|
{
|
||||||
|
not = true;
|
||||||
|
++pi;
|
||||||
|
}
|
||||||
|
auto anymatch = false;
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
const pc2 = pattern[pi];
|
||||||
|
if (pc2 == ']')
|
||||||
|
break;
|
||||||
|
if (!anymatch && (filenameCharCmp!cs(nc, pc2) == 0))
|
||||||
|
anymatch = true;
|
||||||
|
++pi;
|
||||||
|
}
|
||||||
|
if (anymatch == not)
|
||||||
|
return false;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '{':
|
||||||
|
// find end of {} section
|
||||||
|
auto piRemain = pi;
|
||||||
|
for (; piRemain < pattern.length
|
||||||
|
&& pattern[piRemain] != '}'; ++piRemain)
|
||||||
|
{ }
|
||||||
|
|
||||||
|
if (piRemain < pattern.length)
|
||||||
|
++piRemain;
|
||||||
|
++pi;
|
||||||
|
|
||||||
|
while (pi < pattern.length)
|
||||||
|
{
|
||||||
|
const pi0 = pi;
|
||||||
|
C pc3 = pattern[pi];
|
||||||
|
// find end of current alternative
|
||||||
|
for (; pi < pattern.length && pc3 != '}' && pc3 != ','; ++pi)
|
||||||
|
{
|
||||||
|
pc3 = pattern[pi];
|
||||||
|
}
|
||||||
|
|
||||||
|
auto p = path.save;
|
||||||
|
if (pi0 == pi)
|
||||||
|
{
|
||||||
|
if (globMatchEditorConfig!(cs, C)(p, pattern[piRemain..$]))
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
++pi;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Match for:
|
||||||
|
* pattern[pi0..pi-1] ~ pattern[piRemain..$]
|
||||||
|
*/
|
||||||
|
if (pattmp.ptr == null)
|
||||||
|
// Allocate this only once per function invocation.
|
||||||
|
// Should do it with malloc/free, but that would make it impure.
|
||||||
|
pattmp = new C[pattern.length];
|
||||||
|
|
||||||
|
const len1 = pi - 1 - pi0;
|
||||||
|
pattmp[0 .. len1] = pattern[pi0 .. pi - 1];
|
||||||
|
|
||||||
|
const len2 = pattern.length - piRemain;
|
||||||
|
pattmp[len1 .. len1 + len2] = pattern[piRemain .. $];
|
||||||
|
|
||||||
|
if (globMatchEditorConfig!(cs, C)(p, pattmp[0 .. len1 + len2]))
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pc3 == '}')
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (path.empty)
|
||||||
|
return false;
|
||||||
|
if (filenameCharCmp!cs(pc, path.front) != 0)
|
||||||
|
return false;
|
||||||
|
path.popFront();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return path.empty;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unittest
|
||||||
|
{
|
||||||
|
assert (globMatchEditorConfig!(CaseSensitive.no)("foo", "Foo"));
|
||||||
|
assert (!globMatchEditorConfig!(CaseSensitive.yes)("foo", "Foo"));
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("foo", "*"));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"w, "*"w));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"d, "*.*"d));
|
||||||
|
assert(globMatchEditorConfig("foo.bar", "foo*"));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"w, "f*bar"w));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"d, "f*b*r"d));
|
||||||
|
assert(globMatchEditorConfig("foo.bar", "f???bar"));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"w, "[fg]???bar"w));
|
||||||
|
assert(globMatchEditorConfig("foo.bar"d, "[!gh]*bar"d));
|
||||||
|
|
||||||
|
assert(!globMatchEditorConfig("foo", "bar"));
|
||||||
|
assert(!globMatchEditorConfig("foo"w, "*.*"w));
|
||||||
|
assert(!globMatchEditorConfig("foo.bar"d, "f*baz"d));
|
||||||
|
assert(!globMatchEditorConfig("foo.bar", "f*b*x"));
|
||||||
|
assert(!globMatchEditorConfig("foo.bar", "[gh]???bar"));
|
||||||
|
assert(!globMatchEditorConfig("foo.bar"w, "[!fg]*bar"w));
|
||||||
|
assert(!globMatchEditorConfig("foo.bar"d, "[fg]???baz"d));
|
||||||
|
assert(!globMatchEditorConfig("foo.di", "*.d")); // test issue 6634: triggered bad assertion
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("foo.bar", "{foo,bif}.bar"));
|
||||||
|
assert(globMatchEditorConfig("bif.bar"w, "{foo,bif}.bar"w));
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("bar.foo"d, "bar.{foo,bif}"d));
|
||||||
|
assert(globMatchEditorConfig("bar.bif", "bar.{foo,bif}"));
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("bar.fooz"w, "bar.{foo,bif}z"w));
|
||||||
|
assert(globMatchEditorConfig("bar.bifz"d, "bar.{foo,bif}z"d));
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("bar.foo", "bar.{biz,,baz}foo"));
|
||||||
|
assert(globMatchEditorConfig("bar.foo"w, "bar.{biz,}foo"w));
|
||||||
|
assert(globMatchEditorConfig("bar.foo"d, "bar.{,biz}foo"d));
|
||||||
|
assert(globMatchEditorConfig("bar.foo", "bar.{}foo"));
|
||||||
|
|
||||||
|
assert(globMatchEditorConfig("bar.foo"w, "bar.{ar,,fo}o"w));
|
||||||
|
assert(globMatchEditorConfig("bar.foo"d, "bar.{,ar,fo}o"d));
|
||||||
|
assert(globMatchEditorConfig("bar.o", "bar.{,ar,fo}o"));
|
||||||
|
|
||||||
|
assert(!globMatchEditorConfig("foo", "foo?"));
|
||||||
|
assert(!globMatchEditorConfig("foo", "foo[]"));
|
||||||
|
assert(!globMatchEditorConfig("foo", "foob"));
|
||||||
|
assert(!globMatchEditorConfig("foo", "foo{b}"));
|
||||||
|
|
||||||
|
assert (globMatchEditorConfig(`foo/foo\bar`, "f**b**r"));
|
||||||
|
assert(globMatchEditorConfig("foo", "**"));
|
||||||
|
assert(globMatchEditorConfig("foo/bar", "foo/bar"));
|
||||||
|
assert(globMatchEditorConfig("foo/bar", "foo/*"));
|
||||||
|
assert(globMatchEditorConfig("foo/bar", "*/bar"));
|
||||||
|
assert(globMatchEditorConfig("/foo/bar/gluu/sar.png", "**/sar.png"));
|
||||||
|
assert(globMatchEditorConfig("/foo/bar/gluu/sar.png", "**/*.png"));
|
||||||
|
assert(!globMatchEditorConfig("/foo/bar/gluu/sar.png", "*/sar.png"));
|
||||||
|
assert(!globMatchEditorConfig("/foo/bar/gluu/sar.png", "*/*.png"));
|
||||||
|
|
||||||
|
|
||||||
|
static assert(globMatchEditorConfig("foo.bar", "[!gh]*bar"));
|
||||||
|
}
|
Loading…
Reference in New Issue