A blazingly fast, lightweight RDF quad store for .NET 10 β minimal dependencies, pure performance.
β¨ Sub-microsecond queries β Indexed lookups in <200ns (faster than most in-memory operations)
π― Batch load 1M triples in 3.65 seconds β 274K triples/sec throughput
π¦ Lightweight β TriG loading without need for external RDF libraries
π SPARQL support β Full graph pattern matching and variable binding
β
Well tested β 200+ passing tests, W3C TriG compliance
QuadStore is the core in-memory quad store.
QuadStoreStorageProvider adapts it to the standard dotNetRDF storage interfaces.
That means Leviathan can run SPARQL queries against your QuadStore data.
using TripleStore.Core;
using VDS.RDF.Query;
using VDS.RDF.Storage;
// 1) Define a tiny in-memory TriG document with two FOAF name triples.
const string trigContent = """
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
{
<http://example.org/alice> foaf:name "Alice" .
<http://example.org/bob> foaf:name "Bob" .
}
""";
// 2) Persist the TriG content to disk so the loader can read it from a file.
var dataFilePath = Path.Combine(AppContext.BaseDirectory, "data.trig");
File.WriteAllText(dataFilePath, trigContent);
// 3) Create/open the QuadStore data directory.
var storePath = Path.Combine(AppContext.BaseDirectory, "store-data");
using var store = new QuadStore(storePath);
// 4) Load TriG quads directly into QuadStore in a single pass.
var loader = new SinglePassTrigLoader(store);
loader.LoadFromFile(dataFilePath);
// 5) Wrap QuadStore with the dotNetRDF storage adapter so Leviathan can query it.
IStorageProvider provider = new QuadStoreStorageProvider(store);
var queryable = (IQueryableStorage)provider;
// 6) Run a SPARQL SELECT query through Leviathan.
var query = @"SELECT ?person ?name
WHERE {
?person <http://xmlns.com/foaf/0.1/name> ?name .
}";
var results = (SparqlResultSet)queryable.Query(query);
// 7) Print the SPARQL result rows.
Console.WriteLine("Leviathan query results:");
foreach (var row in results)
{
Console.WriteLine($"{row["person"]} -> {row["name"]}");
}This example exposes a simple /sparql endpoint that accepts a SPARQL query and returns:
- SPARQL JSON results for
SELECT/ASK - Turtle for
CONSTRUCT/DESCRIBE
using TripleStore.Core;
using VDS.RDF;
using VDS.RDF.Query;
using VDS.RDF.Storage;
using VDS.RDF.Writing;
var dataDir = args.Length > 0 ? args[0] : "./quadstore-data";
var qs = new QuadStore(dataDir);
var provider = new QuadStoreStorageProvider(qs);
var queryable = (IQueryableStorage)provider;
// ---------------------------------------------------------------------------
// Section B: Seed Data
// ---------------------------------------------------------------------------
// On first run (empty store), load a small set of FOAF triples so the sample
// can be queried immediately without manual data loading.
if (!qs.Query().Any())
{
var seedTriG = """
@prefix foaf: <http://xmlns.com/foaf/0.1/> .
@prefix ex: <http://example.org/> .
{
ex:alice foaf:name "Alice" ;
foaf:knows ex:bob .
ex:bob foaf:name "Bob" ;
foaf:knows ex:alice .
}
""";
var loader = new SinglePassTrigLoader(qs);
loader.LoadFromString(seedTriG);
qs.SaveAll();
}
// ---------------------------------------------------------------------------
// Section C: Host Configuration
// ---------------------------------------------------------------------------
// Build the ASP.NET Core minimal API host and register QuadStore and its
// storage provider as singletons so they are available to the request pipeline.
var builder = WebApplication.CreateBuilder(args);
builder.Services.AddSingleton(qs);
builder.Services.AddSingleton(provider);
builder.Services.AddSingleton<IQueryableStorage>(provider);
var app = builder.Build();
app.MapSparqlEndpoints();
// ---------------------------------------------------------------------------
// Section G: Shutdown
// ---------------------------------------------------------------------------
// Dispose the QuadStore when the application is stopping to release
// memory-mapped file handles cleanly.
app.Lifetime.ApplicationStopping.Register(() =>
{
qs.Dispose();
});
app.Run();QuadStore + QuadStoreStorageProvider is intentionally append-only currently:
- β No graph deletion (
DeleteGraphis not supported) - β No triple removal in
UpdateGraph(additions only) - β No SPARQL Update (
IUpdateableStorage.Updatethrows) - βΉοΈ Per-query in-memory snapshot for Leviathan queries (simple and correct, but can be slower on very large datasets)
| Operation | 10K Triples | 100K Triples | 1M Triples |
|---|---|---|---|
| Subject lookup | <200ns | <200ns | <200ns |
| Predicate query | 36Β΅s | 536Β΅s | 4.4ms |
| Object lookup | <200ns | <200ns | <200ns |
| Graph filter | 138Β΅s | 1.9ms | 17.6ms |
| Multiple patterns | 1.6Β΅s | 21Β΅s | 285Β΅s |
| Full enumeration | 852Β΅s | 7.9ms | 80.3ms |
| Dataset | Time | Throughput |
|---|---|---|
| 10K triples | 370ms | 27K/sec |
| 100K triples | 299ms | 335K/sec |
| 1M triples | 3.65s | 274K/sec |
Benchmarks on Intel Core Ultra 7 265H, .NET 10.0. See detailed benchmark analysis for all metrics.
QuadStore is built for .NET 10.0 and ships as a library. Add it to your project:
# If published to NuGet (coming soon)
dotnet add package QuadStore.CoreFor now, clone and reference locally:
git clone https://github.com/aabs/QuadStore.git
# Add project reference in your .csproj:
# <ProjectReference Include="path/to/src/TripleStore.Core/TripleStore.Core.csproj" />QuadStore maintains roaring bitmap indexes for fast intersection:
- Subject index β O(1) lookup by URI
- Predicate index β O(1) lookup by property
- Object index β O(1) lookup by resource
- Graph index β O(1) lookup by named graph
Multi-dimensional queries use two-pointer intersection over sorted result sets, keeping complex queries sub-millisecond even at scale.
- API Reference β Explore core types and methods
- Benchmark Report β See performance analysis in detail
- Test Suite β Working examples of every operation
- TriG Spec β Supported data format
# Build
dotnet build -c Release
# Run all tests
dotnet test -c Release --nologo
# Run benchmarks
dotnet run -c Release --project benchmark/TripleStore.Benchmarks/TripleStore.Benchmarks.csprojHave ideas? Found a bug? Jump in!
- π Open an issue for bugs or feature requests
- π¬ Start a discussion to share ideas
- π PRs welcome β see DEVELOPMENT.md for setup
All contributions welcome, from bug reports to performance improvements.
MIT β Use freely in commercial and personal projects.
Made with β‘ for .NET developers who demand performance.