Commit 818315b6 authored by Shane Hathaway's avatar Shane Hathaway

Fixed intermittent failures by making MVCCMappingStorage hold a per-connection...

Fixed intermittent failures by making MVCCMappingStorage hold a per-connection snapshot of the database.
parent c160bc59
...@@ -231,7 +231,8 @@ class MappingStorage(object): ...@@ -231,7 +231,8 @@ class MappingStorage(object):
if transactions[tid].pack(oid): if transactions[tid].pack(oid):
del transactions[tid] del transactions[tid]
self._data = new_data self._data.clear()
self._data.update(new_data)
# ZODB.interfaces.IStorage # ZODB.interfaces.IStorage
def registerDB(self, db): def registerDB(self, db):
...@@ -307,6 +308,7 @@ class MappingStorage(object): ...@@ -307,6 +308,7 @@ class MappingStorage(object):
self._ltid = tid self._ltid = tid
self._transactions[tid] = TransactionRecord(tid, transaction, tdata) self._transactions[tid] = TransactionRecord(tid, transaction, tdata)
self._transaction = None self._transaction = None
del self._tdata
self._commit_lock.release() self._commit_lock.release()
# ZEO.interfaces.IServeable # ZEO.interfaces.IServeable
......
...@@ -20,6 +20,8 @@ connection's view. ...@@ -20,6 +20,8 @@ connection's view.
import time import time
import BTrees import BTrees
import ZODB.utils
import ZODB.POSException
from ZODB.interfaces import IMVCCStorage from ZODB.interfaces import IMVCCStorage
from ZODB.MappingStorage import MappingStorage from ZODB.MappingStorage import MappingStorage
from ZODB.TimeStamp import TimeStamp from ZODB.TimeStamp import TimeStamp
...@@ -33,28 +35,63 @@ class MVCCMappingStorage(MappingStorage): ...@@ -33,28 +35,63 @@ class MVCCMappingStorage(MappingStorage):
MappingStorage.__init__(self, name=name) MappingStorage.__init__(self, name=name)
# _polled_tid contains the transaction ID at the last poll. # _polled_tid contains the transaction ID at the last poll.
self._polled_tid = '' self._polled_tid = ''
self._data_snapshot = None # {oid->(state, tid)}
self._main_lock_acquire = self._lock_acquire
self._main_lock_release = self._lock_release
def new_instance(self): def new_instance(self):
"""Returns a storage instance that is a view of the same data. """Returns a storage instance that is a view of the same data.
""" """
res = MVCCMappingStorage(name=self.__name__) inst = MVCCMappingStorage(name=self.__name__)
res._transactions = self._transactions # All instances share the same OID data, transaction log, commit lock,
return res # and OID sequence.
inst._data = self._data
inst._transactions = self._transactions
inst._commit_lock = self._commit_lock
inst.new_oid = self.new_oid
inst.pack = self.pack
inst._main_lock_acquire = self._lock_acquire
inst._main_lock_release = self._lock_release
return inst
@ZODB.utils.locked(MappingStorage.opened)
def sync(self, force=False): def sync(self, force=False):
pass self._data_snapshot = None
def release(self): def release(self):
pass pass
@ZODB.utils.locked(MappingStorage.opened)
def load(self, oid, version=''):
assert not version, "Versions are not supported"
if self._data_snapshot is None:
self.poll_invalidations()
info = self._data_snapshot.get(oid)
if info:
return info
raise ZODB.POSException.POSKeyError(oid)
def poll_invalidations(self): def poll_invalidations(self):
"""Poll the storage for changes by other connections. """Poll the storage for changes by other connections.
""" """
# prevent changes to _transactions and _data during analysis
self._main_lock_acquire()
try:
if self._transactions: if self._transactions:
new_tid = self._transactions.maxKey() new_tid = self._transactions.maxKey()
else: else:
new_tid = '' new_tid = ''
# Copy the current data into a snapshot. This is obviously
# very inefficient for large storages, but it's good for
# tests.
self._data_snapshot = {}
for oid, tid_data in self._data.items():
if tid_data:
tid = tid_data.maxKey()
self._data_snapshot[oid] = tid_data[tid], tid
if self._polled_tid: if self._polled_tid:
if not self._transactions.has_key(self._polled_tid): if not self._transactions.has_key(self._polled_tid):
# This connection is so old that we can no longer enumerate # This connection is so old that we can no longer enumerate
...@@ -64,7 +101,8 @@ class MVCCMappingStorage(MappingStorage): ...@@ -64,7 +101,8 @@ class MVCCMappingStorage(MappingStorage):
changed_oids = set() changed_oids = set()
for tid, txn in self._transactions.items( for tid, txn in self._transactions.items(
self._polled_tid, new_tid, excludemin=True, excludemax=False): self._polled_tid, new_tid,
excludemin=True, excludemax=False):
if txn.status == 'p': if txn.status == 'p':
# This transaction has been packed, so it is no longer # This transaction has been packed, so it is no longer
# possible to enumerate all changed oids. # possible to enumerate all changed oids.
...@@ -73,16 +111,26 @@ class MVCCMappingStorage(MappingStorage): ...@@ -73,16 +111,26 @@ class MVCCMappingStorage(MappingStorage):
if tid == self._ltid: if tid == self._ltid:
# ignore the transaction committed by this connection # ignore the transaction committed by this connection
continue continue
changed_oids.update(txn.data.keys())
changes = txn.data finally:
# pull in changes from the transaction log self._main_lock_release()
for oid, value in changes.iteritems():
tid_data = self._data.get(oid)
if tid_data is None:
tid_data = BTrees.OOBTree.OOBucket()
self._data[oid] = tid_data
tid_data[tid] = changes[oid]
changed_oids.update(changes.keys())
self._polled_tid = new_tid self._polled_tid = new_tid
return list(changed_oids) return list(changed_oids)
def tpc_finish(self, transaction, func = lambda tid: None):
self._data_snapshot = None
MappingStorage.tpc_finish(self, transaction, func)
def tpc_abort(self, transaction):
self._data_snapshot = None
MappingStorage.tpc_abort(self, transaction)
def pack(self, t, referencesf, gc=True):
# prevent all concurrent commits during packing
self._commit_lock.acquire()
try:
MappingStorage.pack(self, t, referencesf, gc)
finally:
self._commit_lock.release()
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