Connections to a database

A Connection encapsulates access to a database. You can have multiple Connections open against the same database file in the same process, across threads and in other processes.

Connection class

class apsw.Connection(filename: str, flags: int = SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE, vfs: str | None = None, statementcachesize: int = 100)

This object wraps a sqlite3 pointer.

Opens the named database. You can use :memory: to get a private temporary in-memory database that is not shared with any other connections.

Parameters:
  • flags – One or more of the open flags orred together

  • vfs – The name of the vfs to use. If None then the default vfs will be used.

  • statementcachesize – Use zero to disable the statement cache, or a number larger than the total distinct SQL statements you execute frequently.

Calls: sqlite3_open_v2

Connection.__enter__() Connection

You can use the database as a context manager as defined in PEP 0343. When you use with a transaction is started. If the block finishes with an exception then the transaction is rolled back, otherwise it is committed. For example:

with connection:
    connection.execute("....")
    with connection:
        # nested is supported
        call_function(connection)
        connection.execute("...")
        with connection as db:
            # You can also use 'as'
            call_function2(db)
            db.execute("...")
Behind the scenes savepoints

are used to provide nested transactions.

Connection.__exit__(etype: type[BaseException] | None, evalue: BaseException | None, etraceback: types.TracebackType | None) bool | None

Implements context manager in conjunction with __enter__(). If no exception happened then the pending transaction is committed, while an exception results in a rollback.

Connection.authorizer: Authorizer | None

While preparing statements, SQLite will call any defined authorizer to see if a particular action is ok to be part of the statement.

Typical usage would be if you are running user supplied SQL and want to prevent harmful operations. You should also set the statementcachesize to zero.

The authorizer callback has 5 parameters:

  • An operation code

  • A string (or None) dependent on the operation (listed as 3rd)

  • A string (or None) dependent on the operation (listed as 4th)

  • A string name of the database (or None)

  • Name of the innermost trigger or view doing the access (or None)

The authorizer callback should return one of SQLITE_OK, SQLITE_DENY or SQLITE_IGNORE. (SQLITE_DENY is returned if there is an error in your Python code).

Calls: sqlite3_set_authorizer

Connection.autovacuum_pages(callable: Callable[[str, int, int, int], int] | None) None

Calls callable to find out how many pages to autovacuum. The callback has 4 parameters:

  • Database name: str. main, temp, the name in ATTACH

  • Database pages: int (how many pages make up the database now)

  • Free pages: int (how many pages could be freed)

  • Page size: int (page size in bytes)

Return how many pages should be freed. Values less than zero or more than the free pages are treated as zero or free page count. On error zero is returned.

Warning

READ THE NOTE IN THE SQLITE DOCUMENTATION.

Calling back into SQLite can result in crashes, corrupt databases, or worse.

Calls: sqlite3_autovacuum_pages

Connection.backup(databasename: str, sourceconnection: Connection, sourcedatabasename: str) Backup

Opens a backup object. All data will be copied from source database to this database.

Parameters:
  • databasename – Name of the database. main, temp, the name in ATTACH

  • sourceconnection – The Connection to copy a database from.

  • sourcedatabasename – Name of the database in the source (eg main).

Return type:

Backup

Calls: sqlite3_backup_init

Connection.blob_open(database: str, table: str, column: str, rowid: int, writeable: bool) Blob

Opens a blob for incremental I/O.

Parameters:
  • database – Name of the database. main, temp, the name in ATTACH.

  • table – The name of the table

  • column – The name of the column

  • rowid – The id that uniquely identifies the row.

  • writeable – If True then you can read and write the blob. If False then you can only read it.

Return type:

Blob

Calls: sqlite3_blob_open

Connection.cache_flush() None

Flushes caches to disk mid-transaction.

Calls: sqlite3_db_cacheflush

Connection.cache_stats(include_entries: bool = False) dict[str, int]

Returns information about the statement cache as dict.

Note

Calling execute with “select a; select b; insert into c …” will result in 3 cache entries corresponding to each of the 3 queries present.

The returned dictionary has the following information.

Key

Explanation

size

Maximum number of entries in the cache

evictions

How many entries were removed (expired) to make space for a newer entry

