Commit 1330a51f authored by Jeremy Hylton's avatar Jeremy Hylton

A bunch of changes to bring into line with ZODB 3.3 naming.

parent c2eebec5
...@@ -33,8 +33,9 @@ this. ...@@ -33,8 +33,9 @@ this.
The downside is that the programmer has to explicitly manage objects, The downside is that the programmer has to explicitly manage objects,
reading an object when it's needed and writing it out to disk when the reading an object when it's needed and writing it out to disk when the
object is no longer required. The ZODB manages objects for you, object is no longer required. The ZODB manages objects for you,
keeping them in a cache and writing them out if they haven't been keeping them in a cache, writing them out to disk when they are
accessed in a while. modified, and dropping them from the cache if they haven't been used
in a while.
\subsection{OODBs vs. Relational DBs} \subsection{OODBs vs. Relational DBs}
...@@ -155,7 +156,7 @@ The ZODB comes with a few different classes that implement the ...@@ -155,7 +156,7 @@ The ZODB comes with a few different classes that implement the
\class{Storage} interface. Such classes handle the job of \class{Storage} interface. Such classes handle the job of
writing out Python objects to a physical storage medium, which can be writing out Python objects to a physical storage medium, which can be
a disk file (the \class{FileStorage} class), a BerkeleyDB file a disk file (the \class{FileStorage} class), a BerkeleyDB file
(\class{BerkeleyStorage}), a relational database (\class{BDBFullStorage}), a relational database
(\class{DCOracleStorage}), or some other medium. ZEO adds (\class{DCOracleStorage}), or some other medium. ZEO adds
\class{ClientStorage}, a new \class{Storage} that doesn't write to \class{ClientStorage}, a new \class{Storage} that doesn't write to
physical media but just forwards all requests across a network to a physical media but just forwards all requests across a network to a
......
...@@ -3,13 +3,6 @@ ...@@ -3,13 +3,6 @@
\section{Resources} \section{Resources}
ZODB HOWTO, by Michel Pelletier:
\\
Goes into slightly more detail about the rules for writing applications using the ZODB.
\\
\url{http://www.zope.org/Members/michel/HowTos/ZODB-How-To}
Introduction to the Zope Object Database, by Jim Fulton: Introduction to the Zope Object Database, by Jim Fulton:
\\ \\
Goes into much greater detail, explaining advanced uses of the ZODB and Goes into much greater detail, explaining advanced uses of the ZODB and
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
The ZODB package includes a number of related modules that provide The ZODB package includes a number of related modules that provide
useful data types such as BTrees. useful data types such as BTrees.
\subsection{\module{ZODB.PersistentMapping}} \subsection{\module{persistent.mapping.PersistentMapping}}
The \class{PersistentMapping} class is a wrapper for mapping objects The \class{PersistentMapping} class is a wrapper for mapping objects
that will set the dirty bit when the mapping is modified by setting or that will set the dirty bit when the mapping is modified by setting or
...@@ -24,7 +24,7 @@ value for \var{container}, a regular Python dictionary is used. ...@@ -24,7 +24,7 @@ value for \var{container}, a regular Python dictionary is used.
\class{PersistentMapping} objects support all the same methods as \class{PersistentMapping} objects support all the same methods as
Python dictionaries do. Python dictionaries do.
\subsection{\module{ZODB.PersistentList}} \subsection{\module{persistent.list.PersistentList}}
The \class{PersistentList} class is a wrapper for mutable sequence objects, The \class{PersistentList} class is a wrapper for mutable sequence objects,
much as \class{PersistentMapping} is a wrapper for mappings. much as \class{PersistentMapping} is a wrapper for mappings.
......
...@@ -13,7 +13,7 @@ ZODB is packaged using the standard distutils tools. ...@@ -13,7 +13,7 @@ ZODB is packaged using the standard distutils tools.
\subsubsection{Requirements} \subsubsection{Requirements}
You will need Python 2.2 or higher. Since the code is packaged using You will need Python 2.3 or higher. Since the code is packaged using
distutils, it is simply a matter of untarring or unzipping the release distutils, it is simply a matter of untarring or unzipping the release
package, and then running \code{python setup.py install}. package, and then running \code{python setup.py install}.
...@@ -34,12 +34,12 @@ the ZODB Wiki at \url{http://www.zope.org/Wikis/ZODB}. ...@@ -34,12 +34,12 @@ the ZODB Wiki at \url{http://www.zope.org/Wikis/ZODB}.
\subsection{How ZODB Works} \subsection{How ZODB Works}
The ZODB is conceptually simple. Python classes subclass a The ZODB is conceptually simple. Python classes subclass a
\class{Persistent} class to become ZODB-aware. \class{persistent.Persistent} class to become ZODB-aware.
Instances of persistent objects are brought in from a permanent Instances of persistent objects are brought in from a permanent
storage medium, such as a disk file, when the program needs them, and storage medium, such as a disk file, when the program needs them, and
remain cached in RAM. The ZODB traps modifications to objects, so remain cached in RAM. The ZODB traps modifications to objects, so
that when a statement such as \code{obj.size = 1} is executed, the that when a statement such as \code{obj.size = 1} is executed, the
modified object is marked as ``dirty''. On request, any dirty objects modified object is marked as ``dirty.'' On request, any dirty objects
are written out to permanent storage; this is called committing a are written out to permanent storage; this is called committing a
transaction. Transactions can also be aborted or rolled back, which transaction. Transactions can also be aborted or rolled back, which
results in any changes being discarded, dirty objects reverting to results in any changes being discarded, dirty objects reverting to
...@@ -97,7 +97,7 @@ implement the \class{Storage} interface. ...@@ -97,7 +97,7 @@ implement the \class{Storage} interface.
storing and retrieving objects from some form of long-term storage. storing and retrieving objects from some form of long-term storage.
A few different types of Storage have been written, such as A few different types of Storage have been written, such as
\class{FileStorage}, which uses regular disk files, and \class{FileStorage}, which uses regular disk files, and
\class{bsddb3Storage}, which uses Sleepycat Software's BerkeleyDB \class{BDBFullStorage}, which uses Sleepycat Software's BerkeleyDB
database. You could write a new Storage that stored objects in a database. You could write a new Storage that stored objects in a
relational database, for example, if that would relational database, for example, if that would
better suit your application. Two example storages, better suit your application. Two example storages,
...@@ -134,27 +134,54 @@ changing the first line that opens a \class{Storage}; the above example uses a ...@@ -134,27 +134,54 @@ changing the first line that opens a \class{Storage}; the above example uses a
\class{FileStorage}. In section~\ref{zeo}, ``How ZEO Works'', \class{FileStorage}. In section~\ref{zeo}, ``How ZEO Works'',
you'll see how ZEO uses this flexibility to good effect. you'll see how ZEO uses this flexibility to good effect.
\subsection{Using a ZODB Configuration File}
ZODB also supports configuration files written in the ZConfig format.
A configuration file can be used to separate the configuration logic
from the application logic. The storages classes and the \class{DB}
class support a variety of keyword arguments; all these options can be
specified in a config file.
The configuration file is simple. The example in the previous section
could use the following example:
\begin{verbatim}
<zodb>
<filestorage>
path /tmp/test-filestorage.fs
</filestorage>
</zodb>
\end{verbatim}
The \module{ZODB.config} module includes several functions for opening
database and storages from configuration files.
\begin{verbatim}
import ZODB.config
db = ZODB.config.databaseFromURL('/tmp/test.conf')
conn = db.open()
\end{verbatim}
The ZConfig documentation, included in the ZODB3 release, explains
the format in detail. Each configuration file is described by a
schema, by convention stored in a \file{component.xml} file. ZODB,
ZEO, zLOG, and zdaemon all have schemas.
\subsection{Writing a Persistent Class} \subsection{Writing a Persistent Class}
Making a Python class persistent is quite simple; it simply needs to Making a Python class persistent is quite simple; it simply needs to
subclass from the \class{Persistent} class, as shown in this example: subclass from the \class{Persistent} class, as shown in this example:
\begin{verbatim} \begin{verbatim}
import ZODB from persistent import Persistent
from Persistence import Persistent
class User(Persistent): class User(Persistent):
pass pass
\end{verbatim} \end{verbatim}
The apparently unnecessary \code{import ZODB} statement is The \class{Persistent} base class is a new-style class implemented in
needed for the following \code{from...import} statement to work C.
correctly, since the ZODB code does some magical tricks with
importing.
The \class{Persistent} base class is an \module{ExtensionClass}
class. As a result, it not compatible with new-style classes or types
in Python 2.2 and up.
For simplicity, in the examples the \class{User} class will For simplicity, in the examples the \class{User} class will
simply be used as a holder for a bunch of attributes. Normally the simply be used as a holder for a bunch of attributes. Normally the
...@@ -246,25 +273,27 @@ in practice it isn't too painful to work around them. ...@@ -246,25 +273,27 @@ in practice it isn't too painful to work around them.
The summary of rules is as follows: The summary of rules is as follows:
\begin{itemize} \begin{itemize}
\item If you modify a mutable object that's the value of an object's \item If you modify a mutable object that's the value of an object's
attribute, the ZODB can't catch that, and won't mark the object as attribute, the ZODB can't catch that, and won't mark the object as
dirty. dirty. The solution is to either set the dirty bit yourself when you
The solution is to either set the dirty bit yourself when you modify modify mutable objects, or use a wrapper for Python's lists and
mutable objects, or use a wrapper for Python's lists and dictionaries dictionaries (\class{PersistentList},
(\class{PersistentList},
\class{PersistentMapping}) \class{PersistentMapping})
that will set the dirty bit properly. that will set the dirty bit properly.
\item Certain of Python's special methods don't work when they're
defined on ExtensionClasses. The most important ones are the
\method{__cmp__} method, and the reversed versions of binary
arithmetic operations: \method{__radd__}, \method{__rsub__}, and so
forth.
\item Recent versions of the ZODB allow writing a class with \item Recent versions of the ZODB allow writing a class with
\method{__setattr__} , \method{__getattr__}, or \method{__delattr__} methods. (Older versions didn't support this at all.) \method{__setattr__} , \method{__getattr__}, or \method{__delattr__}
If you write such a \method{__setattr__} or \method{__delattr__} method, methods. (Older versions didn't support this at all.) If you write
its code has to set the dirty bit manually, such a \method{__setattr__} or \method{__delattr__} method, its code
has to set the dirty bit manually.
\item A persistent class should not have an \method{__del__} method.
The database moves objects freely between memory and storage. If an
object has not been used in a while, it may be released and its
contents loaded from storage the next time it is used. Since the
Python interpreter is unaware of persistence, it would call the
\method{__del__} each time the object was freed.
\end{itemize} \end{itemize}
...@@ -296,13 +325,9 @@ its dirty bit to true. This is done by setting the ...@@ -296,13 +325,9 @@ its dirty bit to true. This is done by setting the
\begin{verbatim} \begin{verbatim}
userobj.friends.append(otherUser) userobj.friends.append(otherUser)
userobj._p_changed = 1 userobj._p_changed = True
\end{verbatim} \end{verbatim}
An obsolete way of doing this that's still supported is calling the
\method{__changed__()} method instead, but setting \member{_p_changed}
is the preferred way.
You can hide the implementation detail of having to mark objects as You can hide the implementation detail of having to mark objects as
dirty by designing your class's API to not use direct attribute dirty by designing your class's API to not use direct attribute
access; instead, you can use the Java-style approach of accessor access; instead, you can use the Java-style approach of accessor
...@@ -328,46 +353,11 @@ and may make it into a future upstream release of Zope. ...@@ -328,46 +353,11 @@ and may make it into a future upstream release of Zope.
% you set an object's _p_changed = None). The __p_deactivate__ method should % you set an object's _p_changed = None). The __p_deactivate__ method should
% not be used (it's also obsolete). % not be used (it's also obsolete).
\subsubsection{Some Special Methods Don't Work}
Don't bother defining certain special methods on
ExtensionClasses, because they won't work. Most notably, the
\method{__cmp__} method on an ExtensionClass will never be called.
Neither will the reversed versions of binary arithmetic operations,
such as \method{__radd__} and \method{__rsub__}.
This is a moderately annoying limitation. It means that the
\class{PersistentList} class can't implement comparisons with regular
sequence objects, and therefore statements such as
\verb|if perslist==[]| don't do what you expect; instead of performing the correct
comparison, they return some arbitrary fixed result, so the \code{if}
statement will always be true or always be false. There is no good
solution to this problem at the moment, so all you can do is design
class interfaces that don't need to overload
\method{__cmp__} or the \method{__r*__} methods.
This limitation is mostly Python's fault. As of Python 2.1, the most
recent version at this writing, the code which handles comparing two
Python objects contains a hard-wired check for objects that are class
instances, which means that \code{type(obj) == types.InstanceType}.
The code inside the Python interpreter looks like this:
\begin{verbatim}
/* Code to compare objects v and w */
if (PyInstance_Check(v) || PyInstance_Check(w))
return PyInstance_DoBinOp(v, w, "__cmp__", "__rcmp__", do_cmp);
/* Do usual Python comparison of v,w */
c = PyObject_Compare(v, w);
\end{verbatim}
While ExtensionClasses try to behave as much like regular Python
instances as possible, they are still not instances, and
\function{type()} doesn't return the \code{InstanceType} object, so
no attempt is ever made to call \method{__cmp__}. Perhaps Python 2.2
will repair this.
\subsubsection{\method{__getattr__}, \method{__delattr__}, and \method{__setattr__}} \subsubsection{\method{__getattr__}, \method{__delattr__}, and \method{__setattr__}}
% XXX This section could be out-of-date. I've got to remember how we
% decided to do this before the beta release.
Recent versions of ZODB allow writing persistent classes that have Recent versions of ZODB allow writing persistent classes that have
\method{__getattr__}, \method{__delattr__}, or \method{__setattr__} \method{__getattr__}, \method{__delattr__}, or \method{__setattr__}
methods. The one minor complication is that the machinery for methods. The one minor complication is that the machinery for
......
...@@ -16,7 +16,7 @@ XXX explain mounting substorages ...@@ -16,7 +16,7 @@ XXX explain mounting substorages
\subsection{FileStorage} \subsection{FileStorage}
\subsection{BerkeleyStorage} \subsection{BDBFullStorage}
\subsection{OracleStorage} \subsection{OracleStorage}
...@@ -18,7 +18,7 @@ them in a distributed fashion without Zope ever entering the picture. ...@@ -18,7 +18,7 @@ them in a distributed fashion without Zope ever entering the picture.
The combination of ZEO and ZODB is essentially a Python-specific The combination of ZEO and ZODB is essentially a Python-specific
object database. object database.
ZEO consists of about 6000 lines of Python code, excluding tests. The ZEO consists of about 12,000 lines of Python code, excluding tests. The
code is relatively small because it contains only code for a TCP/IP code is relatively small because it contains only code for a TCP/IP
server, and for a new type of Storage, \class{ClientStorage}. server, and for a new type of Storage, \class{ClientStorage}.
\class{ClientStorage} simply makes remote procedure calls to the \class{ClientStorage} simply makes remote procedure calls to the
...@@ -45,9 +45,9 @@ writes, and ZEO is therefore better suited for read-intensive ...@@ -45,9 +45,9 @@ writes, and ZEO is therefore better suited for read-intensive
applications. If every \class{ClientStorage} is writing to the applications. If every \class{ClientStorage} is writing to the
database all the time, this will result in a storm of invalidate database all the time, this will result in a storm of invalidate
messages being sent, and this might take up more processing time than messages being sent, and this might take up more processing time than
the actual database operations themselves.\footnote{These messages are the actual database operations themselves. These messages are
small and sent in batches, so there would need to be a lot of writes small and sent in batches, so there would need to be a lot of writes
before it became a problem.} before it became a problem.
On the other hand, for applications that have few writes in comparison On the other hand, for applications that have few writes in comparison
to the number of read accesses, this aggressive caching can be a major to the number of read accesses, this aggressive caching can be a major
...@@ -71,15 +71,15 @@ configure and run a ZEO Storage Server on a machine. ...@@ -71,15 +71,15 @@ configure and run a ZEO Storage Server on a machine.
\subsubsection{Requirements} \subsubsection{Requirements}
The ZEO server software is included in ZODB3. As with the rest of The ZEO server software is included in ZODB3. As with the rest of
ZODB3, you'll need Python 2.1 or higher. ZODB3, you'll need Python 2.3 or higher.
\subsubsection{Running a server} \subsubsection{Running a server}
The start.py script in the ZEO directory can be used to start a The runzeo.py script in the ZEO directory can be used to start a
server. Run it with the -h option to see the various values. If server. Run it with the -h option to see the various values. If
you're just experimenting, a good choise is to use you're just experimenting, a good choise is to use
\code{python ZEO/start.py -D -U /tmp/zeosocket} to run ZEO in \code{python ZEO/runzeo.py -a /tmp/zeosocket -f /tmp/test.fs} to run
debug mode and with a Unix domain socket. ZEO with a Unix domain socket and a \class{FileStorage}.
\subsection{Testing the ZEO Installation} \subsection{Testing the ZEO Installation}
...@@ -101,7 +101,7 @@ from ZEO import ClientStorage ...@@ -101,7 +101,7 @@ from ZEO import ClientStorage
from ZODB import DB from ZODB import DB
# Change next line to connect to your ZEO server # Change next line to connect to your ZEO server
addr = ('kronos.example.com', 1975) addr = 'kronos.example.com', 1975
storage = ClientStorage.ClientStorage(addr) storage = ClientStorage.ClientStorage(addr)
db = DB(storage) db = DB(storage)
conn = db.open() conn = db.open()
...@@ -117,6 +117,26 @@ get_transaction().commit() ...@@ -117,6 +117,26 @@ get_transaction().commit()
If this code runs properly, then your ZEO server is working correctly. If this code runs properly, then your ZEO server is working correctly.
You can also use a configuration file.
\begin{verbatim}
<zodb>
<zeoclient>
server localhost:9100
</zeoclient>
</zodb>
\end{verbatim}
One nice feature of the configuration file is that you don't need to
specify imports for a specific storage. That makes the code a little
shorter and allows you to change storages without changing the code.
\begin{verbatim}
import ZODB.config
db = ZODB.config.databaseFromURL('/tmp/zeo.conf')
\end{verbatim}
\subsection{ZEO Programming Notes} \subsection{ZEO Programming Notes}
ZEO is written using \module{asyncore}, from the Python standard ZEO is written using \module{asyncore}, from the Python standard
......
\documentclass{howto} \documentclass{howto}
\title{ZODB/ZEO Programming Guide} \title{ZODB/ZEO Programming Guide}
\release{0.2} \release{0.3a2}
\date{\today} \date{\today}
\author{A.M.\ Kuchling} \author{A.M.\ Kuchling}
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment