sequel

package module
v1.10.2 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Jun 16, 2026 License: Apache-2.0 Imports: 31 Imported by: 1

README

Sequel

A Go library that enhances database/sql with cross-driver SQL, schema migration, ephemeral test databases, and adaptive connection pooling.

Features at a Glance

  • Connection pool management - Prevents database exhaustion when many consumers in one process share a DSN
  • Schema migration - Concurrency-safe, incremental database migrations
  • Cross-driver support - MySQL, PostgreSQL, CockroachDB, SQL Server, and SQLite with unified API
  • Retrying transactions - Transact runs a closure in a transaction, retries on deadlock/lock contention, and never commits partial work
  • Ephemeral test databases - Isolated databases per test with automatic cleanup

Quick Start

import "github.com/microbus-io/sequel"

// Open a database connection with its own pool
db, err := sequel.Open("", "root:root@tcp(127.0.0.1:3306)/mydb")

// Run migrations
err = db.Migrate("myservice@v1", migrationFilesFS)

// Use db.DB for standard sql.DB operations
rows, err := db.Query("SELECT * FROM users WHERE tenant_id=?", tenantID)

Connection Pool Management

Sequel exposes two constructors so the connection-pool strategy is self-documenting at the call site:

  • Open(driver, dsn) returns a fresh *DB with its own pool. Each call returns a distinct instance; sequel does not coalesce by DSN and does not size the pool automatically. The standard database/sql defaults apply (unlimited open, 2 idle) until the caller adjusts them with SetMaxOpenConns / SetMaxIdleConns. Use this for a single heavy consumer (e.g. a long-running worker pool) where you want to size the pool to the workload.

  • OpenSingleton(driver, dsn) returns a coalesced *DB: multiple calls with the same (driver, dsn) share one *sql.DB and one connection pool. Sequel automatically sizes that pool based on the number of openers using a sqrt-based formula:

    • maxIdle ≈ sqrt(N) where N is the number of openers
    • maxOpen ≈ (sqrt(N) * 2) + 2

    This is the right choice when many parts of the same process each open the same DSN occasionally — the pool grows gently with the number of openers and no caller has to think about pool sizing.

// Single heavy consumer — caller manages the pool.
db, err := sequel.Open("", dsn)
db.SetMaxOpenConns(32)
db.SetMaxIdleConns(8)

// Multiple consumers sharing a DSN — sequel manages one pool across them.
db, err := sequel.OpenSingleton("", dsn)

Schema Migration

Sequel performs incremental schema migration using numbered SQL files (1.sql, 2.sql, etc.). Migrations are:

  • Concurrency-safe - Distributed locking ensures only one replica executes each migration
  • Tracked - A sequel_migrations table records completed migrations
  • Driver-aware - Use -- DRIVER: drivername comments for driver-specific SQL (list multiple, space-separated, to share a statement across drivers)
// Embed migration files
//go:embed sql/*.sql
var migrationFS embed.FS

// Run migrations (safe to call from multiple replicas)
err := db.Migrate("unique-sequence-name", migrationFS)

Example migration file with driver-specific syntax:

-- DRIVER: mysql
ALTER TABLE users MODIFY COLUMN email VARCHAR(384) NOT NULL;

-- DRIVER: pgx cockroachdb
ALTER TABLE users ALTER COLUMN email TYPE VARCHAR(384);

-- DRIVER: mssql
ALTER TABLE users ALTER COLUMN email NVARCHAR(384) NOT NULL;

-- DRIVER: sqlite
-- SQLite does not support ALTER COLUMN; a table rebuild would be needed

Cross-Driver Support

Sequel supports MySQL, PostgreSQL, CockroachDB, SQL Server, and SQLite through a unified API. Write your SQL once using MySQL-style ? placeholders and virtual functions, and Sequel automatically adapts queries for the active driver.

CockroachDB speaks the PostgreSQL wire protocol and shares the pgx driver, but it is exposed as a distinct driver name (cockroachdb) because callers may need to branch on Cockroach-specific behavior — retry semantics and async schema changes in particular. Internally, every PostgreSQL expansion (placeholders, virtual functions, DSN parsing) applies identically to cockroachdb.

Automatic Placeholder Conversion

All query methods (Exec, Query, QueryRow, Prepare, and their Context variants) automatically convert ? placeholders to the driver's native syntax. For PostgreSQL, ? becomes $1, $2, etc. For MySQL, SQL Server, and SQLite, ? is left as-is. Placeholders inside quoted strings are left untouched.

// Works on all drivers - placeholders are converted automatically
rows, err := db.Query("SELECT * FROM users WHERE tenant_id = ? AND active = ?", tenantID, true)
// PostgreSQL receives: SELECT * FROM users WHERE tenant_id = $1 AND active = $2
Virtual Functions

