StringThing.Sqlite
2.0.2
dotnet add package StringThing.Sqlite --version 2.0.2
NuGet\Install-Package StringThing.Sqlite -Version 2.0.2
<PackageReference Include="StringThing.Sqlite" Version="2.0.2" />
<PackageVersion Include="StringThing.Sqlite" Version="2.0.2" />
<PackageReference Include="StringThing.Sqlite" />
paket add StringThing.Sqlite --version 2.0.2
#r "nuget: StringThing.Sqlite, 2.0.2"
#:package StringThing.Sqlite@2.0.2
#addin nuget:?package=StringThing.Sqlite&version=2.0.2
#tool nuget:?package=StringThing.Sqlite&version=2.0.2
StringThing.Sqlite
Injection-safe interpolated SQL for SQLite, built on Microsoft.Data.Sqlite. Part of StringThing.
SQLite is dynamically typed — the compile-time type contract here is light. The value is injection safety by construction, parameter deduplication, and composable fragments.
Install
dotnet add package StringThing.Sqlite
Quick start
var userId = 42;
SqliteSql statement = $"SELECT name FROM users WHERE id = {userId}";
using var command = statement.ToCommand(connection);
Parameters are named automatically using the variable name: @userId, not @p0.
No container or server needed — SQLite is embedded.
Result mapping
Mark row types with [StringThingRow] and declare them partial. A source generator emits an AOT-friendly row materializer — no reflection, no IL emit, no third-party mapper.
using StringThing.Aot;
[StringThingRow]
public partial record User(long Id, string Name, string? Email);
var user = connection.QueryStringSingle<User>(
$"SELECT id AS Id, name AS Name, email AS Email FROM users WHERE id = {userId}");
var users = connection.QueryString<User>(
$"SELECT id AS Id, name AS Name, email AS Email FROM users ORDER BY id");
connection.ExecuteString(
$"DELETE FROM users WHERE id = {userId}");
The full surface: QueryString<T>, QueryStringFirst<T>, QueryStringFirstOrDefault<T>, QueryStringSingle<T>, QueryStringSingleOrDefault<T>, ExecuteString, ExecuteStringScalar (+ T overload), plus Async variants. Column ordinals are resolved once per query; rows are then read by ordinal — name-based binding without per-row name lookup.
Scalar columns map directly. When T is a supported scalar type rather than a [StringThingRow] type, the query reads its first column into that value — no row type or wrapper needed:
var ids = connection.QueryString<int>($"SELECT id FROM users ORDER BY id");
var name = connection.QueryStringSingle<string>($"SELECT name FROM users WHERE id = {id}");
var email = connection.QueryStringSingle<string?>($"SELECT email FROM users WHERE id = {id}");
Nullable scalars read NULL as null/default. A T that is neither a supported scalar nor a [StringThingRow] type is a compile error (ST0002).
Override the column name with [Column] from System.ComponentModel.DataAnnotations.Schema:
[StringThingRow]
public partial record User(
[property: Column("user_id")] long Id,
[property: Column("full_name")] string Name);
Nullable annotations drive IsDBNull checks — string? becomes a null-checked read; string is a direct read.
If the generator can't handle your shape — say, you want to derive a property from a column rather than read it straight, or read columns into a shape the generator couldn't infer from the type — implement IStringThingRow<T> by hand. Same runtime path:
public sealed class UserSummary : IStringThingRow<UserSummary>
{
public long Id { get; init; }
public string Status { get; init; } = "";
public static ReadOnlySpan<string> ColumnBindingOrder => ["id", "email"];
public static UserSummary Read(DbDataReader reader, ReadOnlySpan<int> ordinals) => new()
{
Id = reader.GetInt64(ordinals[0]),
Status = reader.IsDBNull(ordinals[1]) ? "no-email" : "has-email",
};
}
Supported types
bool, byte, short, int, long, float, double, decimal, string, char, Guid, DateTime, DateTimeOffset, DateOnly, TimeOnly, TimeSpan, byte[].
SQLite is dynamically typed — values are stored as TEXT, INTEGER, REAL, BLOB, or NULL regardless of the declared column type. StringThing ensures the .NET value reaches SQLite correctly.
Nullable reference types (string?, byte[]?) map to NULL.
Fragments
var minAge = 18;
var status = "active";
SqliteFragment filter = $"age >= {minAge} AND status = {status}";
SqliteSql statement = $"SELECT * FROM users WHERE {filter}";
Multi-row insert
record InsertUser(int Id, string Name, string? Email) : ISqliteRow
{
public SqliteFragment RowValues => $"({Id}, {Name}, {Email})";
}
var users = new InsertUser[] { new(1, "alice", "alice@example.com"), new(2, "bob", null) };
SqliteSql statement = $"INSERT INTO users (id, name, email) VALUES {SqliteSql.InsertRows(users)}";
IN list
var ids = new List<int> { 1, 2, 3 };
SqliteSql statement = $"SELECT * FROM users WHERE id IN {SqliteSql.InList([.. ids])}";
Unsafe escape hatch
using StringThing.UnsafeSql;
var tableName = Sql.Unsafe("users");
SqliteSql statement = $"SELECT * FROM {tableName} WHERE id = {userId}";
Built by Immersus Machina
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net10.0 is compatible. 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. |
-
net10.0
- Microsoft.Data.Sqlite (>= 10.0.5)
- StringThing.Core (>= 2.0.2)
NuGet packages (1)
Showing the top 1 NuGet packages that depend on StringThing.Sqlite:
| Package | Downloads |
|---|---|
|
StringThing.Sqlite.Dapper
Dapper result mapping for StringThing.Sqlite. Injection-safe interpolated SQL on input, Dapper mapping on output. Dapper is bundled internally. |
GitHub repositories
This package is not used by any popular GitHub repositories.