no_cache

Queries that had can_cache parameter set to False

hits

A match was found in the cache

misses

No match was found in the cache, or the cache couldn’t be used

no_vdbe

The statement was empty (eg a comment) or SQLite took action during parsing (eg some pragmas). These are not cached and also included in the misses count

too_big

UTF8 query size was larger than considered for caching. These are also included in the misses count.

max_cacheable_bytes

Maximum size of query (in bytes of utf8) that will be considered for caching

entries

(Only present if include_entries is True) A list of the cache entries

If entries is present, then each list entry is a dict with the following information.

Key

Explanation

query

Text of the query itself (first statement only)

prepare_flags

Flags passed to sqlite3_prepare_v3 for this query

explain

The value passed to sqlite3_stmt_explain if >= 0

uses

How many times this entry has been (re)used

has_more

Boolean indicating if there was more query text than the first statement

Connection.changes() int

Returns the number of database rows that were changed (or inserted or deleted) by the most recently completed INSERT, UPDATE, or DELETE statement.

Calls: sqlite3_changes64

Connection.close(force: bool = False) None

Closes the database. If there are any outstanding cursors, blobs or backups then they are closed too. It is normally not necessary to call this method as the database is automatically closed when there are no more references. It is ok to call the method multiple times.

If your user defined functions or collations have direct or indirect references to the Connection then it won’t be automatically garbage collected because of circular referencing that can’t be automatically broken. Calling close will free all those objects and what they reference.

SQLite is designed to survive power failures at even the most awkward moments. Consequently it doesn’t matter if it is closed when the process is exited, or even if the exit is graceful or abrupt. In the worst case of having a transaction in progress, that transaction will be rolled back by the next program to open the database, reverting the database to a know good state.

If force is True then any exceptions are ignored.

Calls: sqlite3_close

Connection.collation_needed(callable: Callable[[Connection, str], None] | None) None

callable will be called if a statement requires a collation that hasn’t been registered. Your callable will be passed two parameters. The first is the connection object. The second is the name of the collation. If you have the collation code available then call Connection.create_collation().

This is useful for creating collations on demand. For example you may include the locale in the collation name, but since there are thousands of locales in popular use it would not be useful to prereigster them all. Using collation_needed() tells you when you need to register them.

Calls: sqlite3_collation_needed

Connection.column_metadata(dbname: str | None, table_name: str, column_name: str) tuple[str, str, bool, bool, bool]

dbname is main, temp, the name in ATTACH, or None to search all databases.

The returned tuple has these fields:

0: str - declared data type

1: str - name of default collation sequence

2: bool - True if not null constraint

3: bool - True if part of primary key

4: bool - True if column is autoincrement

Calls: sqlite3_table_column_metadata

Connection.config(op: int, *args: int) int
Parameters:

This is how to get the fkey setting:

val = db.config(apsw.SQLITE_DBCONFIG_ENABLE_FKEY, -1)

A parameter of zero would turn it off, 1 turns on, and negative leaves unaltered. The effective value is always returned.

Calls: sqlite3_db_config

Connection.create_aggregate_function(name: str, factory: AggregateFactory | None, numargs: int = -1, *, flags: int = 0) None

Registers an aggregate function. Aggregate functions operate on all the relevant rows such as counting how many there are.

Parameters:
  • name – The string name of the function. It should be less than 255 characters

  • factory – The function that will be called. Use None to delete the function.

  • numargs – How many arguments the function takes, with -1 meaning any number

  • flagsFunction flags

When a query starts, the factory will be called. It can return an object with a step function called for each matching row, and a final function to provide the final value.

Alternatively a non-class approach can return a tuple of 3 items:

a context object

This can be of any type

a step function

This function is called once for each row. The first parameter will be the context object and the remaining parameters will be from the SQL statement. Any value returned will be ignored.

a final function

This function is called at the very end with the context object as a parameter. The value returned is set as the return for the function. The final function is always called even if an exception was raised by the step function. This allows you to ensure any resources are cleaned up.

Note

You can register the same named function but with different callables and numargs. See create_scalar_function() for an example.

Calls: sqlite3_create_function_v2

Connection.create_collation(name: str, callback: Callable[[str, str], int] | None) None