Virtual functions are driver-agnostic function calls in your SQL that Sequel expands into driver-specific expressions before execution. They are matched case-insensitively and support nesting. Quoted strings inside arguments are handled correctly.

Built-in Virtual Functions

NOW_UTC() returns the current UTC timestamp with millisecond precision.

Driver NOW_UTC() expands to
MySQL (UTC_TIMESTAMP(3))
PostgreSQL (NOW() AT TIME ZONE 'UTC')
SQL Server (CONVERT(DATETIME2(3), SYSUTCDATETIME()))
SQLite STRFTIME('%Y-%m-%d %H:%M:%f', 'now')

On SQL Server the value is rounded to millisecond precision so it matches the other drivers and the precision of a DATETIME2(3) column. SYSUTCDATETIME() alone is 100-nanosecond precision, which rounds up when stored into a millisecond column and can leave a just-written "now" timestamp slightly in the future relative to a later NOW_UTC() comparison.

REGEXP_TEXT_SEARCH(expr IN col1, col2, ...) performs a case-insensitive regular expression search across one or more columns.

Driver REGEXP_TEXT_SEARCH(? IN name, email) expands to
MySQL CONCAT_WS(' ',name,email) REGEXP ?
PostgreSQL REGEXP_LIKE(CONCAT_WS(' ',name,email), ?, 'i')
SQL Server REGEXP_LIKE(CONCAT_WS(' ',name,email), ?, 'i')
SQLite CONCAT_WS(' ',name,email) LIKE '%' || ? || '%'

DATE_ADD_MILLIS(baseExpr, milliseconds) adds milliseconds to a timestamp expression.

Driver DATE_ADD_MILLIS(created_at, ?) expands to
MySQL DATE_ADD(created_at, INTERVAL (?) * 1000 MICROSECOND)
PostgreSQL created_at + MAKE_INTERVAL(secs => (?) / 1000.0)
SQL Server DATEADD(MILLISECOND, ?, created_at)
SQLite STRFTIME('%Y-%m-%d %H:%M:%f', created_at, '+' || ((?) / 1000.0) || ' seconds')

DATE_DIFF_MILLIS(a, b) returns the difference (a - b) in milliseconds.

Driver DATE_DIFF_MILLIS(updated_at, created_at) expands to
MySQL TIMESTAMPDIFF(MICROSECOND, created_at, updated_at) / 1000.0
PostgreSQL EXTRACT(EPOCH FROM (updated_at - created_at)) * 1000.0
SQL Server DATEDIFF_BIG(MILLISECOND, created_at, updated_at)
SQLite (JULIANDAY(updated_at) - JULIANDAY(created_at)) * 86400000.0

LIMIT_OFFSET(limit, offset) provides cross-driver pagination. Note that SQL Server requires an ORDER BY clause.

Driver LIMIT_OFFSET(10, 0) expands to
MySQL LIMIT 10 OFFSET 0
PostgreSQL LIMIT 10 OFFSET 0
SQL Server OFFSET 0 ROWS FETCH NEXT 10 ROWS ONLY
SQLite LIMIT 10 OFFSET 0
db.Query("SELECT * FROM users ORDER BY id LIMIT_OFFSET(?, ?)", limit, offset)
Nesting

Virtual functions can be nested. Inner functions are expanded first across multiple passes:

db.Exec("UPDATE t SET expires_at = DATE_ADD_MILLIS(NOW_UTC(), ?) WHERE id = ?", ttlMs, id)
// MySQL:      UPDATE t SET expires_at = DATE_ADD(UTC_TIMESTAMP(3), INTERVAL (?) * 1000 MICROSECOND) WHERE id = ?
// PostgreSQL: UPDATE t SET expires_at = (NOW() AT TIME ZONE 'UTC') + MAKE_INTERVAL(secs => ($1) / 1000.0) WHERE id = $2
Custom Virtual Functions

Register your own virtual functions with RegisterVirtualFunc:

sequel.RegisterVirtualFunc("BOOL", func(driverName string, args string) (string, error) {
    switch driverName {
    case "mysql", "pgx", "sqlite":
        return args, nil
    case "mssql":
        // SQL Server uses BIT, not BOOL
        return "CAST(" + args + " AS BIT)", nil
    default:
        return "", errors.New("unsupported driver: %s", driverName)
    }
})
UnpackQuery

UnpackQuery is the public method that expands virtual functions and conforms placeholders. It is called automatically by the query shadow methods, but can be used directly if needed:

expanded, err := db.UnpackQuery("SELECT * FROM t WHERE updated_at > DATE_ADD_MILLIS(NOW_UTC(), ?) AND active = ?")
InsertReturnID

