|
1 | | -#nullable disable |
2 | 1 | namespace Microsoft.ComponentDetection.Common; |
3 | 2 |
|
4 | 3 | using System; |
5 | 4 | using System.Collections.Generic; |
6 | 5 | using System.Linq; |
7 | | -using System.Linq.Expressions; |
8 | | -using System.Reflection; |
9 | 6 |
|
10 | 7 | public static class PatternMatchingUtility |
11 | 8 | { |
12 | 9 | public delegate bool FilePatternMatcher(ReadOnlySpan<char> span); |
13 | 10 |
|
14 | 11 | public static FilePatternMatcher GetFilePatternMatcher(IEnumerable<string> patterns) |
15 | 12 | { |
16 | | - var ordinalComparison = Expression.Constant(StringComparison.Ordinal, typeof(StringComparison)); |
17 | | - var asSpan = typeof(MemoryExtensions).GetMethod("AsSpan", BindingFlags.Public | BindingFlags.Static, null, CallingConventions.Standard, [typeof(string)], []); |
18 | | - var equals = typeof(MemoryExtensions).GetMethod("Equals", BindingFlags.Public | BindingFlags.Static, null, CallingConventions.Standard, [typeof(ReadOnlySpan<char>), typeof(ReadOnlySpan<char>), typeof(StringComparison)], []); |
19 | | - var startsWith = typeof(MemoryExtensions).GetMethod("StartsWith", BindingFlags.Public | BindingFlags.Static, null, CallingConventions.Standard, [typeof(ReadOnlySpan<char>), typeof(ReadOnlySpan<char>), typeof(StringComparison)], []); |
20 | | - var endsWith = typeof(MemoryExtensions).GetMethod("EndsWith", BindingFlags.Public | BindingFlags.Static, null, CallingConventions.Standard, [typeof(ReadOnlySpan<char>), typeof(ReadOnlySpan<char>), typeof(StringComparison)], []); |
21 | | - |
22 | | - var predicates = new List<Expression>(); |
23 | | - var left = Expression.Parameter(typeof(ReadOnlySpan<char>), "fileName"); |
24 | | - |
25 | | - foreach (var pattern in patterns) |
| 13 | + var matchers = patterns.Select<string, FilePatternMatcher>(pattern => pattern switch |
26 | 14 | { |
27 | | - if (pattern.StartsWith('*')) |
28 | | - { |
29 | | - var match = Expression.Constant(pattern[1..], typeof(string)); |
30 | | - var right = Expression.Call(null, asSpan, match); |
31 | | - var combine = Expression.Call(null, endsWith, left, right, ordinalComparison); |
32 | | - predicates.Add(combine); |
33 | | - } |
34 | | - else if (pattern.EndsWith('*')) |
35 | | - { |
36 | | - var match = Expression.Constant(pattern[..^1], typeof(string)); |
37 | | - var right = Expression.Call(null, asSpan, match); |
38 | | - var combine = Expression.Call(null, startsWith, left, right, ordinalComparison); |
39 | | - predicates.Add(combine); |
40 | | - } |
41 | | - else |
| 15 | + _ when pattern.StartsWith('*') && pattern.EndsWith('*') => |
| 16 | + span => span.Contains(pattern.AsSpan(1, pattern.Length - 2), StringComparison.Ordinal), |
| 17 | + _ when pattern.StartsWith('*') => |
| 18 | + span => span.EndsWith(pattern.AsSpan(1), StringComparison.Ordinal), |
| 19 | + _ when pattern.EndsWith('*') => |
| 20 | + span => span.StartsWith(pattern.AsSpan(0, pattern.Length - 1), StringComparison.Ordinal), |
| 21 | + _ => span => span.Equals(pattern.AsSpan(), StringComparison.Ordinal), |
| 22 | + }).ToList(); |
| 23 | + |
| 24 | + return span => |
| 25 | + { |
| 26 | + foreach (var matcher in matchers) |
42 | 27 | { |
43 | | - var match = Expression.Constant(pattern, typeof(string)); |
44 | | - var right = Expression.Call(null, asSpan, match); |
45 | | - var combine = Expression.Call(null, equals, left, right, ordinalComparison); |
46 | | - predicates.Add(combine); |
| 28 | + if (matcher(span)) |
| 29 | + { |
| 30 | + return true; |
| 31 | + } |
47 | 32 | } |
48 | | - } |
49 | | - |
50 | | - var aggregateExpression = predicates.Aggregate(Expression.OrElse); |
51 | | - |
52 | | - var func = Expression.Lambda<FilePatternMatcher>(aggregateExpression, left).Compile(); |
53 | 33 |
|
54 | | - return func; |
| 34 | + return false; |
| 35 | + }; |
55 | 36 | } |
56 | 37 | } |
0 commit comments