Development status Active Operating system | ||
Stable release 5.1.1 / November 18, 2016; 4 months ago (2016-11-18) Repository github.com/zopefoundation/ZODB |
The Zope Object Database (ZODB) is an object-oriented database for transparently and persistently storing Python objects. It is included as part of the Zope web application server, but can also be used independently of Zope.
Contents
- History
- Basics
- Example
- Storage unit
- Atomicity
- Class persistence
- ZEO
- Pluggable storages
- Failover technologies
- References
Features of the ZODB include: transactions, history/undo, transparently pluggable storage, built-in caching, multiversion concurrency control (MVCC), and scalability across a network (using ZEO).
History
Basics
A ZODB storage is basically a directed graph of (Python) objects pointing at each other, with a Python dictionary at the root. Objects are accessed by starting at the root, and following pointers until the target object. In this respect, ZODB can be seen as a sophisticated Python persistence layer.
Example
For example, say we have a car described using 3 classes Car
, Wheel
and Screw
. In Python, this could be represented the following way:
If the variable zodb
is the root of persistence, then:
This puts all of the object instances (car, wheel, screws etc.) into storage, which can be retrieved later. If another program gets a connection to the database through the zodb
object, performing:
And retrieves all the objects, the pointer to the car being held in the carzz
variable. Then at some later stage, this object can be altered with a Python code like:
The storage is altered to reflect the change of data (after a commit is ordered).
There is no declaration of the data structure in either Python or ZODB, so new fields can be freely added to any existing object.
Storage unit
For persistence to take place, the Python Car class must be derived from the persistence.Persistent class — this class both holds the data necessary for the persistence machinery to work, such as the internal object id, state of the object, and so on, but also defines the boundary of the persistence in the following sense: every object whose class derives from Persistent is the atomic unit of storage (the whole object is copied to the storage when a field is modified).
In the example above, if Car
is the only class deriving from Persistent, when wheel3
is added to car, all of the objects must be written to the storage. In contrast, if Wheel
also derives from Persistent, then when carzz.wheel3 = Wheel
is performed, a new record is written to the storage to hold the new value of the Car
, but the existing Wheel
are kept, and the new record for the Car
points to the already existing Wheel
record inside the storage.
The ZODB machinery doesn't chase modification down through the graph of pointers. In the example above, carzz.wheel3 = something
is a modification automatically tracked down by the ZODB machinery, because carzz
is of (Persistent) class Car
. The ZODB machinery does this by marking the record as dirty. However, if there is a list, any change inside the list isn't noticed by the ZODB machinery, and the programmer must help by manually adding carzz._p_changed = 1
, notifying ZODB that the record actually changed. Thus, to a certain extent the programmer must be aware of the working of the persistence machinery.
Atomicity
The storage unit (that is, an object whose class derives from Persistent) is also the atomicity unit. In the example above, if Cars
is the only Persistent class, a thread modifies a Wheel (the Car
record must be notified), and another thread modifies another Wheel
inside another transaction, the second commit will fail. If Wheel
is also Persistent, both Wheels
can be modified independently by two different threads in two different transactions.
Class persistence
The class persistence—writing the class of a particular object into the storage—is obtained by writing a kind of "fully qualified" name of the class into each record on the disk. It should be noted that, in Python, the name of the class involves the hierarchy of directory the source file of the class resides in. A consequence is that the source file of persisting object cannot be moved. If it is, the ZODB machinery is unable to locate the class of an object when retrieving it from the storage, resulting into a broken object.
ZEO
Zope Enterprise Objects (ZEO) is a ZODB storage implementation that allows multiple client processes to persist objects to a single ZEO server. This allows transparent scaling.