10 KiB
lyng.io.db — SQL database access for Lyng scripts
This module provides the portable SQL database contract for Lyng. The current shipped providers are SQLite via lyng.io.db.sqlite and a JVM-only JDBC bridge via lyng.io.db.jdbc.
Note:
lyngiois a separate library module. It must be explicitly added as a dependency to your host application and initialized in your Lyng scopes.
Install the module into a Lyng session
For SQLite-backed database access, install both the generic DB module and the SQLite provider:
import net.sergeych.lyng.EvalSession
import net.sergeych.lyng.Scope
import net.sergeych.lyng.io.db.createDbModule
import net.sergeych.lyng.io.db.sqlite.createSqliteModule
suspend fun bootstrapDb() {
val session = EvalSession()
val scope: Scope = session.getScope()
createDbModule(scope)
createSqliteModule(scope)
session.eval("""
import lyng.io.db
import lyng.io.db.sqlite
""".trimIndent())
}
createSqliteModule(...) also registers the sqlite: scheme for generic openDatabase(...).
For JVM JDBC-backed access, install the JDBC provider as well:
import net.sergeych.lyng.EvalSession
import net.sergeych.lyng.Scope
import net.sergeych.lyng.io.db.createDbModule
import net.sergeych.lyng.io.db.jdbc.createJdbcModule
suspend fun bootstrapJdbc() {
val session = EvalSession()
val scope: Scope = session.getScope()
createDbModule(scope)
createJdbcModule(scope)
session.eval("""
import lyng.io.db
import lyng.io.db.jdbc
""".trimIndent())
}
createJdbcModule(...) registers jdbc:, h2:, postgres:, and postgresql: for openDatabase(...).
Using from Lyng scripts
Typed SQLite open helper:
import lyng.io.db.sqlite
val db = openSqlite(":memory:")
val userCount = db.transaction { tx ->
tx.execute("create table user(id integer primary key autoincrement, name text not null)")
tx.execute("insert into user(name) values(?)", "Ada")
tx.execute("insert into user(name) values(?)", "Linus")
tx.select("select count(*) as count from user").toList()[0]["count"]
}
assertEquals(2, userCount)
Generic provider-based open:
import lyng.io.db
import lyng.io.db.sqlite
val db = openDatabase(
"sqlite:./app.db",
Map(
"foreignKeys" => true,
"busyTimeoutMillis" => 5000
)
)
JVM JDBC open with H2:
import lyng.io.db.jdbc
val db = openH2("mem:demo;DB_CLOSE_DELAY=-1")
val names = db.transaction { tx ->
tx.execute("create table person(id bigint auto_increment primary key, name varchar(120) not null)")
tx.execute("insert into person(name) values(?)", "Ada")
tx.execute("insert into person(name) values(?)", "Linus")
tx.select("select name from person order by id").toList()
}
assertEquals("Ada", names[0]["name"])
assertEquals("Linus", names[1]["name"])
Generic JDBC open through openDatabase(...):
import lyng.io.db
import lyng.io.db.jdbc
val db = openDatabase(
"jdbc:h2:mem:demo2;DB_CLOSE_DELAY=-1",
Map()
)
val answer = db.transaction { tx ->
tx.select("select 42 as answer").toList()[0]["answer"]
}
assertEquals(42, answer)
PostgreSQL typed open:
import lyng.io.db.jdbc
val db = openPostgres(
"jdbc:postgresql://127.0.0.1/appdb",
"appuser",
"secret"
)
val titles = db.transaction { tx ->
tx.execute("create table if not exists task(id bigserial primary key, title text not null)")
tx.execute("insert into task(title) values(?)", "Ship JDBC provider")
tx.execute("insert into task(title) values(?)", "Test PostgreSQL path")
tx.select("select title from task order by id").toList()
}
assertEquals("Ship JDBC provider", titles[0]["title"])
Nested transactions use real savepoint semantics:
import lyng.io.db
import lyng.io.db.sqlite
val db = openSqlite(":memory:")
db.transaction { tx ->
tx.execute("create table item(id integer primary key autoincrement, name text not null)")
tx.execute("insert into item(name) values(?)", "outer")
try {
tx.transaction { inner ->
inner.execute("insert into item(name) values(?)", "inner")
throw IllegalStateException("rollback nested")
}
} catch (_: IllegalStateException) {
}
assertEquals(1, tx.select("select count(*) as count from item").toList()[0]["count"])
}
Intentional rollback without treating it as a backend failure:
import lyng.io.db
import lyng.io.db.sqlite
val db = openSqlite(":memory:")
assertThrows(RollbackException) {
db.transaction { tx ->
tx.execute("create table item(id integer primary key autoincrement, name text not null)")
tx.execute("insert into item(name) values(?)", "temporary")
throw RollbackException("stop here")
}
}
Portable API
Database
transaction(block)— opens a transaction, commits on normal exit, rolls back on uncaught failure.
SqlTransaction
select(clause, params...)— execute a statement whose primary result is a row set.execute(clause, params...)— execute a side-effect statement and returnExecutionResult.transaction(block)— nested transaction with real savepoint semantics.
ResultSet
columns— positionalSqlColumnmetadata, available before iteration.size()— result row count.isEmpty()— fast emptiness check where possible.iterator()— normal row iteration while the transaction is active.toList()— materialize detachedSqlRowsnapshots that may be used after the transaction ends.
SqlRow
row[index]— zero-based positional access.row["columnName"]— case-insensitive lookup by output column label.
Name-based access fails with SqlUsageException if the name is missing or ambiguous.
ExecutionResult
affectedRowsCountgetGeneratedKeys()
Statements that return rows directly, such as ... returning ..., should use select(...), not execute(...).
Value mapping
Portable bind values:
nullBoolInt,Double,DecimalStringBufferDate,DateTime,Instant
Unsupported parameter values fail with SqlUsageException.
Portable result metadata categories:
BinaryStringIntDoubleDecimalBoolDateDateTimeInstant
For temporal types, see time functions.
SQLite provider
lyng.io.db.sqlite currently provides the first concrete backend.
Typed helper:
openSqlite(
path: String,
readOnly: Bool = false,
createIfMissing: Bool = true,
foreignKeys: Bool = true,
busyTimeoutMillis: Int = 5000
): Database
Accepted generic URL forms:
sqlite::memory:sqlite:relative/path.dbsqlite:/absolute/path.db
Supported openDatabase(..., extraParams) keys for SQLite:
readOnly: BoolcreateIfMissing: BoolforeignKeys: BoolbusyTimeoutMillis: Int
SQLite write/read policy in v1:
Boolwrites as0/1Decimalwrites as canonical textDatewrites asYYYY-MM-DDDateTimewrites as ISO local timestamp text without timezoneInstantwrites as ISO UTC timestamp text with explicit timezone markerTIME*values stayStringTIMESTAMP/DATETIMEreject timezone-bearing stored text
Open-time validation failures:
- malformed URL or bad option shape ->
IllegalArgumentException - runtime open failure ->
DatabaseException
JDBC provider
lyng.io.db.jdbc is currently implemented on the JVM target only. The lyngio-jvm artifact bundles and explicitly loads these JDBC drivers:
- SQLite
- H2
- PostgreSQL
Typed helpers:
openJdbc(
connectionUrl: String,
user: String? = null,
password: String? = null,
driverClass: String? = null,
properties: Map<String, Object?>? = null
): Database
openH2(
connectionUrl: String,
user: String? = null,
password: String? = null,
properties: Map<String, Object?>? = null
): Database
openPostgres(
connectionUrl: String,
user: String? = null,
password: String? = null,
properties: Map<String, Object?>? = null
): Database
Accepted generic URL forms:
jdbc:h2:mem:test;DB_CLOSE_DELAY=-1h2:mem:test;DB_CLOSE_DELAY=-1jdbc:postgresql://localhost/apppostgres://localhost/apppostgresql://localhost/app
Supported openDatabase(..., extraParams) keys for JDBC:
driverClass: Stringuser: Stringpassword: Stringproperties: Map<String, Object?>
Behavior notes for the JDBC bridge:
- the portable
Database/SqlTransactionAPI stays the same as for SQLite - nested transactions use JDBC savepoints
- JDBC connection properties are built from
user,password, andproperties propertiesvalues are stringified before being passed to JDBC- statements with row-returning clauses still must use
select(...), notexecute(...)
Platform support for this provider:
lyng.io.db.jdbc— JVM onlyopenH2(...)— works out of the box withlyngio-jvmopenPostgres(...)— driver included, but an actual PostgreSQL server is still required
PostgreSQL-specific notes:
openPostgres(...)accepts either a full JDBC URL or shorthand forms such as//localhost/app- local peer/trust setups may use an empty password string
- generated keys work with PostgreSQL
bigserial/ identity columns throughExecutionResult.getGeneratedKeys() - for reproducible automated tests, prefer a disposable PostgreSQL instance such as Docker/Testcontainers instead of a long-lived shared server
Lifetime rules
ResultSet is valid only while its owning transaction is active.
SqlRow values are detached snapshots once materialized, so this pattern is valid:
val rows = db.transaction { tx ->
tx.select("select name from person order by id").toList()
}
assertEquals("Ada", rows[0]["name"])
This means:
- do not keep
ResultSetobjects after the transaction block returns - materialize rows with
toList()inside the transaction when they must outlive it
The same rule applies to generated keys from ExecutionResult.getGeneratedKeys(): the ResultSet is transaction-scoped, but rows returned by toList() are detached.
Platform support
lyng.io.db— generic contract, available when host code installs itlyng.io.db.sqlite— implemented on JVM and Linux Native in the current release treelyng.io.db.jdbc— implemented on JVM in the current release tree
For the broader I/O overview, see lyngio overview.