allow '<' & '>' to be used in strings

This commit is contained in:
Arnout Born 2022-06-03 12:04:24 +02:00
parent b73f748dba
commit 3c00b4decb
3 changed files with 382 additions and 28 deletions

View File

@ -22,19 +22,20 @@ namespace Sieve.Models
{ {
set set
{ {
var filterSplits = Regex.Split(value, EscapeNegPatternForOper).Select(t => t.Trim()).ToArray(); var filterSplits = Regex.Split(value, EscapeNegPatternForOper);
Names = Regex.Split(filterSplits[0], EscapedPipePattern).Select(t => t.Trim()).ToArray(); Names = Regex.Split(filterSplits[0].Trim(), EscapedPipePattern).Select(t => t.Trim()).ToArray();
if (filterSplits.Length > 2) if (filterSplits.Length > 2)
{ {
foreach (var match in Regex.Matches(filterSplits[2], EscapePosPatternForOper)) var filterValue = string.Join("", filterSplits.Skip(2)).Trim();
foreach (var match in Regex.Matches(filterValue, EscapePosPatternForOper))
{ {
var matchStr = match.ToString(); var matchStr = match.ToString();
filterSplits[2] = filterSplits[2].Replace('\\' + matchStr, matchStr); filterValue = filterValue.Replace('\\' + matchStr, matchStr);
} }
Values = Regex.Split(filterSplits[2], EscapedPipePattern) Values = Regex.Split(filterValue, EscapedPipePattern)
.Select(UnEscape) .Select(UnEscape)
.ToArray(); .ToArray();
} }

View File

@ -0,0 +1,334 @@
using System;
using System.Collections.Generic;
using System.Linq;
using Sieve.Exceptions;
using Sieve.Models;
using Sieve.Services;
using SieveUnitTests.Entities;
using SieveUnitTests.Services;
using Xunit;
using Xunit.Abstractions;
namespace SieveUnitTests
{
public class GeneralSpecialChars
{
private readonly ITestOutputHelper _testOutputHelper;
private readonly SieveProcessor _processor;
private readonly SieveProcessor _nullableProcessor;
private readonly IQueryable<Post> _posts;
private readonly IQueryable<Comment> _comments;
public GeneralSpecialChars(ITestOutputHelper testOutputHelper)
{
var nullableAccessor = new SieveOptionsAccessor();
nullableAccessor.Value.IgnoreNullsOnNotEqual = false;
_testOutputHelper = testOutputHelper;
_processor = new ApplicationSieveProcessor(new SieveOptionsAccessor(),
new SieveCustomSortMethods(),
new SieveCustomFilterMethods());
_nullableProcessor = new ApplicationSieveProcessor(nullableAccessor,
new SieveCustomSortMethods(),
new SieveCustomFilterMethods());
_posts = new List<Post>
{
new Post
{
Id = 0,
Title = "A<>1",
LikeCount = 100,
IsDraft = true,
CategoryId = null,
TopComment = new Comment { Id = 0, Text = "A1" },
FeaturedComment = new Comment { Id = 4, Text = "A2" }
},
new Post
{
Id = 1,
Title = "B>2",
LikeCount = 50,
IsDraft = false,
CategoryId = 1,
TopComment = new Comment { Id = 3, Text = "B1" },
FeaturedComment = new Comment { Id = 5, Text = "B2" }
},
new Post
{
Id = 2,
Title = "C==E",
LikeCount = 0,
CategoryId = 1,
TopComment = new Comment { Id = 2, Text = "C1" },
FeaturedComment = new Comment { Id = 6, Text = "C2" }
},
new Post
{
Id = 3,
Title = "D@=k",
LikeCount = 3,
IsDraft = true,
CategoryId = 2,
TopComment = new Comment { Id = 1, Text = "D1" },
FeaturedComment = new Comment { Id = 7, Text = "D2" }
},
new Post
{
Id = 4,
Title = "Yen!=Yin",
LikeCount = 5,
IsDraft = true,
CategoryId = 5,
TopComment = new Comment { Id = 4, Text = "Yen3" },
FeaturedComment = new Comment { Id = 8, Text = "Yen4" }
}
}.AsQueryable();
_comments = new List<Comment>
{
new Comment
{
Id = 0,
DateCreated = DateTimeOffset.UtcNow.AddDays(-20),
Text = "This is an old comment."
},
new Comment
{
Id = 1,
DateCreated = DateTimeOffset.UtcNow.AddDays(-1),
Text = "This is a fairly new comment."
},
new Comment
{
Id = 2,
DateCreated = DateTimeOffset.UtcNow,
Text = "This is a brand new comment. (Text in braces, comma separated)"
},
}.AsQueryable();
}
[Fact]
public void ContainsCanBeCaseInsensitive()
{
var model = new SieveModel
{
Filters = "Title@=*a<"
};
var result = _processor.Apply(model, _posts);
Assert.Equal(0, result.First().Id);
Assert.True(result.Count() == 1);
}
[Fact]
public void NotEqualsCanBeCaseInsensitive()
{
var model = new SieveModel
{
Filters = "Title!=*a<>1"
};
var result = _processor.Apply(model, _posts);
Assert.Equal(1, result.First().Id);
Assert.True(result.Count() == 4);
}
[Fact]
public void EndsWithWorks()
{
var model = new SieveModel
{
Filters = "Title_-=n"
};
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].Values.ToString());
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].Operator);
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].OperatorParsed.ToString());
var result = _processor.Apply(model, _posts);
Assert.Equal(4, result.First().Id);
Assert.True(result.Count() == 1);
}
[Fact]
public void EndsWithCanBeCaseInsensitive()
{
var model = new SieveModel
{
Filters = "Title_-=*N"
};
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].Values.ToString());
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].Operator);
_testOutputHelper.WriteLine(model.GetFiltersParsed()[0].OperatorParsed.ToString());
var result = _processor.Apply(model, _posts);
Assert.Equal(4, result.First().Id);
Assert.True(result.Count() == 1);
}
[Fact]
public void ContainsIsCaseSensitive()
{
var model = new SieveModel
{
Filters = "Title@=a",
};
var result = _processor.Apply(model, _posts);
Assert.True(!result.Any());
}
[Fact]
public void NotContainsWorks()
{
var model = new SieveModel
{
Filters = "Title!@=D",
};
var result = _processor.Apply(model, _posts);
Assert.True(result.Count() == 4);
}
[Theory]
[InlineData(@"Text@=*\,")]
[InlineData(@"Text@=*\, ")]
[InlineData(@"Text@=*braces\,")]
[InlineData(@"Text@=*braces\, comma")]
public void CanFilterWithEscapedComma(string filter)
{
var model = new SieveModel
{
Filters = filter
};
var result = _processor.Apply(model, _comments);
Assert.True(result.Count() == 1);
}
[Fact]
public void CustomFiltersWithOperatorsWork()
{
var model = new SieveModel
{
Filters = "HasInTitle==A",
};
var result = _processor.Apply(model, _posts);
Assert.True(result.Any(p => p.Id == 0));
Assert.True(result.Count() == 1);
}
[Fact]
public void CustomFiltersMixedWithUsualWork1()
{
var model = new SieveModel
{
Filters = "Isnew,CategoryId==2",
};
var result = _processor.Apply(model, _posts);
Assert.True(result.Any(p => p.Id == 3));
Assert.True(result.Count() == 1);
}
[Fact]
public void CombinedAndOrWithSpaceFilteringWorks()
{
var model = new SieveModel
{
Filters = "Title==D@=k, (Title|LikeCount)==3",
};
var result = _processor.Apply(model, _posts);
var entry = result.FirstOrDefault();
var resultCount = result.Count();
Assert.NotNull(entry);
Assert.Equal(1, resultCount);
Assert.Equal(3, entry.Id);
}
[Fact]
public void OrValueFilteringWorks()
{
var model = new SieveModel
{
Filters = "Title==C==E|D@=k",
};
var result = _processor.Apply(model, _posts);
Assert.Equal(2, result.Count());
Assert.True(result.Any(p => p.Id == 2));
Assert.True(result.Any(p => p.Id == 3));
}
[Fact]
public void OrValueFilteringWorks2()
{
var model = new SieveModel
{
Filters = "Text@=(|)",
};
var result = _processor.Apply(model, _comments);
Assert.Equal(1, result.Count());
Assert.Equal(2, result.FirstOrDefault()?.Id);
}
[Fact]
public void NestedFilteringWorks()
{
var model = new SieveModel
{
Filters = "TopComment.Text!@=A",
};
var result = _processor.Apply(model, _posts);
Assert.Equal(4, result.Count());
var posts = result.ToList();
Assert.Contains("B", posts[0].TopComment.Text);
Assert.Contains("C", posts[1].TopComment.Text);
Assert.Contains("D", posts[2].TopComment.Text);
Assert.Contains("Yen", posts[3].TopComment.Text);
}
[Fact]
public void FilteringNullsWorks()
{
var posts = new List<Post>
{
new Post
{
Id = 1,
Title = null,
LikeCount = 0,
IsDraft = false,
CategoryId = null,
TopComment = null,
FeaturedComment = null
},
}.AsQueryable();
var model = new SieveModel
{
Filters = "FeaturedComment.Text!@=Some value",
};
var result = _processor.Apply(model, posts);
Assert.Equal(0, result.Count());
}
}
}

