mirror of
https://github.com/Biarity/Sieve.git
synced 2024-11-21 13:02:46 +01:00
Release Sieve 2.5.0 (#151)
* Setup release 2.5.0 with automated build and pre-releases * #80 added support for escaping pipe control characters (#113) * #80 added support for escaping comma and pipe control characters * Update SieveModel.cs Fix build. Accidentally broken by resolving conflicts. * Migrate UnitTests to xUnit Co-authored-by: Clayton Andersen <candersen@restaurant365.com> Co-authored-by: ITDancer13 <kevin@ksommer.eu> Co-authored-by: ITDancer139 <kevinitdancersommer@gmail.com> * SieveProcessor.Options made protected property (#134) Mapper assignment in constructor is moved to a null-coalescing member pair (a field and a property) "IncludeScopes" switch is removed from appSettings.{env}.json files * Revert to _mapper assignment in constructor. (#140) * reverting fix (#142) * Revert to _mapper assignment in constructor. * reverting fix * pass filter values as parameters (#112) make GetClosureOverConstant really work * Make ApplyFiltering, ApplySorting and ApplyPagination protected virtual #139 (#144) * stop excluding null values when filtering using notEqual (#114) * stop excluding null values when filtering using notEqual * add IgnoreNullsOnNotEqual config field, to enable/disable the new feature Co-authored-by: AnasZakarneh <a.zakarneh@foothillsolutions.com> Co-authored-by: Clayton Andersen <tunaman65@gmail.com> Co-authored-by: Clayton Andersen <candersen@restaurant365.com> Co-authored-by: ITDancer139 <kevinitdancersommer@gmail.com> Co-authored-by: Hasan Manzak <hasan.manzak@gmail.com> Co-authored-by: alicak <alicak@users.noreply.github.com> Co-authored-by: AnasZakarneh <Zakarnehanas1@gmail.com> Co-authored-by: AnasZakarneh <a.zakarneh@foothillsolutions.com>
This commit is contained in:
parent
83a2c1ab18
commit
e83d213181
4
.github/workflows/ci.yml
vendored
4
.github/workflows/ci.yml
vendored
@ -17,12 +17,10 @@
|
|||||||
name: ci
|
name: ci
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
pull_request:
|
pull_request:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
|
- 'releases/*'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ubuntu-latest:
|
ubuntu-latest:
|
||||||
|
33
.github/workflows/ci_publish.yml
vendored
Normal file
33
.github/workflows/ci_publish.yml
vendored
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# <auto-generated>
|
||||||
|
#
|
||||||
|
# This code was generated.
|
||||||
|
#
|
||||||
|
# - To turn off auto-generation set:
|
||||||
|
#
|
||||||
|
# [GitHubActions (AutoGenerate = false)]
|
||||||
|
#
|
||||||
|
# - To trigger manual generation invoke:
|
||||||
|
#
|
||||||
|
# nuke --generate-configuration GitHubActions_ci_publish --host GitHubActions
|
||||||
|
#
|
||||||
|
# </auto-generated>
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
name: ci_publish
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- 'releases/*'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
ubuntu-latest:
|
||||||
|
name: ubuntu-latest
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v1
|
||||||
|
- name: Run './build.cmd CiPublish'
|
||||||
|
run: ./build.cmd CiPublish
|
||||||
|
env:
|
||||||
|
NUGET_API_KEY: ${{ secrets.NUGET_API_KEY }}
|
@ -43,6 +43,9 @@
|
|||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"description": "Disables displaying the NUKE logo"
|
"description": "Disables displaying the NUKE logo"
|
||||||
},
|
},
|
||||||
|
"NUGET_API_KEY": {
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
"Plan": {
|
"Plan": {
|
||||||
"type": "boolean",
|
"type": "boolean",
|
||||||
"description": "Shows the execution plan (HTML)"
|
"description": "Shows the execution plan (HTML)"
|
||||||
@ -65,9 +68,11 @@
|
|||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": [
|
"enum": [
|
||||||
"Ci",
|
"Ci",
|
||||||
|
"CiPublish",
|
||||||
"Clean",
|
"Clean",
|
||||||
"Compile",
|
"Compile",
|
||||||
"Package",
|
"Package",
|
||||||
|
"Publish",
|
||||||
"Restore",
|
"Restore",
|
||||||
"Test"
|
"Test"
|
||||||
]
|
]
|
||||||
@ -84,9 +89,11 @@
|
|||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": [
|
"enum": [
|
||||||
"Ci",
|
"Ci",
|
||||||
|
"CiPublish",
|
||||||
"Clean",
|
"Clean",
|
||||||
"Compile",
|
"Compile",
|
||||||
"Package",
|
"Package",
|
||||||
|
"Publish",
|
||||||
"Restore",
|
"Restore",
|
||||||
"Test"
|
"Test"
|
||||||
]
|
]
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
branches:
|
||||||
|
release:
|
||||||
|
mode: ContinuousDeployment
|
@ -128,7 +128,8 @@ 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,6 +1,5 @@
|
|||||||
{
|
{
|
||||||
"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,6 +9,7 @@ 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[] {
|
||||||
"!@=*",
|
"!@=*",
|
||||||
@ -36,7 +37,11 @@ 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 ? Regex.Split(filterSplits[1], EscapedPipePattern).Select(t => t.Trim()).ToArray() : null;
|
Values = filterSplits.Length > 1
|
||||||
|
? 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("*");
|
||||||
@ -90,6 +95,5 @@ 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;
|
||||||
|
@ -9,5 +9,7 @@
|
|||||||
public int MaxPageSize { get; set; } = 0;
|
public int MaxPageSize { get; set; } = 0;
|
||||||
|
|
||||||
public bool ThrowExceptions { get; set; } = false;
|
public bool ThrowExceptions { get; set; } = false;
|
||||||
|
|
||||||
|
public bool IgnoreNullsOnNotEqual { get; set; } = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -68,7 +68,6 @@ 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();
|
||||||
@ -78,7 +77,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;
|
||||||
}
|
}
|
||||||
@ -87,7 +86,7 @@ namespace Sieve.Services
|
|||||||
ISieveCustomSortMethods customSortMethods)
|
ISieveCustomSortMethods customSortMethods)
|
||||||
{
|
{
|
||||||
_mapper = MapProperties(_mapper);
|
_mapper = MapProperties(_mapper);
|
||||||
_options = options;
|
Options = options;
|
||||||
_customSortMethods = customSortMethods;
|
_customSortMethods = customSortMethods;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,16 +94,18 @@ 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>
|
||||||
@ -148,7 +149,7 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
if (!_options.Value.ThrowExceptions)
|
if (!Options.Value.ThrowExceptions)
|
||||||
{
|
{
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -162,7 +163,7 @@ namespace Sieve.Services
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private IQueryable<TEntity> ApplyFiltering<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
protected virtual IQueryable<TEntity> ApplyFiltering<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
||||||
object[] dataForCustomMethods = null)
|
object[] dataForCustomMethods = null)
|
||||||
{
|
{
|
||||||
if (model?.GetFiltersParsed() == null)
|
if (model?.GetFiltersParsed() == null)
|
||||||
@ -216,11 +217,14 @@ 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
|
||||||
@ -253,8 +257,7 @@ 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,
|
private static Expression GetFilterValueNullCheck(Expression parameter, string fullPropertyName, bool isFilterTermValueNull)
|
||||||
bool isFilterTermValueNull)
|
|
||||||
{
|
{
|
||||||
var (propertyValue, nullCheck) = GetPropertyValueAndNullCheckExpression(parameter, fullPropertyName);
|
var (propertyValue, nullCheck) = GetPropertyValueAndNullCheckExpression(parameter, fullPropertyName);
|
||||||
|
|
||||||
@ -336,15 +339,13 @@ 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)
|
||||||
{
|
{
|
||||||
return Expression.Constant(constant, targetType);
|
Expression<Func<T>> hoistedConstant = () => constant;
|
||||||
|
return Expression.Convert(hoistedConstant.Body, targetType);
|
||||||
}
|
}
|
||||||
|
|
||||||
private IQueryable<TEntity> ApplySorting<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
protected virtual IQueryable<TEntity> ApplySorting<TEntity>(TSieveModel model, IQueryable<TEntity> result,
|
||||||
object[] dataForCustomMethods = null)
|
object[] dataForCustomMethods = null)
|
||||||
{
|
{
|
||||||
if (model?.GetSortsParsed() == null)
|
if (model?.GetSortsParsed() == null)
|
||||||
@ -373,11 +374,11 @@ namespace Sieve.Services
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private IQueryable<TEntity> ApplyPagination<TEntity>(TSieveModel model, IQueryable<TEntity> result)
|
protected virtual 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;
|
var maxPageSize = Options.Value.MaxPageSize > 0 ? Options.Value.MaxPageSize : pageSize;
|
||||||
|
|
||||||
if (pageSize <= 0)
|
if (pageSize <= 0)
|
||||||
{
|
{
|
||||||
@ -399,14 +400,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);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -426,7 +427,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>));
|
||||||
@ -437,7 +438,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<>));
|
||||||
@ -481,11 +482,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,16 +15,24 @@ 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
|
||||||
@ -180,8 +188,25 @@ 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]
|
||||||
@ -613,5 +638,61 @@ namespace SieveUnitTests
|
|||||||
Assert.Equal(1,posts[2].Id);
|
Assert.Equal(1,posts[2].Id);
|
||||||
Assert.Equal(0,posts[3].Id);
|
Assert.Equal(0,posts[3].Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void CanFilter_WithEscapeCharacter()
|
||||||
|
{
|
||||||
|
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
|
||||||
|
{
|
||||||
|
Filters = "Text==Here is\\, another comment"
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = _processor.Apply(model, comments);
|
||||||
|
Assert.Equal(1, result.Count());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void OrEscapedPipeValueFilteringWorks()
|
||||||
|
{
|
||||||
|
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()
|
||||||
|
{
|
||||||
|
Filters = "Text==Here is \\| a comment|Here is \\| another comment",
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = _processor.Apply(model, comments);
|
||||||
|
Assert.Equal(2, result.Count());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,16 +16,24 @@ 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
|
||||||
@ -181,8 +189,25 @@ 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]
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using GlobExpressions;
|
||||||
using Nuke.Common;
|
using Nuke.Common;
|
||||||
using Nuke.Common.CI;
|
using Nuke.Common.CI;
|
||||||
using Nuke.Common.CI.GitHubActions;
|
using Nuke.Common.CI.GitHubActions;
|
||||||
@ -8,17 +9,23 @@ using Nuke.Common.IO;
|
|||||||
using Nuke.Common.ProjectModel;
|
using Nuke.Common.ProjectModel;
|
||||||
using Nuke.Common.Tools.DotNet;
|
using Nuke.Common.Tools.DotNet;
|
||||||
using Nuke.Common.Tools.GitVersion;
|
using Nuke.Common.Tools.GitVersion;
|
||||||
|
using Nuke.Common.Utilities.Collections;
|
||||||
using static Nuke.Common.IO.FileSystemTasks;
|
using static Nuke.Common.IO.FileSystemTasks;
|
||||||
using static Nuke.Common.Tools.DotNet.DotNetTasks;
|
using static Nuke.Common.Tools.DotNet.DotNetTasks;
|
||||||
|
|
||||||
[CheckBuildProjectConfigurations]
|
[CheckBuildProjectConfigurations]
|
||||||
[ShutdownDotNetAfterServerBuild]
|
[ShutdownDotNetAfterServerBuild]
|
||||||
[GitHubActions("ci", GitHubActionsImage.UbuntuLatest,
|
[GitHubActions("ci", GitHubActionsImage.UbuntuLatest,
|
||||||
OnPushBranches = new[] {"master"},
|
OnPullRequestBranches = new[] {"master", "releases/*"},
|
||||||
OnPullRequestBranches = new[] {"master"},
|
|
||||||
AutoGenerate = true,
|
AutoGenerate = true,
|
||||||
InvokedTargets = new[] {nameof(Ci)},
|
InvokedTargets = new[] {nameof(Ci)},
|
||||||
CacheKeyFiles = new string[0])]
|
CacheKeyFiles = new string[0])]
|
||||||
|
[GitHubActions("ci_publish", GitHubActionsImage.UbuntuLatest,
|
||||||
|
OnPushBranches = new[] {"releases/*"},
|
||||||
|
AutoGenerate = true,
|
||||||
|
InvokedTargets = new[] {nameof(CiPublish)},
|
||||||
|
CacheKeyFiles = new string[0],
|
||||||
|
ImportSecrets = new[] {"NUGET_API_KEY"})]
|
||||||
class Build : NukeBuild
|
class Build : NukeBuild
|
||||||
{
|
{
|
||||||
[Parameter("Configuration to build - Default is 'Debug' (local) or 'Release' (server)")]
|
[Parameter("Configuration to build - Default is 'Debug' (local) or 'Release' (server)")]
|
||||||
@ -30,6 +37,9 @@ class Build : NukeBuild
|
|||||||
|
|
||||||
[Solution] readonly Solution Solution;
|
[Solution] readonly Solution Solution;
|
||||||
|
|
||||||
|
// ReSharper disable once InconsistentNaming
|
||||||
|
[Parameter] string NUGET_API_KEY;
|
||||||
|
|
||||||
Project SieveProject => Solution.AllProjects.First(p => p.Name == "Sieve");
|
Project SieveProject => Solution.AllProjects.First(p => p.Name == "Sieve");
|
||||||
|
|
||||||
AbsolutePath OutputDirectory => RootDirectory / "output";
|
AbsolutePath OutputDirectory => RootDirectory / "output";
|
||||||
@ -83,13 +93,29 @@ class Build : NukeBuild
|
|||||||
.EnableNoBuild());
|
.EnableNoBuild());
|
||||||
});
|
});
|
||||||
|
|
||||||
Target Ci => _ => _
|
Target Publish => _ => _
|
||||||
.DependsOn(Package);
|
.DependsOn(Package)
|
||||||
|
.Requires(() => IsServerBuild)
|
||||||
|
.Requires(() => NUGET_API_KEY)
|
||||||
|
.Requires(() => Configuration.Equals(Configuration.Release))
|
||||||
|
.Executes(() =>
|
||||||
|
{
|
||||||
|
Glob.Files(OutputDirectory, "*.nupkg")
|
||||||
|
.NotEmpty()
|
||||||
|
.ForEach(x =>
|
||||||
|
{
|
||||||
|
DotNetNuGetPush(s => s
|
||||||
|
.SetTargetPath(OutputDirectory / x)
|
||||||
|
.SetSource("https://api.nuget.org/v3/index.json")
|
||||||
|
.SetApiKey(NUGET_API_KEY));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
Target Ci => _ => _
|
||||||
|
.DependsOn(Test);
|
||||||
|
|
||||||
|
Target CiPublish => _ => _
|
||||||
|
.DependsOn(Publish);
|
||||||
|
|
||||||
/// Support plugins are available for:
|
|
||||||
/// - JetBrains ReSharper https://nuke.build/resharper
|
|
||||||
/// - JetBrains Rider https://nuke.build/rider
|
|
||||||
/// - Microsoft VisualStudio https://nuke.build/visualstudio
|
|
||||||
/// - Microsoft VSCode https://nuke.build/vscode
|
|
||||||
public static int Main() => Execute<Build>(x => x.Package);
|
public static int Main() => Execute<Build>(x => x.Package);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user