CustomDapperEntityActions 1.0.33
dotnet add package CustomDapperEntityActions --version 1.0.33
NuGet\Install-Package CustomDapperEntityActions -Version 1.0.33
<PackageReference Include="CustomDapperEntityActions" Version="1.0.33" />
<PackageVersion Include="CustomDapperEntityActions" Version="1.0.33" />
<PackageReference Include="CustomDapperEntityActions" />
paket add CustomDapperEntityActions --version 1.0.33
#r "nuget: CustomDapperEntityActions, 1.0.33"
#:package CustomDapperEntityActions@1.0.33
#addin nuget:?package=CustomDapperEntityActions&version=1.0.33
#tool nuget:?package=CustomDapperEntityActions&version=1.0.33
CustomDapperEntityActions
A .NET package that provides Entity Framework Core-like actions using Dapper, designed for developers whose companies restrict the use of EF Core.
Overview
CustomDapperEntityActions bridges the gap between Dapper's lightweight performance and Entity Framework Core's convenient entity management features. It offers familiar entity action patterns while maintaining Dapper's speed and flexibility.
Features
- EF Core-style entity operations
- Built on top of Dapper for optimal performance
- Compatible with .NET 9.0
- Support for common CRUD operations
Installation
Install via NuGet Package Manager: dotnet add package CustomDapperEntityActions --version 1.0.6
Link to nuget: https://www.nuget.org/packages/CustomDapperEntityActions/
How to use
- Define entities
[Table("[schema].[table]")]
public class Plant
{
[Key]
[Column("id")]
public int Id { get; set; }
[Column("description")]
public string? Description { get; set; }
[Column("is_active")]
public bool IsAcive { get; set; }
}
1.1 Define Entities with 1..n relationship
// Parent table
[Table("parent")]
public class Parent
{
[Key]
[Column("parent_id")]
public int ParentId { get; set; }
[Column("name")]
public required string Name { get; set; }
[NotMapped]
public List<Child> Children { get; set; } = new List<Child>(); // initialise is important and needed
}
// Child table
using System.ComponentModel.DataAnnotations.Schema;
using System.ComponentModel.DataAnnotations;
[Table("child")]
public class Child
{
[Key]
[Column("child_id")]
public int ChildId { get; set; }
// foreign key to parent necessary
[Column("parent_id")]
[ForeignKey("Parent")]
public int ParentId { get; set; }
[Column("name")]
public required string Name { get; set; }
[Column("is_active")]
public bool IsActive { get; set; }
}
- Create DbContext
public class ApplicationDbContext : IDisposable
{
// if using postgres, sql also possible
private readonly NpgsqlConnection _connection;
private IDbTransaction _transaction;
// Expose the connection for custom transaction management
public IDbConnection Connection => _connection;
// Expose the current transaction
public IDbTransaction Transaction => _transaction;
// The entity we just made
public DapperSet<Plant> Plants { get; }
public ApplicationDbContext(string connectionString)
{
_connection = new NpgsqlConnection(connectionString);
_connection.Open();
_transaction = _connection.BeginTransaction();
// assign them
Plants = new DapperSet<Plant>(_connection, _transaction);
}
public void Commit() => _transaction?.Commit();
public void Rollback() => _transaction?.Rollback();
public void Dispose()
{
_transaction?.Dispose();
_connection?.Dispose();
}
}
- Update program.cs
bld.Services.AddScoped<ApplicationDbContext>(provider => new ApplicationDbContext(connectionString));
- usage
// insert
var insert = await dbContext.Tests.InsertAsync<string>(new TestLalala
{
Description = "Test",
IsAcive = true
});
// update
var update = await dbContext.Tests.UpdateAsync(new TestLalala
{
Id = 5,
Description = "Luksass",
IsAcive = true
});
// Get byid
await dbContext.Tests.GetByIdAsync(5);
await dbContext.Tests.DeleteAsync(5);
// IMPORTANT: You must call Commit() to persist changes to the database
// after Insert, Update, or Delete operations
dbContext.Commit();
// Linq like query NEW version
var specificTests = await dbContext.Tests
.Where(x => x.TestCd == "Zoko")
.OrderBy(x => x.Description)
.ExecuteAsync();
// without where also possible
var specificTests = await dbContext.Tests
.OrderBy(x => x.Description)
.ExecuteAsync();
// only where also possible
var specificTests = await dbContext.Tests
.Where(x => x.TestCd == "Zoko")
// descending
var specificTests = await dbContext.Tests
.OrderBy(x => x.Description, descending: true)
.ExecuteAsync();
// Chaining statements
var specificTests = await dbContext.Tests
.Where( x => x.TestMode == "Offline")
.Where(x => x.IsActive )
.OrderBy(x => x.TestType)
.OrderBy( x => x.TestMode)
.ExecuteAsync();
// Linq like query OLD version
var specificTests = await dbContext.Tests.WhereAsync(x =>x.IsAcive );
// GetAll
DbContext.Tests.GetAllAsync
Persisting Changes
When using Insert, Update, or Delete operations, you must call dbContext.Commit()
to persist your changes to the database:
// Insert a new entity
await dbContext.Plants.InsertAsync<string>(new Plant {
PlantCd = "PLANT5",
Description = "New Plant",
IsAcive = true
});
// Update an existing entity
await dbContext.Plants.UpdateAsync(new Plant {
PlantCd = "PLANT5",
Description = "Updated Plant",
IsAcive = false
});
// Delete an entity
await dbContext.Plants.DeleteAsync<string>("PLANT5");
// Commit the transaction to persist all changes
dbContext.Commit();
// If you need to rollback changes instead
// dbContext.Rollback();
The library uses transactions to ensure data consistency. Without calling Commit()
, your changes will be rolled back when the DbContext is disposed.
Transaction Management
The library supports two ways to manage transactions:
Default Transaction: By default, all operations use the transaction provided in the DbContext constructor.
Custom Transaction: You can now provide a specific transaction to individual CRUD operations, which will override the default transaction.
Using Custom Transactions
Each CRUD method (ExecuteAsync
, InsertAsync
, UpdateAsync
, and DeleteAsync
) accepts an optional transaction parameter:
// Using the default transaction from DbContext
await dbContext.Plants.InsertAsync<string>(newPlant);
// Using a custom transaction
using var customTransaction = dbContext.Connection.BeginTransaction();
await dbContext.Plants.InsertAsync<string>(newPlant, customTransaction);
customTransaction.Commit();
// Multiple operations within the same custom transaction
using var transaction = dbContext.Connection.BeginTransaction();
await dbContext.Plants.InsertAsync<string>(newPlant, transaction);
await dbContext.Tests.UpdateAsync(existingTest, transaction);
await dbContext.Measurements.DeleteAsync<int>(measurementId, transaction);
// Query operations also support transactions
var plant = await dbContext.Plants.GetByIdAsync<string>("PLANT1", transaction);
var allPlants = await dbContext.Plants.GetAllAsync(transaction);
transaction.Commit();
This feature is useful when you need to:
- Perform operations across multiple DbContext instances with the same transaction
- Control transaction scope more precisely
- Implement custom transaction management logic
Database Provider Support
The library now supports different database providers with provider-specific SQL syntax generation. This is particularly important for pagination, where different databases use different syntax.
Supported Database Providers
- SQLite: Uses
LIMIT/OFFSET
syntax for pagination - SQL Server: Uses
OFFSET/FETCH
syntax for pagination - PostgreSQL: Uses
LIMIT/OFFSET
syntax for pagination
Configuring the Database Provider
You can specify the database provider when creating a DbContext. The DatabaseProvider
parameter is now nullable, defaulting to SQLite if null:
// Default (SQLite) - these are equivalent:
var dbContext1 = new ApplicationDbContext(connection);
var dbContext2 = new ApplicationDbContext(connection, null);
var dbContext3 = new ApplicationDbContext(connection, DatabaseProvider.SQLite);
// SQL Server
var sqlServerContext = new ApplicationDbContext(connection, DatabaseProvider.SqlServer);
// PostgreSQL
var postgresContext = new ApplicationDbContext(connection, DatabaseProvider.PostgreSQL);
Example: Pagination with Different Providers
The pagination syntax is automatically adjusted based on the configured database provider:
// SQLite/PostgreSQL: Generates "LIMIT 10 OFFSET 0"
var sqlitePage = await sqliteContext.Plants
.OrderBy(p => p.PlantCd)
.Paginate(0, 10)
.ExecuteAsync();
// SQL Server: Generates "OFFSET 0 ROWS FETCH NEXT 10 ROWS ONLY"
var sqlServerPage = await sqlServerContext.Plants
.OrderBy(p => p.PlantCd)
.Paginate(0, 10)
.ExecuteAsync();
Note that SQL Server requires an ORDER BY clause for OFFSET/FETCH pagination. If you don't provide one, a default ORDER BY (SELECT NULL)
will be added automatically.
Testing with Different Database Providers
To test your code with different database providers, you can use the following approach:
1. For SQLite (In-Memory Testing)
// Create an in-memory SQLite connection
using var connection = new SqliteConnection("DataSource=:memory:");
connection.Open();
// Create a context with SQLite provider
using var dbContext = new ApplicationDbContext(connection, DatabaseProvider.SQLite);
// Run your tests
var results = await dbContext.YourEntity
.Paginate(0, 10)
.ExecuteAsync();
2. For SQL Server
// Create a SQL Server connection
using var connection = new SqlConnection("Your SQL Server Connection String");
connection.Open();
// Create a context with SQL Server provider
using var dbContext = new ApplicationDbContext(connection, DatabaseProvider.SqlServer);
// Run your tests
var results = await dbContext.YourEntity
.OrderBy(e => e.Id) // Required for SQL Server pagination
.Paginate(0, 10)
.ExecuteAsync();
3. For PostgreSQL
// Create a PostgreSQL connection
using var connection = new NpgsqlConnection("Your PostgreSQL Connection String");
connection.Open();
// Create a context with PostgreSQL provider
using var dbContext = new ApplicationDbContext(connection, DatabaseProvider.PostgreSQL);
// Run your tests
var results = await dbContext.YourEntity
.Paginate(0, 10)
.ExecuteAsync();
4. Testing SQL Generation Without Real Databases
You can also test the SQL generation logic without connecting to actual database servers by using SQLite with different database providers, as shown in the PaginationProviderTests.cs
file:
// Using SQLite but with SqlServer provider to test the SQL generation
using var connection = new SqliteConnection("DataSource=:memory:");
connection.Open();
// Create tables and seed data
// ...
// Test with SQL Server syntax
using var dbContext = new ApplicationDbContext(connection, DatabaseProvider.SqlServer);
var results = await dbContext.YourEntity
.OrderBy(e => e.Id)
.Paginate(0, 10)
.ExecuteAsync();
// The query will use SQL Server syntax (OFFSET/FETCH) but execute against SQLite
// This works because SQLite is flexible with SQL syntax
This approach allows you to verify that the correct SQL syntax is being generated without needing to set up multiple database servers.
5. Integration Testing with Docker Containers
For more realistic testing with actual database servers, you can use Docker containers. This project includes Docker configuration and helper scripts to make this process easier:
# Start Docker containers and run integration tests
./run-integration-tests.ps1
The Docker setup includes:
- docker-compose.yml: Configures SQL Server and PostgreSQL containers
- DatabaseConnectionFactory: Creates connections to different database providers
- RealDatabasePaginationTests: Integration tests that use real database connections
To manually set up and run the tests:
# Start the containers
docker-compose up -d
# Install required packages
dotnet add DapperOrmCore.Tests/DapperOrmCore.Tests.csproj package System.Data.SqlClient
dotnet add DapperOrmCore.Tests/DapperOrmCore.Tests.csproj package Npgsql
# Run the integration tests
dotnet test DapperOrmCore.Tests/DapperOrmCore.Tests.csproj --filter "FullyQualifiedName~RealDatabasePaginationTests"
# Clean up when done
docker-compose down
This approach provides the most accurate testing of your database provider-specific code, as it tests against actual database servers rather than just simulating the SQL syntax.
Pagination
var specificTests = await dbContext.Tests
// Other linq queries
.Paginate(0,50) // paginate(pageIndex, pageSize), this gives records 0-50
.ExecuteAsync();
Select
The Select feature allows you to project entities to a different type, similar to Entity Framework Core's Select method. This optimizes your queries by only retrieving the columns you need from the database.
// Select specific properties
var descriptions = await dbContext.Plants
.Select(p => p.Description)
.ExecuteAsync();
// Project to an anonymous type
var plantInfo = await dbContext.Plants
.Select(p => new { p.PlantCd, p.Description })
.ExecuteAsync();
// Combine with Where (filtering happens at the database level)
var activePlants = await dbContext.Plants
.Select(p => new { p.PlantCd, p.Description })
.Where(p => p.IsAcive)
.ExecuteAsync();
// Combine with OrderBy
var orderedPlants = await dbContext.Plants
.Select(p => new { p.PlantCd, p.Description })
.OrderBy(p => p.Description)
.ExecuteAsync();
// Combine with Paginate
var pagedPlants = await dbContext.Plants
.Select(p => new { p.PlantCd, p.Description })
.Paginate(0, 10)
.ExecuteAsync();
The Select method modifies the SQL SELECT statement to only include the specified columns, which is more efficient than retrieving all columns. It also supports filtering on properties that are not included in the Select projection, with the filtering happening at the database level.
Logging
If you want to log the queries, you need to put miminum log level to "Information" and use serilog.
JOIN
Joins can be done with the include tag (only one to one mapping currently one to many is for future features )
var measurements = await dbContext.Measurements
.Include(x => x.Test)
.Include(x => x.Plant)
// then you can also filter only on the parent
.Where(x => x.Value > 100)
.OrderBy(x => x.MeasurementDate)
.Paginate(0, 2)
.ExecuteAsync();
Interceptors
Interceptors provide a powerful way to hook into the entity lifecycle, allowing you to execute custom logic before or after insert and update operations. This feature gives you complete control over what happens at these critical points.
Basic Usage
// Create a DbContext with interceptors
public class ApplicationDbContext : IDisposable
{
private readonly NpgsqlConnection _connection;
private IDbTransaction _transaction;
public DapperSet<TestLalala> Tests { get; }
public ApplicationDbContext(string connectionString)
{
_connection = new NpgsqlConnection(connectionString);
_connection.Open();
_transaction = _connection.BeginTransaction();
// Create your custom interceptor
var yourInterceptor = new YourCustomInterceptor();
// Use the interceptor with a DapperSet
Tests = new DapperSet<TestLalala>(_connection, _transaction, yourInterceptor);
}
// ... Commit, Rollback, Dispose methods
}
Creating Interceptors
You can create interceptors by implementing ISaveChangesInterceptor
or extending the SaveChangesInterceptor
base class. This gives you complete freedom to decide what happens before or after insert/update operations:
public class YourCustomInterceptor : SaveChangesInterceptor
{
// Called before an entity is inserted
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
// Your custom logic here
if (entity is TestLalala test)
{
// For example, set creation date
test.CreatedDate = DateTime.UtcNow;
}
return Task.CompletedTask;
}
// Called after an entity is inserted
public override Task AfterInsertAsync(object entity, CancellationToken cancellationToken = default)
{
// Post-insert logic here
return Task.CompletedTask;
}
// Called before an entity is updated
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
// Your custom update logic here
if (entity is TestLalala test)
{
// For example, set last edit date
test.LastEditDate = DateTime.UtcNow;
}
return Task.CompletedTask;
}
// Called after an entity is updated
public override Task AfterUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
// Post-update logic here
return Task.CompletedTask;
}
}
Example: Audit Interceptor
Here's an example of an interceptor that handles both creation and modification dates:
public class AuditInterceptor : SaveChangesInterceptor
{
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is TestLalala test)
{
// Set creation date on insert
test.CreatedDate = DateTime.UtcNow;
// Initialize last edit date to same value
test.LastEditDate = test.CreatedDate;
}
return Task.CompletedTask;
}
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is TestLalala test)
{
// Only update the last edit date, not the creation date
test.LastEditDate = DateTime.UtcNow;
}
return Task.CompletedTask;
}
}
Multiple Interceptors
You can use multiple interceptors together, and they will be executed in the order they are provided:
// Create multiple interceptors
var auditInterceptor = new AuditInterceptor();
var validationInterceptor = new ValidationInterceptor();
var loggingInterceptor = new LoggingInterceptor();
// Use them together
var entitySet = new DapperSet<TestLalala>(
_connection,
_transaction,
auditInterceptor, // Executed first
validationInterceptor, // Executed second
loggingInterceptor // Executed third
);
Example: Validation Interceptor
public class ValidationInterceptor : SaveChangesInterceptor
{
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
ValidateEntity(entity);
return Task.CompletedTask;
}
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
ValidateEntity(entity);
return Task.CompletedTask;
}
private void ValidateEntity(object entity)
{
if (entity is TestLalala test)
{
// Perform validation
if (string.IsNullOrEmpty(test.Description))
{
throw new ArgumentException("Description cannot be empty");
}
if (test.Precision < 0 || test.Precision > 100)
{
throw new ArgumentException("Precision must be between 0 and 100");
}
}
}
}
Example: Logging Interceptor
public class LoggingInterceptor : SaveChangesInterceptor
{
private readonly ILogger _logger;
public LoggingInterceptor(ILogger logger)
{
_logger = logger;
}
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
_logger.LogInformation($"Inserting entity of type {entity.GetType().Name}");
return Task.CompletedTask;
}
public override Task AfterInsertAsync(object entity, CancellationToken cancellationToken = default)
{
_logger.LogInformation($"Successfully inserted entity of type {entity.GetType().Name}");
return Task.CompletedTask;
}
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
_logger.LogInformation($"Updating entity of type {entity.GetType().Name}");
return Task.CompletedTask;
}
public override Task AfterUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
_logger.LogInformation($"Successfully updated entity of type {entity.GetType().Name}");
return Task.CompletedTask;
}
}
Example: User Tracking Interceptor
This advanced interceptor tracks which user created or last modified an entity, providing accountability in multi-user systems:
public class UserTrackingInterceptor : SaveChangesInterceptor
{
// Current user context
public string CurrentUser { get; set; }
public string CurrentRole { get; set; }
public UserTrackingInterceptor(string initialUser, string initialRole)
{
CurrentUser = initialUser;
CurrentRole = initialRole;
}
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is IUserTrackable trackable)
{
// Set both created and last edited fields on insert
trackable.CreatedBy = CurrentUser;
trackable.CreatedRole = CurrentRole;
trackable.LastEditedBy = CurrentUser;
trackable.LastEditedRole = CurrentRole;
}
return Task.CompletedTask;
}
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is IUserTrackable trackable)
{
// Only update the last edited fields, preserve creation info
trackable.LastEditedBy = CurrentUser;
trackable.LastEditedRole = CurrentRole;
}
return Task.CompletedTask;
}
}
// Interface for trackable entities
public interface IUserTrackable
{
string CreatedBy { get; set; }
string CreatedRole { get; set; }
string LastEditedBy { get; set; }
string LastEditedRole { get; set; }
}
// Example entity implementing the interface
public class UserTrackableEntity : IUserTrackable
{
[Key]
[Column("id")]
public int Id { get; set; }
[Column("name")]
public string Name { get; set; }
[Column("created_by")]
public string CreatedBy { get; set; }
[Column("created_role")]
public string CreatedRole { get; set; }
[Column("last_edited_by")]
public string LastEditedBy { get; set; }
[Column("last_edited_role")]
public string LastEditedRole { get; set; }
}
Example: Versioning Interceptor
This interceptor automatically manages version numbers for entities, which is useful for optimistic concurrency and tracking changes:
public class VersioningInterceptor : SaveChangesInterceptor
{
public override Task BeforeInsertAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is IVersionable versionable)
{
// Start at version 1 for new entities
versionable.Version = 1;
}
return Task.CompletedTask;
}
public override Task BeforeUpdateAsync(object entity, CancellationToken cancellationToken = default)
{
if (entity is IVersionable versionable)
{
// Increment version on each update
versionable.Version++;
}
return Task.CompletedTask;
}
}
// Interface for versionable entities
public interface IVersionable
{
int Version { get; set; }
}
// Example entity implementing the interface
public class VersionedEntity : IVersionable
{
[Key]
[Column("id")]
public int Id { get; set; }
[Column("name")]
public string Name { get; set; }
[Column("version")]
public int Version { get; set; }
}
Example: Combining Multiple Interceptors
This example shows how to combine multiple interceptors to create a comprehensive entity lifecycle management system:
// Create a DbContext with multiple interceptors
public class ApplicationDbContext : IDisposable
{
private readonly NpgsqlConnection _connection;
private IDbTransaction _transaction;
public DapperSet<ComplexEntity> ComplexEntities { get; }
public ApplicationDbContext(string connectionString, string currentUser, string currentRole)
{
_connection = new NpgsqlConnection(connectionString);
_connection.Open();
_transaction = _connection.BeginTransaction();
// Create multiple interceptors
var auditInterceptor = new AuditInterceptor();
var userTrackingInterceptor = new UserTrackingInterceptor(currentUser, currentRole);
var versioningInterceptor = new VersioningInterceptor();
var validationInterceptor = new ValidationInterceptor();
// Use them together with a DapperSet
ComplexEntities = new DapperSet<ComplexEntity>(
_connection,
_transaction,
auditInterceptor, // Handles creation and modification dates
userTrackingInterceptor, // Tracks who created/modified the entity
versioningInterceptor, // Manages version numbers
validationInterceptor // Validates the entity before saving
);
}
// ... Commit, Rollback, Dispose methods
}
// A complex entity that implements multiple interfaces
public class ComplexEntity : IUserTrackable, IVersionable
{
[Key]
[Column("id")]
public int Id { get; set; }
[Column("name")]
public string Name { get; set; }
[Column("created_date")]
public DateTime CreatedDate { get; set; }
[Column("last_edit_date")]
public DateTime LastEditDate { get; set; }
[Column("created_by")]
public string CreatedBy { get; set; }
[Column("created_role")]
public string CreatedRole { get; set; }
[Column("last_edited_by")]
public string LastEditedBy { get; set; }
[Column("last_edited_role")]
public string LastEditedRole { get; set; }
[Column("version")]
public int Version { get; set; }
}
This approach allows you to build a modular, maintainable system where each interceptor handles a specific concern, following the Single Responsibility Principle.
Classes need to be annotated like this with the NotMapped and ForeignKey tag
[Table("ipc.measurement")]
public class CoolMeasurement
{
[Key]
[Column("id")]
public int Id { get; set; }
[Column("test_cd")]
public required string TestCd { get; set; }
[Column("plant_cd")]
public required string PlantCd { get; set; }
[Column("avg_value")]
public double Value { get; set; }
[Column("measurement_date")]
public DateTime MeasurementDate { get; set; } = DateTime.UtcNow;
[NotMapped]
[ForeignKey("test_cd")]
public TestLalala Test { get; set; }
[NotMapped]
[ForeignKey("plant_cd")]
public Plant Plant { get; set; }
}
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net9.0 is compatible. 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. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
-
net9.0
- Dapper (>= 2.1.66)
- Serilog (>= 4.2.0)
- Serilog.Sinks.Console (>= 6.0.0)
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 |
---|---|---|
1.0.33 | 136 | 7/15/2025 |
1.0.31 | 146 | 7/10/2025 |
1.0.29 | 414 | 6/11/2025 |
1.0.27 | 266 | 6/10/2025 |
1.0.25 | 265 | 6/10/2025 |
1.0.23 | 266 | 6/10/2025 |
1.0.21 | 121 | 6/6/2025 |
1.0.19 | 107 | 6/6/2025 |
1.0.18 | 227 | 4/30/2025 |
1.0.16 | 143 | 4/30/2025 |
1.0.14 | 155 | 4/29/2025 |
1.0.13 | 117 | 4/25/2025 |
1.0.12-alpha.0.1 | 103 | 4/25/2025 |
1.0.11 | 117 | 4/25/2025 |
1.0.11-alpha.0.16.2.1 | 101 | 4/25/2025 |
1.0.11-alpha.0.16.2 | 97 | 4/25/2025 |
1.0.11-alpha.0.16 | 101 | 4/25/2025 |
1.0.11-alpha.0.15 | 97 | 4/25/2025 |
1.0.11-alpha.0.13 | 102 | 4/25/2025 |
1.0.11-alpha.0.11 | 99 | 4/25/2025 |
1.0.11-alpha.0.9 | 100 | 4/25/2025 |
1.0.11-alpha.0.8 | 136 | 4/24/2025 |
1.0.11-alpha.0.7 | 130 | 4/24/2025 |
1.0.10 | 165 | 4/23/2025 |
1.0.9 | 161 | 4/23/2025 |
1.0.8 | 195 | 4/14/2025 |
1.0.7 | 168 | 4/8/2025 |
1.0.6 | 164 | 4/8/2025 |
1.0.5 | 162 | 4/7/2025 |
1.0.4 | 139 | 4/4/2025 |
1.0.3 | 162 | 4/2/2025 |
1.0.2 | 153 | 4/1/2025 |
1.0.1 | 158 | 4/1/2025 |
1.0.0 | 157 | 4/1/2025 |
1.0.0-alpha.0.29 | 129 | 4/24/2025 |
0.0.0-alpha.0.28 | 138 | 4/23/2025 |