You can control how SQLite sorts (termed collation) when giving the COLLATE term to a SELECT. For example your collation could take into account locale or do numeric sorting.

The callback will be called with two items. It should return -1 if the first is less then the second, 0 if they are equal, and 1 if first is greater:

def mycollation(first: str, two: str) -> int:
    if first < second:
        return -1
    if first == second:
        return 0
    if first > second:
        return 1

Passing None as the callback will unregister the collation.

Calls: sqlite3_create_collation_v2

Connection.create_module(name: str, datasource: VTModule | None, *, use_bestindex_object: bool = False, use_no_change: bool = False, iVersion: int = 1, eponymous: bool = False, eponymous_only: bool = False, read_only: bool = False) None

Registers a virtual table, or drops it if datasource is None. See Virtual Tables for details.

Parameters:

See also

Calls: sqlite3_create_module_v2

Connection.create_scalar_function(name: str, callable: ScalarProtocol | None, numargs: int = -1, *, deterministic: bool = False, flags: int = 0) None

Registers a scalar function. Scalar functions operate on one set of parameters once.

Parameters:
  • name – The string name of the function. It should be less than 255 characters

  • callable – The function that will be called. Use None to unregister.

  • numargs – How many arguments the function takes, with -1 meaning any number

  • deterministic – When True this means the function always returns the same result for the same input arguments. SQLite’s query planner can perform additional optimisations for deterministic functions. For example a random() function is not deterministic while one that returns the length of a string is.

  • flags – Additional function flags

Note

You can register the same named function but with different callable and numargs. For example:

connection.create_scalar_function("toip", ipv4convert, 4)
connection.create_scalar_function("toip", ipv6convert, 16)
connection.create_scalar_function("toip", strconvert, -1)

The one with the correct numargs will be called and only if that doesn’t exist then the one with negative numargs will be called.

Calls: sqlite3_create_function_v2

Connection.create_window_function(name: str, factory: WindowFactory | None, numargs: int = -1, *, flags: int = 0) None

Registers a window function

Parameters:
  • name – The string name of the function. It should be less than 255 characters

  • factory – Called to start a new window. Use None to delete the function.

  • numargs – How many arguments the function takes, with -1 meaning any number

  • flagsFunction flags

You need to provide callbacks for the step, final, value and inverse methods. This can be done by having factory as a class, returning an instance with the corresponding method names, or by having factory return a sequence of a first parameter, and then each of the 4 functions.

Debugging note SQlite always calls the final method to allow for cleanup. If you have an exception in one of the other methods, then final will also be called, and you may see both methods in tracebacks.

Calls: sqlite3_create_window_function

Connection.cursor() Cursor

Creates a new Cursor object on this database.

Return type:

Cursor

Connection.cursor_factory: Callable[[Connection], Any]

Defaults to Cursor

Called with a Connection as the only parameter when a cursor is needed such as by the cursor() method, or Connection.execute().

Note that whatever is returned doesn’t have to be an actual Cursor instance, and just needs to have the methods present that are actually called. These are likely to be execute, executemany, close etc.

Connection.data_version(schema: str | None = None) int

Unlike pragma data_version this value updates when changes are made by other connections, AND this one.

Parameters:

schemaschema is main, temp, the name in ATTACH, defaulting to main if not supplied.

Calls: sqlite3_file_control

Connection.db_filename(name: str) str

Returns the full filename of the named (attached) database. The main is main, temp, the name in ATTACH

Calls: sqlite3_db_filename

Connection.db_names() list[str]

Returns the list of database names. For example the first database is named ‘main’, the next ‘temp’, and the rest with the name provided in ATTACH

Calls: sqlite3_db_name

Connection.deserialize(name: str, contents: bytes) None

Replaces the named database with an in-memory copy of contents. name is main, temp, the name in ATTACH

The resulting database is in-memory, read-write, and the memory is owned, resized, and freed by SQLite.

Calls: sqlite3_deserialize

Connection.drop_modules(keep: Iterable[str] | None) None

If keep is None then all registered virtual tables are dropped.

Otherwise keep is a sequence of strings, naming the virtual tables that are kept, dropping all others.

