mirror of
https://github.com/Biarity/Sieve.git
synced 2025-09-19 14:49:39 +02:00
Compare commits
15 Commits
v2.5.0
...
releases/3
Author | SHA1 | Date | |
---|---|---|---|
|
aedbc1ed96 | ||
|
2c9d907764 | ||
|
8bd9ce85d9 | ||
|
f738e3bf1e | ||
|
dd1b0a9edc | ||
|
27838b062c | ||
|
38af9af982 | ||
|
d188bed4f0 | ||
|
1e29271fd9 | ||
|
034730bffb | ||
|
79c825cb7a | ||
|
028ab1d196 | ||
|
9277690e96 | ||
|
d5474478b3 | ||
|
4c5510772a |
38
.github/ISSUE_TEMPLATE/bug_report.md
vendored
38
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -1,38 +0,0 @@
|
|||||||
---
|
|
||||||
name: Bug report
|
|
||||||
about: Create a report to help us improve
|
|
||||||
title: ''
|
|
||||||
labels: ''
|
|
||||||
assignees: ''
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
**Describe the bug**
|
|
||||||
A clear and concise description of what the bug is.
|
|
||||||
|
|
||||||
**To Reproduce**
|
|
||||||
Steps to reproduce the behavior:
|
|
||||||
1. Go to '...'
|
|
||||||
2. Click on '....'
|
|
||||||
3. Scroll down to '....'
|
|
||||||
4. See error
|
|
||||||
|
|
||||||
**Expected behavior**
|
|
||||||
A clear and concise description of what you expected to happen.
|
|
||||||
|
|
||||||
**Screenshots**
|
|
||||||
If applicable, add screenshots to help explain your problem.
|
|
||||||
|
|
||||||
**Desktop (please complete the following information):**
|
|
||||||
- OS: [e.g. iOS]
|
|
||||||
- Browser [e.g. chrome, safari]
|
|
||||||
- Version [e.g. 22]
|
|
||||||
|
|
||||||
**Smartphone (please complete the following information):**
|
|
||||||
- Device: [e.g. iPhone6]
|
|
||||||
- OS: [e.g. iOS8.1]
|
|
||||||
- Browser [e.g. stock browser, safari]
|
|
||||||
- Version [e.g. 22]
|
|
||||||
|
|
||||||
**Additional context**
|
|
||||||
Add any other context about the problem here.
|
|
5
.github/ISSUE_TEMPLATE/config.yml
vendored
5
.github/ISSUE_TEMPLATE/config.yml
vendored
@@ -1,5 +0,0 @@
|
|||||||
blank_issues_enabled: false
|
|
||||||
contact_links:
|
|
||||||
- name: Feature Request
|
|
||||||
url: https://github.com/biarity/sieve/discussions/new
|
|
||||||
about: Share your ideas on how to make Sieve better.
|
|
1
.github/workflows/ci_publish.yml
vendored
1
.github/workflows/ci_publish.yml
vendored
@@ -19,7 +19,6 @@ name: ci_publish
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
|
||||||
- 'releases/*'
|
- 'releases/*'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
@@ -2,8 +2,7 @@
|
|||||||
⚗️ Sieve is a simple, clean, and extensible framework for .NET Core that **adds sorting, filtering, and pagination functionality out of the box**.
|
⚗️ Sieve is a simple, clean, and extensible framework for .NET Core that **adds sorting, filtering, and pagination functionality out of the box**.
|
||||||
Most common use case would be for serving ASP.NET Core GET queries.
|
Most common use case would be for serving ASP.NET Core GET queries.
|
||||||
|
|
||||||
[](https://www.nuget.org/packages/Sieve)
|
[](https://www.nuget.org/packages/Sieve)
|
||||||
[](https://www.nuget.org/packages/Sieve)
|
|
||||||
|
|
||||||
[Get Sieve on nuget](https://www.nuget.org/packages/Sieve/)
|
[Get Sieve on nuget](https://www.nuget.org/packages/Sieve/)
|
||||||
|
|
||||||
@@ -75,7 +74,7 @@ Where `SieveCustomSortMethodsOfPosts` for example is:
|
|||||||
```C#
|
```C#
|
||||||
public class SieveCustomSortMethods : ISieveCustomSortMethods
|
public class SieveCustomSortMethods : ISieveCustomSortMethods
|
||||||
{
|
{
|
||||||
public IQueryable<Post> Popularity(IQueryable<Post> source, bool useThenBy, bool desc) // The method is given an indicator of whether to use ThenBy(), and if the query is descending
|
public IQueryable<Post> Popularity(IQueryable<Post> source, bool useThenBy, bool desc) // The method is given an indicator of weather to use ThenBy(), and if the query is descending
|
||||||
{
|
{
|
||||||
var result = useThenBy ?
|
var result = useThenBy ?
|
||||||
((IOrderedQueryable<Post>)source).ThenBy(p => p.LikeCount) : // ThenBy only works on IOrderedQueryable<TEntity>
|
((IOrderedQueryable<Post>)source).ThenBy(p => p.LikeCount) : // ThenBy only works on IOrderedQueryable<TEntity>
|
||||||
@@ -128,8 +127,7 @@ Then you can add the configuration:
|
|||||||
"CaseSensitive": "boolean: should property names be case-sensitive? Defaults to false",
|
"CaseSensitive": "boolean: should property names be case-sensitive? Defaults to false",
|
||||||
"DefaultPageSize": "int number: optional number to fallback to when no page argument is given. Set <=0 to disable paging if no pageSize is specified (default).",
|
"DefaultPageSize": "int number: optional number to fallback to when no page argument is given. Set <=0 to disable paging if no pageSize is specified (default).",
|
||||||
"MaxPageSize": "int number: maximum allowed page size. Set <=0 to make infinite (default)",
|
"MaxPageSize": "int number: maximum allowed page size. Set <=0 to make infinite (default)",
|
||||||
"ThrowExceptions": "boolean: should Sieve throw exceptions instead of silently failing? Defaults to false",
|
"ThrowExceptions": "boolean: should Sieve throw exceptions instead of silently failing? Defaults to false"
|
||||||
"IgnoreNullsOnNotEqual": "boolean: ignore null values when filtering using is not equal operator? Default to true"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
{
|
{
|
||||||
"Logging": {
|
"Logging": {
|
||||||
|
"IncludeScopes": false,
|
||||||
"LogLevel": {
|
"LogLevel": {
|
||||||
"Default": "Debug",
|
"Default": "Debug",
|
||||||
"System": "Information",
|
"System": "Information",
|
||||||
|
@@ -4,10 +4,10 @@
|
|||||||
},
|
},
|
||||||
"Sieve": {
|
"Sieve": {
|
||||||
"CaseSensitive": false,
|
"CaseSensitive": false,
|
||||||
"DefaultPageSize": 10,
|
"DefaultPageSize": 10
|
||||||
"IgnoreNullsOnNotEqual": true
|
|
||||||
},
|
},
|
||||||
"Logging": {
|
"Logging": {
|
||||||
|
"IncludeScopes": false,
|
||||||
"Debug": {
|
"Debug": {
|
||||||
"LogLevel": {
|
"LogLevel": {
|
||||||
"Default": "Warning"
|
"Default": "Warning"
|
||||||
|
@@ -9,7 +9,6 @@ namespace Sieve.Models
|
|||||||
public FilterTerm() { }
|
public FilterTerm() { }
|
||||||
|
|
||||||
private const string EscapedPipePattern = @"(?<!($|[^\\])(\\\\)*?\\)\|";
|
private const string EscapedPipePattern = @"(?<!($|[^\\])(\\\\)*?\\)\|";
|
||||||
private const string PipeToEscape = @"\|";
|
|
||||||
|
|
||||||
private static readonly string[] Operators = new string[] {
|
private static readonly string[] Operators = new string[] {
|
||||||
"!@=*",
|
"!@=*",
|
||||||
@@ -37,11 +36,7 @@ namespace Sieve.Models
|
|||||||
var filterSplits = value.Split(Operators, StringSplitOptions.RemoveEmptyEntries)
|
var filterSplits = value.Split(Operators, StringSplitOptions.RemoveEmptyEntries)
|
||||||
.Select(t => t.Trim()).ToArray();
|
.Select(t => t.Trim()).ToArray();
|
||||||
Names = Regex.Split(filterSplits[0], EscapedPipePattern).Select(t => t.Trim()).ToArray();
|
Names = Regex.Split(filterSplits[0], EscapedPipePattern).Select(t => t.Trim()).ToArray();
|
||||||
Values = filterSplits.Length > 1
|
Values = filterSplits.Length > 1 ? Regex.Split(filterSplits[1], EscapedPipePattern).Select(t => t.Trim()).ToArray() : null;
|
||||||
? Regex.Split(filterSplits[1], EscapedPipePattern)
|
|
||||||
.Select(t => t.Replace(PipeToEscape, "|").Trim())
|
|
||||||
.ToArray()
|
|
||||||
: null;
|
|
||||||
Operator = Array.Find(Operators, o => value.Contains(o)) ?? "==";
|
Operator = Array.Find(Operators, o => value.Contains(o)) ?? "==";
|
||||||
OperatorParsed = GetOperatorParsed(Operator);
|
OperatorParsed = GetOperatorParsed(Operator);
|
||||||
OperatorIsCaseInsensitive = Operator.EndsWith("*");
|
OperatorIsCaseInsensitive = Operator.EndsWith("*");
|
||||||
@@ -95,5 +90,6 @@ namespace Sieve.Models
|
|||||||
&& Values.SequenceEqual(other.Values)
|
&& Values.SequenceEqual(other.Values)
|
||||||
&& Operator == other.Operator;
|
&& Operator == other.Operator;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
@@ -2,14 +2,27 @@
|
|||||||
{
|
{
|
||||||
public class SieveOptions
|
public class SieveOptions
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// If flag is set, property names have to match including case sensitivity.
|
||||||
|
/// </summary>
|
||||||
public bool CaseSensitive { get; set; } = false;
|
public bool CaseSensitive { get; set; } = false;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Fallback value of no page size is specified in the request.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>Values less or equal to 0 disable paging.</remarks>
|
||||||
public int DefaultPageSize { get; set; } = 0;
|
public int DefaultPageSize { get; set; } = 0;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Specifies the upper limit of a page size to be requested.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>Values less or equal to 0 are ignored.</remarks>
|
||||||
public int MaxPageSize { get; set; } = 0;
|
public int MaxPageSize { get; set; } = 0;
|
||||||
|
|
||||||
public bool ThrowExceptions { get; set; } = false;
|
/// <summary>
|
||||||
|
/// If flag is set, Sieve throws exception otherwise exceptions are caught and the already processed
|
||||||
public bool IgnoreNullsOnNotEqual { get; set; } = true;
|
/// result is returned.
|
||||||
|
/// </summary>
|
||||||
|
public bool ThrowExceptions { get; set; } = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -68,6 +68,7 @@ namespace Sieve.Services
|
|||||||
where TSortTerm : ISortTerm, new()
|
where TSortTerm : ISortTerm, new()
|
||||||
{
|
{
|
||||||
private const string NullFilterValue = "null";
|
private const string NullFilterValue = "null";
|
||||||
|
private readonly IOptions<SieveOptions> _options;
|
||||||
private readonly ISieveCustomSortMethods _customSortMethods;
|
private readonly ISieveCustomSortMethods _customSortMethods;
|
||||||
private readonly ISieveCustomFilterMethods _customFilterMethods;
|
private readonly ISieveCustomFilterMethods _customFilterMethods;
|
||||||
private readonly SievePropertyMapper _mapper = new SievePropertyMapper();
|
private readonly SievePropertyMapper _mapper = new SievePropertyMapper();
|
||||||
@@ -77,7 +78,7 @@ namespace Sieve.Services
|
|||||||
ISieveCustomFilterMethods customFilterMethods)
|
ISieveCustomFilterMethods customFilterMethods)
|
||||||
{
|
{
|
||||||
_mapper = MapProperties(_mapper);
|
_mapper = MapProperties(_mapper);
|
||||||
Options = options;
|
_options = options;
|
||||||
_customSortMethods = customSortMethods;
|
_customSortMethods = customSortMethods;
|
||||||
_customFilterMethods = customFilterMethods;
|
_customFilterMethods = customFilterMethods;
|
||||||
}
|
}
|
||||||
@@ -86,7 +87,7 @@ namespace Sieve.Services
|
|||||||
ISieveCustomSortMethods customSortMethods)
|
ISieveCustomSortMethods customSortMethods)
|
||||||
{
|
{
|
||||||
_mapper = MapProperties(_mapper);
|
_mapper = MapProperties(_mapper);
|
||||||
Options = options;
|
_options = options;
|
||||||
_customSortMethods = customSortMethods;
|
_customSortMethods = customSortMethods;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,18 +95,16 @@ namespace Sieve.Services
|
|||||||
ISieveCustomFilterMethods customFilterMethods)
|
ISieveCustomFilterMethods customFilterMethods)
|
||||||
{
|
{
|
||||||
_mapper = MapProperties(_mapper);
|
_mapper = MapProperties(_mapper);
|
||||||
Options = options;
|
_options = options;
|
||||||
_customFilterMethods = customFilterMethods;
|
_customFilterMethods = customFilterMethods;
|
||||||
}
|
}
|
||||||
|
|
||||||
public SieveProcessor(IOptions<SieveOptions> options)
|
public SieveProcessor(IOptions<SieveOptions> options)
|
||||||
{
|
{
|
||||||
_mapper = MapProperties(_mapper);
|
_mapper = MapProperties(_mapper);
|
||||||
Options = options;
|
_options = options;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected IOptions<SieveOptions> Options { get; }
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Apply filtering, sorting, and pagination parameters found in `model` to `source`
|
/// Apply filtering, sorting, and pagination parameters found in `model` to `source`
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -149,7 +148,7 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
if (!Options.Value.ThrowExceptions)
|
if (!_options.Value.ThrowExceptions)
|
||||||
{
|
{
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -163,7 +162,7 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual IQueryable<TEntity> ApplyFiltering<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
private IQueryable<TEntity> ApplyFiltering<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
||||||
object[] dataForCustomMethods = null)
|
object[] dataForCustomMethods = null)
|
||||||
{
|
{
|
||||||
if (model?.GetFiltersParsed() == null)
|
if (model?.GetFiltersParsed() == null)
|
||||||
@@ -217,14 +216,11 @@ namespace Sieve.Services
|
|||||||
expression = Expression.Not(expression);
|
expression = Expression.Not(expression);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (expression.NodeType != ExpressionType.NotEqual || Options.Value.IgnoreNullsOnNotEqual)
|
|
||||||
{
|
|
||||||
var filterValueNullCheck = GetFilterValueNullCheck(parameter, fullPropertyName, isFilterTermValueNull);
|
var filterValueNullCheck = GetFilterValueNullCheck(parameter, fullPropertyName, isFilterTermValueNull);
|
||||||
if (filterValueNullCheck != null)
|
if (filterValueNullCheck != null)
|
||||||
{
|
{
|
||||||
expression = Expression.AndAlso(filterValueNullCheck, expression);
|
expression = Expression.AndAlso(filterValueNullCheck, expression);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
innerExpression = innerExpression == null
|
innerExpression = innerExpression == null
|
||||||
? expression
|
? expression
|
||||||
@@ -257,7 +253,8 @@ namespace Sieve.Services
|
|||||||
: result.Where(Expression.Lambda<Func<TEntity, bool>>(outerExpression, parameter));
|
: result.Where(Expression.Lambda<Func<TEntity, bool>>(outerExpression, parameter));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Expression GetFilterValueNullCheck(Expression parameter, string fullPropertyName, bool isFilterTermValueNull)
|
private static Expression GetFilterValueNullCheck(Expression parameter, string fullPropertyName,
|
||||||
|
bool isFilterTermValueNull)
|
||||||
{
|
{
|
||||||
var (propertyValue, nullCheck) = GetPropertyValueAndNullCheckExpression(parameter, fullPropertyName);
|
var (propertyValue, nullCheck) = GetPropertyValueAndNullCheckExpression(parameter, fullPropertyName);
|
||||||
|
|
||||||
@@ -339,13 +336,15 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Workaround to ensure that the filter value gets passed as a parameter in generated SQL from EF Core
|
// Workaround to ensure that the filter value gets passed as a parameter in generated SQL from EF Core
|
||||||
|
// See https://github.com/aspnet/EntityFrameworkCore/issues/3361
|
||||||
|
// Expression.Constant passed the target type to allow Nullable comparison
|
||||||
|
// See http://bradwilson.typepad.com/blog/2008/07/creating-nullab.html
|
||||||
private static Expression GetClosureOverConstant<T>(T constant, Type targetType)
|
private static Expression GetClosureOverConstant<T>(T constant, Type targetType)
|
||||||
{
|
{
|
||||||
Expression<Func<T>> hoistedConstant = () => constant;
|
return Expression.Constant(constant, targetType);
|
||||||
return Expression.Convert(hoistedConstant.Body, targetType);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual IQueryable<TEntity> ApplySorting<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
private IQueryable<TEntity> ApplySorting<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
||||||
object[] dataForCustomMethods = null)
|
object[] dataForCustomMethods = null)
|
||||||
{
|
{
|
||||||
if (model?.GetSortsParsed() == null)
|
if (model?.GetSortsParsed() == null)
|
||||||
@@ -374,11 +373,15 @@ namespace Sieve.Services
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual IQueryable<TEntity> ApplyPagination<TEntity>(TSieveModel model, IQueryable<TEntity> result)
|
private IQueryable<TEntity> ApplyPagination<TEntity>(TSieveModel model, IQueryable<TEntity> result)
|
||||||
{
|
{
|
||||||
var page = model?.Page ?? 1;
|
var page = model?.Page ?? 1;
|
||||||
var pageSize = model?.PageSize ?? Options.Value.DefaultPageSize;
|
var pageSize = model?.PageSize ?? _options.Value.DefaultPageSize;
|
||||||
var maxPageSize = Options.Value.MaxPageSize > 0 ? Options.Value.MaxPageSize : pageSize;
|
|
||||||
|
if (_options.Value.MaxPageSize > 0)
|
||||||
|
{
|
||||||
|
pageSize = Math.Min(pageSize, _options.Value.MaxPageSize);
|
||||||
|
}
|
||||||
|
|
||||||
if (pageSize <= 0)
|
if (pageSize <= 0)
|
||||||
{
|
{
|
||||||
@@ -386,7 +389,7 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
|
|
||||||
result = result.Skip((page - 1) * pageSize);
|
result = result.Skip((page - 1) * pageSize);
|
||||||
result = result.Take(Math.Min(pageSize, maxPageSize));
|
result = result.Take(pageSize);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -400,14 +403,14 @@ namespace Sieve.Services
|
|||||||
string name)
|
string name)
|
||||||
{
|
{
|
||||||
var property = _mapper.FindProperty<TEntity>(canSortRequired, canFilterRequired, name,
|
var property = _mapper.FindProperty<TEntity>(canSortRequired, canFilterRequired, name,
|
||||||
Options.Value.CaseSensitive);
|
_options.Value.CaseSensitive);
|
||||||
if (property.Item1 != null)
|
if (property.Item1 != null)
|
||||||
{
|
{
|
||||||
return property;
|
return property;
|
||||||
}
|
}
|
||||||
|
|
||||||
var prop = FindPropertyBySieveAttribute<TEntity>(canSortRequired, canFilterRequired, name,
|
var prop = FindPropertyBySieveAttribute<TEntity>(canSortRequired, canFilterRequired, name,
|
||||||
Options.Value.CaseSensitive);
|
_options.Value.CaseSensitive);
|
||||||
return (prop?.Name, prop);
|
return (prop?.Name, prop);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -427,7 +430,7 @@ namespace Sieve.Services
|
|||||||
{
|
{
|
||||||
var customMethod = parent?.GetType()
|
var customMethod = parent?.GetType()
|
||||||
.GetMethodExt(name,
|
.GetMethodExt(name,
|
||||||
Options.Value.CaseSensitive
|
_options.Value.CaseSensitive
|
||||||
? BindingFlags.Default
|
? BindingFlags.Default
|
||||||
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance,
|
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance,
|
||||||
typeof(IQueryable<TEntity>));
|
typeof(IQueryable<TEntity>));
|
||||||
@@ -438,7 +441,7 @@ namespace Sieve.Services
|
|||||||
// Find generic methods `public IQueryable<T> Filter<T>(IQueryable<T> source, ...)`
|
// Find generic methods `public IQueryable<T> Filter<T>(IQueryable<T> source, ...)`
|
||||||
var genericCustomMethod = parent?.GetType()
|
var genericCustomMethod = parent?.GetType()
|
||||||
.GetMethodExt(name,
|
.GetMethodExt(name,
|
||||||
Options.Value.CaseSensitive
|
_options.Value.CaseSensitive
|
||||||
? BindingFlags.Default
|
? BindingFlags.Default
|
||||||
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance,
|
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance,
|
||||||
typeof(IQueryable<>));
|
typeof(IQueryable<>));
|
||||||
@@ -482,11 +485,11 @@ namespace Sieve.Services
|
|||||||
var incompatibleCustomMethods =
|
var incompatibleCustomMethods =
|
||||||
parent?
|
parent?
|
||||||
.GetType()
|
.GetType()
|
||||||
.GetMethods(Options.Value.CaseSensitive
|
.GetMethods(_options.Value.CaseSensitive
|
||||||
? BindingFlags.Default
|
? BindingFlags.Default
|
||||||
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance)
|
: BindingFlags.IgnoreCase | BindingFlags.Public | BindingFlags.Instance)
|
||||||
.Where(method => string.Equals(method.Name, name,
|
.Where(method => string.Equals(method.Name, name,
|
||||||
Options.Value.CaseSensitive
|
_options.Value.CaseSensitive
|
||||||
? StringComparison.InvariantCulture
|
? StringComparison.InvariantCulture
|
||||||
: StringComparison.InvariantCultureIgnoreCase))
|
: StringComparison.InvariantCultureIgnoreCase))
|
||||||
.ToList()
|
.ToList()
|
||||||
|
@@ -15,24 +15,16 @@ namespace SieveUnitTests
|
|||||||
{
|
{
|
||||||
private readonly ITestOutputHelper _testOutputHelper;
|
private readonly ITestOutputHelper _testOutputHelper;
|
||||||
private readonly SieveProcessor _processor;
|
private readonly SieveProcessor _processor;
|
||||||
private readonly SieveProcessor _nullableProcessor;
|
|
||||||
private readonly IQueryable<Post> _posts;
|
private readonly IQueryable<Post> _posts;
|
||||||
private readonly IQueryable<Comment> _comments;
|
private readonly IQueryable<Comment> _comments;
|
||||||
|
|
||||||
public General(ITestOutputHelper testOutputHelper)
|
public General(ITestOutputHelper testOutputHelper)
|
||||||
{
|
{
|
||||||
var nullableAccessor = new SieveOptionsAccessor();
|
|
||||||
nullableAccessor.Value.IgnoreNullsOnNotEqual = false;
|
|
||||||
|
|
||||||
_testOutputHelper = testOutputHelper;
|
_testOutputHelper = testOutputHelper;
|
||||||
_processor = new ApplicationSieveProcessor(new SieveOptionsAccessor(),
|
_processor = new ApplicationSieveProcessor(new SieveOptionsAccessor(),
|
||||||
new SieveCustomSortMethods(),
|
new SieveCustomSortMethods(),
|
||||||
new SieveCustomFilterMethods());
|
new SieveCustomFilterMethods());
|
||||||
|
|
||||||
_nullableProcessor = new ApplicationSieveProcessor(nullableAccessor,
|
|
||||||
new SieveCustomSortMethods(),
|
|
||||||
new SieveCustomFilterMethods());
|
|
||||||
|
|
||||||
_posts = new List<Post>
|
_posts = new List<Post>
|
||||||
{
|
{
|
||||||
new Post
|
new Post
|
||||||
@@ -188,25 +180,8 @@ namespace SieveUnitTests
|
|||||||
};
|
};
|
||||||
|
|
||||||
var result = _processor.Apply(model, _posts);
|
var result = _processor.Apply(model, _posts);
|
||||||
var nullableResult = _nullableProcessor.Apply(model, _posts);
|
|
||||||
|
|
||||||
Assert.True(result.Count() == 2);
|
Assert.True(result.Count() == 2);
|
||||||
Assert.True(nullableResult.Count() == 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void CanFilterNullableIntsWithNotEqual()
|
|
||||||
{
|
|
||||||
var model = new SieveModel()
|
|
||||||
{
|
|
||||||
Filters = "CategoryId!=1"
|
|
||||||
};
|
|
||||||
|
|
||||||
var result = _processor.Apply(model, _posts);
|
|
||||||
var nullableResult = _nullableProcessor.Apply(model, _posts);
|
|
||||||
|
|
||||||
Assert.True(result.Count() == 1);
|
|
||||||
Assert.True(nullableResult.Count() == 2);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Theory]
|
[Theory]
|
||||||
@@ -639,60 +614,43 @@ namespace SieveUnitTests
|
|||||||
Assert.Equal(0,posts[3].Id);
|
Assert.Equal(0,posts[3].Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Theory]
|
||||||
public void CanFilter_WithEscapeCharacter()
|
[InlineData(1)]
|
||||||
|
[InlineData(2)]
|
||||||
|
[InlineData(3)]
|
||||||
|
[InlineData(4)]
|
||||||
|
public void Paging_DifferentPages(int page)
|
||||||
{
|
{
|
||||||
var comments = new List<Comment>
|
|
||||||
{
|
|
||||||
new Comment
|
|
||||||
{
|
|
||||||
Id = 0,
|
|
||||||
DateCreated = DateTimeOffset.UtcNow,
|
|
||||||
Text = "Here is, a comment"
|
|
||||||
},
|
|
||||||
new Comment
|
|
||||||
{
|
|
||||||
Id = 1,
|
|
||||||
DateCreated = DateTimeOffset.UtcNow.AddDays(-1),
|
|
||||||
Text = "Here is, another comment"
|
|
||||||
},
|
|
||||||
}.AsQueryable();
|
|
||||||
|
|
||||||
var model = new SieveModel
|
var model = new SieveModel
|
||||||
{
|
{
|
||||||
Filters = "Text==Here is\\, another comment"
|
Page = page,
|
||||||
|
PageSize = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
var result = _processor.Apply(model, comments);
|
var result = _processor.Apply(model, _posts);
|
||||||
Assert.Equal(1, result.Count());
|
var posts = result.ToList();
|
||||||
|
|
||||||
|
Assert.Single(posts);
|
||||||
|
var expectedId = page - 1;
|
||||||
|
Assert.Equal(expectedId, posts.First().Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Theory]
|
||||||
public void OrEscapedPipeValueFilteringWorks()
|
[InlineData(1)]
|
||||||
|
[InlineData(2)]
|
||||||
|
[InlineData(3)]
|
||||||
|
[InlineData(4)]
|
||||||
|
public void Paging_DifferentPageSizes(int pageSize)
|
||||||
{
|
{
|
||||||
var comments = new List<Comment>
|
var model = new SieveModel
|
||||||
{
|
{
|
||||||
new Comment
|
Page = 1,
|
||||||
{
|
PageSize = pageSize,
|
||||||
Id = 0,
|
|
||||||
DateCreated = DateTimeOffset.UtcNow,
|
|
||||||
Text = "Here is | a comment"
|
|
||||||
},
|
|
||||||
new Comment
|
|
||||||
{
|
|
||||||
Id = 1,
|
|
||||||
DateCreated = DateTimeOffset.UtcNow.AddDays(-1),
|
|
||||||
Text = "Here is | another comment"
|
|
||||||
},
|
|
||||||
}.AsQueryable();
|
|
||||||
|
|
||||||
var model = new SieveModel()
|
|
||||||
{
|
|
||||||
Filters = "Text==Here is \\| a comment|Here is \\| another comment",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var result = _processor.Apply(model, comments);
|
var result = _processor.Apply(model, _posts);
|
||||||
Assert.Equal(2, result.Count());
|
|
||||||
|
Assert.Equal(pageSize, result.Count());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -16,24 +16,16 @@ namespace SieveUnitTests
|
|||||||
{
|
{
|
||||||
private readonly ITestOutputHelper _testOutputHelper;
|
private readonly ITestOutputHelper _testOutputHelper;
|
||||||
private readonly SieveProcessor _processor;
|
private readonly SieveProcessor _processor;
|
||||||
private readonly SieveProcessor _nullableProcessor;
|
|
||||||
private readonly IQueryable<IPost> _posts;
|
private readonly IQueryable<IPost> _posts;
|
||||||
private readonly IQueryable<Comment> _comments;
|
private readonly IQueryable<Comment> _comments;
|
||||||
|
|
||||||
public GeneralWithInterfaces(ITestOutputHelper testOutputHelper)
|
public GeneralWithInterfaces(ITestOutputHelper testOutputHelper)
|
||||||
{
|
{
|
||||||
var nullableAccessor = new SieveOptionsAccessor();
|
|
||||||
nullableAccessor.Value.IgnoreNullsOnNotEqual = false;
|
|
||||||
|
|
||||||
_testOutputHelper = testOutputHelper;
|
_testOutputHelper = testOutputHelper;
|
||||||
_processor = new ApplicationSieveProcessor(new SieveOptionsAccessor(),
|
_processor = new ApplicationSieveProcessor(new SieveOptionsAccessor(),
|
||||||
new SieveCustomSortMethods(),
|
new SieveCustomSortMethods(),
|
||||||
new SieveCustomFilterMethods());
|
new SieveCustomFilterMethods());
|
||||||
|
|
||||||
_nullableProcessor = new ApplicationSieveProcessor(nullableAccessor,
|
|
||||||
new SieveCustomSortMethods(),
|
|
||||||
new SieveCustomFilterMethods());
|
|
||||||
|
|
||||||
_posts = new List<IPost>
|
_posts = new List<IPost>
|
||||||
{
|
{
|
||||||
new Post
|
new Post
|
||||||
@@ -189,25 +181,8 @@ namespace SieveUnitTests
|
|||||||
};
|
};
|
||||||
|
|
||||||
var result = _processor.Apply(model, _posts);
|
var result = _processor.Apply(model, _posts);
|
||||||
var nullableResult = _nullableProcessor.Apply(model, _posts);
|
|
||||||
|
|
||||||
Assert.True(result.Count() == 2);
|
Assert.True(result.Count() == 2);
|
||||||
Assert.True(nullableResult.Count() == 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Fact]
|
|
||||||
public void CanFilterNullableIntsWithNotEqual()
|
|
||||||
{
|
|
||||||
var model = new SieveModel()
|
|
||||||
{
|
|
||||||
Filters = "CategoryId!=1"
|
|
||||||
};
|
|
||||||
|
|
||||||
var result = _processor.Apply(model, _posts);
|
|
||||||
var nullableResult = _nullableProcessor.Apply(model, _posts);
|
|
||||||
|
|
||||||
Assert.True(result.Count() == 1);
|
|
||||||
Assert.True(nullableResult.Count() == 2);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
@@ -21,7 +21,7 @@ using static Nuke.Common.Tools.DotNet.DotNetTasks;
|
|||||||
InvokedTargets = new[] {nameof(Ci)},
|
InvokedTargets = new[] {nameof(Ci)},
|
||||||
CacheKeyFiles = new string[0])]
|
CacheKeyFiles = new string[0])]
|
||||||
[GitHubActions("ci_publish", GitHubActionsImage.UbuntuLatest,
|
[GitHubActions("ci_publish", GitHubActionsImage.UbuntuLatest,
|
||||||
OnPushBranches = new[] {"master", "releases/*"},
|
OnPushBranches = new[] {"releases/*"},
|
||||||
AutoGenerate = true,
|
AutoGenerate = true,
|
||||||
InvokedTargets = new[] {nameof(CiPublish)},
|
InvokedTargets = new[] {nameof(CiPublish)},
|
||||||
CacheKeyFiles = new string[0],
|
CacheKeyFiles = new string[0],
|
||||||
|
Reference in New Issue
Block a user