Commit 1c519e6e authored by Shane Hathaway's avatar Shane Hathaway

Merged shane-oid-length-branch.

The ZEO client cache now works with OIDs of length up to 65535 bytes.
Also added some sanity checks that check the size of the data being written,
since there are some hard limits.
parent a2381f21
This diff is collapsed.
...@@ -35,8 +35,9 @@ Offset Size Contents ...@@ -35,8 +35,9 @@ Offset Size Contents
0 4 timestamp (seconds since 1/1/1970) 0 4 timestamp (seconds since 1/1/1970)
4 3 data size, in 256-byte increments, rounded up 4 3 data size, in 256-byte increments, rounded up
7 1 code (see below) 7 1 code (see below)
8 8 object id 8 2 object id length
16 8 serial number 10 8 serial number
18 variable object id
The code at offset 7 packs three fields: The code at offset 7 packs three fields:
...@@ -56,6 +57,7 @@ import sys ...@@ -56,6 +57,7 @@ import sys
import time import time
import getopt import getopt
import struct import struct
from types import StringType
def usage(msg): def usage(msg):
print >>sys.stderr, msg print >>sys.stderr, msg
...@@ -155,12 +157,16 @@ def main(): ...@@ -155,12 +157,16 @@ def main():
if not quiet: if not quiet:
print "Skipping 8 bytes at offset", offset-8 print "Skipping 8 bytes at offset", offset-8
continue continue
r = f_read(16) r = f_read(10)
if len(r) < 16: if len(r) < 10:
break break
offset += 16 offset += 10
records += 1 records += 1
oid, serial = struct_unpack(">8s8s", r) oidlen, serial = struct_unpack(">H8s", r)
oid = f_read(oidlen)
if len(oid) != oidlen:
break
offset += oidlen
if t0 is None: if t0 is None:
t0 = ts t0 = ts
thisinterval = t0 / interval thisinterval = t0 / interval
...@@ -197,11 +203,11 @@ def main(): ...@@ -197,11 +203,11 @@ def main():
bysizew[dlen] = d = bysizew.get(dlen) or {} bysizew[dlen] = d = bysizew.get(dlen) or {}
d[oid] = d.get(oid, 0) + 1 d[oid] = d.get(oid, 0) + 1
if verbose: if verbose:
print "%s %d %02x %016x %016x %1s %s" % ( print "%s %d %02x %s %016x %1s %s" % (
time.ctime(ts)[4:-5], time.ctime(ts)[4:-5],
current, current,
code, code,
U64(oid), oid_repr(oid),
U64(serial), U64(serial),
version, version,
dlen and str(dlen) or "") dlen and str(dlen) or "")
...@@ -346,6 +352,12 @@ def U64(s): ...@@ -346,6 +352,12 @@ def U64(s):
h, v = struct.unpack(">II", s) h, v = struct.unpack(">II", s)
return (long(h) << 32) + v return (long(h) << 32) + v
def oid_repr(oid):
if isinstance(oid, StringType) and len(oid) == 8:
return '%16x' % U64(oid)
else:
return repr(oid)
def addcommas(n): def addcommas(n):
sign, s = '', str(n) sign, s = '', str(n)
if s[0] == '-': if s[0] == '-':
......
...@@ -27,6 +27,10 @@ from ZEO.ClientCache import ClientCache ...@@ -27,6 +27,10 @@ from ZEO.ClientCache import ClientCache
class ClientCacheTests(unittest.TestCase): class ClientCacheTests(unittest.TestCase):
_oid = 'abcdefgh'
_oid2 = 'bcdefghi'
_oid3 = 'cdefghij'
def setUp(self): def setUp(self):
unittest.TestCase.setUp(self) unittest.TestCase.setUp(self)
self.cachesize = 10*1000*1000 self.cachesize = 10*1000*1000
...@@ -42,7 +46,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -42,7 +46,7 @@ class ClientCacheTests(unittest.TestCase):
def testStoreLoad(self): def testStoreLoad(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -51,7 +55,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -51,7 +55,7 @@ class ClientCacheTests(unittest.TestCase):
def testMissingLoad(self): def testMissingLoad(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -60,7 +64,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -60,7 +64,7 @@ class ClientCacheTests(unittest.TestCase):
def testInvalidate(self): def testInvalidate(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -72,7 +76,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -72,7 +76,7 @@ class ClientCacheTests(unittest.TestCase):
def testVersion(self): def testVersion(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
vname = 'myversion' vname = 'myversion'
...@@ -86,7 +90,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -86,7 +90,7 @@ class ClientCacheTests(unittest.TestCase):
def testVersionOnly(self): def testVersionOnly(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '' data = ''
serial = '' serial = ''
vname = 'myversion' vname = 'myversion'
...@@ -100,7 +104,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -100,7 +104,7 @@ class ClientCacheTests(unittest.TestCase):
def testInvalidateNonVersion(self): def testInvalidateNonVersion(self):
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
vname = 'myversion' vname = 'myversion'
...@@ -122,7 +126,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -122,7 +126,7 @@ class ClientCacheTests(unittest.TestCase):
# Invalidating a version should not invalidate the non-version data. # Invalidating a version should not invalidate the non-version data.
# (This tests for the same bug as testInvalidatePersists below.) # (This tests for the same bug as testInvalidatePersists below.)
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -139,7 +143,7 @@ class ClientCacheTests(unittest.TestCase): ...@@ -139,7 +143,7 @@ class ClientCacheTests(unittest.TestCase):
results.append((oid, serial, vserial)) results.append((oid, serial, vserial))
cache.verify(verifier) cache.verify(verifier)
self.assertEqual(results, []) self.assertEqual(results, [])
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -151,12 +155,12 @@ class ClientCacheTests(unittest.TestCase): ...@@ -151,12 +155,12 @@ class ClientCacheTests(unittest.TestCase):
# Make sure that cache._index[oid] is erased for oids that are # Make sure that cache._index[oid] is erased for oids that are
# stored in the cache file that's rewritten after a flip. # stored in the cache file that's rewritten after a flip.
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '1234'*100 data = '1234'*100
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
cache.checkSize(10*self.cachesize) # Force a file flip cache.checkSize(10*self.cachesize) # Force a file flip
oid2 = 'abcdefgz' oid2 = self._oid2
data2 = '1234'*10 data2 = '1234'*10
serial2 = 'ABCDEFGZ' serial2 = 'ABCDEFGZ'
cache.store(oid2, data2, serial2, '', '', '') cache.store(oid2, data2, serial2, '', '', '')
...@@ -178,17 +182,17 @@ class ClientCacheTests(unittest.TestCase): ...@@ -178,17 +182,17 @@ class ClientCacheTests(unittest.TestCase):
cache = self.cache cache = self.cache
# Create some objects # Create some objects
oid1 = 'abcdefgh' oid1 = self._oid
data1 = '1234' * 100 data1 = '1234' * 100
serial1 = 'ABCDEFGH' serial1 = 'ABCDEFGH'
oid2 = 'bcdefghi' oid2 = self._oid2
data2 = '2345' * 200 data2 = '2345' * 200
serial2 = 'BCDEFGHI' serial2 = 'BCDEFGHI'
version2 = 'myversion' version2 = 'myversion'
nonversion = 'nada' nonversion = 'nada'
vdata2 = '5432' * 250 vdata2 = '5432' * 250
vserial2 = 'IHGFEDCB' vserial2 = 'IHGFEDCB'
oid3 = 'cdefghij' oid3 = self._oid3
data3 = '3456' * 300 data3 = '3456' * 300
serial3 = 'CDEFGHIJ' serial3 = 'CDEFGHIJ'
...@@ -276,6 +280,8 @@ class ClientCacheTests(unittest.TestCase): ...@@ -276,6 +280,8 @@ class ClientCacheTests(unittest.TestCase):
class PersistentClientCacheTests(unittest.TestCase): class PersistentClientCacheTests(unittest.TestCase):
_oid = 'abcdefgh'
def setUp(self): def setUp(self):
unittest.TestCase.setUp(self) unittest.TestCase.setUp(self)
self.vardir = os.getcwd() # Don't use /tmp, it's a security risk self.vardir = os.getcwd() # Don't use /tmp, it's a security risk
...@@ -323,13 +329,12 @@ class PersistentClientCacheTests(unittest.TestCase): ...@@ -323,13 +329,12 @@ class PersistentClientCacheTests(unittest.TestCase):
# 'current' file when a persistent cache was opened. # 'current' file when a persistent cache was opened.
cache = self.cache cache = self.cache
self.assertEqual(cache._current, 0) # Check that file 0 is current self.assertEqual(cache._current, 0) # Check that file 0 is current
oid = 'abcdefgh' oid = self._oid
data = '1234' data = '1234'
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
cache.checkSize(10*self.cachesize) # Force a file flip cache.checkSize(10*self.cachesize) # Force a file flip
self.assertEqual(cache._current, 1) # Check that the flip worked self.assertEqual(cache._current, 1) # Check that the flip worked
oid = 'abcdefgh'
data = '123' data = '123'
serial = 'ABCDEFGZ' serial = 'ABCDEFGZ'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -348,7 +353,7 @@ class PersistentClientCacheTests(unittest.TestCase): ...@@ -348,7 +353,7 @@ class PersistentClientCacheTests(unittest.TestCase):
cache = self.cache cache = self.cache
magicsize = (ord('i') + 1) << 16 magicsize = (ord('i') + 1) << 16
cache = self.cache cache = self.cache
oid = 'abcdefgh' oid = self._oid
data = '!'*magicsize data = '!'*magicsize
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -367,7 +372,7 @@ class PersistentClientCacheTests(unittest.TestCase): ...@@ -367,7 +372,7 @@ class PersistentClientCacheTests(unittest.TestCase):
ltid = 'pqrstuvw' ltid = 'pqrstuvw'
cache.setLastTid(ltid) cache.setLastTid(ltid)
self.assertEqual(cache.getLastTid(), ltid) self.assertEqual(cache.getLastTid(), ltid)
oid = 'abcdefgh' oid = self._oid
data = '1234' data = '1234'
serial = 'ABCDEFGH' serial = 'ABCDEFGH'
cache.store(oid, data, serial, '', '', '') cache.store(oid, data, serial, '', '', '')
...@@ -381,10 +386,23 @@ class PersistentClientCacheTests(unittest.TestCase): ...@@ -381,10 +386,23 @@ class PersistentClientCacheTests(unittest.TestCase):
cache.checkSize(10*self.cachesize) # Force a file flip cache.checkSize(10*self.cachesize) # Force a file flip
self.failUnless(cache.getLastTid() is None) self.failUnless(cache.getLastTid() is None)
class ClientCacheLongOIDTests(ClientCacheTests):
_oid = 'abcdefghijklmnop' * 2
_oid2 = 'bcdefghijklmnopq' * 2
_oid3 = 'cdefghijklmnopqr' * 2
class PersistentClientCacheLongOIDTests(PersistentClientCacheTests):
_oid = 'abcdefghijklmnop' * 2
def test_suite(): def test_suite():
suite = unittest.TestSuite() suite = unittest.TestSuite()
suite.addTest(unittest.makeSuite(ClientCacheTests)) suite.addTest(unittest.makeSuite(ClientCacheTests))
suite.addTest(unittest.makeSuite(ClientCacheLongOIDTests))
suite.addTest(unittest.makeSuite(PersistentClientCacheTests)) suite.addTest(unittest.makeSuite(PersistentClientCacheTests))
suite.addTest(unittest.makeSuite(PersistentClientCacheLongOIDTests))
return suite return suite
if __name__ == '__main__': if __name__ == '__main__':
......
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