Rystem.RepositoryFramework.Abstractions 0.8.1

There is a newer version of this package available.
See the version list below for details.
dotnet add package Rystem.RepositoryFramework.Abstractions --version 0.8.1
NuGet\Install-Package Rystem.RepositoryFramework.Abstractions -Version 0.8.1
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="Rystem.RepositoryFramework.Abstractions" Version="0.8.1" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Rystem.RepositoryFramework.Abstractions --version 0.8.1
#r "nuget: Rystem.RepositoryFramework.Abstractions, 0.8.1"
#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 Rystem.RepositoryFramework.Abstractions as a Cake Addin
#addin nuget:?package=Rystem.RepositoryFramework.Abstractions&version=0.8.1

// Install Rystem.RepositoryFramework.Abstractions as a Cake Tool
#tool nuget:?package=Rystem.RepositoryFramework.Abstractions&version=0.8.1

Repository Framework

Contribute: https://www.buymeacoffee.com/keyserdsoze

Showcase (youtube)

Showcase (code)

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

Document to read before using this library:

Basic knowledge

Interfaces

Based on CQRS we could split our repository pattern in two main interfaces, one for update (write, delete) and one for read.

Command (Write-Delete)
public interface ICommand<T, TKey> : ICommandPattern
    where TKey : notnull
{
    Task<bool> InsertAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<bool> UpdateAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<bool> DeleteAsync(TKey key, CancellationToken cancellationToken = default);
}
Query (Read)
public interface IQuery<T, TKey> : IQueryPattern
    where TKey : notnull
{
    Task<T?> GetAsync(TKey key, CancellationToken cancellationToken = default);
    Task<IEnumerable<T>> QueryAsync(Expression<Func<T, bool>>? predicate = null, int? top = null, int? skip = null, CancellationToken cancellationToken = default);
}
Repository Pattern (Write-Delete-Read)
public interface IRepository<T, TKey> : ICommand<T, TKey>, IQuery<T, TKey>, IRepositoryPattern, ICommandPattern, IQueryPattern
    where TKey : notnull
{
}

Examples

Model
public class User
{
    public string Id { get; set; }
    public string Name { get; set; }
    public string Email { get; set; }
}
Command
public class UserWriter : ICommand<User, string>
{
    public Task<bool> DeleteAsync(string key, CancellationToken cancellationToken = default)
    {
        //delete on with DB or storage context
        throw new NotImplementedException();
    }
    public Task<bool> InsertAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //insert on DB or storage context
        throw new NotImplementedException();
    }
    public Task<bool> UpdateAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //update on DB or storage context
        throw new NotImplementedException();
    }
}
Query
public class UserReader : IQuery<User, string>
{
    public Task<User?> GetAsync(string key, CancellationToken cancellationToken = default)
    {
        //get an item by key from DB or storage context
        throw new NotImplementedException();
    }
    public Task<IEnumerable<User>> QueryAsync(Expression<Func<User, bool>>? predicate = null, int? top = null, int? skip = null, CancellationToken cancellationToken = default)
    {
        //get a list of items by a predicate with top and skip from DB or storage context
        throw new NotImplementedException();
    }
}
Alltogether as repository pattern

if you don't have CQRS infrastructure (usually it's correct to use CQRS when you have minimum two infrastructures one for write and delete and at least one for read)

public class UserRepository : IRepository<User, string>, IQuery<User, string>, ICommand<User, string>
{
    public Task<bool> DeleteAsync(string key, CancellationToken cancellationToken = default)
    {
        //delete on with DB or storage context
        throw new NotImplementedException();
    }
    public Task<bool> InsertAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //insert on DB or storage context
        throw new NotImplementedException();
    }
    public Task<bool> UpdateAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //update on DB or storage context
        throw new NotImplementedException();
    }
    public Task<User?> GetAsync(string key, CancellationToken cancellationToken = default)
    {
        //get an item by key from DB or storage context
        throw new NotImplementedException();
    }
    public Task<IEnumerable<User>> QueryAsync(Expression<Func<User, bool>>? predicate = null, int? top = null, int? skip = null, CancellationToken cancellationToken = default)
    {
        //get a list of items by a predicate with top and skip from DB or storage context
        throw new NotImplementedException();
    }
}

Specific key interfaces

