Rystem.RepositoryFramework.Abstractions 6.0.4

dotnet add package Rystem.RepositoryFramework.Abstractions --version 6.0.4
NuGet\Install-Package Rystem.RepositoryFramework.Abstractions -Version 6.0.4
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="6.0.4" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Rystem.RepositoryFramework.Abstractions --version 6.0.4
#r "nuget: Rystem.RepositoryFramework.Abstractions, 6.0.4"
#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=6.0.4

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

What is Rystem?

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 ICommandPattern<T, TKey> : ICommandPattern
    where TKey : notnull
{
    Task<State<T, TKey>> InsertAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<State<T, TKey>> UpdateAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<State<T, TKey>> DeleteAsync(TKey key, CancellationToken cancellationToken = default);
    IAsyncEnumerable<BatchResult<T, TKey>> BatchAsync(BatchOperations<T, TKey> operations, CancellationToken cancellationToken = default);
}
Query (Read)
public interface IQueryPattern<T, TKey> : IQueryPattern
    where TKey : notnull
{
    Task<State<T, TKey>> ExistAsync(TKey key, CancellationToken cancellationToken = default);
    Task<T?> GetAsync(TKey key, CancellationToken cancellationToken = default);
    IAsyncEnumerable<IEntity<T, TKey>> QueryAsync(IFilterExpression filter, CancellationToken cancellationToken = default);
    ValueTask<TProperty> OperationAsync<TProperty>(OperationType<TProperty> operation, IFilterExpression filter, CancellationToken cancellationToken = default);
}
Repository Pattern (Write-Delete-Read)

Repository pattern is a sum of CQRS interfaces.

public interface IRepositoryPattern<T, TKey> : ICommandPattern<T, TKey>, IQueryPattern<T, TKey>, IRepositoryPattern, ICommandPattern, IQueryPattern
    where TKey : notnull
{
    Task<State<T, TKey>> InsertAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<State<T, TKey>> UpdateAsync(TKey key, T value, CancellationToken cancellationToken = default);
    Task<State<T, TKey>> DeleteAsync(TKey key, CancellationToken cancellationToken = default);
    IAsyncEnumerable<BatchResult<T, TKey>> BatchAsync(BatchOperations<T, TKey> operations, CancellationToken cancellationToken = default);
    Task<State<T, TKey>> ExistAsync(TKey key, CancellationToken cancellationToken = default);
    Task<T?> GetAsync(TKey key, CancellationToken cancellationToken = default);
    IAsyncEnumerable<IEntity<T, TKey>> QueryAsync(IFilterExpression filter, CancellationToken cancellationToken = default);
    ValueTask<TProperty> OperationAsync<TProperty>(OperationType<TProperty> operation, IFilterExpression filter, CancellationToken cancellationToken = default);
}

Examples

Model
public class User
{
    public string Id { get; set; }
    public string Name { get; set; }
    public string Email { get; set; }
}
Command

Your storage class has to extend ICommand, and use it on injection

public class UserWriter : ICommand<User, string>
{
    public Task<State<User, string>> DeleteAsync(string key, CancellationToken cancellationToken = default)
    {
        //delete on with DB or storage context
        throw new NotImplementedException();
    }
    public Task<State<User, string>> InsertAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //insert on DB or storage context
        throw new NotImplementedException();
    }
    public Task<State<User, string>> UpdateAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //update on DB or storage context
        throw new NotImplementedException();
    }
    public Task<BatchResults<User, string>> BatchAsync(BatchOperations<User, string> operations, CancellationToken cancellationToken = default)
    {
        //insert, update or delete some items on DB or storage context
        throw new NotImplementedException();
    }
}
Query

Your storage class has to extend IQuery, and use it on injection

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<State<User, string>> ExistAsync(string key, CancellationToken cancellationToken = default)
    {
        //check if an item by key exists in DB or storage context
        throw new NotImplementedException();
    }
    public IAsyncEnumerable<IEntity<User, string>> QueryAsync(IFilterExpression filter, CancellationToken cancellationToken = default)
    {
        //get a list of items by a predicate with top and skip from DB or storage context
        throw new NotImplementedException();
    }
    public ValueTask<TProperty> OperationAsync<TProperty>(OperationType<TProperty> operation, IFilterExpression filter, CancellationToken cancellationToken = default)
    {
        //get an items count by a predicate with top and skip from DB or storage context or max or min or some other operations
        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). You may choose to extend IRepository, but when you inject you have to use IRepository