Connection.enable_load_extension(enable: bool) None

Enables/disables extension loading which is disabled by default.

Parameters:

enable – If True then extension loading is enabled, else it is disabled.

See also

Calls: sqlite3_enable_load_extension

Connection.exec_trace: ExecTracer | None

Called with the cursor, statement and bindings for each execute() or executemany() on this Connection, unless the Cursor installed its own tracer. Your execution tracer can also abort execution of a statement.

If callable is None then any existing execution tracer is removed.

Connection.execute(statements: str, bindings: Bindings | None = None, *, can_cache: bool = True, prepare_flags: int = 0, explain: int = -1) Cursor

Executes the statements using the supplied bindings. Execution returns when the first row is available or all statements have completed. (A cursor is automatically obtained).

For pragmas you should use pragma() which handles quoting and caching correctly.

See Cursor.execute() for more details, and the example.

Connection.executemany(statements: str, sequenceofbindings: Iterable[Bindings], *, can_cache: bool = True, prepare_flags: int = 0, explain: int = -1) Cursor

This method is for when you want to execute the same statements over a sequence of bindings, such as inserting into a database. (A cursor is automatically obtained).

See Cursor.executemany() for more details, and the example.

Connection.file_control(dbname: str, op: int, pointer: int) bool

Calls the xFileControl() method on the Virtual File System (VFS) implementing file access for the database.

Parameters:
  • dbname – The name of the database to affect. main, temp, the name in ATTACH

  • op – A numeric code with values less than 100 reserved for SQLite internal use.

  • pointer – A number which is treated as a void pointer at the C level.

Returns:

True or False indicating if the VFS understood the op.

The example shows getting SQLITE_FCNTL_DATA_VERSION.

If you want data returned back then the pointer needs to point to something mutable. Here is an example using ctypes of passing a Python dictionary to xFileControl() which can then modify the dictionary to set return values:

obj={"foo": 1, 2: 3}                 # object we want to pass
objwrap=ctypes.py_object(obj)        # objwrap must live before and after the call else
                                     # it gets garbage collected
connection.file_control(
         "main",                     # which db
         123,                        # our op code
         ctypes.addressof(objwrap))  # get pointer

The xFileControl() method then looks like this:

def xFileControl(self, op, pointer):
    if op==123:                      # our op code
        obj=ctypes.py_object.from_address(pointer).value
        # play with obj - you can use id() to verify it is the same
        print(obj["foo"])
        obj["result"]="it worked"
        return True
    else:
        # pass to parent/superclass
        return super().xFileControl(op, pointer)

This is how you set the chunk size by which the database grows. Do not combine it into one line as the c_int would be garbage collected before the file control call is made:

chunksize=ctypes.c_int(32768)
connection.file_control("main", apsw.SQLITE_FCNTL_CHUNK_SIZE, ctypes.addressof(chunksize))

Calls: sqlite3_file_control

Connection.filename: str

The filename of the database.

Calls: sqlite3_db_filename

Connection.filename_journal: str

The journal filename of the database,

Calls: sqlite3_filename_journal

Connection.filename_wal: str

The WAL filename of the database,

Calls: sqlite3_filename_wal

Connection.fts5_tokenizer(name: str, args: list[str] | None = None) apsw.FTS5Tokenizer

Returns the named tokenizer initialized with args. Names are case insensitive.

Connection.fts5_tokenizer_available(name: str) bool

Checks if the named tokenizer is registered.

Connection.get_autocommit() bool

Returns if the Connection is in auto commit mode (ie not in a transaction).

Calls: sqlite3_get_autocommit

Connection.get_exec_trace() ExecTracer | None

Returns the currently installed execution tracer

Connection.get_row_trace() RowTracer | None

Returns the currently installed row tracer

Connection.in_transaction: bool

True if currently in a transaction, else False

Calls: sqlite3_get_autocommit

Connection.interrupt() None

Causes all pending operations on the database to abort at the earliest opportunity. You can call this from any thread. For example you may have a long running query when the user presses the stop button in your user interface. InterruptError will be raised in the queries that got interrupted.

Calls: sqlite3_interrupt

Connection.is_interrupted: bool

Indicates if this connection has been interrupted.