You may use specific interfaces for 4 specific classic keys: Int, Long, Guid, String

Command for Guid (Write-Delete)
public interface IGuidableCommand<T> : ICommand<T, Guid>, ICommandPattern
{
}
Query for Guid (Read)
public interface IGuidableQuery<T> : IQuery<T, Guid>, IQueryPattern
{
}
Repository Pattern for Guid (Write-Delete-Read)

public interface IGuidableRepository<T> : IRepository<T, Guid>, IGuidableCommand<T>, IGuidableQuery<T>, IRepositoryPattern, IQueryPattern, ICommandPattern { }

String key
interface IStringableCommand<T> : ICommand<T,string>, ICommandPattern
interface IStringableQuery<T> : IQuery<T, string>, IQueryPattern
interface IStringableRepository<T> : IRepository<T, string>, IStringableCommand<T>, IStringableQuery<T>, IRepositoryPattern, ICommandPattern, IQueryPattern
Int key
interface IIntableCommand<T> : ICommand<T, int>, ICommandPattern
interface IIntableQuery<T> : IQuery<T, int>, IQueryPattern
interface IIntableRepository<T> : IRepository<T, int>, IIntableCommand<T>, IIntableQuery<T>, IRepositoryPattern, IQueryPattern, ICommandPattern
Long key
interface ILongableCommand<T> : ICommand<T, long>, ICommandPattern
interface ILongableQuery<T> : IQuery<T, long>, IQueryPattern
interface ILongableRepository<T> : IRepository<T, long>, ILongableCommand<T>, ILongableQuery<T>, IRepositoryPattern

Dependency Injection

IServiceCollection extension methods

You can use these extension methods to have a simple mechanic to integrate with .Net DI pattern, by default it's integrated with scoped service life time but you can choose another one at your discretion. You may find them in Microsoft.Extensions.DependencyInjection namespace.

