transaction
Documentation¶
A general transaction support library for Python.
The transaction package offers a two-phase commit protocol which allows multiple backends of any kind to participate in a transaction and commit their changes only if all of them can successfully do so. It also offers support for savepoints, so that part of a transaction can be rolled back without having to abort it completely.
There are already transaction backends for SQLAlchemy, ZODB, email, filesystem, and others. in addition, there are packages like pyramid_tm, which allows all the code in a web request to run inside of a transaction, and aborts the transaction automatically if an error occurs. It’s also not difficult to create your own backends if necessary.
Additional Documentation
- Changes
- Transaction convenience support
- Dooming Transactions
- Savepoints
- Hooking the Transaction Machinery
- Writing a Data Manager
- Transaction integrations / Data Manager Implementations
- Using transactions with SQLAlchemy
- Managing more than one backend
transaction
API Referencetransaction
Developer Notes
Getting the transaction package¶
To install the transaction package you can use pip:
$ pip install transaction
After this, the package can be imported in your Python code, but there are a few things that we need to explain before doing that.
Using transactions¶
At its simplest, the developer will use an existing transaction backend, and will at most require to commit or abort a transaction now and then. For example:
1import transaction
2
3try:
4 # some code that uses one or more backends
5 .
6 .
7 .
8 transaction.commit()
9except SomeError:
10 transaction.abort()
Things you need to know about the transaction machinery¶
Transactions
A consists of one or more operations that we want to perform as a single action. It’s an all or nothing proposition: either all the operations that are part of the transaction are completed successfully or none of them have any effect.
In the transaction package, a transaction object
represents a running transaction that can be committed or aborted in
the end.
Transaction managers
Applications interact with a transaction using a transaction manager
, which is responsible for establishing the
transaction boundaries. Basically this means that it creates the
transactions and keeps track of the current one. Whenever an
application wants to use the transaction machinery, it gets the
current transaction from the transaction manager before starting any
operations
The default transaction manager, transaction.manager
, is thread
local. You use it as a global variable, but every thread has it’s own
copy. [1]
Application developers will most likely never need to create their own transaction managers.
Data Managers
A data manager
handles the
interaction between the transaction manager and the data storage
mechanism used by the application, which can be an object storage like
the ZODB, a relational database, a file or any other storage mechanism
that the application needs to control.
The data manager provides a common interface for the transaction
manager to use while a transaction is running. To be part of a
specific transaction, a data manager has to join
it. Any number of data
managers can join a transaction, which means that you could for
example perform writing operations on a ZODB storage and a relational
database as part of the same transaction. The transaction manager will
make sure that both data managers can commit the transaction or none
of them does.
An application developer will need to write a data manager for each different type of storage that the application uses. There are also third party data managers that can be used instead.
The two phase commit protocol
The transaction machinery uses a two phase commit protocol for coordinating all participating data managers in a transaction. The two phases work like follows:
The commit process is started.
Each associated data manager prepares the changes to be persistent.
Each data manager verifies that no errors or other exceptional conditions occurred during the attempt to persist the changes. If that happens, an exception should be raised. This is called ‘voting’. A data manager votes ‘no’ by raising an exception if something goes wrong; otherwise, its vote is counted as a ‘yes’.
If any of the associated data managers votes ‘no’, the transaction is aborted; otherwise, the changes are made permanent.
The two phase commit sequence requires that all the storages being used are capable of rolling back or aborting changes.
Savepoints
A savepoint allows supported data managers
to save work to their
storage without committing the full transaction. In other words, the
transaction will go on, but if a rollback is needed we can get back to
this point instead of starting all over.
Savepoints are also useful to free memory that would otherwise be used to keep the whole state of the transaction. This can be very important when a transaction attempts a large number of changes.