181b81b4a4
Former-commit-id: cf92446697332992ec36726e78eb8703e1f259d7
222 lines
5.1 KiB
C#
222 lines
5.1 KiB
C#
//
|
|
// System.IO.SearchPattern2.cs: Filename glob support.
|
|
//
|
|
// Author:
|
|
// Dan Lewis (dihlewis@yahoo.co.uk)
|
|
//
|
|
// (C) 2002
|
|
//
|
|
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
// Copied from corlib/System.IO/SearchPatter.cs
|
|
using System;
|
|
|
|
namespace System.IO {
|
|
|
|
// FIXME: there's a complication with this algorithm under windows.
|
|
// the pattern '*.*' matches all files (i think . matches the extension),
|
|
// whereas under UNIX it should only match files containing the '.' character.
|
|
|
|
class SearchPattern2 {
|
|
public SearchPattern2 (string pattern) : this (pattern, false) { }
|
|
|
|
public SearchPattern2 (string pattern, bool ignore)
|
|
{
|
|
this.ignore = ignore;
|
|
this.pattern = pattern;
|
|
Compile (pattern);
|
|
}
|
|
|
|
// OSX has a case-insensitive yet case-aware filesystem
|
|
// so we need a overload in here for the Kqueue watcher
|
|
public bool IsMatch (string text, bool ignorecase)
|
|
{
|
|
if (!hasWildcard) {
|
|
bool match = String.Compare (pattern, text, ignorecase) == 0;
|
|
if (match)
|
|
return true;
|
|
}
|
|
|
|
// This is a special case for FSW. It needs to match e.g. subdir/file.txt
|
|
// when the pattern is "file.txt"
|
|
var fileName = Path.GetFileName (text);
|
|
|
|
if (!hasWildcard)
|
|
return (String.Compare (pattern, fileName, ignorecase) == 0);
|
|
|
|
|
|
return Match (ops, fileName, 0);
|
|
}
|
|
|
|
public bool IsMatch (string text)
|
|
{
|
|
return IsMatch (text, ignore);
|
|
}
|
|
|
|
public bool HasWildcard {
|
|
get { return hasWildcard; }
|
|
}
|
|
// private
|
|
|
|
Op ops; // the compiled pattern
|
|
bool ignore; // ignore case
|
|
bool hasWildcard;
|
|
string pattern;
|
|
|
|
private void Compile (string pattern)
|
|
{
|
|
if (pattern == null || pattern.IndexOfAny (InvalidChars) >= 0)
|
|
throw new ArgumentException ("Invalid search pattern: '" + pattern + "'");
|
|
|
|
if (pattern == "*") { // common case
|
|
ops = new Op (OpCode.True);
|
|
hasWildcard = true;
|
|
return;
|
|
}
|
|
|
|
ops = null;
|
|
|
|
int ptr = 0;
|
|
Op last_op = null;
|
|
while (ptr < pattern.Length) {
|
|
Op op;
|
|
|
|
switch (pattern [ptr]) {
|
|
case '?':
|
|
op = new Op (OpCode.AnyChar);
|
|
++ ptr;
|
|
hasWildcard = true;
|
|
break;
|
|
|
|
case '*':
|
|
op = new Op (OpCode.AnyString);
|
|
++ ptr;
|
|
hasWildcard = true;
|
|
break;
|
|
|
|
default:
|
|
op = new Op (OpCode.ExactString);
|
|
int end = pattern.IndexOfAny (WildcardChars, ptr);
|
|
if (end < 0)
|
|
end = pattern.Length;
|
|
|
|
op.Argument = pattern.Substring (ptr, end - ptr);
|
|
if (ignore)
|
|
op.Argument = op.Argument.ToLower ();
|
|
|
|
ptr = end;
|
|
break;
|
|
}
|
|
|
|
if (last_op == null)
|
|
ops = op;
|
|
else
|
|
last_op.Next = op;
|
|
|
|
last_op = op;
|
|
}
|
|
|
|
if (last_op == null)
|
|
ops = new Op (OpCode.End);
|
|
else
|
|
last_op.Next = new Op (OpCode.End);
|
|
}
|
|
|
|
private bool Match (Op op, string text, int ptr)
|
|
{
|
|
while (op != null) {
|
|
switch (op.Code) {
|
|
case OpCode.True:
|
|
return true;
|
|
|
|
case OpCode.End:
|
|
if (ptr == text.Length)
|
|
return true;
|
|
|
|
return false;
|
|
|
|
case OpCode.ExactString:
|
|
int length = op.Argument.Length;
|
|
if (ptr + length > text.Length)
|
|
return false;
|
|
|
|
string str = text.Substring (ptr, length);
|
|
if (ignore)
|
|
str = str.ToLower ();
|
|
|
|
if (str != op.Argument)
|
|
return false;
|
|
|
|
ptr += length;
|
|
break;
|
|
|
|
case OpCode.AnyChar:
|
|
if (++ ptr > text.Length)
|
|
return false;
|
|
break;
|
|
|
|
case OpCode.AnyString:
|
|
while (ptr <= text.Length) {
|
|
if (Match (op.Next, text, ptr))
|
|
return true;
|
|
|
|
++ ptr;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
op = op.Next;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// private static
|
|
|
|
internal static readonly char [] WildcardChars = { '*', '?' };
|
|
internal static readonly char [] InvalidChars = { Path.DirectorySeparatorChar, Path.AltDirectorySeparatorChar };
|
|
|
|
private class Op {
|
|
public Op (OpCode code)
|
|
{
|
|
this.Code = code;
|
|
this.Argument = null;
|
|
this.Next = null;
|
|
}
|
|
|
|
public OpCode Code;
|
|
public string Argument;
|
|
public Op Next;
|
|
}
|
|
|
|
private enum OpCode {
|
|
ExactString, // literal
|
|
AnyChar, // ?
|
|
AnyString, // *
|
|
End, // end of pattern
|
|
True // always succeeds
|
|
};
|
|
}
|
|
}
|