Transactions
Manage atomic, durable transactions using ekoDB's REST API with support for savepoints and full rollback capabilities.
ekoDB transactions today guarantee atomicity (all-or-nothing, via a per-transaction undo log) and durability (WAL). They do not currently enforce isolation between concurrent transactions. There is no locking, no MVCC, and no conflict detection. The isolation_level parameter is accepted by the API and recorded on the transaction, but it does not change runtime behavior. Every level behaves as ReadUncommitted today, so concurrent transactions can observe each other's uncommitted writes regardless of the level requested. Enforced isolation is on the roadmap. Do not rely on any isolation guarantee yet; serialize correctness-critical concurrent work at the application layer. See the Isolation Levels section for the precise current status.
For detailed information about transactions architecture, isolation levels, and best practices, see the Transactions Architecture Reference.
When using the REST API directly (without client libraries), be aware of these important limitations:
- Manual Transaction ID Management - You must track and include the
transaction_idwith every operation within a transaction - Connection Stateless - REST is stateless; transaction state is maintained server-side with timeout expiration
- No Automatic Rollback - If your client crashes, the transaction will remain pending until timeout (default: 5 minutes)
- Network Failures - A failed commit response doesn't guarantee the commit failed; use idempotency keys for critical operations
Recommendation: Use the official client libraries which handle transaction lifecycle automatically.
Isolation Levels
The four levels below are defined in the IsolationLevel enum and can be passed to the API. They are recorded on the transaction and surfaced in status responses, but they do not currently change runtime behavior. ekoDB has no concurrency control (no read/write locks, no MVCC, no snapshot versioning, no conflict detection), so every transaction effectively runs at ReadUncommitted semantics regardless of the level requested. The parameter exists for forward compatibility. Enforced isolation (Serializable is not yet implemented) is roadmap work.
ekoDB defines four isolation-level names. The descriptions below are the intended (not-yet-enforced) semantics:
- ReadUncommitted - Intended: dirty reads allowed, no read locks. This is also the de-facto behavior of every level today.
- ReadCommitted (default) - Intended: no dirty reads. Not yet enforced; behaves as ReadUncommitted today.
- RepeatableRead - Intended: consistent reads within a transaction. Not yet enforced; behaves as ReadUncommitted today.
- Serializable - Intended: highest consistency via full MVCC / conflict detection. Not yet implemented; behaves as ReadUncommitted today.
Begin Transaction
Start a new transaction with optional isolation level and timeout.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
let tx_id = client.begin_transaction(Some(TransactionOptions {
isolation_level: IsolationLevel::Serializable,
timeout_seconds: Some(600),
})).await?;
println!("{}", tx_id); // 'tx-abc-123'
tx_id = await client.begin_transaction(
isolation_level='Serializable'
)
print(tx_id) # 'tx-abc-123'
const txId = await client.beginTransaction({
isolation_level: "Serializable",
});
console.log(txId); // 'tx-abc-123'
const txId = await client.beginTransaction({
isolation_level: "Serializable",
});
console.log(txId); // 'tx-abc-123'
val txId = client.beginTransaction(
isolationLevel = "Serializable"
)
println(txId) // 'tx-abc-123'
txId, err := client.BeginTransaction(ekodb.TransactionOptions{
IsolationLevel: "Serializable",
})
fmt.Println(txId) // 'tx-abc-123'
curl -X POST https://{EKODB_API_URL}/api/transactions \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-H "Content-Type: application/json" \
-d '{
"isolation_level": "Serializable",
"timeout_seconds": 600
}'
# Response
{
"transaction_id": "tx-abc-123",
"isolation_level": "Serializable",
"created_at": 1699814400
}
Prefer a client library? See examples in Rust, Python, TypeScript, Go, or Kotlin
Want to use the REST API directly? See examples in JavaScript, Python, Go, or Rust
Parameters:
isolation_level(optional) -ReadUncommitted,ReadCommitted,RepeatableRead,Serializabletimeout_seconds(optional) - Default: 300 (5 minutes)
Commit Transaction
Commit all operations within the transaction.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
client.commit_transaction(&tx_id).await?;
await client.commit_transaction(tx_id)
await client.commitTransaction(txId);
await client.commitTransaction(txId);
client.commitTransaction(txId)
err := client.CommitTransaction(txId)
curl -X POST https://{EKODB_API_URL}/api/transactions/{transaction_id}/commit \
-H "Authorization: Bearer {YOUR_API_TOKEN}"
# Response
{
"status": "committed",
"transaction_id": "tx-abc123",
"operations_count": 5
}
Rollback Transaction
Rollback all operations and abort the transaction.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
client.rollback_transaction(&tx_id).await?;
await client.rollback_transaction(tx_id)
await client.rollbackTransaction(txId);
await client.rollbackTransaction(txId);
client.rollbackTransaction(txId)
err := client.RollbackTransaction(txId)
curl -X POST https://{EKODB_API_URL}/api/transactions/{transaction_id}/rollback \
-H "Authorization: Bearer {YOUR_API_TOKEN}"
# Response
{
"status": "rolled_back",
"transaction_id": "tx-abc123",
"operations_rolled_back": 5
}
Create Savepoint
Create a savepoint for partial rollback within a transaction.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
client.create_savepoint(&tx_id, "checkpoint1").await?;
await client.create_savepoint(tx_id, 'checkpoint1')
await client.createSavepoint(txId, "checkpoint1");
await client.createSavepoint(txId, "checkpoint1");
client.createSavepoint(txId, "checkpoint1")
err := client.CreateSavepoint(txId, "checkpoint1")
curl -X POST https://{EKODB_API_URL}/api/transactions/{transaction_id}/savepoints \
-H "Authorization: Bearer {ADMIN_TOKEN}" \
-H "Content-Type: application/json" \
-d '{
"name": "checkpoint1"
}'
# Response
{
"status": "created",
"transaction_id": "tx-abc123",
"savepoint_name": "checkpoint1"
}
Rollback to Savepoint
Rollback to a specific savepoint, undoing operations after it.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
client.rollback_to_savepoint(&tx_id, "checkpoint1").await?;
await client.rollback_to_savepoint(tx_id, 'checkpoint1')
await client.rollbackToSavepoint(txId, "checkpoint1");
await client.rollbackToSavepoint(txId, "checkpoint1");
client.rollbackToSavepoint(txId, "checkpoint1")
err := client.RollbackToSavepoint(txId, "checkpoint1")
curl -X POST https://{EKODB_API_URL}/api/transactions/{transaction_id}/savepoints/{savepoint_name}/rollback \
-H "Authorization: Bearer {ADMIN_TOKEN}"
# Response
{
"status": "rolled_back",
"transaction_id": "tx-abc123",
"savepoint_name": "checkpoint1",
"operations_rolled_back": 3
}
Release Savepoint
Remove a savepoint that's no longer needed.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
client.release_savepoint(&tx_id, "checkpoint1").await?;
await client.release_savepoint(tx_id, 'checkpoint1')
await client.releaseSavepoint(txId, "checkpoint1");
await client.releaseSavepoint(txId, "checkpoint1");
client.releaseSavepoint(txId, "checkpoint1")
err := client.ReleaseSavepoint(txId, "checkpoint1")
curl -X DELETE https://{EKODB_API_URL}/api/transactions/{transaction_id}/savepoints/{savepoint_name} \
-H "Authorization: Bearer {ADMIN_TOKEN}"
# Response
{
"status": "released",
"transaction_id": "tx-abc123",
"savepoint_name": "checkpoint1"
}
Get Transaction Status
Check the current status of a transaction.
- Client Libraries (Recommended)
- Direct API
- 🦀 Rust
- 🐍 Python
- 📘 TypeScript
- 📦 JavaScript
- 🟣 Kotlin
- 🔷 Go
let status = client.get_transaction_status(&tx_id).await?;
println!("{:?}", status);
status = await client.get_transaction_status(tx_id)
print(status['state'])
print(status['operations_count'])
const status = await client.getTransactionStatus(txId);
console.log(status.state);
console.log(status.operations_count);
const status = await client.getTransactionStatus(txId);
console.log(status.state);
console.log(status.operations_count);
val status = client.getTransactionStatus(txId)
println(status)
status, err := client.GetTransactionStatus(txId)
fmt.Println(status)
curl https://{EKODB_API_URL}/api/transactions/{transaction_id} \
-H "Authorization: Bearer {ADMIN_TOKEN}"
# Response
{
"transaction_id": "tx-abc123",
"state": "Active",
"isolation_level": "Serializable",
"operations_count": 5,
"savepoints_count": 2,
"created_at": 1699814400,
"last_activity": 1699814450,
"timeout_seconds": 600
}
List Active Transactions
List all currently active transactions.
- Client Libraries (Recommended)
- Direct API
- 📘 TypeScript
- 📦 JavaScript
- 🐍 Python
- 🦀 Rust
- 🟣 Kotlin
- 🔷 Go
const transactions = await client.listActiveTransactions();
console.log(transactions);
/*
[
{ transaction_id: 'tx-abc-123', isolation_level: 'Serializable', created_at: 1699814400 },
{ transaction_id: 'tx-def-456', isolation_level: 'ReadCommitted', created_at: 1699814500 }
]
*/
const transactions = await client.listActiveTransactions();
console.log(transactions);
transactions = await client.list_active_transactions()
print(transactions)
# [{'transaction_id': 'tx-abc-123', ...}]
let transactions = client.list_active_transactions().await?;
println!("{:?}", transactions);
val transactions = client.listActiveTransactions()
println(transactions)
transactions, err := client.ListActiveTransactions()
fmt.Println(transactions)
curl https://{EKODB_API_URL}/api/transactions \
-H "Authorization: Bearer {ADMIN_TOKEN}"
# Response
{
"transactions": [
{
"transaction_id": "tx-abc123",
"state": "Active",
"isolation_level": "Serializable",
"operations_count": 5,
"savepoints_count": 2,
"created_at": 1699814400,
"last_activity": 1699814450,
"timeout_seconds": 600
},
{
"transaction_id": "tx-def456",
"state": "Active",
"isolation_level": "ReadCommitted",
"operations_count": 2,
"savepoints_count": 0,
"created_at": 1699814500,
"last_activity": 1699814510,
"timeout_seconds": 300
}
],
"count": 2
}
Complete Example
Here's a complete example of a multi-step transaction with savepoints:
# 1. Begin transaction
curl -X POST https://{EKODB_API_URL}/api/transactions \
-H "Content-Type: application/json" \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-d '{"isolation_level": "Serializable", "timeout_seconds": 600}'
# Response: {"transaction_id": "tx-001"}
# 2. Create initial savepoint
curl -X POST https://{EKODB_API_URL}/api/transactions/tx-001/savepoints \
-H "Content-Type: application/json" \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-d '{"name": "start"}'
# 3. Debit source account
curl -X PUT https://{EKODB_API_URL}/api/update/accounts/ACC001 \
-H "Content-Type: application/json" \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-d '{"balance": 900}'
# 4. Create savepoint after debit
curl -X POST https://{EKODB_API_URL}/api/transactions/tx-001/savepoints \
-H "Content-Type: application/json" \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-d '{"name": "after_debit"}'
# 5. Credit destination account
curl -X PUT https://{EKODB_API_URL}/api/update/accounts/ACC002 \
-H "Content-Type: application/json" \
-H "Authorization: Bearer {YOUR_API_TOKEN}" \
-d '{"balance": 1100}'
# 6. If credit fails, rollback to after_debit
# curl -X POST https://{EKODB_API_URL}/api/transactions/tx-001/savepoints/after_debit/rollback \
# -H "Authorization: Bearer {YOUR_API_TOKEN}"
# Then retry credit operation...
# 7. Commit transaction
curl -X POST https://{EKODB_API_URL}/api/transactions/tx-001/commit \
-H "Authorization: Bearer {YOUR_API_TOKEN}"
Best Practices
Declare Intended Isolation Level
The table below describes which level to declare so intent is captured today and enforced behavior applies automatically once isolation enforcement ships. Until then, no level provides isolation (see Isolation Levels), so a correctness-critical concurrent workload must not depend on the database for isolation yet. Serialize such work at the application layer or run it single-writer.
| Use Case | Declare | Why (intent — see note above) |
|---|---|---|
| Analytics/Reporting | ReadUncommitted | Performance over consistency |
| General CRUD | ReadCommitted | Intended default |
| Financial Calculations | RepeatableRead | Consistent reads |
| Banking/Critical | Serializable | Maximum safety (when enforced) |
Use Savepoints for Multi-Stage Operations
Savepoints allow partial rollback without aborting the entire transaction:
# Create savepoint before risky operation
POST /api/transactions/{id}/savepoints
{"name": "before_risky_op"}
# Perform risky operation
PUT /api/update/...
# If operation fails, rollback to savepoint
POST /api/transactions/{id}/savepoints/before_risky_op/rollback
# Retry or continue with alternative approach
Set Appropriate Timeouts
// Short transaction (1 minute)
{"timeout_seconds": 60}
// Long-running workflow (30 minutes)
{"timeout_seconds": 1800}
// Critical operation (5 minutes, default)
{"timeout_seconds": 300}
Handle Errors Gracefully
Always implement proper error handling with rollback:
# Begin transaction
curl -X POST .../api/transactions ...
# -> {"transaction_id": "tx-001"}
# Try operations
if operation_fails; then
# Rollback on error
curl -X POST .../api/transactions/tx-001/rollback ...
else
# Commit on success
curl -X POST .../api/transactions/tx-001/commit ...
fi
Monitor Active Transactions
Track and debug active transactions:
# List all active transactions
GET /api/transactions
# Check specific transaction status
GET /api/transactions/{id}
# Useful for:
# - Debugging stuck transactions
# - Monitoring long-running operations
# - Identifying transaction bottlenecks
Use Cases
Financial Transfers
Multi-account updates with atomicity guarantees:
# Begin transaction
POST /api/transactions
{"isolation_level": "Serializable"}
# Debit source
PUT /api/update/accounts/{from_id}
{"$decrement": {"balance": 100}}
# Credit destination
PUT /api/update/accounts/{to_id}
{"$increment": {"balance": 100}}
# Commit both or rollback both
POST /api/transactions/{id}/commit
E-Commerce Orders
Order creation with inventory reservation:
# Begin transaction
POST /api/transactions
{"isolation_level": "RepeatableRead"}
# Create savepoint before inventory check
POST /api/transactions/{id}/savepoints
{"name": "before_inventory"}
# Reserve inventory
PUT /api/update/inventory/{product_id}
{"$decrement": {"stock": 1}}
# If insufficient stock, rollback to savepoint
# Otherwise create order
POST /api/insert/orders
{...}
# Commit transaction
POST /api/transactions/{id}/commit
Batch Processing
Process multiple records with error recovery:
# Begin transaction
POST /api/transactions
# Process records in batches
for batch in batches:
# Create savepoint before batch
POST /api/transactions/{id}/savepoints
{"name": "batch_{i}"}
# Process batch
POST /api/batch/insert/...
# If batch fails, rollback and continue
if error:
POST /api/transactions/{id}/savepoints/batch_{i}/rollback
continue
# Commit all successful batches
POST /api/transactions/{id}/commit
Related Documentation
- Transactions Architecture Reference - Detailed architecture, isolation levels, and best practices
- Functions Architecture - Function-based transactions (REST API)
- Functions via Client Libraries - Function-based transactions with client SDKs
- Basic Operations - Standard CRUD operations
- Batch Operations - Bulk operations within transactions
- White Paper - ACID compliance and architecture
Example Code
Direct HTTP/REST API Examples
Raw HTTP examples demonstrating the REST API directly:
- JavaScript:
transactions.js - Python:
transactions.py - Go:
transactions.go - Rust:
transactions.rs
Client Library Examples
Production-ready examples using official client libraries:
- Rust:
client_transactions.rs - Python:
client_transactions.py - TypeScript:
client_transactions.ts - Go:
client_transactions.go - Kotlin:
ClientTransactions.kt - JavaScript:
client_transactions.js