Add Command pattern
public static IServiceCollection AddCommand<T, TKey, TStorage>(this IServiceCollection services,
    ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
    where TStorage : class, ICommand<T, TKey>
    where TKey : notnull
Add Query pattern
public static IServiceCollection AddQuery<T, TKey, TStorage>(this IServiceCollection services,
   ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
   where TStorage : class, IQuery<T, TKey>
   where TKey : notnull
       
Add Repository pattern

As before you may choose to use repository pattern instead CQRS.

public static IServiceCollection AddRepository<T, TKey, TStorage>(this IServiceCollection services,
  ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
  where TStorage : class, IRepository<T, TKey>
  where TKey : notnull
      
Add Command&Query or Repository with specific keys

An example with Guid, valid also for String, Int, Long

 public static IServiceCollection AddRepositoryWithGuidKey<T, TStorage>(this IServiceCollection services,
   ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
   where TStorage : class, IGuidableRepository<T>
       
public static IServiceCollection AddCommandWithGuidKey<T, TStorage>(this IServiceCollection services,
    ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
    where TStorage : class, IGuidableCommand<T>
        
public static IServiceCollection AddQueryWithGuidKey<T, TStorage>(this IServiceCollection services,
   ServiceLifetime serviceLifetime = ServiceLifetime.Scoped)
   where TStorage : class, IGuidableQuery<T>

How to use it in DI?

You have only to inject the interface you added everywhere you need it. Below an example from unit test.

public class RandomCreationTest
{
    private readonly IStringableRepository<PopulationTest> _test;
    private readonly IRepository<RegexPopulationTest, double> _population;
    private readonly IStringableQuery<DelegationPopulation> _delegation;

    public RandomCreationTest(IStringableRepository<PopulationTest> test,
        IRepository<RegexPopulationTest, double> population,
        IStringableQuery<DelegationPopulation> delegation)
    {
        _test = test;
        _population = population;
        _delegation = delegation;
    }
}

Api auto-generated

In your web application you have only to add one row.

app.AddApiForRepositoryFramework();

Startup example

In the example below you may find the DI for repository with string key for User model, populated with random data in memory, swagger to test the solution, the population method just after the build and the configuration of your API based on repository framework.

using RepositoryFramework.WebApi.Models;

var builder = WebApplication.CreateBuilder(args);
builder.Services
    .AddRepositoryInMemoryStorageWithStringKey<User>()
    .PopulateWithRandomData(x => x.Email!, 120, 5);
    builder.Services.AddEndpointsApiExplorer();
    builder.Services.AddSwaggerGen();

var app = builder.Build();
app.Services.Populate();
// Configure the HTTP request pipeline.
if (app.Environment.IsDevelopment())
{
    app.UseSwagger();
    app.UseSwaggerUI();
}
app.UseHttpsRedirection();
app.AddApiForRepositoryFramework();
app.Run();

HttpClient to use your API (example)

You can add a client for a specific url

.AddRepositoryClient<User, string>("localhost:7058");

and use it in DI with

IRepositoryClient<User, string>
Integration for specific key

Here an example for Guid, you may find the same Client for Int, Long, String

.AddRepositoryClientWithGuidKey<User>("localhost:7058");

that injects

IGuidableRepositoryClient<User>

Integration for Command or Query

.AddCommandClient<User, string>("localhost:7058");

or

.AddQueryClient<User, string>("localhost:7058");

In memory integration by default

With this library you can add in memory integration with the chance to create random data with random values, random based on regular expressions and delegated methods

public static RepositoryInMemoryBuilder<T, TKey> AddRepositoryInMemoryStorage<T, TKey>(
    this IServiceCollection services,
    Action<RepositoryBehaviorSettings<T, TKey>>? settings = default)
    where TKey : notnull 

Populate with specific key (example with Guid)

public RepositoryInMemoryBuilder<TNext, Guid> AddRepositoryInMemoryStorageWithGuidKey<TNext>(Action<RepositoryBehaviorSettings<TNext, Guid>>? settings = default)
    

How to populate with random data?

Simple random (example)

Populate your in memory storage with 120 users

var builder = WebApplication.CreateBuilder(args);
builder.Services.AddRepositoryInMemoryStorageWithStringKey<User>()
.PopulateWithRandomData(x => x.Email!, 120);

and in app after build during startup of your application

var app = builder.Build();
app.Services.Populate();
Simple random with regex (example)

Populate your in memory storage with 100 users and property Email with a random regex @"[a-z]{4,10}@gmail.com"

.AddRepositoryInMemoryStorage<User, string>()
.PopulateWithRandomData()
.WithPattern(x => x.Email, @"[a-z]{4,10}@gmail\.com")
.Populate(x => x.Id!, 100)
Where can I use the regex pattern?

You can use regex pattern on all primitives type and most used structs.

Complete list:
int, uint, byte, sbyte, short, ushort, long, ulong, nint, nuint, float, double, decimal, bool, char, Guid, DateTime, TimeSpan, Range, string, int?, uint?, byte?, sbyte?, short?, ushort?, long?, ulong?, nint?, nuint?, float?, double?, decimal?, bool?, char?, Guid?, DateTime?, TimeSpan?, Range?, string?

You can use the pattern in Class, IEnumerable, IDictionary, or Array, and in everything that extends IEnumerable or IDictionary

Important!! You can override regex service in your DI

public static IServiceCollection AddRegexService<T>(
        this IServiceCollection services)
        where T : class, IRegexService
IEnumerable or Array one-dimension (example)

You have your model x (User) that has a property Groups as IEnumerable or something that extends IEnumerable, Groups is a class with a property Id as string. In the code below you are creating a list of class Groups with 8 elements in each 100 User instances, in each element of Groups you randomize based on this regex "[a-z]{4,5}". You may take care of use First() linq method to set correctly the Id property.

.AddRepositoryInMemoryStorage<User, string>()
.PopulateWithRandomData(x => x.Id!, 100, 8)
.WithPattern(x => x.Groups!.First().Id, "[a-z]{4,5}")

and after build in your startup

app.Services.Populate()
IDictionary (example)

Similar to IEnumerable population you may populate your Claims property (a dictionary) with random key but with values based on regular expression "[a-z]{4,5}". As well as IEnumerable implementation you will have 6 elements (because I choose to create 6 elements in Populate method)

.AddRepositoryInMemoryStorage<User, string>()
.PopulateWithRandomData(x => x.Id!, 100, 6)
.WithPattern(x => x.Claims!.First().Value, "[a-z]{4,5}")

and after build in your startup

app.Services.Populate()

or if you have in Value an object

AddRepositoryInMemoryStorage<User, string>()
.PopulateWithRandomData(x => x.Id!, 100, 6)
.WithPattern(x => x.Claims!.First().Value.SomeProperty, "[a-z]{4,5}")

and after build in your startup

app.Services.Populate()

Populate with delegation

Similar to regex pattern, you can use a delegation to populate something.

Dictionary (example)

Here you can see that all 6 elements in each 100 users are populated in Value with string "A"

.AddRepositoryPatternInMemoryStorage<User, string>()
.PopulateWithRandomData(x => x.Id!, 100, 6)
.WithPattern(x => x.Claims!.First().Value, () => "A"))

and after build in your startup

app.Services.Populate()

Populate with Implementation

If you have an interface or abstraction in your model, you can specify an implementation type for population. You have two different methods, with typeof

.AddRepositoryInMemoryStorageWithStringKey<PopulationTest>()
.PopulateWithRandomData()
.WithImplementation(x => x.I, typeof(MyInnerInterfaceImplementation))

or generics

.AddRepositoryInMemoryStorageWithStringKey<PopulationTest>()
.PopulateWithRandomData()
.WithImplementation<IInnerInterface, MyInnerInterfaceImplementation>(x => x.I!)

In Memory, simulate real implementation

If you want to test with possible exceptions (for your reliability tests) and waiting time (for your load tests) you may do it with this library and in memory behavior settings.

Add random exceptions

You can set different custom exceptions and different percentage for each operation: Delete, Get, Insert, Update, Query. In the code below I'm adding three exceptions with a percentage of throwing them, they are the same for each operation. I have a 0.45% for normal Exception, 0.1% for "Big Exception" and 0.548% for "Great Exception"

.AddRepositoryInMemoryStorage<User, string>(options =>
{
    var customExceptions = new List<ExceptionOdds>
    {
        new ExceptionOdds()
        {
            Exception = new Exception(),
            Percentage = 0.45
        },
        new ExceptionOdds()
        {
            Exception = new Exception("Big Exception"),
            Percentage = 0.1
        },
        new ExceptionOdds()
        {
            Exception = new Exception("Great Exception"),
            Percentage = 0.548
        }
    };
    options.ExceptionOddsForDelete.AddRange(customExceptions);
    options.ExceptionOddsForGet.AddRange(customExceptions);
    options.ExceptionOddsForInsert.AddRange(customExceptions);
    options.ExceptionOddsForUpdate.AddRange(customExceptions);
    options.ExceptionOddsForQuery.AddRange(customExceptions);
})

Add random waiting time

You can set different range in milliseconds for each operation. In the code below I'm adding a same custom range for Delete, Insert, Update, Get between 1000ms and 2000ms, and a unique custom range for Query between 3000ms and 7000ms.

 .AddRepositoryInMemoryStorage<User, string>(options =>
  {
      var customRange = new Range(1000, 2000);
      options.MillisecondsOfWaitForDelete = customRange;
      options.MillisecondsOfWaitForInsert = customRange;
      options.MillisecondsOfWaitForUpdate = customRange;
      options.MillisecondsOfWaitForGet = customRange;
      options.MillisecondsOfWaitForQuery = new Range(3000, 7000);
  })   

Add random waiting time before an exception

You can set different range in milliseconds for each operation before to throw an exception. In the code below I'm adding a same custom range for Delete, Insert, Update, Get between 1000ms and 2000ms, and a unique custom range for Query between 3000ms and 7000ms in case of exception.

.AddRepositoryInMemoryStorage<User, string>(options =>
{
    var customRange = new Range(1000, 2000);
    options.MillisecondsOfWaitBeforeExceptionForDelete = customRange;
    options.MillisecondsOfWaitBeforeExceptionForInsert = customRange;
    options.MillisecondsOfWaitBeforeExceptionForUpdate = customRange;
    options.MillisecondsOfWaitBeforeExceptionForGet = customRange;
    options.MillisecondsOfWaitBeforeExceptionForQuery = new Range(3000, 7000);
    var customExceptions = new List<ExceptionOdds>
    {
        new ExceptionOdds()
        {
            Exception = new Exception(),
            Percentage = 0.45
        },
        new ExceptionOdds()
        {
            Exception = new Exception("Big Exception"),
            Percentage = 0.1
        },
        new ExceptionOdds()
        {
            Exception = new Exception("Great Exception"),
            Percentage = 0.548
        }
    };
    options.ExceptionOddsForDelete.AddRange(customExceptions);
    options.ExceptionOddsForGet.AddRange(customExceptions);
    options.ExceptionOddsForInsert.AddRange(customExceptions);
    options.ExceptionOddsForUpdate.AddRange(customExceptions);
    options.ExceptionOddsForQuery.AddRange(customExceptions);
})
Product Compatible and additional computed target framework versions.
.NET 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 was computed.  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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • net6.0

    • No dependencies.

NuGet packages (12)

Showing the top 5 NuGet packages that depend on Rystem.RepositoryFramework.Abstractions:

Package Downloads
Rystem.RepositoryFramework.Api.Server

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

Rystem.RepositoryFramework.Api.Client

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

Rystem.RepositoryFramework.Infrastructure.Azure.Storage.Blob

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

Rystem.RepositoryFramework.Infrastructure.Azure.Storage.Table

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

Rystem.RepositoryFramework.Cache

Rystem.RepositoryFramework allows you to use correctly concepts like repository pattern, CQRS and DDD. You have interfaces for your domains, auto-generated api, auto-generated HttpClient to simplify connection "api to front-end", a functionality for auto-population in memory of your models, a functionality to simulate exceptions and waiting time from external sources to improve your implementation/business test and load test.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
6.0.21 180 6/18/2024
6.0.20 93,986 6/16/2024
6.0.19 30,454 6/14/2024
6.0.18 167 6/14/2024
6.0.17 163 6/14/2024
6.0.16 50,008 6/10/2024
6.0.15 163 6/9/2024
6.0.14 94,376 5/24/2024
6.0.13 178 5/23/2024
6.0.12 137 5/23/2024
6.0.11 200 5/20/2024
6.0.9 207 5/19/2024
6.0.7 174 5/18/2024
6.0.6 177 5/10/2024
6.0.5 162 5/10/2024
6.0.4 550,136 4/3/2024
6.0.3 734 3/25/2024
6.0.2 378,095 3/11/2024
6.0.1 51,140 3/8/2024
6.0.0 1,171,915 11/21/2023
6.0.0-rc.6 128 10/25/2023
6.0.0-rc.5 96 10/25/2023
6.0.0-rc.4 82 10/23/2023
6.0.0-rc.3 73 10/19/2023
6.0.0-rc.2 110 10/18/2023
6.0.0-rc.1 121 10/16/2023
5.0.20 641,296 9/25/2023
5.0.19 3,087 9/10/2023
5.0.18 2,341 9/6/2023
5.0.17 2,269 9/6/2023
5.0.16 2,307 9/5/2023
5.0.15 2,246 9/5/2023
5.0.14 2,262 9/5/2023
5.0.13 2,352 9/1/2023
5.0.12 2,205 8/31/2023
5.0.11 2,217 8/30/2023
5.0.10 2,263 8/29/2023
5.0.9 2,306 8/24/2023
5.0.8 2,363 8/24/2023
5.0.7 451,828 8/23/2023
5.0.6 19,730 8/21/2023
5.0.5 6,489 8/21/2023
5.0.4 2,383 8/16/2023
5.0.3 214,916 8/2/2023
5.0.2 4,201 8/2/2023
5.0.1 13,960 8/1/2023
5.0.0 14,358 7/31/2023
4.1.26 143,237 7/20/2023
4.1.25 27,039 7/16/2023
4.1.24 400,439 6/13/2023
4.1.23 48,130 6/13/2023
4.1.22 132,007 5/30/2023
4.1.21 58,107 5/20/2023
4.1.20 407,484 4/19/2023
4.1.19 98,238 3/20/2023
4.1.18 2,784 3/20/2023
4.1.17 3,043 3/16/2023
4.1.16 2,804 3/16/2023
4.1.15 2,753 3/15/2023
4.1.14 10,460 3/9/2023
4.1.13 2,897 3/7/2023
4.1.12 3,348 2/9/2023
4.1.11 2,972 1/26/2023
4.1.10 3,178 1/22/2023
4.1.9 2,836 1/20/2023
4.1.8 3,104 1/18/2023
4.1.7 3,079 1/18/2023
4.1.6 3,142 1/17/2023
4.1.1 3,095 1/4/2023
4.1.0 2,981 1/1/2023
3.1.5 2,934 12/21/2022
3.1.4 1,505 12/21/2022
3.1.3 3,237 12/12/2022
3.1.2 2,987 12/7/2022
3.1.1 3,075 12/7/2022
3.1.0 3,062 12/1/2022
3.0.29 3,038 12/1/2022
3.0.28 3,828 12/1/2022
3.0.27 3,225 11/23/2022
3.0.25 6,402 11/23/2022
3.0.24 4,433 11/18/2022
3.0.23 4,111 11/18/2022
3.0.22 4,246 11/15/2022
3.0.21 4,301 11/14/2022
3.0.20 4,354 11/13/2022
3.0.19 4,656 11/2/2022
3.0.18 4,384 11/2/2022
3.0.17 4,465 10/29/2022
3.0.16 4,530 10/29/2022
3.0.15 1,605 10/29/2022
3.0.14 7,300 10/24/2022
3.0.13 4,591 10/24/2022
3.0.12 4,595 10/17/2022
3.0.11 4,551 10/10/2022
3.0.10 4,126 10/6/2022
3.0.9 4,061 10/6/2022
3.0.8 4,024 10/6/2022
3.0.7 4,146 10/6/2022
3.0.6 4,141 10/5/2022
3.0.5 4,037 10/5/2022
3.0.4 4,112 10/5/2022
3.0.3 4,097 10/3/2022
3.0.2 4,136 9/30/2022
3.0.1 4,095 9/29/2022
3.0.0 1,649 9/29/2022
2.0.17 3,718 9/29/2022
2.0.16 4,185 9/27/2022
2.0.15 4,321 9/27/2022
2.0.14 4,214 9/26/2022
2.0.13 4,142 9/26/2022
2.0.12 4,150 9/26/2022
2.0.11 4,080 9/25/2022
2.0.10 4,286 9/25/2022
2.0.9 4,159 9/22/2022
2.0.8 4,084 9/22/2022
2.0.7 1,660 9/22/2022
2.0.6 4,098 9/20/2022
2.0.5 4,300 9/20/2022
2.0.4 4,165 9/20/2022
2.0.2 4,188 9/20/2022
2.0.1 4,396 9/13/2022
2.0.0 4,264 8/19/2022
1.1.24 4,294 7/30/2022
1.1.23 4,235 7/29/2022
1.1.22 4,048 7/29/2022
1.1.21 4,442 7/29/2022
1.1.20 4,217 7/29/2022
1.1.19 4,230 7/27/2022
1.1.17 4,256 7/27/2022
1.1.16 4,232 7/26/2022
1.1.15 4,281 7/25/2022
1.1.14 4,258 7/25/2022
1.1.13 4,139 7/22/2022
1.1.12 4,127 7/19/2022
1.1.11 4,225 7/19/2022
1.1.10 4,191 7/19/2022
1.1.9 4,244 7/19/2022
1.1.8 4,287 7/18/2022
1.1.7 4,125 7/18/2022
1.1.6 4,204 7/18/2022
1.1.5 4,209 7/17/2022
1.1.4 1,627 7/17/2022
1.1.3 6,663 7/17/2022
1.1.2 4,264 7/17/2022
1.1.1 1,662 7/17/2022
1.1.0 4,216 7/17/2022
1.0.2 4,220 7/15/2022
1.0.1 2,926 7/15/2022
1.0.0 5,398 7/8/2022
0.10.7 4,222 7/7/2022
0.10.6 1,722 7/7/2022
0.10.3 2,162 7/7/2022
0.10.2 5,640 7/2/2022
0.10.1 4,228 7/1/2022
0.10.0 4,019 7/1/2022
0.9.10 5,278 6/20/2022
0.9.9 4,269 6/11/2022
0.9.8 1,648 6/10/2022
0.9.7 4,155 6/9/2022
0.9.6 4,147 6/5/2022
0.9.5 2,787 6/3/2022
0.9.3 4,047 6/3/2022
0.9.2 2,409 5/31/2022
0.9.1 2,347 5/31/2022
0.9.0 2,350 5/31/2022
0.8.3-beta.1 101 5/31/2022
0.8.2 1,683 5/30/2022
0.8.1 1,703 5/29/2022