Calls: sqlite3_is_interrupted

Connection.last_insert_rowid() int

Returns the integer key of the most recent insert in the database.

Calls: sqlite3_last_insert_rowid

Connection.limit(id: int, newval: int = -1) int

If called with one parameter then the current limit for that id is returned. If called with two then the limit is set to newval.

Parameters:
  • id – One of the runtime limit ids

  • newval – The new limit. This is a 32 bit signed integer even on 64 bit platforms.

Returns:

The limit in place on entry to the call.

See also

Calls: sqlite3_limit

Connection.load_extension(filename: str, entrypoint: str | None = None) None

Loads filename as an extension

Parameters:
  • filename – The file to load.

  • entrypoint – The initialization method to call. If this parameter is not supplied then the SQLite default of sqlite3_extension_init is used.

Raises:

ExtensionLoadingError – If the extension could not be loaded. The exception string includes more details.

Calls: sqlite3_load_extension

Connection.open_flags: int

The combination of flags used to open the database.

Connection.open_vfs: str

The string name of the vfs used to open the database.

Connection.overload_function(name: str, nargs: int) None

Registers a placeholder function so that a virtual table can provide an implementation via VTTable.FindFunction().

Parameters:
  • name – Function name

  • nargs – How many arguments the function takes

Calls: sqlite3_overload_function

Connection.pragma(name: str, value: SQLiteValue | None = None, *, schema: str | None = None) Any

Issues the pragma (with the value if supplied) and returns the result with the least amount of structure. For example pragma("user_version") will return just the number, while pragma("journal_mode", "WAL") will return the journal mode now in effect.

Pragmas do not support bindings, so this method is a convenient alternative to composing SQL text. Pragmas are often executed while being prepared, instead of when run like regular SQL. They may also contain encryption keys. This method ensures they are not cached to avoid problems.

Use the schema parameter to run the pragma against a different attached database (eg temp).

Connection.read(schema: str, which: int, offset: int, amount: int) tuple[bool, bytes]

Invokes the underlying VFS method to read data from the database. It is strongly recommended to read aligned complete pages, since that is only what SQLite does.

schema is main, temp, the name in ATTACH

which is 0 for the database file, 1 for the journal.

The return value is a tuple of a boolean indicating a complete read if True, and the bytes read which will always be the amount requested in size.

SQLITE_IOERR_SHORT_READ will give a False value for the boolean, and there is no way of knowing how much was read.

Implemented using SQLITE_FCNTL_FILE_POINTER and SQLITE_FCNTL_JOURNAL_POINTER. Errors will usually be generic SQLITE_ERROR with no message.

Calls: sqlite3_file_control

Connection.readonly(name: str) bool

True or False if the named (attached) database was opened readonly or file permissions don’t allow writing. The name is main, temp, the name in ATTACH

An exception is raised if the database doesn’t exist.

Calls: sqlite3_db_readonly

Connection.register_fts5_function(name: str, function: FTS5Function) None

Registers the (case insensitive) named function used as an auxiliary function.

The first parameter to the function will be FTS5ExtensionApi and the rest will be the function arguments at the SQL level.

Connection.register_fts5_tokenizer(name: str, tokenizer_factory: FTS5TokenizerFactory) None

Registers a tokenizer factory. Names are case insensitive. It is not possible to unregister a tokenizer.

Connection.release_memory() None

Attempts to free as much heap memory as possible used by this connection.

Calls: sqlite3_db_release_memory

Connection.row_trace: RowTracer | None

Called with the cursor and row being returned for cursors associated with this Connection, unless the Cursor installed its own tracer. You can change the data that is returned or cause the row to be skipped altogether.

If callable is None then any existing row tracer is removed.

Connection.serialize(name: str) bytes

Returns a memory copy of the database. name is main, temp, the name in ATTACH

The memory copy is the same as if the database was backed up to disk.

If the database name doesn’t exist, then None is returned, not an exception (this is SQLite’s behaviour). One exception is than an empty temp will result in a None return.

Calls: sqlite3_serialize

Connection.set_authorizer(callable: Authorizer | None) None

Sets the authorizer

Connection.set_busy_handler(callable: Callable[[int], bool] | None) None

