Commit 18c6f729 authored by Guido van Rossum's avatar Guido van Rossum

- Make the remaining functions in this module (read_index(), log(),

  rilog()) into methods, so the log message can include the storage
  name.

- Change the trace filename to include a hyphen and the storage name.

This addresses ZODB SF bug 641925.
parent 729ba6d0
...@@ -111,9 +111,6 @@ from ZODB.utils import U64 ...@@ -111,9 +111,6 @@ from ZODB.utils import U64
import zLOG import zLOG
from ZEO.ICache import ICache from ZEO.ICache import ICache
def log(msg, level=zLOG.INFO):
zLOG.LOG("ZEC", level, msg)
magic='ZEC0' magic='ZEC0'
class ClientCache: class ClientCache:
...@@ -122,11 +119,14 @@ class ClientCache: ...@@ -122,11 +119,14 @@ class ClientCache:
def __init__(self, storage='1', size=20000000, client=None, var=None): def __init__(self, storage='1', size=20000000, client=None, var=None):
# Arguments: # Arguments:
# storage -- storage name (used in persistent cache file names only) # storage -- storage name (used in filenames and log messages)
# size -- size limit in bytes of both files together # size -- size limit in bytes of both files together
# client -- if not None, use a persistent cache file and use this name # client -- if not None, use a persistent cache file and use this name
# var -- directory where to create persistent cache files # var -- directory where to create persistent cache files
self._storage = storage
self._limit = size / 2
# Allocate locks: # Allocate locks:
L = allocate_lock() L = allocate_lock()
self._acquire = L.acquire self._acquire = L.acquire
...@@ -182,10 +182,9 @@ class ClientCache: ...@@ -182,10 +182,9 @@ class ClientCache:
f[0].write(magic) f[0].write(magic)
current = 0 current = 0
log("%s: storage=%r, size=%r; file[%r]=%r" % self.log("%s: storage=%r, size=%r; file[%r]=%r" %
(self.__class__.__name__, storage, size, current, p[current])) (self.__class__.__name__, storage, size, current, p[current]))
self._limit = size / 2
self._current = current self._current = current
self._setup_trace() self._setup_trace()
...@@ -203,8 +202,8 @@ class ClientCache: ...@@ -203,8 +202,8 @@ class ClientCache:
f = self._f f = self._f
current = self._current current = self._current
if f[not current] is not None: if f[not current] is not None:
read_index(index, serial, f[not current], not current) self.read_index(serial, not current)
self._pos = read_index(index, serial, f[current], current) self._pos = self.read_index(serial, current)
return serial.items() return serial.items()
finally: finally:
...@@ -240,13 +239,13 @@ class ClientCache: ...@@ -240,13 +239,13 @@ class ClientCache:
f.seek(ap) f.seek(ap)
h = f.read(27) h = f.read(27)
if len(h) != 27: if len(h) != 27:
log("invalidate: short record for oid %16x " self.log("invalidate: short record for oid %16x "
"at position %d in cache file %d" "at position %d in cache file %d"
% (U64(oid), ap, p < 0)) % (U64(oid), ap, p < 0))
del self._index[oid] del self._index[oid]
return None return None
if h[:8] != oid: if h[:8] != oid:
log("invalidate: oid mismatch: expected %16x read %16x " self.log("invalidate: oid mismatch: expected %16x read %16x "
"at position %d in cache file %d" "at position %d in cache file %d"
% (U64(oid), U64(h[:8]), ap, p < 0)) % (U64(oid), U64(h[:8]), ap, p < 0))
del self._index[oid] del self._index[oid]
...@@ -281,7 +280,7 @@ class ClientCache: ...@@ -281,7 +280,7 @@ class ClientCache:
else: else:
tlen = -1 tlen = -1
if tlen <= 0 or vlen < 0 or dlen < 0 or vlen+dlen > tlen: if tlen <= 0 or vlen < 0 or dlen < 0 or vlen+dlen > tlen:
log("load: bad record for oid %16x " self.log("load: bad record for oid %16x "
"at position %d in cache file %d" "at position %d in cache file %d"
% (U64(oid), ap, p < 0)) % (U64(oid), ap, p < 0))
del self._index[oid] del self._index[oid]
...@@ -452,7 +451,7 @@ class ClientCache: ...@@ -452,7 +451,7 @@ class ClientCache:
else: else:
tlen = -1 tlen = -1
if tlen <= 0 or vlen < 0 or dlen < 0 or vlen+dlen > tlen: if tlen <= 0 or vlen < 0 or dlen < 0 or vlen+dlen > tlen:
log("modifiedInVersion: bad record for oid %16x " self.log("modifiedInVersion: bad record for oid %16x "
"at position %d in cache file %d" "at position %d in cache file %d"
% (U64(oid), ap, p < 0)) % (U64(oid), ap, p < 0))
del self._index[oid] del self._index[oid]
...@@ -481,7 +480,7 @@ class ClientCache: ...@@ -481,7 +480,7 @@ class ClientCache:
current = not self._current current = not self._current
self._current = current self._current = current
self._trace(0x70) self._trace(0x70)
log("flipping cache files. new current = %d" % current) self.log("flipping cache files. new current = %d" % current)
# Delete the half of the index that's no longer valid # Delete the half of the index that's no longer valid
index = self._index index = self._index
for oid in index.keys(): for oid in index.keys():
...@@ -551,19 +550,21 @@ class ClientCache: ...@@ -551,19 +550,21 @@ class ClientCache:
def _setup_trace(self): def _setup_trace(self):
# See if cache tracing is requested through $ZEO_CACHE_TRACE. # See if cache tracing is requested through $ZEO_CACHE_TRACE.
# A dash and the storage name are appended to get the filename.
# If not, or if we can't write to the trace file, # If not, or if we can't write to the trace file,
# disable tracing by setting self._trace to a dummy function. # disable tracing by setting self._trace to a dummy function.
self._tracefile = None self._tracefile = None
tfn = os.environ.get("ZEO_CACHE_TRACE") tfn = os.environ.get("ZEO_CACHE_TRACE")
if tfn: if tfn:
tfn = tfn + "-" + self._storage
try: try:
self._tracefile = open(tfn, "ab") self._tracefile = open(tfn, "ab")
self._trace(0x00) self._trace(0x00)
except IOError, msg: except IOError, msg:
self._tracefile = None self._tracefile = None
log("cannot write tracefile %s (%s)" % (tfn, msg)) self.log("cannot write tracefile %s (%s)" % (tfn, msg))
else: else:
log("opened tracefile %s" % tfn) self.log("opened tracefile %s" % tfn)
if self._tracefile is None: if self._tracefile is None:
def notrace(*args): def notrace(*args):
pass pass
...@@ -587,7 +588,9 @@ class ClientCache: ...@@ -587,7 +588,9 @@ class ClientCache:
oid, oid,
serial)) serial))
def read_index(index, serial, f, fileindex): def read_index(self, serial, fileindex):
index = self._index
f = self._f[fileindex]
seek = f.seek seek = f.seek
read = f.read read = f.read
pos = 4 pos = 4
...@@ -599,7 +602,7 @@ def read_index(index, serial, f, fileindex): ...@@ -599,7 +602,7 @@ def read_index(index, serial, f, fileindex):
if len(h) != 27: if len(h) != 27:
# An empty read is expected, anything else is suspect # An empty read is expected, anything else is suspect
if h: if h:
rilog("truncated header", pos, fileindex) self.rilog("truncated header", pos, fileindex)
break break
if h[8] in 'vni': if h[8] in 'vni':
...@@ -607,7 +610,7 @@ def read_index(index, serial, f, fileindex): ...@@ -607,7 +610,7 @@ def read_index(index, serial, f, fileindex):
else: else:
tlen = -1 tlen = -1
if tlen <= 0 or vlen < 0 or dlen < 0 or vlen + dlen > tlen: if tlen <= 0 or vlen < 0 or dlen < 0 or vlen + dlen > tlen:
rilog("invalid header data", pos, fileindex) self.rilog("invalid header data", pos, fileindex)
break break
oid = h[:8] oid = h[:8]
...@@ -616,24 +619,24 @@ def read_index(index, serial, f, fileindex): ...@@ -616,24 +619,24 @@ def read_index(index, serial, f, fileindex):
seek(dlen+vlen, 1) seek(dlen+vlen, 1)
vdlen = read(4) vdlen = read(4)
if len(vdlen) != 4: if len(vdlen) != 4:
rilog("truncated record", pos, fileindex) self.rilog("truncated record", pos, fileindex)
break break
vdlen = unpack(">i", vdlen)[0] vdlen = unpack(">i", vdlen)[0]
if vlen+dlen+43+vdlen != tlen: if vlen+dlen+43+vdlen != tlen:
rilog("inconsistent lengths", pos, fileindex) self.rilog("inconsistent lengths", pos, fileindex)
break break
seek(vdlen, 1) seek(vdlen, 1)
vs = read(8) vs = read(8)
if read(4) != h[9:13]: if read(4) != h[9:13]:
rilog("inconsistent tlen", pos, fileindex) self.rilog("inconsistent tlen", pos, fileindex)
break break
else: else:
if h[8] in 'vn' and vlen == 0: if h[8] in 'vn' and vlen == 0:
if dlen+31 != tlen: if dlen+31 != tlen:
rilog("inconsistent nv lengths", pos, fileindex) self.rilog("inconsistent nv lengths", pos, fileindex)
seek(dlen, 1) seek(dlen, 1)
if read(4) != h[9:13]: if read(4) != h[9:13]:
rilog("inconsistent nv tlen", pos, fileindex) self.rilog("inconsistent nv tlen", pos, fileindex)
break break
vs = None vs = None
...@@ -660,12 +663,15 @@ def read_index(index, serial, f, fileindex): ...@@ -660,12 +663,15 @@ def read_index(index, serial, f, fileindex):
pass pass
if count: if count:
log("read_index: cache file %d has %d records and %d bytes" self.log("read_index: cache file %d has %d records and %d bytes"
% (fileindex, count, pos)) % (fileindex, count, pos))
return pos return pos
def rilog(msg, pos, fileindex): def rilog(self, msg, pos, fileindex):
# Helper to log messages from read_index # Helper to log messages from read_index
log("read_index: %s at position %d in cache file %d" self.log("read_index: %s at position %d in cache file %d"
% (msg, pos, fileindex)) % (msg, pos, fileindex))
def log(self, msg, level=zLOG.INFO):
zLOG.LOG("ZEC:%s" % self._storage, level, msg)
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