InsertReturnID executes an INSERT statement and returns the auto-generated ID for the named ID column. Each driver uses its native mechanism:

Driver Mechanism
MySQL LastInsertId() from the result
PostgreSQL Appends RETURNING <idColumn> to the query
SQL Server Injects OUTPUT INSERTED.<idColumn> before VALUES
SQLite LastInsertId() from the result
id, err := db.InsertReturnID(ctx, "id", "INSERT INTO users (name, email) VALUES (?, ?)", name, email)
DriverName()

DriverName() returns the active driver name ("mysql", "pgx", "mssql", or "sqlite") for cases where you need driver-specific logic in Go code.

Transactions

db.BeginTx returns a sequel.Tx that shadows sql.Tx with virtual-function expansion and placeholder conforming — use it exactly like sql.Tx.

For transactions that must survive contention, db.Transact runs a closure in a transaction, commits on success, and retries the whole closure on a deadlock or lock-contention error with a short jittered backoff:

err := db.Transact(ctx, func(tx *sequel.Tx) error {
    if _, err := tx.ExecContext(ctx, "UPDATE accounts SET balance = balance - ? WHERE id = ?", amt, from); err != nil {
        return err
    }
    _, err := tx.ExecContext(ctx, "UPDATE accounts SET balance = balance + ? WHERE id = ?", amt, to)
    return err
})
  • Retry-safe by re-running. A retried attempt re-executes the closure from the start in a new transaction (the previous attempt is rolled back), so the closure must be safe to run more than once — any non-transactional side effects it performs may repeat. Because retries re-run the Go code rather than replay recorded statements, a transaction whose control flow depends on data committed by another transaction between attempts stays correct.
  • No partial commits. The Tx passed to the closure records the first statement error and short-circuits the rest, so the transaction never commits half its work even if the closure forgets to check a statement's error.
  • SQL Server XACT_ABORT ON. Applied automatically inside Transact so any statement error aborts the whole transaction server-side.

A Tx from BeginTx does neither error-recording nor retry — it behaves exactly like sql.Tx.

Ephemeral Test Databases

Provisioning a per-test database is a separate step from opening a connection. CreateTestingDatabase(driver, baseDSN, uniqueTestID) creates (or reuses) a uniquely-named database and returns its DSN; pass that DSN to Open or OpenSingleton to connect.

// Test fixture
func TestUserService(t *testing.T) {
    dsn, err := sequel.CreateTestingDatabase("", "root:root@tcp(127.0.0.1:3306)/mydb", t.Name())
    if err != nil { t.Fatal(err) }
    db, err := sequel.OpenSingleton("", dsn)
    if err != nil { t.Fatal(err) }
    defer db.Close()  // also drops the testing database
}

The same helper can be invoked from production startup paths that want to swap in a per-test database without rewriting the rest of the wiring:

func startup(cfg Config) (*sequel.DB, error) {
    dsn := cfg.DSN
    if cfg.Testing {
        var err error
        dsn, err = sequel.CreateTestingDatabase("", cfg.DSN, cfg.TestID)
        if err != nil { return nil, err }
    }
    return sequel.OpenSingleton("", dsn)
}

Repeated calls within the same process with the same (driver, baseDSN, uniqueTestID) reuse the same testing database — the DROP+CREATE runs only once. The returned DSN points at a database whose name has the testing_NN_ prefix; sequel inspects this on Close and drops the database automatically when the last referencing *DB is closed. There is no separate cleanup call to remember. If a process exits before Close runs, the leftover-cleanup sweep on the next CreateTestingDatabase call removes stale databases older than 1–2 hours.

Choosing the server with SEQUEL_TESTING_DSN

When CreateTestingDatabase is called with neither a driver nor a base DSN, it falls back to the SEQUEL_TESTING_DSN environment variable. This lets you run the same test suite against any supported server without touching test code — leave the variable unset to use in-memory SQLite (the default, no server required), or set it to a base DSN to run against that server instead, with the driver inferred from the DSN:

func TestUserService(t *testing.T) {
    // "" driver + "" DSN → SEQUEL_TESTING_DSN, or in-memory SQLite if it is unset.
    dsn, err := sequel.CreateTestingDatabase("", "", t.Name())
    if err != nil { t.Fatal(err) }
    db, err := sequel.OpenSingleton("", dsn)
    if err != nil { t.Fatal(err) }
    defer db.Close()
}
# Same tests, different engine — no code change.
go test ./...                                                          # SQLite (default)
SEQUEL_TESTING_DSN='postgres://user:pw@127.0.0.1:5432/' go test ./...  # PostgreSQL
SEQUEL_TESTING_DSN='root:pw@tcp(127.0.0.1:3306)/'       go test ./...  # MySQL

