Clara.Analysis.Snowball
0.1.25
See the version list below for details.
dotnet add package Clara.Analysis.Snowball --version 0.1.25
NuGet\Install-Package Clara.Analysis.Snowball -Version 0.1.25
<PackageReference Include="Clara.Analysis.Snowball" Version="0.1.25" />
paket add Clara.Analysis.Snowball --version 0.1.25
#r "nuget: Clara.Analysis.Snowball, 0.1.25"
// Install Clara.Analysis.Snowball as a Cake Addin #addin nuget:?package=Clara.Analysis.Snowball&version=0.1.25 // Install Clara.Analysis.Snowball as a Cake Tool #tool nuget:?package=Clara.Analysis.Snowball&version=0.1.25
Clara
Simple, yet feature complete, in memory search engine.
Highlights
This library is meant for relatively small document sets (up to tenths of thousands) while maintaining fast query times (around 1 milisecond). Updating index requires reindexing, which means building new index, replacing in memory reference and discarding old one.
- Inspired by commonly known Lucene design
- Fast in memory searching
- Low memory allocation for search execution
- Stemmers and stopwords handling for 30 languages
- Text, keyword, hierarchy and range (any comparable structure values) fields
- Synonym graph with multi word synonym support
- Fully configurable and extendable text analysis pipeline
- Searching with BM25 weighted document scoring
- Filtering on any field type by values or range
- Faceting without restricting facet value list by filtered values
- Result sorting by document score or range field values
- Fluent query builder
Supported languages
Internally
Porter (English)
via Snowball
English, Arabic, Armenian, Basque, Catalan, Danish, Dutch, Finnish, French, German, Greek, Hindi, Hungarian, Indonesian, Irish, Italian, Lithuanian, Nepali, Norwegian, Portuguese, Romanian, Russian, Serbian, Spanish, Swedish, Tamil, Turkish, Yiddish
via Morfologik
Polish
Getting started
Given sample product data set from https://dummyjson.com/products.
[
{
"id": 1,
"title": "iPhone 9",
"description": "An apple mobile which is nothing like apple",
"price": 549,
"discountPercentage": 12.96,
"rating": 4.69,
"stock": 94,
"brand": "Apple",
"category": "smartphones"
}
]
We define data model.
public class Product
{
public int Id { get; set; }
public string Title { get; set; }
public string Description { get; set; }
public decimal? Price { get; set; }
public double? DiscountPercentage { get; set; }
public double? Rating { get; set; }
public int? Stock { get; set; }
public string Brand { get; set; }
public string Category { get; set; }
}
Then we define model to index mapper. Mapper is a definition of how our index will be built from source documents and what capabilities will it provide afterwards.
We only support single field searching, all text that is to be indexed has to be combined into single field. We can provide more text fields, for example when we want to provide multiple language support from single index. In such case we would combine text for each language and use adequate analyzer.
For simple fields we define delegates that provide raw values for indexing. Each field can provide none, one or more values, null values are automatically skipped during indexing. All simple fields can be marked as filterable or facetable, while only range fields can be made sortable.
Built indexes have no persistence and reside only in memory. If index needs updating, it should be rebuild and old one should be discarded. This is why fields have no names and can be referenced only by their usually static definition.
IIndexMapper<TSource>
interface is straightforward. It provides all fields collection, method to access document
key and method to access indexed document value. Indexed document value, which is provided in query results
can be different than index source document. To indicate such distinction use IIndexMapper<TSouce, TDocument>
type instead and return proper document type in GetDocument
method implementation.
public class ProductMapper : IIndexMapper<Product>
{
private static readonly StringBuilder builder = new();
public static TextField<Product> Text { get; } = new(x => GetText(x), new PorterAnalyzer());
public static DecimalField<Product> Price { get; } = new(x => x.Price, isFilterable: true, isFacetable: true, isSortable: true);
public static DoubleField<Product> DiscountPercentage { get; } = new(x => x.DiscountPercentage, isFilterable: true, isFacetable: true, isSortable: true);
public static DoubleField<Product> Rating { get; } = new(x => x.Rating, isFilterable: true, isFacetable: true, isSortable: true);
public static DoubleField<Product> Stock { get; } = new(x => x.Stock, isFilterable: true, isFacetable: true, isSortable: true);
public static KeywordField<Product> Brand { get; } = new(x => x.Brand, isFilterable: true, isFacetable: true);
public static KeywordField<Product> Category { get; } = new(x => x.Category, isFilterable: true, isFacetable: true);
public IEnumerable<Field> GetFields()
{
yield return Text;
yield return Price;
yield return DiscountPercentage;
yield return Rating;
yield return Stock;
yield return Brand;
yield return Category;
}
public string GetDocumentKey(Product item) => item.Id.ToString();
public Product GetDocument(Product item) => item;
private static string GetText(Product product)
{
builder.Clear();
builder.AppendLine(product.Title);
builder.AppendLine(product.Description);
builder.AppendLine(product.Brand);
builder.AppendLine(product.Category);
return builder.ToString();
}
}
Then we build our index.
public Index<Product> BuildIndex(IEnumerable<Product> items)
{
var builder =
new IndexBuilder<Product, Product>(
new ProductMapper());
foreach (var item in Product.Items)
{
builder.Index(item);
}
return builder.Build();
}
With index built, can run queries on it. Result documents can be accessed with Documents
property and
facet results via Facets
. Documents are not paged, since engine has to build whole result set each time
for facet values computation, while using pooled buffers for result construction. If paging is needed,
it can be added by simple Skip
/Take
logic on top Documents
collection.
// Query result must always be disposed in order to return pooled buffers for reuse
using var result = index.Query(
index.QueryBuilder()
.Search(ProductMapper.Text, "smartphone")
.Filter(ProductMapper.Brand, Values.Any("Apple", "Samsung"))
.Filter(ProductMapper.Price, from: 1, to: 1000)
.Facet(ProductMapper.Brand)
.Facet(ProductMapper.Category)
.Facet(ProductMapper.Price)
.Sort(ProductMapper.Price, SortDirection.Descending));
foreach (var document in result.Documents)
{
Console.WriteLine($"{document.Key} => {document.Score} [{document.Document.Title}]");
}
// Access facets with result.Facets
Advanced scenarios
Custom analyzers
Above code uses PorterAnalyzer
which provides basic English language stemming. For other languages
Clara.Analysis.Snowball
or Clara.Analysis.Morfologik
packages can be used. Those packages provide
stem and stop token filters for all supported languages.
For example you could define PolishAnalyzer
like this.
public static readonly IAnalyzer PolishAnalyzer =
new Analyzer(
new BasicTokenizer(numberDecimalSeparator: ','), // Splits text into tokens
new LowerInvariantTokenFilter(), // Transforms into lower case
new CachingTokenFilter(), // Prevents new string instance creation
new PolishStopTokenFilter(), // Language specific stop words default exclusion set
new KeywordLengthTokenFilter(), // Exclude from stemming tokens with length 2 or less
new KeywordDigitsTokenFilter(), // Exclude from stemming tokens containing digits
new PolishStemTokenFilter()); // Language specific token stemming
And then use it for index mapper field definition.
public static TextField<Product> TextPolish = new(x => GetTextPolish(x), PolishAnalyzer);
Custom range fields
Range fields represent index fields for struct
values with IComparable<T>
interface implementation.
By default DateTime
, Decimal
, Double
and Int32
types are supported. Implementors can support any
type that fullfills requirements by directly using RangeField<T>
and providing min
and max
values
for a given type or by providing their own concrete implementation.
Below is example implementation for DateOnly
structure type.
public sealed class DateOnlyField<TSource> : RangeField<TSource, int>
{
public DateOnlyField(Func<TSource, DateOnly?> valueMapper, bool isFilterable = false, bool isFacetable = false, bool isSortable = false)
: base(
valueMapper: valueMapper,
minValue: DateOnly.MinValue,
maxValue: DateOnly.MaxValue,
isFilterable: isFilterable,
isFacetable: isFacetable,
isSortable: isSortable)
{
}
public DateOnlyField(Func<TSource, IEnumerable<DateOnly>?> valueMapper, bool isFilterable = false, bool isFacetable = false, bool isSortable = false)
: base(
valueMapper: valueMapper,
minValue: DateOnly.MinValue,
maxValue: DateOnly.MaxValue,
isFilterable: isFilterable,
isFacetable: isFacetable,
isSortable: isSortable)
{
}
}
Synonym maps
TODO
Benchmarks
Query benchmarks and tests are performed using sample 100 product data set.
BenchmarkDotNet v0.13.8, Windows 11 (10.0.22621.2283/22H2/2022Update/SunValley2)
12th Gen Intel Core i9-12900K, 1 CPU, 24 logical and 16 physical cores
.NET SDK 7.0.308
[Host] : .NET 7.0.11 (7.0.1123.42427), X64 RyuJIT AVX2 DEBUG
DefaultJob : .NET 7.0.11 (7.0.1123.42427), X64 RyuJIT AVX2
Method | Mean | Error | StdDev | Gen0 | Allocated |
---|---|---|---|---|---|
SearchFilterFacetSortQueryX100 | 579.123 μs | 6.9726 μs | 6.1810 μs | - | 1473 B |
SearchFilterFacetSortQuery | 12.778 μs | 0.2520 μs | 0.4347 μs | 0.0916 | 1472 B |
SearchQuery | 7.449 μs | 0.1412 μs | 0.1681 μs | 0.0305 | 704 B |
FilterQuery | 1.462 μs | 0.0244 μs | 0.0228 μs | 0.0458 | 720 B |
FacetQuery | 9.634 μs | 0.1914 μs | 0.1880 μs | 0.0305 | 536 B |
SortQuery | 3.453 μs | 0.0391 μs | 0.0347 μs | 0.0229 | 408 B |
Query | 1.503 μs | 0.0300 μs | 0.0492 μs | 0.0191 | 312 B |
Benchmark variants with suffix "X100" use instead 100 times more product data. As observed due to internal structure pooling memory allocation per search execution is constant and independent from amount of indexed documents after initial allocation of pooled buffers.
License
Released under the MIT License
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 is compatible. |
.NET Framework | net461 was computed. net462 is compatible. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETFramework 4.6.2
- Clara (>= 0.1.25)
- libstemmer.net (>= 2.2.3)
-
.NETStandard 2.0
- Clara (>= 0.1.25)
- libstemmer.net (>= 2.2.3)
-
.NETStandard 2.1
- Clara (>= 0.1.25)
- libstemmer.net (>= 2.2.3)
-
net6.0
- Clara (>= 0.1.25)
- libstemmer.net (>= 2.2.3)
-
net7.0
- Clara (>= 0.1.25)
- libstemmer.net (>= 2.2.3)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
0.1.39 | 185 | 10/14/2024 |
0.1.38 | 107 | 10/3/2024 |
0.1.37 | 375 | 11/23/2023 |
0.1.36 | 177 | 11/1/2023 |
0.1.35 | 151 | 10/28/2023 |
0.1.34 | 129 | 10/26/2023 |
0.1.33 | 137 | 10/26/2023 |
0.1.32 | 133 | 10/13/2023 |
0.1.31 | 132 | 10/12/2023 |
0.1.30 | 138 | 10/4/2023 |
0.1.29 | 133 | 9/28/2023 |
0.1.28 | 132 | 9/27/2023 |
0.1.27 | 129 | 9/24/2023 |
0.1.26 | 130 | 9/24/2023 |
0.1.25 | 126 | 9/23/2023 |
0.1.24 | 127 | 9/21/2023 |
0.1.23 | 125 | 9/19/2023 |
0.1.22 | 125 | 9/19/2023 |
0.1.21 | 120 | 9/18/2023 |
0.1.20 | 114 | 9/17/2023 |