Extends ReHackt.Linq with object-object mapping using AutoMapper profiles.
$ dotnet add package ReHackt.Linq.AutoMapperSome useful System.Linq.IQueryable extensions such as filtering, ordering, paging...
Get it on NuGet
QueryableFilter<T> allows to dynamically filter an IQueryable<T> with a query string. For example, this can be useful for an API whose clients can filter a collection of entities on any of its properties, or create complex logical queries.
For example
string query = @"Name eq ""Bond"" and (""james"" in Email or (Status in [1, 2] and ""007"" in Codes)) and (Amount lt 1000 or IsEnabled eq false)";
if(QueryableFilter.TryParse(query, out QueryableFilter<Agent> filter) {
IQueryable<Agent> agents = _agentManager.Agents.Filter(filter);
}
else { /* Handle invalid query */ };
Is equivalent to
IQueryable<Agent> agents = _agentManager.Agents
.Where(u => u.Name == "Bond"
&& (u.Email.Contains("james")
|| (new int[] { 1, 2 }.Contains(Status) && u.Codes.Contains("007")))
&& (u.Amount < 1000 || u.IsEnabled == false);
string.Contains or IList.Contains)Filter allows to apply a QueryableFilter<T> to the input sequence using LINQ method syntax.
source.Filter(filter) // filter is a QueryableFilter<T>
Is syntactic sugar for
filter.Apply(source)
This method also allows you to directly filter the input sequence with a query string (implicitly creating a QueryableFilter<T>). Be careful, this can throw an argument exception if the query string is not valid.
source.Filter(filterQuery) // filterQuery is a string
Is syntactic sugar for
QueryableFilter.TryParse(filterQuery, out QueryableFilter<T> filter) ?
source.Filter(filter) :
throw new ArgumentException("Invalid filter query", nameof(filterQuery))
source.WhereIf(condition, predicate)
Is syntactic sugar for
condition ? source.Where(predicate) : source
This allows you to keep the LINQ method syntax to apply filters according to a condition that does not depend on the element being tested.
For example
return source
.Join(...)
.Where(...)
.WhereIf(condition1, predicate1)
.WhereIf(condition2, predicate2)
.OrderBy(...)
.Select(...);
Is equivalent to
source = source
.Join(...)
.Where(...);
if(condition1) {
source = source.Where(predicate1);
}
if(condition2) {
source = source.Where(predicate2);
}
return source
.OrderBy(...)
.Select(...);
// TODO Write documentation of Sort(...)
source.PageBy(page, pageSize)
Is syntactic sugar for
source.Skip(((page < 1 ? 1 : page) - 1) * pageSize).Take(pageSize)