From de01038d13383ae9fa9a6b371325fd0bb2b494aa Mon Sep 17 00:00:00 2001
From: Martijn Bodeman <11424653+skwasjer@users.noreply.github.com>
Date: Fri, 6 Sep 2024 00:25:15 +0200
Subject: [PATCH] fix: special regex characters were not escaped properly when
derived a regex from a pattern with wildcard. (#106)
---
.../Matchers/Patterns/RegexPatternMatcher.cs | 10 ++--
.../Patterns/WildcardPatternMatcher.cs | 20 +++++++-
.../Patterns/WildcardPatternMatcherTests.cs | 46 +++++++++++++++++++
3 files changed, 70 insertions(+), 6 deletions(-)
create mode 100644 test/MockHttp.Tests/Matchers/Patterns/WildcardPatternMatcherTests.cs
diff --git a/src/MockHttp/Matchers/Patterns/RegexPatternMatcher.cs b/src/MockHttp/Matchers/Patterns/RegexPatternMatcher.cs
index 1b179c01..2ce2116e 100644
--- a/src/MockHttp/Matchers/Patterns/RegexPatternMatcher.cs
+++ b/src/MockHttp/Matchers/Patterns/RegexPatternMatcher.cs
@@ -5,8 +5,6 @@ namespace MockHttp.Matchers.Patterns;
internal class RegexPatternMatcher : PatternMatcher
{
- private readonly Regex _regex;
-
public RegexPatternMatcher
(
#if NET7_0_OR_GREATER
@@ -20,18 +18,20 @@ string regex
public RegexPatternMatcher(Regex regex)
{
- _regex = regex ?? throw new ArgumentNullException(nameof(regex));
+ Regex = regex ?? throw new ArgumentNullException(nameof(regex));
}
+ internal Regex Regex { get; }
+
///
public override bool IsMatch(string value)
{
- return _regex.IsMatch(value);
+ return Regex.IsMatch(value);
}
///
public override string ToString()
{
- return _regex.ToString();
+ return Regex.ToString();
}
}
diff --git a/src/MockHttp/Matchers/Patterns/WildcardPatternMatcher.cs b/src/MockHttp/Matchers/Patterns/WildcardPatternMatcher.cs
index 941e5425..8ce1e300 100644
--- a/src/MockHttp/Matchers/Patterns/WildcardPatternMatcher.cs
+++ b/src/MockHttp/Matchers/Patterns/WildcardPatternMatcher.cs
@@ -4,6 +4,8 @@ namespace MockHttp.Matchers.Patterns;
internal sealed class WildcardPatternMatcher : RegexPatternMatcher
{
+ private static readonly char[] SpecialRegexChars = ['.', '+', '*', '?', '^', '$', '(', ')', '[', ']', '{', '}', '|', '\\'];
+
private readonly string _pattern;
public WildcardPatternMatcher(string pattern)
@@ -47,7 +49,7 @@ private static string GetMatchPattern(string pattern)
IEnumerable matchGroups = pattern
.Split('*')
.Where(s => !string.IsNullOrEmpty(s))
- .Select(s => $"({s})");
+ .Select(s => $"({RegexUriEscape(s)})");
sb.Append(string.Join(".+", matchGroups));
@@ -55,4 +57,20 @@ private static string GetMatchPattern(string pattern)
return sb.ToString();
}
+
+ private static string RegexUriEscape(string s)
+ {
+ var sb = new StringBuilder();
+ foreach (char ch in s)
+ {
+ if (SpecialRegexChars.Contains(ch))
+ {
+ sb.Append('\\');
+ }
+
+ sb.Append(ch);
+ }
+
+ return sb.ToString();
+ }
}
diff --git a/test/MockHttp.Tests/Matchers/Patterns/WildcardPatternMatcherTests.cs b/test/MockHttp.Tests/Matchers/Patterns/WildcardPatternMatcherTests.cs
new file mode 100644
index 00000000..5e1b9754
--- /dev/null
+++ b/test/MockHttp.Tests/Matchers/Patterns/WildcardPatternMatcherTests.cs
@@ -0,0 +1,46 @@
+namespace MockHttp.Matchers.Patterns;
+
+public sealed class WildcardPatternMatcherTests
+{
+ [Theory]
+ [InlineData("test", "^(test)$")]
+ [InlineData("test*", "^(test).*")]
+ [InlineData("*test", ".*(test)$")]
+ [InlineData("*test*", ".*(test).*")]
+ [InlineData("*test1*test2*", ".*(test1).+(test2).*")]
+ [InlineData("/path?*", "^(/path\\?).*")]
+ [InlineData("/path/file.jpg?q=*&p=1&m=[a,b]", "^(/path/file\\.jpg\\?q=).+(&p=1&m=\\[a,b\\])$")]
+ [InlineData(".+?^$()[]{}|\\", "^(\\.\\+\\?\\^\\$\\(\\)\\[\\]\\{\\}\\|\\\\)$")]
+ public void When_creating_matcher_it_should_generate_expected_regex(string wildcardPattern, string expectedRegex)
+ {
+ var sut = new WildcardPatternMatcher(wildcardPattern);
+ sut.Regex.ToString().Should().Be(expectedRegex);
+ }
+
+ [Theory]
+ [InlineData("test", "test", true)]
+ [InlineData("test", "testing", false)]
+ [InlineData("test*", "test", true)]
+ [InlineData("test*", "testing", true)]
+ [InlineData("test*", "stress testing", false)]
+ [InlineData("*test", "test", true)]
+ [InlineData("*test", "testing", false)]
+ [InlineData("*test", "stress test", true)]
+ [InlineData("*test*", "test", true)]
+ [InlineData("*test*", "testing", true)]
+ [InlineData("*test*", "stress testing", true)]
+ [InlineData("*test*", "tes", false)]
+ [InlineData("*test1*test2*", "test1 test2", true)]
+ [InlineData("*test1*test2*", "test0 test1 test2 test3", true)]
+ [InlineData("*test1*test2*", "test test2", false)]
+ [InlineData("/path?q=*", "/path?q=1", true)]
+ [InlineData("/path?q=*", "/path?v=1&q=1", false)]
+ [InlineData("/path/file.jpg?q=*&p=1&m=[a,b]", "/path/file.jpg?q=search%20term&p=1&m=[a,b]", true)]
+ [InlineData("/path/file.jpg?q=*&p=1&m=[a,b]", "/path/file.jpg?q=search%20term&p=1&m=(a,b)", false)]
+ [InlineData("/path/file.jpg?q=*&p=*&m=*", "/path/file.jpg?q=search%20term&p=1&m=[a,b]", true)]
+ public void Given_that_value_matches_pattern_when_matching_it_should_pass(string wildcardPattern, string value, bool expected)
+ {
+ var sut = new WildcardPatternMatcher(wildcardPattern);
+ sut.IsMatch(value).Should().Be(expected);
+ }
+}