Source code for relstorage.adapters.sqlite.locker

# -*- coding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2019 Zope Foundation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.1 (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.
#
##############################################################################

from __future__ import absolute_import
from __future__ import division
from __future__ import print_function

from zope.interface import implementer

from ..interfaces import ILocker
from ..locker import AbstractLocker


[docs] @implementer(ILocker) class Sqlite3Locker(AbstractLocker): supports_row_lock_nowait = False _lock_share_clause = '<This should not be used>' _lock_share_clause_nowait = '<This should not be used>' _lock_current_clause = '<This Should not be used>' def _lock_readCurrent_oids_for_share(self, cursor, current_oids, shared_locks_block): """ This is a no-op. Once this is called, the parent class will call the after_lock_share callback, and the parent adapter will use that to check the recorded TIDs against the desired TIDs. Our store connection is still in autocommit mode at this point, so it will read the current data. """ _lock_current_objects_query = 'UPDATE commit_row_lock SET tid = tid WHERE tid < 0' def _lock_rows_being_modified(self, cursor): # We exclusively lock the whole database at this point. There's no way to # avoid it. We also switch out of autocommit mode and start a transaction, # meaning our store cursor is no longer updating...but since we have an exclusive # lock, that's fine. We use a statement that matches no rows, but still takes # locks. conn = cursor.connection # If we pre-allocated the TID due to a restore, we could already be # in a tranasction. In which case there's nothing to do (but for safety # we still execute the query) AbstractLocker._lock_rows_being_modified(self, cursor) assert conn.in_transaction or conn.in_transaction is None, repr(conn) # Py2 def hold_commit_lock(self, cursor, ensure_current=False, nowait=False): # We may or may not actually have locked rows; if we're doing a restore # we allocate the tid first. Likewise, an undo also # locks upfront. if ensure_current: cursor.execute(self._lock_current_objects_query) in_transaction = cursor.connection.in_transaction assert in_transaction or in_transaction is None # Py2 def release_commit_lock(self, cursor): # no action needed, locks released with transaction. pass def release_pack_lock(self, cursor): pass def hold_pack_lock(self, cursor): pass