Clara 0.1.28

There is a newer version of this package available.
See the version list below for details.
dotnet add package Clara --version 0.1.28                
NuGet\Install-Package Clara -Version 0.1.28                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Clara" Version="0.1.28" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Clara --version 0.1.28                
#r "nuget: Clara, 0.1.28"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Clara as a Cake Addin
#addin nuget:?package=Clara&version=0.1.28

// Install Clara as a Cake Tool
#tool nuget:?package=Clara&version=0.1.28                

Clara

CI NuGet NuGet License

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 fields
  • Synonym tree with multi word synonym support
  • Fully configurable and extendable text analysis pipeline
  • Document scring with BM25 weight
  • 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
  • Fluent query builder

Supported languages

  • Internally

    Porter (English)

  • Using Clara.Analysis.Snowball package

    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

  • Using Clara.Analysis.Morfologik package

    Polish

Getting started

Given sample product data set from dummyjson.com.

[
  {
    "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 sealed class ProductMapper : IIndexMapper<Product>
{
    public static IAnalyzer Analyzer { get; } = new PorterAnalyzer();

    public static TextField<Product> Text { get; } = new(x => GetText(x), Analyzer);
    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 Int32Field<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)
    {
        yield return product.Id.ToString(CultureInfo.InvariantCulture);
        yield return product.Title;
        yield return product.Description;
        yield return product.Brand;
        yield return product.Category;
    }
}

Then we build our index.

// When rebuilding index, this reference should be reused
var sharedTokenEncoderStore = new SharedTokenEncoderStore();

var index = IndexBuilder.Build(Product.Items, new ProductMapper(), sharedTokenEncoderStore);

With index built, can run queries against it. Result documents can be accessed with Documents property and facet results via Facets. Documents are not paged, since search engine builds whole result set each time, in order to perform 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: 300, to: 1500)
        .Facet(ProductMapper.Brand)
        .Facet(ProductMapper.Category)
        .Facet(ProductMapper.Price)
        .Sort(ProductMapper.Price, SortDirection.Descending));

Console.WriteLine("Documents:");

foreach (var document in result.Documents.Take(10))
{
    Console.WriteLine($"  [{document.Document.Title}] => {document.Score}");
}

Console.WriteLine("Brands:");

foreach (var value in result.Facets.Field(ProductMapper.Brand).Values.Take(5))
{
    Console.WriteLine($"  {(value.IsSelected ? "(x)" : "( )")} [{value.Value}] => {value.Count}");
}

Console.WriteLine("Categories:");

foreach (var value in result.Facets.Field(ProductMapper.Category).Values.Take(5))
{
    Console.WriteLine($"  {(value.IsSelected ? "(x)" : "( )")} [{value.Value}] => {value.Count}");
}

var priceFacet = result.Facets.Field(ProductMapper.Price);

Console.WriteLine("Price:");
Console.WriteLine($"  [Min] => {priceFacet.Min}");
Console.WriteLine($"  [Max] => {priceFacet.Max}");

Running this query against sample data results in following output.

Documents:
  [Samsung Universe 9] => 3,3160777
  [iPhone X] => 2,9904046
  [iPhone 9] => 3,5479112
Brands:
  (x) [Apple] => 2
  (x) [Samsung] => 1
  ( ) [Huawei] => 1
Categries:
  ( ) [smartphones] => 3
Price:
  [Min] => 549
  [Max] => 1249

Advanced scenarios

Field mapping

Keyword fields

TODO

Hierarchy fields

TODO

Range fields

Range fields represent index fields for struct values with IComparable<T> interface implementation. Internally DateTime, Decimal, Double and Int32 types are supported. Implementors can support any type that fullfills requirements by directly using RangeField<T> and providing minValue and maxValue 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)
    {
    }
}
Text fields

TODO

Analysis

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.

Below is shown example analyzer definition PolishAnalyzer using Morfologik.

var polishAnalyzer =
    new Analyzer(
        new BasicTokenizer(),             // 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 LengthKeywordTokenFilter(),   // Exclude from stemming tokens with length 2 or less
        new DigitsKeywordTokenFilter(),   // Exclude from stemming tokens containing digits
        new KeywordTokenFilter(keywords), // Exclude from stemming provided keyword tokens
        new PolishStemTokenFilter());     // Language specific token stemming