View File

@ -30,23 +30,30 @@ namespace SieveUnitTests
new Comment new Comment
{ {
Id = 1, Id = 1,
DateCreated = DateTimeOffset.UtcNow, DateCreated = DateTimeOffset.UtcNow,
Text = "null is here twice in the text ending by null", Text = "null is here twice in the text ending by null",
Author = "Cat", Author = "Cat",
}, },
new Comment new Comment
{ {
Id = 2, Id = 2,
DateCreated = DateTimeOffset.UtcNow, DateCreated = DateTimeOffset.UtcNow,
Text = "Regular comment without n*ll", Text = "Regular comment without n*ll",
Author = "Mouse", Author = "Mouse",
}, },
new Comment new Comment
{ {
Id = 100, Id = 100,
DateCreated = DateTimeOffset.UtcNow, DateCreated = DateTimeOffset.UtcNow,
Text = null, Text = null,
Author = "null", Author = "null",
},
new Comment
{
Id = 105,
DateCreated = DateTimeOffset.UtcNow,
Text = "The duck wrote this",
Author = "Duck <5",
} }
}.AsQueryable(); }.AsQueryable();
} }
@ -56,23 +63,35 @@ namespace SieveUnitTests
[InlineData("Text==*null")] [InlineData("Text==*null")]
public void Filter_Equals_Null(string filter) public void Filter_Equals_Null(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(100, result.Single().Id); Assert.Equal(100, result.Single().Id);
} }
[Theory]
[InlineData("Author==Duck <5")]
[InlineData("Text|Author==Duck <5")]
public void Filter_Equals_WithSpecialChar(string filter)
{
var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments);
Assert.Equal(105, result.Single().Id);
}
[Theory] [Theory]
[InlineData("Text!=null")] [InlineData("Text!=null")]
[InlineData("Text!=*null")] [InlineData("Text!=*null")]
public void Filter_NotEquals_Null(string filter) public void Filter_NotEquals_Null(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] {0, 1, 2}, result.Select(p => p.Id)); Assert.Equal(new[] { 0, 1, 2, 105 }, result.Select(p => p.Id));
} }
[Theory] [Theory]
@ -83,13 +102,13 @@ namespace SieveUnitTests
[InlineData("Text@=*null|text")] [InlineData("Text@=*null|text")]
public void Filter_Contains_NullString(string filter) public void Filter_Contains_NullString(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] {0, 1}, result.Select(p => p.Id)); Assert.Equal(new[] { 0, 1 }, result.Select(p => p.Id));
} }
[Theory] [Theory]
[InlineData("Text|Author==null", 100)] [InlineData("Text|Author==null", 100)]
[InlineData("Text|Author@=null", 0, 1, 100)] [InlineData("Text|Author@=null", 0, 1, 100)]
@ -98,7 +117,7 @@ namespace SieveUnitTests
[InlineData("Text|Author_=*null", 1, 100)] [InlineData("Text|Author_=*null", 1, 100)]
public void MultiFilter_Contains_NullString(string filter, params int[] expectedIds) public void MultiFilter_Contains_NullString(string filter, params int[] expectedIds)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
@ -109,12 +128,12 @@ namespace SieveUnitTests
[InlineData(@"Author==\null", 100)] [InlineData(@"Author==\null", 100)]
[InlineData(@"Author==*\null", 100)] [InlineData(@"Author==*\null", 100)]
[InlineData(@"Author==*\NuLl", 100)] [InlineData(@"Author==*\NuLl", 100)]
[InlineData(@"Author!=*\null", 0, 1, 2)] [InlineData(@"Author!=*\null", 0, 1, 2, 105)]
[InlineData(@"Author!=*\NulL", 0, 1, 2)] [InlineData(@"Author!=*\NulL", 0, 1, 2, 105)]
[InlineData(@"Author!=\null", 0, 1, 2)] [InlineData(@"Author!=\null", 0, 1, 2, 105)]
public void SingleFilter_Equals_NullStringEscaped(string filter, params int[] expectedIds) public void SingleFilter_Equals_NullStringEscaped(string filter, params int[] expectedIds)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
@ -129,11 +148,11 @@ namespace SieveUnitTests
[InlineData("Text_=*null|text")] [InlineData("Text_=*null|text")]
public void Filter_StartsWith_NullString(string filter) public void Filter_StartsWith_NullString(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] {1}, result.Select(p => p.Id)); Assert.Equal(new[] { 1 }, result.Select(p => p.Id));
} }
[Theory] [Theory]
@ -159,11 +178,11 @@ namespace SieveUnitTests
[InlineData("Text!@=*null|text")] [InlineData("Text!@=*null|text")]
public void Filter_DoesNotContain_NullString(string filter) public void Filter_DoesNotContain_NullString(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] {2}, result.Select(p => p.Id)); Assert.Equal(new[] { 2, 105 }, result.Select(p => p.Id));
} }
[Theory] [Theory]
@ -173,11 +192,11 @@ namespace SieveUnitTests
[InlineData("Text!_=*NulL")] [InlineData("Text!_=*NulL")]
public void Filter_DoesNotStartsWith_NullString(string filter) public void Filter_DoesNotStartsWith_NullString(string filter)
{ {
var model = new SieveModel {Filters = filter}; var model = new SieveModel { Filters = filter };
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] {0, 2}, result.Select(p => p.Id)); Assert.Equal(new[] { 0, 2, 105 }, result.Select(p => p.Id));
} }
[Theory] [Theory]
@ -191,7 +210,7 @@ namespace SieveUnitTests
var result = _processor.Apply(model, _comments); var result = _processor.Apply(model, _comments);
Assert.Equal(new[] { 0, 2 }, result.Select(p => p.Id)); Assert.Equal(new[] { 0, 2, 105 }, result.Select(p => p.Id));
} }
} }
} }