Sets the busy handler to callable. callable will be called with one integer argument which is the number of prior calls to the busy callback for the same lock. If the busy callback returns False, then SQLite returns SQLITE_BUSY to the calling code. If the callback returns True, then SQLite tries to open the table again and the cycle repeats.

If you previously called set_busy_timeout() then calling this overrides that.

Passing None unregisters the existing handler.

Calls: sqlite3_busy_handler

Connection.set_busy_timeout(milliseconds: int) None

If the database is locked such as when another connection is making changes, SQLite will keep retrying. This sets the maximum amount of time SQLite will keep retrying before giving up. If the database is still busy then apsw.BusyError will be returned.

Parameters:

milliseconds – Maximum thousandths of a second to wait.

If you previously called set_busy_handler() then calling this overrides that.

Calls: sqlite3_busy_timeout

Connection.set_commit_hook(callable: CommitHook | None) None

callable will be called just before a commit. It should return False for the commit to go ahead and True for it to be turned into a rollback. In the case of an exception in your callable, a True (rollback) value is returned. Pass None to unregister the existing hook.

See also

Calls: sqlite3_commit_hook

Connection.set_exec_trace(callable: ExecTracer | None) None

Method to set Connection.exec_trace

Connection.set_last_insert_rowid(rowid: int) None

Sets the value calls to last_insert_rowid() will return.

Calls: sqlite3_set_last_insert_rowid

Connection.set_profile(callable: Callable[[str, int], None] | None) None

Sets a callable which is invoked at the end of execution of each statement and passed the statement string and how long it took to execute. (The execution time is in nanoseconds.) Note that it is called only on completion. If for example you do a SELECT and only read the first result, then you won’t reach the end of the statement.

Calls: sqlite3_trace_v2

Connection.set_progress_handler(callable: Callable[[], bool] | None, nsteps: int = 20) None

Sets a callable which is invoked every nsteps SQLite inststructions. The callable should return True to abort or False to continue. (If there is an error in your Python callable then True/abort will be returned).

See also

Calls: sqlite3_progress_handler

Connection.set_rollback_hook(callable: Callable[[], None] | None) None

Sets a callable which is invoked during a rollback. If callable is None then any existing rollback hook is unregistered.

The callable is called with no parameters and the return value is ignored.

Calls: sqlite3_rollback_hook

Connection.set_row_trace(callable: RowTracer | None) None

Method to set Connection.row_trace

Connection.set_update_hook(callable: Callable[[int, str, str, int], None] | None) None

Calls callable whenever a row is updated, deleted or inserted. If callable is None then any existing update hook is unregistered. The update hook cannot make changes to the database while the query is still executing, but can record them for later use or apply them in a different connection.

The update hook is called with 4 parameters:

type (int)

SQLITE_INSERT, SQLITE_DELETE or SQLITE_UPDATE

database name (str)

main, temp, the name in ATTACH

table name (str)

The table on which the update happened

rowid (int)

The affected row

See also

Calls: sqlite3_update_hook

Connection.set_wal_hook(callable: Callable[[Connection, str, int], int] | None) None

callable will be called just after data is committed in Write Ahead Logging mode. It should return SQLITE_OK or an error code. The callback is called with 3 parameters:

  • The Connection

  • The database name. main, temp, the name in ATTACH

  • The number of pages in the wal log

You can pass in None in order to unregister an existing hook.

Calls: sqlite3_wal_hook

Connection.sqlite3_pointer() int

Returns the underlying sqlite3 * for the connection. This method is useful if there are other C level libraries in the same process and you want them to use the APSW connection handle. The value is returned as a number using PyLong_FromVoidPtr under the hood. You should also ensure that you increment the reference count on the Connection for as long as the other libraries are using the pointer. It is also a very good idea to call sqlite_lib_version() and ensure it is the same as the other libraries.

Connection.status(op: int, reset: bool = False) tuple[int, int]

Returns current and highwater measurements for the database.

Parameters:
  • op – A status parameter

  • reset – If True then the highwater is set to the current value

Returns:

A tuple of current value and highwater value

See also

Calls: sqlite3_db_status

Connection.system_errno: int

The underlying system error code for the most recent I/O error.