It can be used for index mapper field definition as follows.

public static TextField<Product> TextPolish = new(x => GetTextPolish(x), PolishAnalyzer);
Synonym maps

TODO

Extending analysis pipeline

TODO

Benchmarks

Index and query benchmarks and tests are performed using sample 100 product data set. Benchmark variants with x100 suffix use data set combined 100 times.

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

Index benchmarks

Method Mean Error StdDev Gen0 Gen1 Gen2 Allocated
Index_x100 71,140.7 μs 1,730.65 μs 5,102.87 μs 2428.5714 2285.7143 1000.0000 31207.54 KB
IndexSynonym 530.7 μs 7.55 μs 7.06 μs 36.1328 12.6953 - 558.23 KB
Index 466.4 μs 4.78 μs 4.47 μs 34.6680 13.1836 - 538.11 KB
SharedIndex_x100 66,836.8 μs 1,482.05 μs 4,369.86 μs 2250.0000 2000.0000 875.0000 29919.28 KB
SharedIndexSynonym 508.8 μs 7.71 μs 7.21 μs 32.2266 10.7422 - 505.52 KB
SharedIndex 439.6 μs 5.44 μs 5.09 μs 31.2500 10.7422 - 485.41 KB

Query benchmarks

Method Mean Error StdDev Gen0 Allocated
ComplexQuery_x100 562.813 μs 7.9706 μs 6.2229 μs - 1601 B
ComplexQuery 11.902 μs 0.0440 μs 0.0368 μs 0.0916 1600 B
SearchQuery 7.100 μs 0.0211 μs 0.0197 μs 0.0458 720 B
FilterQuery 1.326 μs 0.0134 μs 0.0125 μs 0.0458 744 B
FacetQuery 8.738 μs 0.0782 μs 0.0653 μs 0.0305 600 B
SortQuery 3.483 μs 0.0105 μs 0.0082 μs 0.0229 408 B
BasicQuery 1.437 μs 0.0180 μs 0.0168 μs 0.0191 312 B

Due to internal buffer structures pooling, memory allocation per search execution is constant after initial allocation of pooled buffers.

License

Released under the MIT License

Product 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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • .NETFramework 4.6.2

  • .NETStandard 2.0

  • .NETStandard 2.1

    • No dependencies.
  • net6.0

    • No dependencies.
  • net7.0

    • No dependencies.

NuGet packages (2)

Showing the top 2 NuGet packages that depend on Clara:

Package Downloads
Clara.Analysis.Morfologik

Simple, yet feature complete, in memory search engine.

Clara.Analysis.Snowball

Simple, yet feature complete, in memory search engine.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.1.39 249 10/14/2024
0.1.38 160 10/3/2024
0.1.37 451 11/23/2023
0.1.36 284 11/1/2023
0.1.35 227 10/28/2023
0.1.34 196 10/26/2023
0.1.33 226 10/26/2023
0.1.32 236 10/13/2023
0.1.31 223 10/12/2023
0.1.30 250 10/4/2023
0.1.29 231 9/28/2023
0.1.28 189 9/27/2023
0.1.27 258 9/24/2023
0.1.26 249 9/24/2023
0.1.25 210 9/23/2023
0.1.24 222 9/21/2023
0.1.23 194 9/19/2023
0.1.22 193 9/19/2023
0.1.21 218 9/18/2023
0.1.20 204 9/17/2023
0.1.18 477 11/9/2022
0.1.15 332 11/7/2022
0.1.14 489 6/5/2022
0.1.13 419 6/3/2022
0.1.12 430 6/2/2022
0.1.11 415 5/31/2022
0.1.10 421 5/29/2022
0.1.9 421 5/28/2022
0.1.8 414 5/27/2022
0.1.7 417 5/25/2022
0.1.6 412 5/25/2022
0.1.5 397 5/25/2022
0.1.4 403 5/24/2022
0.1.3 423 5/23/2022
0.1.2 412 5/21/2022
0.1.1 413 5/20/2022
0.1.0 418 5/20/2022