public class UserRepository : IRepository<User, string>, IQuery<User, string>, ICommand<User, string>
{
    public Task<State<User, string>> DeleteAsync(string key, CancellationToken cancellationToken = default)
    {
        //delete on with DB or storage context
        throw new NotImplementedException();
    }
    public Task<State<User, string>> InsertAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //insert on DB or storage context
        throw new NotImplementedException();
    }
    public Task<State<User, string>> UpdateAsync(string key, User value, CancellationToken cancellationToken = default)
    {
        //update on DB or storage context
        throw new NotImplementedException();
    }
    public Task<BatchResults<User, string>> BatchAsync(BatchOperations<User, string> operations, CancellationToken cancellationToken = default)
    {
        //insert, update or delete some items 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<State<User, string>> ExistAsync(string key, CancellationToken cancellationToken = default)
    {
        //check if an item by key exists in DB or storage context
        throw new NotImplementedException();
    }
    public IAsyncEnumerable<IEntity<User, string>> QueryAsync(IFilterExpression filter, CancellationToken cancellationToken = default)
    {
        //get a list of items by a predicate with top and skip from DB or storage context
        throw new NotImplementedException();
    }
    public ValueTask<TProperty> OperationAsync<TProperty>(OperationType<TProperty> operation, IFilterExpression filter, CancellationToken cancellationToken = default)
    {
        //get an items count by a predicate with top and skip from DB or storage context or max or min or some other operations
        throw new NotImplementedException();
    }
}

How to use it

In DI you install the service. Here an example on how to set a custom storage, prepare a translation (to translate name of your properties for query during filtering), and AddBusiness for your integration. Furthermore you may use the factory integration from Rystem.

var factoryName = "storage";
 services.AddRepository<AppUser, AppUserKey>(builder =>
{
    builder.SetStorage<AppUserStorage>(factoryName);
    builder.Translate<User>()
        .With(x => x.Id, x => x.Identificativo)
        .With(x => x.Username, x => x.Nome)
        .With(x => x.Email, x => x.IndirizzoElettronico);
    builder
        .AddBusiness()
            .AddBusinessBeforeInsert<AppUserBeforeInsertBusiness>()
            .AddBusinessBeforeInsert<AppUserBeforeInsertBusiness2>();
});

And you may inject the object

Please, use IRepository and not IRepositoryPattern

IRepository<AppUser, AppUserKey> repository

Query and Command

In DI you install the services

services.AddCommand<AppUser, AppUserKey>(...);
services.AddQuery<AppUser, AppUserKey>(...);

And you may inject the objects

Please, use ICommand, IQuery and not ICommandPattern, IQueryPattern

ICommand<AppUser, AppUserKey> command
IQuery<AppUser, AppUserKey> query

TKey when it's not a primitive

You can use a class or record. Record is better in my opinion, for example, if you want to use the Equals operator from key, with record you don't check it by the refence but by the value of the properties in the record. My key:

public class MyKey 
{
    public int Id { get; set; }
    public int Id2 { get; set; }
}

the DI

services.AddRepository<User, MyKey>(...);

and you may inject (for ICommand and IQuery is the same)

IRepository<User, MyKey> repository

IKey interface

You may implement the IKey interface to decide how to work with your key. Here an example with Parse and AsString method and custom implementation with separator $.

public class ClassicKey : IKey
{
    public string A { get; set; }
    public int B { get; set; }
    public double C { get; set; }

    public static IKey Parse(string keyAsString)
    {
        var splitted = keyAsString.Split('$');
        return new ClassicKey { A = splitted[0], B = int.Parse(splitted[1]), C = double.Parse(splitted[2]) };
    }

    public string AsString()
    {
        return $"{A}${B}${C}";
    }
}

IDefaultKey

You may implement IDefaultKey if you want a simple key preconstructed parser.

public class DefaultKey : IDefaultKey
{
    public string A { get; set; }
    public int B { get; set; }
    public double C { get; set; }
}