Calls: sqlite3_system_errno

Connection.table_exists(dbname: str | None, table_name: str) bool

Returns True if the named table exists, else False.

dbname is main, temp, the name in ATTACH, or None to search all databases

Calls: sqlite3_table_column_metadata

Connection.total_changes() int

Returns the total number of database rows that have be modified, inserted, or deleted since the database connection was opened.

Calls: sqlite3_total_changes64

Connection.trace_v2(mask: int, callback: Callable[[dict], None] | None = None, *, id: Any | None = None) None

Registers a trace callback. Multiple traces can be active at once (implemented by APSW). A callback of None unregisters a trace. Registered callbacks are distinguished by their id - an equality test is done to match ids.

The callback is called with a dict of relevant values based on the code.

Key

Type

Explanation

code

int

One of the trace event codes

connection

Connection

Connection this trace event belongs to

sql

str

SQL text (except SQLITE_TRACE_ROW and SQLITE_TRACE_CLOSE).

id

int

An opaque key to correlate events on the same statement. The id can be reused after SQLITE_TRACE_PROFILE.

trigger

bool

If trigger SQL is executing then this is True and the SQL is of the trigger. Virtual table nested queries also come through as trigger activity.

total_changes

int

Value of total_changes() (SQLITE_TRACE_STMT and SQLITE_TRACE_PROFILE only)

nanoseconds

int

nanoseconds SQL took to execute (SQLITE_TRACE_PROFILE only)

stmt_status

dict

SQLITE_TRACE_PROFILE only: Keys are names from status parameters - eg “SQLITE_STMTSTATUS_VM_STEP” and corresponding integer values. The counters are reset each time a statement starts execution. This includes any changes made by triggers.

Note that SQLite ignores any errors from the trace callbacks, so whatever was being traced will still proceed. Exceptions will be delivered when your Python code resumes.

If you register for all trace types, the following sequence will happen.

  • SQLITE_TRACE_STMT with trigger False and an id and sql of the statement.

  • Multiple times: SQLITE_TRACE_STMT with the same id and trigger True if a trigger is executed. The first time the sql will be TRIGGER name and then subsequent calls will be lines of the trigger. This also happens for virtual tables that make queries.

  • Multiple times: SQLITE_TRACE_ROW with the same id for each time execution stopped at a row. (Rows visited by triggers do not cause thie event)

  • SQLITE_TRACE_PROFILE with the same id for any virtual table queries - the sql will be of those queries

  • SQLITE_TRACE_PROFILE with the same id for the initial SQL.

Calls:
Connection.txn_state(schema: str | None = None) int

Returns the current transaction state of the database, or a specific schema if provided. apsw.mapping_txn_state contains the values returned.

Calls: sqlite3_txn_state

Connection.vfsname(dbname: str) str | None

Issues the SQLITE_FCNTL_VFSNAME file control against the named database (main, temp, attached name).

This is useful to see which VFS is in use, and if inheritance is used then / will separate the names. If you have a VFSFile in use then its fully qualified class name will also be included.

If SQLITE_FCNTL_VFSNAME is not implemented, dbname is not a database name, or an error occurred then None is returned.

Connection.vtab_config(op: int, val: int = 0) None

Callable during virtual table Connect()/Create().

Calls: sqlite3_vtab_config

Connection.vtab_on_conflict() int

Callable during virtual table insert or update

Calls: sqlite3_vtab_on_conflict

Connection.wal_autocheckpoint(n: int) None

Sets how often the Write Ahead Logging checkpointing is run.

Parameters:

n – A number representing the checkpointing interval or zero/negative to disable auto checkpointing.

Calls: sqlite3_wal_autocheckpoint

Connection.wal_checkpoint(dbname: str | None = None, mode: int = apsw.SQLITE_CHECKPOINT_PASSIVE) tuple[int, int]

Does a WAL checkpoint. Has no effect if the database(s) are not in WAL mode.

Parameters:
  • dbname – The name of the database or all databases if None

  • mode – One of the checkpoint modes.

Returns:

A tuple of the size of the WAL log in frames and the number of frames checkpointed as described in the documentation.

Calls: sqlite3_wal_checkpoint_v2