Transactor
Transactor is an injectable type making DB transactions seamless.
Install / Use
/learn @Thiht/TransactorREADME
transactor
The transactor pattern is a way to manage transactions seamlessly. You can inject your transactor in your services to make transactions completely transparently.
It relies mostly on the Transactor interface:
type Transactor interface {
WithinTransaction(context.Context, func(context.Context) error) error
}
WithinTransaction starts a new transaction and adds it to the context. Any repository method can then retrieve a transaction from the context or fallback to the initial DB handler. The transaction is committed if the provided function doesn't return an error. It's rollbacked otherwise.
Usage
Installation
go get github.com/Thiht/transactor
The database/sql default implementation (stdlib) is included in the github.com/Thiht/transactor package.
Additional implementations are available in separate modules:
- the
pgximplementation is available ingithub.com/Thiht/transactor/pgx, - the
sqlximplementation is available ingithub.com/Thiht/transactor/sqlx.
The following examples use the stdlib implementation, but the code isn't too different with the other implementations.
Initialize a transactor
This example uses database/sql with the pgx driver, but any database/sql driver can be used.
import stdlibTransactor "github.com/Thiht/transactor/stdlib"
db, _ := sql.Open("pgx", dsn)
transactor, dbGetter := stdlibTransactor.NewTransactor(
db,
stdlibTransactor.NestedTransactionsSavepoints,
)
The currently available strategies for nested transactions with the stdlib implementation are:
- NestedTransactionsSavepoints, an implementation using
SAVEPOINTSand compatible with PostgreSQL, MySQL, MariaDB, and SQLite, - NestedTransactionsOracle, an implementation using Oracle savepoints,
- NestedTransactionsMSSQL, an implementation using Microsoft SQL Server savepoints,
- NestedTransactionsNone, an implementation that prevents using nested transactions.
Use the dbGetter in your repositories
Instead of injecting the *sql.DB handler directly to your repositories, you now have to inject the dbGetter. It will return the appropriate DB handler depending on whether the current execution is in a transaction.
type store struct {
- db *sql.DB
+ dbGetter stdlibTransactor.DBGetter
}
func (s store) GetBalance(ctx context.Context, account string) (int, error) {
var balance int
- err := s.db.QueryRowContext(
+ err := s.dbGetter(ctx).QueryRowContext(
ctx,
`SELECT balance FROM accounts WHERE account = $1`,
account,
).Scan(&balance)
return balance, err
}
You can use the IsWithinTransaction helper if you need to implement different behaviours depending on whether a transaction is running.
For example with PostgreSQL, you could add FOR UPDATE conditionally:
func (s store) GetBalance(ctx context.Context, account string) (int, error) {
query := `SELECT balance FROM accounts WHERE account = $1`
if stdlibTransactor.IsWithinTransaction(ctx) {
query += ` FOR UPDATE`
}
// ...
}
Use the transactor in your services
type service struct {
balanceStore stores.Balance
transactor transactor.Transactor
}
func (s service) IncreaseBalance(ctx context.Context, account string, amount int) error {
return s.transactor.WithinTransaction(ctx, func(ctx context.Context) error {
balance, err := s.balanceStore.GetBalance(ctx, account)
if err != nil {
return err
}
balance += amount
err = s.balanceStore.SetBalance(ctx, account, balance)
if err != nil {
return err
}
return nil
})
}
Thanks to nested transactions support, you can even call your services within a transaction:
type service struct {
balanceStore stores.Balance
transactor transactor.Transactor
}
func (s service) TransferBalance(
ctx context.Context,
fromAccount, toAccount string,
amount int,
) error {
return s.transactor.WithinTransaction(ctx, func(ctx context.Context) error {
err := s.DecreaseBalance(ctx, fromAccount, amount)
if err != nil {
return err
}
err = s.IncreaseBalance(ctx, toAccount, amount)
if err != nil {
return err
}
return nil
})
}
func (s service) IncreaseBalance(ctx context.Context, account string, amount int) error {
return s.transactor.WithinTransaction(ctx, func(ctx context.Context) error {
// ...
})
}
func (s service) DecreaseBalance(ctx context.Context, account string, amount int) error {
return s.transactor.WithinTransaction(ctx, func(ctx context.Context) error {
// ...
})
}
[!WARNING] Transactions are not thread safe, so make sure not to call code making concurrent database access inside
WithinTransaction
Testing
In your tests, you can inject a fake transactor and dbGetter, using NewFakeTransactor:
transactor, dbGetter := stdlibTransactor.NewFakeTransactor(db)
The fake transactor will simply execute its callback function, and the dbGetter will return the provided db handler.
This strategy works because when using this library, you don't have to worry about how transactions are made, just about returning errors appropriately in WithinTransaction.
Related Skills
oracle
343.3kBest practices for using the oracle CLI (prompt + file bundling, engines, sessions, and file attachment patterns).
xurl
343.3kA CLI tool for making authenticated requests to the X (Twitter) API. Use this skill when you need to post tweets, reply, quote, search, read posts, manage followers, send DMs, upload media, or interact with any X API v2 endpoint.
prose
343.3kOpenProse VM skill pack. Activate on any `prose` command, .prose files, or OpenProse mentions; orchestrates multi-agent workflows.
Command Development
92.1kThis skill should be used when the user asks to "create a slash command", "add a command", "write a custom command", "define command arguments", "use command frontmatter", "organize commands", "create command with file references", "interactive command", "use AskUserQuestion in command", or needs guidance on slash command structure, YAML frontmatter fields, dynamic arguments, bash execution in commands, user interaction patterns, or command development best practices for Claude Code.