Automatically you can call AsString to receive a string composed by all properties separated by triple |, for instance {A}|||{B}|||{C}. You can decide during startup the separator in two ways. One with ServiceCollectionExtensions

builder.Services.AddDefaultSeparatorForDefaultKeyInterface("$$$");

the other one with a static method offered by IDefaultKey interface

IDefaultKey.SetDefaultSeparator("$$$");

Default TKey record

You may use the default record key in repository framework namespace. It's not really useful when used with no-primitive or no-struct objects (in terms of memory usage [Heap]). For 1 value (it's not really useful I know, but I liked to create it).

new Key<int>(2);

or for 2 values (useful)

new Key<int, int>(2, 4);

or for 3 values (unuseful)

new Key<int, int, string>(2, 4, "312");

or for 4 values (useful)

new Key<int, int, double, Guid>(2, 4, 3, Guid.NewGuid());

or for 5 values (unuseful)

new Key<int, int, string, Guid, string>(2, 4, "312", Guid.NewGuid(), "3232");

the DI

services.AddRepository<User, Key<int, int>, UserRepository>();

and you may inject (for ICommand and IQuery is the same)

IRepository<User, Key<int, int>> repository

Translation

In some cases you need to "translate" your query for your database context query, for example in case of EF integration.

services.AddDbContext<SampleContext>(options =>
{
    options.UseSqlServer(configuration["ConnectionString:Database"]);
}, ServiceLifetime.Scoped);
services.AddRepository<AppUser, AppUserKey>(repositoryBuilder =>
{
    repositoryBuilder.SetStorage<AppUserStorage>();
    repositoryBuilder.Translate<User>()
        .With(x => x.Id, x => x.Identificativo)
        .With(x => x.Username, x => x.Nome)
        .With(x => x.Email, x => x.IndirizzoElettronico);
});

In this case I'm helping the Filter class to understand how to transform itself when used in a different context. Use Filter methods to help to translate and apply to your context the right query.

await foreach (var user in filter.ApplyAsAsyncEnumerable(_context.Users))
    yield return new AppUser(user.Identificativo, user.Nome, user.IndirizzoElettronico, new());

You may use Filter for queryable, FilterAsEnumerable for Enumerable and FilterAsAsyncEnumerable for async enumerable context.

You can add more translations for the same model

services.AddDbContext<SampleContext>(options =>
{
    options.UseSqlServer(configuration["ConnectionString:Database"]);
}, ServiceLifetime.Scoped);
services.AddRepository<AppUser, AppUserKey>(repositoryBuilder =>
{
    repositoryBuilder.SetStorage<AppUserStorage>();
    repositoryBuilder.Translate<User>()
        .With(x => x.Id, x => x.Identificativo)
        .With(x => x.Username, x => x.Nome)
        .With(x => x.Email, x => x.IndirizzoElettronico);
    repositoryBuilder
        .AddBusiness()
            .AddBusinessBeforeInsert<AppUserBeforeInsertBusiness>()
            .AddBusinessBeforeInsert<AppUserBeforeInsertBusiness2>();
});

Entity framework examples

Here you may find the example Repository pattern applied Unit test flow

Business Manager

You have the chance to write your business methods to execute them before or after a command or query. For instance, you have to check before an update or insert the value of an entity and deny the final insert/update on the database.

Example

In this example BeforeInsertAsync runs before InsertAsync of IRepository/ICommand and AfterInsertAsync runs after InsertAsync of IRepository/ICommand.

.AddRepository<Animal, long>(builder => {
    builder.
        WithInMemory();
    builder
        .AddBusiness()
            .AddBusinessAfterInsert<AnimalBusiness>()
            .AddBusinessBeforeInsert<AnimalBusiness>();
});
    

more interesting usage comes to move business in another project, you can add to your infrastructure in the following way

.AddBusinessForRepository<Animal, long>(builder => {
    builder
        .AddBusiness()
            .AddBusinessAfterInsert<AnimalBusiness>()
            .AddBusinessBeforeInsert<AnimalBusiness>();
});
   

Then, you could have a library for infrastructure (or more than one) and a library for business to separate furthermore the concepts.

The animal business to inject will be the following one.

