RocksDB.Extensions is a .NET library that provides an opinionated integration of RocksDB with the .NET dependency injection system. This library enables you to easily create key/value data stores backed by RocksDB, and provides several built-in serializer factories to serialize your data.
NuGet | Status |
---|---|
RocksDb.Extensions | |
RocksDb.Extensions.Protobuf | |
RocksDb.Extensions.System.Text.Json | |
RocksDb.Extensions.ProtoBufNet |
To use RocksDb.Extensions in your project, add the NuGet package using the dotnet CLI:
dotnet add package RocksDb.Extensions
The library abstracts away the internals of RocksDb behind the concept of key/value data stores. To create your first store, you need a class that inherits from the base class RocksDbStore<TKey, TValue>
. For example:
public class UsersStore : RocksDbStore<string, User>
{
public UsersStore(IRocksDbAccessor<string, User> rocksDbAccessor) : base(rocksDbAccessor)
{
}
}
In this example, we're using string
as the key and a custom type User
as the value. The library is flexible regarding how you can serialize your keys and values. There are several serialization options available, and you can add your own by inheriting from the ISerializerFactory
interface.
The library provides 4 built-in serializer factories:
- PrimitiveTypesSerializerFactory
- SystemTextJsonSerializerFactory
- ProtoBufNetSerializerFactory
- ProtobufSerializerFactory
You can use one or more of these factories to serialize your data. For example, if you want to use PrimitiveTypesSerializerFactory
to serialize your keys and SystemTextJsonSerializerFactory
to serialize your values, you can configure your RocksDb instance as follows:
var rocksDbBuilder = builder.Services.AddRocksDb(options =>
{
// Specify the path for the RocksDb database
options.Path = "./my-rocks-db";
// Clear pre-configured SerializerFactories so we can have full control over the order in which serializers will be resolved.
options.SerializerFactories.Clear();
// Add the serializer factories for the key and value types
options.SerializerFactories.Add(new PrimitiveTypesSerializerFactory());
options.SerializerFactories.Add(new SystemTextJsonSerializerFactory());
});
Before your store can be used, you need to register it with RocksDb. You can do this as follows:
rocksDbBuilder.AddStore<string, User, UsersStore>("users-store");
This registers an instance of UsersStore
with RocksDb under the name "users-store".
You can also resolve your store as a keyed service using the column family name:
var usersStore = serviceProvider.GetRequiredKeyedService<UsersStore>("users-store");
This approach allows you to register and retrieve multiple stores of the same type, each differentiated by their column family name.
Once you have registered your store, you can use it to add, get, and remove data from RocksDb. For example:
[ApiController]
[Route("[controller]")]
public class UsersController : ControllerBase
{
private readonly UsersStore _usersStore;
public UsersController(UsersStore usersStore)
{
_usersStore = usersStore;
}
[HttpPost]
public IActionResult Post([FromBody] User user)
{
var id = Guid.NewGuid().ToString();
user.Id = id;
_usersStore.Put(id, user);
return Ok();
}
[HttpGet]
public IEnumerable<User> Get()
{
return _usersStore.GetAll();
}
[HttpDelete]
public IActionResult Delete(string userId)
{
_usersStore.Remove(userId);
return Ok();
}
}
For certain scenarios, such as automated testing, it may be useful to remove the existing RocksDB database on startup to ensure a clean start. This can be achieved using the DeleteExistingDatabaseOnStartup
configuration option.
To use this option, set it to true when configuring the RocksDbOptions
object:
var rocksDbBuilder = builder.Services.AddRocksDb(options =>
{
// Your RocksDb configuration...
options.DeleteExistingDatabaseOnStartup = true;
});
When this option is set to true, the existing database will be deleted on startup and a new one will be created. Note that all data in the existing database will be lost when this option is used.
By default, the DeleteExistingDatabaseOnStartup
option is set to false to preserve the current behavior of not automatically deleting the database. If you need to ensure a clean start for your application, set this option to true in your configuration.
RocksDb.Extensions provides built-in support for collections across different serialization packages:
The RocksDb.Extensions.System.Text.Json
and RocksDb.Extensions.ProtoBufNet
packages support collections out of the box. You can use any collection type like List<T>
or arrays without additional configuration.
The library includes specialized support for collections when working with:
- Protocol Buffer message types
- Primitive types (int, long, string, etc.)
When using IList<T>
with these types, the library automatically handles serialization/deserialization without requiring wrapper message types. This is particularly useful for Protocol Buffers, where RepeatedField<T>
typically cannot be serialized as a standalone entity.
The serialization format varies depending on the element type:
[4 bytes: List length][Contiguous array of serialized elements]
[4 bytes: List length][For each element: [4 bytes: Element size][N bytes: Element data]]
Example types that work automatically with this support:
- Protocol Buffer message types:
IList<YourProtobufMessage>
- Primitive types:
IList<int>
,IList<long>
,IList<string>
, etc.