Passing an explicit driver — even with an empty DSN, which just selects that driver's localhost default — opts out of the fallback, so a test that deliberately targets a specific engine keeps using it regardless of the environment. Because the variable is read inside CreateTestingDatabase, any project that provisions its test databases through sequel inherits this behavior with no additional wiring.

Observability

Sequel emits OpenTelemetry traces and metrics and slog logs when you supply the corresponding providers. Everything is opt-in and off by default — a *DB with no providers configured does no extra work beyond a single atomic-pointer check on the hot path.

Providers are attached after Open/OpenSingleton (which keep the standard database/sql signature) rather than at construction. Nothing is lost by this: sql.Open does no I/O — it only prepares a lazy pool — so there is no work inside Open worth instrumenting; every operation that does real work happens later on the returned *DB.

db, _ := sequel.Open("", dsn)
db.SetTracerProvider(tracerProvider) // trace.TracerProvider — client spans per query/transaction/migration
db.SetMeterProvider(meterProvider)   // metric.MeterProvider — sequel_* metrics
db.SetLogger(logger)                 // *slog.Logger — migration events; per-query in verbose mode
db.SetVerbose(true)                  // optional: add statement text to spans, log each query at Debug

Configure once, before the *DB is used concurrently. For an OpenSingleton-shared *DB, the providers are process-wide for that pool; set them from the owning caller (last writer wins). Pass nil to any setter to disable that signal.

Spans

Each query, Transact, and Migrate gets a client span following OpenTelemetry database semantic conventions:

  • db.system.name — the driver (mysql, pgx, cockroachdb, mssql, sqlite)
  • db.operation.name — the SQL verb (SELECT, INSERT, …)
  • db.collection.name — the table, only when it can be determined unambiguously (omitted for joins, multi-table FROM lists, and subqueries, so a present value is trustworthy)
  • db.query.text — the parameterized statement, only in verbose mode (placeholders only; argument values are never captured)

The span name is "{operation} {table}" (e.g. SELECT users), or just the operation when no table is captured.

Metrics

All metric names carry the sequel_ prefix:

Metric Type Notes
sequel_query_duration histogram (s) attrs: db.system.name, db.operation.name, status (ok/error)
sequel_transaction_duration histogram (s) attrs: db.system.name, outcome (committed/rolledback)
sequel_lock_contention_total counter incremented once per surfaced lock-contention/deadlock error
sequel_migration_runs_total counter counts migrations that actually ran (skipped ones excluded); attrs include status
sequel_pool_open_connections gauge from sql.DBStats, attr database (never the raw DSN)
sequel_pool_in_use_connections gauge
sequel_pool_idle_connections gauge
sequel_pool_wait_count gauge
sequel_pool_wait_duration_seconds gauge
Logs

The library does not log operation errors — every error is returned to the caller, who is best placed to log it. Logging is reserved for:

  • Info — one-off events: each schema migration as it is attempted (regardless of outcome).
  • Debug — every query, but only when SetVerbose(true) is set.
QueryRow returns *sequel.Row

To instrument single-row queries, QueryRow/QueryRowContext return a *sequel.Row (which embeds *sql.Row) rather than *sql.Row. database/sql defers a QueryRow error to Scan, so the shadow captures it there. The common db.QueryRow(q).Scan(&x) call site is unchanged; only code that explicitly stores the result as *sql.Row needs adjustment.

Sequel is the copyrighted work of various contributors. It is licensed to you free of charge by Microbus LLC - a Delaware limited liability company formed to hold rights to the combined intellectual property of all contributors - under the Apache License 2.0.

Documentation

Index

Constants

This section is empty.

Variables

This section is empty.

Functions

func ApplyBindings

func ApplyBindings(args ...any) (err error)

ApplyBindings should be called after scanning values from the result set to perform all late binding.

func CreateTestingDatabase added in v1.6.1

func CreateTestingDatabase(driverName string, baseDataSourceName string, uniqueTestID string) (dsn string, err error)

CreateTestingDatabase provisions a uniquely-named database (or returns a SQLite in-memory DSN) for testing and returns the resolved data source name. Pass the result to Open or OpenSingleton to open a connection.

The returned DSN points at a database whose name has the testing_NN_ prefix. When the last *DB referencing that database is Closed, sequel drops it automatically — no separate cleanup call is required.

uniqueTestID scopes the database so that independent tests don't collide. Pass t.Name() from a test, or an equivalent identifier from production startup code that wants a per-run database:

dsn := cfg.DSN
if cfg.Testing {
    dsn, err = sequel.CreateTestingDatabase("", cfg.DSN, cfg.TestID)
    if err != nil { return err }
}
db, err := sequel.OpenSingleton("", dsn)

Within a single process, repeated calls with the same (driverName, baseDataSourceName, uniqueTestID) reuse the same testing database — the underlying DROP+CREATE only happens on the first call.