public sealed class AnimalBusiness : IRepositoryBusinessBeforeInsert<Animal, long>, IRepositoryBusinessAfterInsert<Animal, long>
{
    public static int After;
    public Task<State<Animal>> AfterInsertAsync(State<Animal, long> state, Entity<Animal, long> entity, CancellationToken cancellationToken = default)
    {
        After++;
        return Task.FromResult(state);
    }

    public static int Before;
    public Task<State<Animal, long>> BeforeInsertAsync(Entity<Animal, long> entity, CancellationToken cancellationToken = default)
    {
        Before++;
        return Task.FromResult(State.Ok(entity));
    }
}

You have to create the class as public to allow the dependency injection to instastiate it. It's added directly to DI.

Factory

Integration with factory from rystem is hidden in the framework, and it's ready to be used. For instance here i'm installing two different repositories for the same model and key.

var builder = WebApplication.CreateBuilder(args);
builder.Services
.AddRepository<SuperUser, string>(settins =>
{
    settins.WithInMemory(builder =>
    {
        builder
            .PopulateWithRandomData(120, 5)
            .WithPattern(x => x.Value!.Email, @"[a-z]{5,10}@gmail\.com");
    });
    settins.WithInMemory(builder =>
    {
        builder
            .PopulateWithRandomData(2, 5)
            .WithPattern(x => x.Value!.Email, @"[a-z]{5,10}@gmail\.com");
    }, "inmemory");
});

Usage

var serviceProvider = ....;
IFactory<IRepository<SuperUser, string>> superUserFactory = serviceProvider.GetRequiredService<IFactory<IRepository<SuperUser, string>>>();
var firstIntegration = superUserFactory.Create();
var secondIntegration = superUserFactory.Create("inmemory");

By default is injected directly the last one repository integration installed.

var serviceProvider = ....;
IRepository<SuperUser, string> secondIntegration = serviceProvider.GetRequiredService<IRepository<SuperUser, string>>();

