Commit 3b20f9da authored by Guido van Rossum's avatar Guido van Rossum

A bouquet of minor code and comment cleanups.

Note that the checkModifyAfterAbortVersion() test still fails; Jeremy
wants to change abortVersion in FileStorage and DemoStorage to revert
the serialno to the last nonversion serialno instead of assigning a
brand new serialno, so I'm holding off on invalidating the cache in
ClientStorage.abortVersion.

Also note that on Red Hat 7.2, the ZEO test suite hangs in
checkCommitLock2OnAbort or checkCommitLock2OnCommit; we'll try to
debug this later today.
parent ec536904
...@@ -20,14 +20,12 @@ or temporary. It is persistent if the client argument is not None. ...@@ -20,14 +20,12 @@ or temporary. It is persistent if the client argument is not None.
Persistent cache files live in the var directory and are named Persistent cache files live in the var directory and are named
'c<storage>-<client>-<digit>.zec' where <storage> is the storage 'c<storage>-<client>-<digit>.zec' where <storage> is the storage
argument (default ''), <client> is the client argument, and <digit> is argument (default '1'), <client> is the client argument, and <digit> is
0 or 1. Temporary cache files are unnamed files in the standard 0 or 1. Temporary cache files are unnamed files in the standard
temporary directory as determined by the tempfile module. temporary directory as determined by the tempfile module.
The ClientStorage overrides some of the defaults; in particular, its The ClientStorage overrides the client name default to the value of
storage name defaults to '1' and its client name defaults to the value the environment variable ZEO_CLIENT, if it exists.
of the environment variable ZEO_CLIENT, if it exists, otherwise to
None.
Each cache file has a 4-byte magic number followed by a sequence of Each cache file has a 4-byte magic number followed by a sequence of
records of the form: records of the form:
...@@ -37,6 +35,7 @@ records of the form: ...@@ -37,6 +35,7 @@ records of the form:
0: oid -- 8-byte object id 0: oid -- 8-byte object id
8: status -- 1-byte status 'v': valid, 'n': non-version valid, 'i': invalid 8: status -- 1-byte status 'v': valid, 'n': non-version valid, 'i': invalid
('n' means only the non-version data in the record is valid)
9: tlen -- 4-byte (unsigned) record length 9: tlen -- 4-byte (unsigned) record length
...@@ -93,10 +92,9 @@ file 0 and file 1. ...@@ -93,10 +92,9 @@ file 0 and file 1.
""" """
__version__ = "$Revision: 1.28 $"[11:-2] __version__ = "$Revision: 1.29 $"[11:-2]
import os import os
import sys
import tempfile import tempfile
from struct import pack, unpack from struct import pack, unpack
from thread import allocate_lock from thread import allocate_lock
...@@ -113,7 +111,12 @@ class ClientCache: ...@@ -113,7 +111,12 @@ class ClientCache:
__implements__ = ICache __implements__ = ICache
def __init__(self, storage='', size=20000000, client=None, var=None): def __init__(self, storage='1', size=20000000, client=None, var=None):
# Arguments:
# storage -- storage name (used in persistent cache file names only)
# size -- size limit in bytes of both files together
# client -- if not None, use a persistent cache file and use this name
# var -- directory where to create persistent cache files
# Allocate locks: # Allocate locks:
L = allocate_lock() L = allocate_lock()
...@@ -122,6 +125,7 @@ class ClientCache: ...@@ -122,6 +125,7 @@ class ClientCache:
if client is not None: if client is not None:
# Create a persistent cache # Create a persistent cache
# CLIENT_HOME and INSTANCE_HOME are builtins set by App.FindHomes
if var is None: if var is None:
try: try:
var = CLIENT_HOME var = CLIENT_HOME
...@@ -175,8 +179,11 @@ class ClientCache: ...@@ -175,8 +179,11 @@ class ClientCache:
self._current = current self._current = current
def open(self): def open(self):
# XXX open is overloaded to perform two tasks for # Two tasks:
# optimization reasons # - Set self._index, self._get, and self._pos.
# - Read and validate both cache files, returning a list of
# serials to be used by verify().
# This may be called more than once (by the cache verification code).
self._acquire() self._acquire()
try: try:
self._index=index={} self._index=index={}
...@@ -219,11 +226,11 @@ class ClientCache: ...@@ -219,11 +226,11 @@ class ClientCache:
f = self._f[p < 0] f = self._f[p < 0]
ap = abs(p) ap = abs(p)
f.seek(ap) f.seek(ap)
h = f.read(8) h = f.read(27)
if h != oid: if h[:8] != oid:
return return
f.seek(p+8) # Switch from reading to writing f.seek(p+8) # Switch from reading to writing
if version: if version and h[15:19] != '\0\0\0\0':
f.write('n') f.write('n')
else: else:
del self._index[oid] del self._index[oid]
...@@ -266,16 +273,19 @@ class ClientCache: ...@@ -266,16 +273,19 @@ class ClientCache:
if dlen: if dlen:
seek(dlen, 1) seek(dlen, 1)
v = read(vlen) vheader = read(vlen+4)
v = vheader[:-4]
if version != v: if version != v:
if dlen: if dlen:
seek(-dlen-vlen, 1) seek(p+27)
return read(dlen), h[19:] return read(dlen), h[19:]
else: else:
return None return None
dlen = unpack(">i", read(4))[0] vdlen = unpack(">i", vheader[-4:])[0]
return read(dlen), read(8) vdata = read(vdlen)
vserial = read(8)
return vdata, vserial
finally: finally:
self._release() self._release()
...@@ -302,12 +312,12 @@ class ClientCache: ...@@ -302,12 +312,12 @@ class ClientCache:
return self._store(oid, '', '', version, data, serial) return self._store(oid, '', '', version, data, serial)
if dlen: if dlen:
p = read(dlen) nvdata = read(dlen)
s = h[19:] nvserial = h[19:]
else: else:
return self._store(oid, '', '', version, data, serial) return self._store(oid, '', '', version, data, serial)
self._store(oid, p, s, version, data, serial) self._store(oid, nvdata, nvserial, version, data, serial)
else: else:
# Simple case, just store new data: # Simple case, just store new data:
self._store(oid, data, serial, '', None, None) self._store(oid, data, serial, '', None, None)
...@@ -315,6 +325,10 @@ class ClientCache: ...@@ -315,6 +325,10 @@ class ClientCache:
self._release() self._release()
def modifiedInVersion(self, oid): def modifiedInVersion(self, oid):
# This should return:
# - The version from the record for oid, if there is one.
# - '', if there is no version in the record and its status is 'v'.
# - None, if we don't know: no valid record or status is 'n'.
self._acquire() self._acquire()
try: try:
p = self._get(oid, None) p = self._get(oid, None)
...@@ -345,10 +359,10 @@ class ClientCache: ...@@ -345,10 +359,10 @@ class ClientCache:
self._release() self._release()
def checkSize(self, size): def checkSize(self, size):
self._acquire()
try:
# Make sure we aren't going to exceed the target size. # Make sure we aren't going to exceed the target size.
# If we are, then flip the cache. # If we are, then flip the cache.
self._acquire()
try:
if self._pos + size > self._limit: if self._pos + size > self._limit:
current = not self._current current = not self._current
self._current = current self._current = current
...@@ -358,10 +372,9 @@ class ClientCache: ...@@ -358,10 +372,9 @@ class ClientCache:
if (index[oid] < 0) == current: if (index[oid] < 0) == current:
del index[oid] del index[oid]
if self._p[current] is not None: if self._p[current] is not None:
# Persistent cache file: # Persistent cache file: remove the old file
# Note that due to permission madness, waaa, # before opening the new one, because the old file
# we need to remove the old file before # may be owned by root (created before setuid()).
# we open the new one. Waaaaaaaaaa.
if self._f[current] is not None: if self._f[current] is not None:
self._f[current].close() self._f[current].close()
try: try:
...@@ -373,7 +386,7 @@ class ClientCache: ...@@ -373,7 +386,7 @@ class ClientCache:
# Temporary cache file: # Temporary cache file:
self._f[current] = tempfile.TemporaryFile(suffix='.zec') self._f[current] = tempfile.TemporaryFile(suffix='.zec')
self._f[current].write(magic) self._f[current].write(magic)
self._pos = pos = 4 self._pos = 4
finally: finally:
self._release() self._release()
...@@ -417,12 +430,10 @@ class ClientCache: ...@@ -417,12 +430,10 @@ class ClientCache:
self._pos += tlen self._pos += tlen
def read_index(index, serial, f, current): def read_index(index, serial, f, fileindex):
seek = f.seek seek = f.seek
read = f.read read = f.read
pos = 4 pos = 4
seek(0, 2)
size = f.tell()
while 1: while 1:
f.seek(pos) f.seek(pos)
...@@ -443,7 +454,7 @@ def read_index(index, serial, f, current): ...@@ -443,7 +454,7 @@ def read_index(index, serial, f, current):
if len(vdlen) != 4: if len(vdlen) != 4:
break break
vdlen = unpack(">i", vdlen)[0] vdlen = unpack(">i", vdlen)[0]
if vlen+dlen+42+vdlen > tlen: if vlen+dlen+43+vdlen != tlen:
break break
seek(vdlen, 1) seek(vdlen, 1)
vs = read(8) vs = read(8)
...@@ -453,7 +464,7 @@ def read_index(index, serial, f, current): ...@@ -453,7 +464,7 @@ def read_index(index, serial, f, current):
vs = None vs = None
if h[8] in 'vn': if h[8] in 'vn':
if current: if fileindex:
index[oid] = -pos index[oid] = -pos
else: else:
index[oid] = pos index[oid] = pos
......
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