If a driver name is not provided, it is inferred from the data source name on a best-effort basis. Drivers currently supported: "mysql" (MySQL), "pgx" (Postgres), "cockroachdb" (CockroachDB), "mssql" (SQL Server) or "sqlite" (SQLite).

If neither a driver name nor a base data source name is provided, it falls back to the SEQUEL_TESTING_DSN environment variable. This lets any consumer that builds ephemeral test databases through sequel redirect its entire suite at a real server without changing test code: leave SEQUEL_TESTING_DSN unset to keep the SQLite default, or set it to a base DSN to run against that server instead, with the driver inferred from it. Naming a driver — even with an empty DSN — opts out of the fallback, so a test that explicitly asks for SQLite keeps running on SQLite regardless of the environment.

If neither the arguments nor SEQUEL_TESTING_DSN select a server, the following localhost defaults are used based on the driver name:

  • (empty): SQLite in-memory database
  • sqlite: SQLite in-memory database
  • mysql: root:root@tcp(127.0.0.1:3306)/
  • pgx: postgres://postgres:postgres@127.0.0.1:5432/
  • cockroachdb: postgres://root@127.0.0.1:26257/?sslmode=disable
  • mssql: sqlserver://sa:Password123@127.0.0.1:1433

func IsLockContentionError added in v1.5.7

func IsLockContentionError(err error) bool

IsLockContentionError returns true if the error indicates database lock contention or a deadlock. Such errors are transient and the operation can typically be retried. Recognizes lock errors from SQLite, MySQL, PostgreSQL, SQL Server, and CockroachDB.

Classification prefers the driver's native error code (immune to message wording, localization, and user data appearing in error messages); a substring match is used as a fallback for errors whose driver type is not present in the chain (e.g. some wrapped or text-only CockroachDB retry errors).

func Nullify

func Nullify[T comparable](value T) any

Nullify returns nil if the value equals to the zero value of its Go data type, else it returns the value. Use this construct to convert zero values to nil when writing to a nullable database column.

Example:

db.Exec(
	"INSERT INTO my_table (id, desc, modified_time) VALUES (?,?,?)",
	obj.ID,
	sequel.Nullify(obj.Description),
	sequel.Nullify(obj.ModifiedTime),
)

func RegisterVirtualFunc added in v1.2.0

func RegisterVirtualFunc(name string, handler func(driverName string, args string) (string, error))

RegisterVirtualFunc registers a virtual SQL function that will be replaced in queries before execution. The name is matched case-insensitively, e.g. registering "NOW_UTC" matches NOW_UTC(), now_utc(), Now_Utc(), etc. The handler receives the driver name and the string found between the parentheses, and returns the replacement SQL expression, or an error.

Types

type Binder

type Binder[T any] struct {
	sql.Null[T]
	// contains filtered or unexported fields
}

Binder is a thin wrapper over sql.Null that allows for late-binding of its value.

func Bind

func Bind[T any](binder func(value T) (err error)) *Binder[T]

Bind applies a binding function to the scanned value.

Example:

var obj Object
args := []any{
	&obj.ID,
	sequel.Bind(func(tags string) {
		return json.Unmarshal([]byte(tags), &obj.Tags)
	}),
	sequel.Bind(func(modifiedTime time.Time) {
		obj.Year, obj.Month, obj.Day = modifiedTime.Date()
		return nil
	}),
}
db.QueryRow("SELECT id, tags, modified_time FROM my_table WHERE id=?", id).Scan(args...)
sequel.ApplyBindings(args...)

func (*Binder[T]) Apply

func (n *Binder[T]) Apply() (err error)

Apply should be called after scanning the columns from the result set.

type DB

type DB struct {
	*sql.DB
	// contains filtered or unexported fields
}

DB is an enhanced database connection that

  • Limits the size of the connection pool to each server to approx the sqrt of the number of clients
  • Performs schema migration
  • Automatically creates and connects to a localhost database while testing

func Open

func Open(driverName string, dataSourceName string) (db *DB, err error)

Open returns a database connection to the named data source with a dedicated connection pool. Each call returns a distinct *DB; sequel does not coalesce by DSN. The caller is responsible for sizing the pool via SetMaxOpenConns / SetMaxIdleConns if the database/sql defaults (unlimited open, 2 idle) don't fit.

Use OpenSingleton when multiple consumers in the same process share a DSN and you want sequel to manage one pool across all of them.

If a driver name is not provided, it is inferred from the data source name on a best-effort basis. Drivers currently supported: "mysql" (MySQL), "pgx" (Postgres), "cockroachdb" (CockroachDB), "mssql" (SQL Server) or "sqlite" (SQLite).