Here you find the "inmemory" integration. You can use the decorator pattern offered by Rystem for your integration to decorate an IRepository<T, TKey> or ICommand<T, TKey> or IQuery<T, TKey>.

Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  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.

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.4 213,153 4/3/2024
6.0.3 543 3/25/2024
6.0.2 378,087 3/11/2024
6.0.1 51,137 3/8/2024
6.0.0 1,171,904 11/21/2023
6.0.0-rc.6 127 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,267 9/25/2023
5.0.19 3,079 9/10/2023
5.0.18 2,338 9/6/2023
5.0.17 2,266 9/6/2023
5.0.16 2,304 9/5/2023
5.0.15 2,243 9/5/2023
5.0.14 2,259 9/5/2023
5.0.13 2,349 9/1/2023
5.0.12 2,195 8/31/2023
5.0.11 2,214 8/30/2023
5.0.10 2,259 8/29/2023
5.0.9 2,299 8/24/2023
5.0.8 2,360 8/24/2023
5.0.7 451,825 8/23/2023
5.0.6 19,727 8/21/2023
5.0.5 6,486 8/21/2023
5.0.4 2,375 8/16/2023
5.0.3 214,908 8/2/2023
5.0.2 4,198 8/2/2023
5.0.1 13,957 8/1/2023
5.0.0 14,354 7/31/2023
4.1.26 143,233 7/20/2023
4.1.25 27,036 7/16/2023
4.1.24 400,435 6/13/2023
4.1.23 48,122 6/13/2023
4.1.22 131,925 5/30/2023
4.1.21 58,104 5/20/2023
4.1.20 407,478 4/19/2023
4.1.19 98,234 3/20/2023
4.1.18 2,781 3/20/2023
4.1.17 3,039 3/16/2023
4.1.16 2,801 3/16/2023
4.1.15 2,750 3/15/2023
4.1.14 10,454 3/9/2023
4.1.13 2,894 3/7/2023
4.1.12 3,329 2/9/2023
4.1.11 2,966 1/26/2023
4.1.10 3,173 1/22/2023
4.1.9 2,816 1/20/2023
4.1.8 3,099 1/18/2023
4.1.7 3,068 1/18/2023
4.1.6 3,122 1/17/2023
4.1.1 3,092 1/4/2023
4.1.0 2,978 1/1/2023
3.1.5 2,915 12/21/2022
3.1.4 1,502 12/21/2022
3.1.3 3,227 12/12/2022
3.1.2 2,984 12/7/2022
3.1.1 3,072 12/7/2022
3.1.0 3,059 12/1/2022
3.0.29 3,019 12/1/2022
3.0.28 3,825 12/1/2022
3.0.27 3,216 11/23/2022
3.0.25 5,933 11/23/2022
3.0.24 4,430 11/18/2022
3.0.23 4,108 11/18/2022
3.0.22 4,245 11/15/2022
3.0.21 4,300 11/14/2022
3.0.20 4,353 11/13/2022
3.0.19 4,655 11/2/2022
3.0.18 4,383 11/2/2022
3.0.17 4,461 10/29/2022
3.0.16 4,529 10/29/2022
3.0.15 1,602 10/29/2022
3.0.14 7,295 10/24/2022
3.0.13 4,588 10/24/2022
3.0.12 4,588 10/17/2022
3.0.11 4,550 10/10/2022
3.0.10 4,125 10/6/2022
3.0.9 4,058 10/6/2022
3.0.8 4,021 10/6/2022
3.0.7 4,143 10/6/2022
3.0.6 4,138 10/5/2022
3.0.5 4,036 10/5/2022
3.0.4 4,111 10/5/2022
3.0.3 4,094 10/3/2022
3.0.2 4,134 9/30/2022
3.0.1 4,094 9/29/2022
3.0.0 1,648 9/29/2022
2.0.17 3,716 9/29/2022
2.0.16 4,182 9/27/2022
2.0.15 4,320 9/27/2022
2.0.14 4,211 9/26/2022
2.0.13 4,139 9/26/2022
2.0.12 4,131 9/26/2022
2.0.11 4,077 9/25/2022
2.0.10 4,283 9/25/2022
2.0.9 4,156 9/22/2022
2.0.8 4,083 9/22/2022
2.0.7 1,659 9/22/2022
2.0.6 4,097 9/20/2022
2.0.5 4,299 9/20/2022
2.0.4 4,155 9/20/2022
2.0.2 4,187 9/20/2022
2.0.1 4,389 9/13/2022
2.0.0 4,262 8/19/2022
1.1.24 4,291 7/30/2022
1.1.23 4,230 7/29/2022
1.1.22 4,047 7/29/2022
1.1.21 4,441 7/29/2022
1.1.20 4,216 7/29/2022
1.1.19 4,229 7/27/2022
1.1.17 4,255 7/27/2022
1.1.16 4,231 7/26/2022
1.1.15 4,280 7/25/2022
1.1.14 4,257 7/25/2022
1.1.13 4,138 7/22/2022
1.1.12 4,126 7/19/2022
1.1.11 4,224 7/19/2022
1.1.10 4,190 7/19/2022
1.1.9 4,243 7/19/2022
1.1.8 4,279 7/18/2022
1.1.7 4,124 7/18/2022
1.1.6 4,203 7/18/2022
1.1.5 4,201 7/17/2022
1.1.4 1,626 7/17/2022
1.1.3 6,660 7/17/2022
1.1.2 4,263 7/17/2022
1.1.1 1,661 7/17/2022
1.1.0 4,215 7/17/2022
1.0.2 4,212 7/15/2022
1.0.1 2,912 7/15/2022
1.0.0 5,397 7/8/2022
0.10.7 4,219 7/7/2022
0.10.6 1,721 7/7/2022
0.10.3 2,161 7/7/2022
0.10.2 5,637 7/2/2022
0.10.1 4,227 7/1/2022
0.10.0 4,018 7/1/2022
0.9.10 5,276 6/20/2022
0.9.9 4,268 6/11/2022
0.9.8 1,647 6/10/2022
0.9.7 4,154 6/9/2022
0.9.6 4,146 6/5/2022
0.9.5 2,786 6/3/2022
0.9.3 4,046 6/3/2022
0.9.2 2,407 5/31/2022
0.9.1 2,346 5/31/2022
0.9.0 2,349 5/31/2022
0.8.3-beta.1 100 5/31/2022
0.8.2 1,680 5/30/2022
0.8.1 1,702 5/29/2022