[Zope-Checkins] CVS: ZODB3/ZEO - cache.py:1.1.2.1
Jeremy Hylton
cvs-admin at zope.org
Tue Nov 4 15:04:38 EST 2003
Update of /cvs-repository/ZODB3/ZEO
In directory cvs.zope.org:/tmp/cvs-serv12826
Added Files:
Tag: ZODB3-mvcc-2-branch
cache.py
Log Message:
Strawman implementation of new cache API and simple unit tests.
=== Added File ZODB3/ZEO/cache.py ===
##############################################################################
#
# Copyright (c) 2003 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE.
#
##############################################################################
"""Example client cache that stores multiple revisions of an object."""
##
# A disk-based cache for ZEO clients.
# <p>
# This class provides an interface to a persistent, disk-based cache
# used by ZEO clients to store copies of database records from the
# server.
# <p>
# The details of the constructor as unspecified at this point.
class Cache:
"""A simple in-memory cache."""
def __init__(self):
self.tid = None
self.current = {}
self.current_tid = {}
self.version = {}
self.noncurrent = {}
##
# Set the last transaction seen by the cache.
# @param tid a transaction id
# @exception ValueError attempt to set a new tid less than the current tid
def setLastTid(self, tid):
if self.tid is not None:
if tid < self.tid:
raise ValueError(
"new last tid must be greater that previous one")
self.tid = tid
##
# Return the last transaction seen by the cache.
# @return a transaction id
# @defreturn string
def getLastTid(self):
return self.tid
##
# Return the current data record for oid and version
# @param oid object id
# @param version a version string
# @return data record and serial number
# @defreturn 2-tuple: (string, string)
def load(self, oid, version=""):
if version:
triple = self.version.get(oid)
if triple is not None:
stored_version, data, serial = triple
if version == stored_version:
return data, serial
return self.current.get(oid)
##
# Return a non-current revision of oid that was current before tid.
# @param oid object id
# @param tid id of transaction that wrote next revision of oid
# @return data record, serial number, start tid, and end tid
# @defreturn 4-tuple: (string, string, string, string)
def loadNonCurrent(self, oid, tid):
for lo, hi, data, serial in self.noncurrent.get(oid, []):
if lo < tid <= hi:
return data, serial, lo, hi
return None
##
# Return the version an object is modified in or None for an
# object that is not modified in a version.
# @param oid object id
# @return name of version in which the object is modified
# @defreturn string or None
def modifiedInVersion(self, oid):
pair = self.version.get(oid)
if pair is None:
return None
return pair[0]
##
# Store a new data record in the cache.
# @param oid object id
# @param version name of version that oid was modified in. The cache
# only stores current version data, so end_tid should
# be None.
# @param start_tid the id of the transaction that wrote this revision
# @param end_tid the id of the transaction that created the next
# revision of oid. If end_tid is None, the data is
# current.
# @param data the actual data
# @exception ValueError tried to store non-current version data
def store(self, oid, version, serial, start_tid, end_tid, data):
if version:
if end_tid is not None:
raise ValueError("cache only stores current version data")
self.version[oid] = version, data, serial
return
# XXX If there was a previous revision, we need to invalidate it.
if end_tid is None:
self.current[oid] = data, serial
self.current_tid[oid] = start_tid
else:
# XXX could use bisect and keep the list sorted
L = self.noncurrent.setdefault(oid, [])
L.append((start_tid, end_tid, data, serial))
##
# Mark the current data for oid as non-current. If there is no
# current data for oid, do nothing.
# @param oid object id
# @param version name of version to invalidate.
# @param tid the id of the transaction that wrote a now revision of oid
def invalidate(self, oid, version, tid):
if tid > self.tid:
self.tid = tid
if version:
if oid in self.version:
del self.version[oid]
data = self.current.get(oid)
if data is None:
return
del self.current[oid]
start_tid = self.current_tid.pop(oid)
L = self.noncurrent.setdefault(oid, [])
L.append((start_tid, tid, data[0], data[1]))
More information about the Zope-Checkins
mailing list