Example data source name for each of the supported drivers:

  • mysql: username:password@tcp(hostname:3306)/
  • pgx: postgres://username:password@hostname:5432/
  • cockroachdb: postgres://username:password@hostname:26257/
  • mssql: sqlserver://username:password@hostname:1433
  • sqlite: file:path/to/database.sqlite

func OpenSingleton added in v1.6.1

func OpenSingleton(driverName string, dataSourceName string) (db *DB, err error)

OpenSingleton returns a per-DSN coalesced *DB whose connection pool sequel manages automatically based on the number of openers (sqrt-based growth, see [DB.adjustConnectionLimits]). Multiple OpenSingleton calls with the same (driverName, dataSourceName) return the same *DB and share its connection pool. This is the right choice when many parts of the same process each access the database occasionally.

Use Open when you want a dedicated pool with explicit caller-managed sizing.

Driver inference, DSN defaults, and supported drivers are the same as Open.

func (*DB) Begin added in v1.4.0

func (db *DB) Begin() (*Tx, error)

Begin starts a transaction and returns a sequel.Tx that applies virtual function expansion and placeholder conforming.

func (*DB) BeginTx added in v1.4.0

func (db *DB) BeginTx(ctx context.Context, opts *sql.TxOptions) (*Tx, error)

BeginTx starts a transaction with the given options and returns a sequel.Tx that applies virtual function expansion and placeholder conforming.

func (*DB) Close

func (db *DB) Close() (err error)

Close closes the database connection.

When the last reference closes and the underlying database name matches the testing pattern (testing_NN_…), sequel drops the database from the server as a best-effort cleanup. This makes CreateTestingDatabase-provisioned databases self-cleaning on test teardown.

func (*DB) ConformArgPlaceholders deprecated

func (db *DB) ConformArgPlaceholders(stmt string) string

Deprecated: ConformArgPlaceholders is applied automatically by the query shadow methods. Use ? placeholders directly in queries passed to Exec, Query, QueryRow, and Prepare.

func (*DB) DriverName

func (db *DB) DriverName() string

DriverName is the name of the driver: "mysql", "pgx", "cockroachdb", "mssql" or "sqlite".

func (*DB) Exec added in v1.2.0

func (db *DB) Exec(query string, args ...any) (sql.Result, error)

Exec shadows sql.DB.Exec and conforms arg placeholders for the driver.

func (*DB) ExecContext added in v1.2.0

func (db *DB) ExecContext(ctx context.Context, query string, args ...any) (sql.Result, error)

ExecContext shadows sql.DB.ExecContext and conforms arg placeholders for the driver.

func (*DB) InsertReturnID added in v1.3.0

func (db *DB) InsertReturnID(ctx context.Context, idColumn string, stmt string, args ...any) (int64, error)

InsertReturnID executes an INSERT statement and returns the auto-generated ID for the named ID column.

func (*DB) Migrate

func (db *DB) Migrate(sequenceName string, fileSys fs.FS) (err error)

Migrate reads all #.sql files from the FS, and executes any new migrations in order of their file name. The order of execution is guaranteed only within the context of a sequence name.

func (*DB) NowUTC deprecated

func (db *DB) NowUTC() string

Deprecated: Use the NOW_UTC() virtual function directly in queries instead.

func (*DB) Prepare added in v1.2.0

func (db *DB) Prepare(query string) (*sql.Stmt, error)

Prepare shadows sql.DB.Prepare and conforms arg placeholders for the driver.

func (*DB) PrepareContext added in v1.2.0

func (db *DB) PrepareContext(ctx context.Context, query string) (*sql.Stmt, error)

PrepareContext shadows sql.DB.PrepareContext and conforms arg placeholders for the driver.

func (*DB) Query added in v1.2.0

func (db *DB) Query(query string, args ...any) (*sql.Rows, error)

Query shadows sql.DB.Query and conforms arg placeholders for the driver.

func (*DB) QueryContext added in v1.2.0

func (db *DB) QueryContext(ctx context.Context, query string, args ...any) (*sql.Rows, error)

QueryContext shadows sql.DB.QueryContext and conforms arg placeholders for the driver.

func (*DB) QueryRow added in v1.2.0

func (db *DB) QueryRow(query string, args ...any) *Row

QueryRow shadows sql.DB.QueryRow and conforms arg placeholders for the driver. It returns a Row, which embeds *sql.Row so existing QueryRow(...).Scan(...) call sites are unchanged.

func (*DB) QueryRowContext added in v1.2.0

func (db *DB) QueryRowContext(ctx context.Context, query string, args ...any) *Row

QueryRowContext shadows sql.DB.QueryRowContext and conforms arg placeholders for the driver. It returns a Row, which embeds *sql.Row so existing QueryRowContext(...).Scan(...) call sites are unchanged.

func (*DB) RegexpTextSearch deprecated

func (db *DB) RegexpTextSearch(searchableColumns ...string) string

Deprecated: Use the REGEXP_TEXT_SEARCH() virtual function directly in queries instead.

func (*DB) SetLogger added in v1.9.0

func (db *DB) SetLogger(logger *slog.Logger)

SetLogger attaches an slog.Logger. The library does not log operation errors (they are returned to the caller, who logs them); it logs one-off events such as schema migrations at Info, and — when the logger is enabled at Debug level — each query at Debug. Per-query logging is therefore controlled by the logger's own level, not a separate switch. A freshly opened *DB uses a discard logger; pass nil here to revert to that discard logger (disabling logging).

func (*DB) SetMeterProvider added in v1.9.0

func (db *DB) SetMeterProvider(mp metric.MeterProvider)

SetMeterProvider attaches an OpenTelemetry MeterProvider so sequel emits sequel_ metrics (query and transaction duration, lock-contention count, migration count, and connection-pool gauges). A freshly opened *DB already uses the process-wide otel.GetMeterProvider(); call this to override it, or pass nil to revert to that global provider (whose default is a no-op). See DB.SetTracerProvider for when to call.

func (*DB) SetTracerProvider added in v1.9.0

func (db *DB) SetTracerProvider(tp trace.TracerProvider)

SetTracerProvider attaches an OpenTelemetry TracerProvider so sequel emits a client span around each query, transaction, and migration. A freshly opened *DB already uses the process-wide otel.GetTracerProvider(); call this to override it, or pass nil to revert to that global provider (whose default is a no-op).

Observability is configured after Open/OpenSingleton (which keep the standard database/sql signature) rather than at construction. This loses nothing: sql.Open does no I/O — it only prepares a lazy pool — so there is no work inside Open worth a span; every operation that does real work happens later on the returned *DB.

Configure before the *DB is used concurrently. For an OpenSingleton-shared *DB the providers are process- wide for that pool; the last setter wins, so configure once from the owning caller.

func (*DB) Transact added in v1.8.0

func (db *DB) Transact(ctx context.Context, fn func(tx *Tx) error) (err error)

Transact runs fn inside a transaction, committing on success and rolling back on error. If the transaction fails on lock contention or a deadlock, it is retried with a short jittered backoff. Because a retry re-executes fn from the start in a new transaction, fn must be safe to run more than once; any non-transactional side effects it performs (in-memory changes, channel sends) may repeat.

The Tx passed to fn records the first statement error and short-circuits the remaining statements, so fn cannot commit partial work even if it does not check every statement's error. For SQL Server, SET XACT_ABORT ON is applied so that any statement error aborts the whole transaction.

func (*DB) UnpackQuery added in v1.2.0

func (db *DB) UnpackQuery(query string) (string, error)

UnpackQuery expands virtual functions (e.g. NOW_UTC(), REGEXP_TEXT_SEARCH()) into driver-specific SQL expressions, and conforms arg placeholders to the syntax expected by the driver (e.g. ? to $1, $2 for PostgreSQL).

type Executor added in v1.4.1

type Executor interface {
	Exec(query string, args ...any) (sql.Result, error)
	ExecContext(ctx context.Context, query string, args ...any) (sql.Result, error)
	Query(query string, args ...any) (*sql.Rows, error)
	QueryContext(ctx context.Context, query string, args ...any) (*sql.Rows, error)
	QueryRow(query string, args ...any) *Row
	QueryRowContext(ctx context.Context, query string, args ...any) *Row
	Prepare(query string) (*sql.Stmt, error)
	PrepareContext(ctx context.Context, query string) (*sql.Stmt, error)
	InsertReturnID(ctx context.Context, idColumn string, stmt string, args ...any) (int64, error)
	DriverName() string
	UnpackQuery(query string) (string, error)
}

Executor is the interface satisfied by both DB and Tx.

type Null

type Null[T any] struct {
	*Binder[T]
}

Null is a thin wrapper over sql.Null that allows for reading NULL values.

func Nullable

func Nullable[T any](ptr *T) *Null[T]

Nullable is a simple binder that interprets NULL values to be the zero value of their Go data type.

Example:

var obj Object
args := []any{
	&obj.ID,
	sequel.Nullable(&obj.Description),
	sequel.Nullable(&obj.ModifiedTime),
}
db.QueryRow("SELECT id, desc, modified_time FROM my_table WHERE id=?", id).Scan(args...)
sequel.ApplyBindings(args...)

type Row added in v1.9.0

type Row struct {
	*sql.Row
	// contains filtered or unexported fields
}

Row shadows *sql.Row so sequel can observe a single-row query. database/sql does not surface a QueryRow error until Scan, so Row records the operation's duration, classifies lock contention, and ends the span when the caller calls Scan (or Err). It embeds *sql.Row, so the common QueryRow(...).Scan(...) call site is unchanged; only code that explicitly stores the result as *sql.Row needs adjustment.

As with *sql.Row, a Row whose Scan/Err is never called holds resources open — and here, leaves its span unended. Call Scan (or Err) exactly as you would with *sql.Row.

func (*Row) Err added in v1.9.0

func (r *Row) Err() error

Err shadows sql.Row.Err and finishes instrumentation with the query error, so a caller that inspects Err instead of scanning still closes the span.

func (*Row) Scan added in v1.9.0

func (r *Row) Scan(dest ...any) error

Scan shadows sql.Row.Scan and finishes instrumentation with the scan error.

type Tx added in v1.4.0

type Tx struct {
	*sql.Tx
	// contains filtered or unexported fields
}

Tx is an in-progress database transaction that shadows sql.Tx methods to apply virtual function expansion and placeholder conforming.

When created by DB.Transact, a Tx records the first statement error and short-circuits subsequent statements (returning that error without touching the database). This guarantees a transaction cannot commit partial state when a caller forgets to check a statement's error, and it surfaces a deadlock (rather than masking it as a later "COMMIT has no corresponding BEGIN" on some drivers) so Transact can retry. A Tx obtained from DB.BeginTx does not do this — its statement methods behave exactly like the underlying sql.Tx.

func (*Tx) DriverName added in v1.4.0

func (tx *Tx) DriverName() string

DriverName is the name of the driver: "mysql", "pgx", "cockroachdb", "mssql" or "sqlite".

func (*Tx) Err added in v1.8.0

func (tx *Tx) Err() error

Err returns the first statement error recorded in Transact mode, or nil. Always nil for a Tx obtained from BeginTx.

func (*Tx) Exec added in v1.4.0

func (tx *Tx) Exec(query string, args ...any) (sql.Result, error)

Exec shadows sql.Tx.Exec and conforms arg placeholders for the driver.

func (*Tx) ExecContext added in v1.4.0

func (tx *Tx) ExecContext(ctx context.Context, query string, args ...any) (sql.Result, error)

ExecContext shadows sql.Tx.ExecContext and conforms arg placeholders for the driver.

func (*Tx) InsertReturnID added in v1.4.0

func (tx *Tx) InsertReturnID(ctx context.Context, idColumn string, stmt string, args ...any) (int64, error)

InsertReturnID executes an INSERT statement and returns the auto-generated ID for the named ID column.

func (*Tx) Prepare added in v1.4.0

func (tx *Tx) Prepare(query string) (*sql.Stmt, error)

Prepare shadows sql.Tx.Prepare and conforms arg placeholders for the driver.

func (*Tx) PrepareContext added in v1.4.0

func (tx *Tx) PrepareContext(ctx context.Context, query string) (*sql.Stmt, error)

PrepareContext shadows sql.Tx.PrepareContext and conforms arg placeholders for the driver.

func (*Tx) Query added in v1.4.0

func (tx *Tx) Query(query string, args ...any) (*sql.Rows, error)

Query shadows sql.Tx.Query and conforms arg placeholders for the driver.

func (*Tx) QueryContext added in v1.4.0

func (tx *Tx) QueryContext(ctx context.Context, query string, args ...any) (*sql.Rows, error)

QueryContext shadows sql.Tx.QueryContext and conforms arg placeholders for the driver.

func (*Tx) QueryRow added in v1.4.0

func (tx *Tx) QueryRow(query string, args ...any) *Row

QueryRow shadows sql.Tx.QueryRow and conforms arg placeholders for the driver. It returns a Row, which embeds *sql.Row so existing QueryRow(...).Scan(...) call sites are unchanged.

func (*Tx) QueryRowContext added in v1.4.0

func (tx *Tx) QueryRowContext(ctx context.Context, query string, args ...any) *Row

QueryRowContext shadows sql.Tx.QueryRowContext and conforms arg placeholders for the driver. It returns a Row, which embeds *sql.Row so existing QueryRowContext(...).Scan(...) call sites are unchanged.

func (*Tx) UnpackQuery added in v1.4.0

func (tx *Tx) UnpackQuery(query string) (string, error)

UnpackQuery expands virtual functions (e.g. NOW_UTC(), REGEXP_TEXT_SEARCH()) into driver-specific SQL expressions, and conforms arg placeholders to the syntax expected by the driver (e.g. ? to $1, $2 for PostgreSQL).

type UnsafeSQL

type UnsafeSQL string

UnsafeSQL wraps a string to indicate not to use an argument placeholder when inserting it into a SQL statement. It should be used to insert values such as NOW() or calculation of other fields. Use with caution to avoid SQL injection.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL