Working with Transactions
CRUD operations in Exposed must be called from within a transaction. Transactions encapsulate a set of DSL operations. To create and execute a transaction with default parameters, simply pass a function block to the transaction
function:
Transactions are executed synchronously on the current thread, so they will block other parts of your application! If you need to execute a transaction asynchronously, consider running it on a separate Thread
.
Accessing returned values
Although you can modify variables from your code within the transaction block, transaction
supports returning a value directly, enabling immutability:
Working with multiple databases
If you want to work with different databases, you have to store the database reference returned by Database.connect()
and provide it to transaction
function as the first parameter. The transaction
block without parameters will work with the latest connected database.
Entities stick
to a transaction that was used to load that entity. That means that all changes persist to the same database and what cross-database references are prohibited and will throw exceptions.
Setting default database
transaction
block without parameters will use the default database. As before 0.10.1 this will be the latest connected database. It is also possible to set the default database explicitly.
Using nested transactions
By default, a nested transaction
block shares the transaction resources of its parent transaction
block, so any effect on the child affects the parent:
Since Exposed 0.16.1 it is possible to use nested transactions as separate transactions by setting useNestedTransactions = true
on the desired Database
instance.
After that any exception or rollback operation that happens within a transaction
block will not roll back the whole transaction but only the code inside the current transaction
. Exposed uses SQL SAVEPOINT
functionality to mark the current transaction at the beginning of a transaction
block and release it on exit.
Using SAVEPOINT
could affect performance, so please read the documentation of the DBMS you use for more details.
Working with Coroutines
In the modern world, non-blocking and asynchronous code is popular. Kotlin has Coroutines, which provide an imperative way to write asynchronous code. Most Kotlin frameworks (like Ktor) have built-in support for coroutines, while Exposed is mostly blocking.
Why?
Because Exposed interacts with databases using JDBC API, which was designed in an era of blocking APIs. Additionally, Exposed stores some values in thread-local variables while coroutines could (and will) be executed in different threads.
Starting from Exposed 0.15.1, bridge functions are available that give you a safe way to interact with Exposed within suspend
blocks: newSuspendedTransaction()
and Transaction.withSuspendTransaction()
. These have the same parameters as a blocking transaction()
but allow you to provide a CoroutineContext
argument that explicitly specifies the CoroutineDispatcher
in which the function will be executed. If context is not provided your code will be executed within the current CoroutineContext
.
Here is an example that uses these three types of transactions:
Please note that such code remains blocking (as it still uses JDBC) and you should not try to share a transaction between multiple threads as it may lead to undefined behavior.
If you desire to execute some code asynchronously and use the result later, take a look at suspendedTransactionAsync()
:
This function will accept the same parameters as newSuspendedTransaction()
above but returns its future result as an implementation of Deferred
, which you can await
on to achieve your result.
Advanced parameters and usage
For specific functionality, transactions can be created with the additional parameters: transactionIsolation
, readOnly
, and db
:
transactionIsolation
The transactionIsolation
parameter, defined in the SQL standard, specifies what is supposed to happen when multiple transactions execute concurrently on the database. This value does NOT affect Exposed operation directly, but is sent to the database, where it is expected to be obeyed. Allowable values are defined in java.sql.Connection
and are as follows:
TRANSACTION_NONE: Transactions are not supported.
TRANSACTION_READ_UNCOMMITTED: The most lenient setting. Allows uncommitted changes from one transaction to affect a read in another transaction (a "dirty read").
TRANSACTION_READ_COMMITTED: This setting prevents dirty reads from occurring, but still allows non-repeatable reads to occur. A non-repeatable read is when a transaction ("Transaction A") reads a row from the database, another transaction ("Transaction B") changes the row, and Transaction A reads the row again, resulting in an inconsistency.
TRANSACTION_REPEATABLE_READ: The default setting for Exposed transactions. Prevents both dirty and non-repeatable reads, but still allows for phantom reads. A phantom read is when a transaction ("Transaction A") selects a list of rows through a
WHERE
clause, another transaction ("Transaction B") performs anINSERT
orDELETE
with a row that satisfies Transaction A'sWHERE
clause, and Transaction A selects using the same WHERE clause again, resulting in an inconsistency.TRANSACTION_SERIALIZABLE: The strictest setting. Prevents dirty reads, non-repeatable reads, and phantom reads.
readOnly
The readOnly
parameter indicates whether any database connection used by the transaction is in read-only mode, and is set to false
by default. Much like with transactionIsolation
, this value is not directly used by Exposed, but is simply relayed to the database.
db
The db
parameter is optional and is used to select the database where the transaction should be settled (see the section above).
maxAttempts
Transactions also provide a property, maxAttempts
, which sets the maximum number of attempts that should be made to perform a transaction block. If this value is set to 1 and an SQLException occurs inside the transaction block, the exception will be thrown without performing a retry. If this property is not set, any default value provided in DatabaseConfig
will be used instead:
If this property is set to a value greater than 1, minRetryDelay
and maxRetryDelay
can also be set in the transaction block to indicate the minimum and maximum number of milliseconds to wait before retrying.
queryTimeout
Another advanced property available in a transaction block is queryTimeout
. This sets the number of seconds to wait for each statement in the block to execute before timing out:
Statement Interceptors
DSL operations within a transaction create SQL statements, on which commands like Execute, Commit, and Rollback are issued. Exposed provides the StatementInterceptor
interface that allows you to implement your own logic before and after these specific steps in a statement's lifecycle.
registerInterceptor()
and unregisterInterceptor()
can be used to enable and disable a custom interceptor in a single transaction.
To use a custom interceptor that acts on all transactions, extend the GlobalStatementInterceptor
class instead. Exposed uses the Java SPI ServiceLoader to discover and load any implementations of this class. In this situation, a new file should be created in the resources folder named:
The contents of this file should be the fully qualified class names of all custom implementations.