{
  "RepoName": "https://github.com/piskvorky/sqlitedict.git",
  "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n#\n# This code was inspired by:\n#  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n#  * http://code.activestate.com/recipes/526618/\n\n\"\"\"\nA lightweight wrapper around Python's sqlite3 database, with a dict-like interface\nand multi-thread access support::\n\n>>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n>>> mydict['some_key'] = any_picklable_object\n>>> print mydict['some_key']\n>>> print len(mydict) # etc... all dict functions work\n\nPickle is used internally to serialize the values. Keys are strings.\n\nIf you don't use autocommit (default is no autocommit for performance), then\ndon't forget to call `mydict.commit()` when done with a transaction.\n\n\"\"\"\n\nimport sqlite3\nimport os\nimport sys\nimport tempfile\nimport threading\nimport logging\nimport traceback\nfrom base64 import b64decode, b64encode\nimport weakref\n\n__version__ = '2.1.0'\n\n\ndef reraise(tp, value, tb=None):\n    if value is None:\n        value = tp()\n    if value.__traceback__ is not tb:\n        raise value.with_traceback(tb)\n    raise value\n\n\ntry:\n    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\nexcept ImportError:\n    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n\n# some Python 3 vs 2 imports\ntry:\n    from collections import UserDict as DictClass\nexcept ImportError:\n    from UserDict import DictMixin as DictClass\n\ntry:\n    from queue import Queue\nexcept ImportError:\n    from Queue import Queue\n\n\nlogger = logging.getLogger(__name__)\n\n#\n# There's a thread that holds the actual SQL connection (SqliteMultithread).\n# We communicate with this thread via queues (request and responses).\n# The requests can either be SQL commands or one of the \"special\" commands\n# below:\n#\n# _REQUEST_CLOSE: request that the SQL connection be closed\n# _REQUEST_COMMIT: request that any changes be committed to the DB\n#\n# Responses are either SQL records (e.g. results of a SELECT) or the magic\n# _RESPONSE_NO_MORE command, which indicates nothing else will ever be written\n# to the response queue.\n#\n_REQUEST_CLOSE = '--close--'\n_REQUEST_COMMIT = '--commit--'\n_RESPONSE_NO_MORE = '--no more--'\n\n#\n# We work with weak references for better memory efficiency.\n# Dereferencing, checking the referent queue still exists, and putting to it\n# is boring and repetitive, so we have a _put function to handle it for us.\n#\n_PUT_OK, _PUT_REFERENT_DESTROYED, _PUT_NOOP = 0, 1, 2\n\n\ndef _put(queue_reference, item):\n    if queue_reference is not None:\n        queue = queue_reference()\n        if queue is None:\n            #\n            # We got a reference to a queue, but that queue no longer exists\n            #\n            retval = _PUT_REFERENT_DESTROYED\n        else:\n            queue.put(item)\n            retval = _PUT_OK\n\n        del queue\n        return retval\n\n    #\n    # We didn't get a reference to a queue, so do nothing (no-op).\n    #\n    return _PUT_NOOP\n\n\ndef open(*args, **kwargs):\n    \"\"\"See documentation of the SqliteDict class.\"\"\"\n    return SqliteDict(*args, **kwargs)\n\n\ndef encode(obj):\n    \"\"\"Serialize an object using pickle to a binary format accepted by SQLite.\"\"\"\n    return sqlite3.Binary(dumps(obj, protocol=PICKLE_PROTOCOL))\n\n\ndef decode(obj):\n    \"\"\"Deserialize objects retrieved from SQLite.\"\"\"\n    return loads(bytes(obj))\n\n\ndef encode_key(key):\n    \"\"\"Serialize a key using pickle + base64 encoding to text accepted by SQLite.\"\"\"\n    return b64encode(dumps(key, protocol=PICKLE_PROTOCOL)).decode(\"ascii\")\n\n\ndef decode_key(key):\n    \"\"\"Deserialize a key retrieved from SQLite.\"\"\"\n    return loads(b64decode(key.encode(\"ascii\")))\n\n\ndef identity(obj):\n    \"\"\"Identity f(x) = x function for encoding/decoding.\"\"\"\n    return obj\n\n\nclass SqliteDict(DictClass):\n    VALID_FLAGS = ['c', 'r', 'w', 'n']\n\n    def __init__(self, filename=None, tablename='unnamed', flag='c',\n                 autocommit=False, journal_mode=\"DELETE\", encode=encode,\n                 decode=decode, encode_key=identity, decode_key=identity,\n                 timeout=5, outer_stack=True):\n        \"\"\"\n        Initialize a thread-safe sqlite-backed dictionary. The dictionary will\n        be a table `tablename` in database file `filename`. A single file (=database)\n        may contain multiple tables.\n\n        If no `filename` is given, a random file in temp will be used (and deleted\n        from temp once the dict is closed/deleted).\n\n        If you enable `autocommit`, changes will be committed after each operation\n        (more inefficient but safer). Otherwise, changes are committed on `self.commit()`,\n        `self.clear()` and `self.close()`.\n\n        Set `journal_mode` to 'OFF' if you're experiencing sqlite I/O problems\n        or if you need performance and don't care about crash-consistency.\n\n        Set `outer_stack` to False to disable the output of the outer exception\n        to the error logs.  This may improve the efficiency of sqlitedict\n        operation at the expense of a detailed exception trace.\n\n        The `flag` parameter. Exactly one of:\n          'c': default mode, open for read/write, creating the db/table if necessary.\n          'w': open for r/w, but drop `tablename` contents first (start with empty table)\n          'r': open as read-only\n          'n': create a new database (erasing any existing tables, not just `tablename`!).\n\n        The `encode` and `decode` parameters are used to customize how the values\n        are serialized and deserialized.\n        The `encode` parameter must be a function that takes a single Python\n        object and returns a serialized representation.\n        The `decode` function must be a function that takes the serialized\n        representation produced by `encode` and returns a deserialized Python\n        object.\n        The default is to use pickle.\n\n        The `timeout` defines the maximum time (in seconds) to wait for initial Thread startup.\n\n        \"\"\"\n        self.in_temp = filename is None\n        if self.in_temp:\n            fd, filename = tempfile.mkstemp(prefix='sqldict')\n            os.close(fd)\n\n        if flag not in SqliteDict.VALID_FLAGS:\n            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n        self.flag = flag\n\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n\n        dirname = os.path.dirname(filename)\n        if dirname:\n            if not os.path.exists(dirname):\n                raise RuntimeError('Error! The directory does not exist, %s' % dirname)\n\n        self.filename = filename\n\n        # Use standard SQL escaping of double quote characters in identifiers, by doubling them.\n        # See https://github.com/RaRe-Technologies/sqlitedict/pull/113\n        self.tablename = tablename.replace('\"', '\"\"')\n\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        self.encode = encode\n        self.decode = decode\n        self.encode_key = encode_key\n        self.decode_key = decode_key\n        self._outer_stack = outer_stack\n\n        logger.debug(\"opening Sqlite table %r in %r\" % (tablename, filename))\n        self.conn = self._new_conn()\n        if self.flag == 'r':\n            if self.tablename not in SqliteDict.get_tablenames(self.filename):\n                msg = 'Refusing to create a new table \"%s\" in read-only DB mode' % tablename\n                raise RuntimeError(msg)\n        else:\n            MAKE_TABLE = 'CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % self.tablename\n            self.conn.execute(MAKE_TABLE)\n            self.conn.commit()\n        if flag == 'w':\n            self.clear()\n\n    def _new_conn(self):\n        return SqliteMultithread(\n            self.filename,\n            autocommit=self.autocommit,\n            journal_mode=self.journal_mode,\n            outer_stack=self._outer_stack,\n        )\n\n    def __enter__(self):\n        if not hasattr(self, 'conn') or self.conn is None:\n            self.conn = self._new_conn()\n        return self\n\n    def __exit__(self, *exc_info):\n        self.close()\n\n    def __str__(self):\n        return \"SqliteDict(%s)\" % (self.filename)\n\n    def __repr__(self):\n        return str(self)  # no need of something complex\n\n    def __len__(self):\n        # `select count (*)` is super slow in sqlite (does a linear scan!!)\n        # As a result, len() is very slow too once the table size grows beyond trivial.\n        # We could keep the total count of rows ourselves, by means of triggers,\n        # but that seems too complicated and would slow down normal operation\n        # (insert/delete etc).\n        GET_LEN = 'SELECT COUNT(*) FROM \"%s\"' % self.tablename\n        rows = self.conn.select_one(GET_LEN)[0]\n        return rows if rows is not None else 0\n\n    def __bool__(self):\n        # No elements is False, otherwise True\n        GET_MAX = 'SELECT MAX(ROWID) FROM \"%s\"' % self.tablename\n        m = self.conn.select_one(GET_MAX)[0]\n        # Explicit better than implicit and bla bla\n        return True if m is not None else False\n\n    def iterkeys(self):\n        GET_KEYS = 'SELECT key FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key in self.conn.select(GET_KEYS):\n            yield self.decode_key(key[0])\n\n    def itervalues(self):\n        GET_VALUES = 'SELECT value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for value in self.conn.select(GET_VALUES):\n            yield self.decode(value[0])\n\n    def iteritems(self):\n        GET_ITEMS = 'SELECT key, value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key, value in self.conn.select(GET_ITEMS):\n            yield self.decode_key(key), self.decode(value)\n\n    def keys(self):\n        return self.iterkeys()\n\n    def values(self):\n        return self.itervalues()\n\n    def items(self):\n        return self.iteritems()\n\n    def __contains__(self, key):\n        HAS_ITEM = 'SELECT 1 FROM \"%s\" WHERE key = ?' % self.tablename\n        return self.conn.select_one(HAS_ITEM, (self.encode_key(key),)) is not None\n\n    def __getitem__(self, key):\n        GET_ITEM = 'SELECT value FROM \"%s\" WHERE key = ?' % self.tablename\n        item = self.conn.select_one(GET_ITEM, (self.encode_key(key),))\n        if item is None:\n            raise KeyError(key)\n        return self.decode(item[0])\n\n    def __setitem__(self, key, value):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to write to read-only SqliteDict')\n\n        ADD_ITEM = 'REPLACE INTO \"%s\" (key, value) VALUES (?,?)' % self.tablename\n        self.conn.execute(ADD_ITEM, (self.encode_key(key), self.encode(value)))\n        if self.autocommit:\n            self.commit()\n\n    def __delitem__(self, key):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to delete from read-only SqliteDict')\n\n        if key not in self:\n            raise KeyError(key)\n        DEL_ITEM = 'DELETE FROM \"%s\" WHERE key = ?' % self.tablename\n        self.conn.execute(DEL_ITEM, (self.encode_key(key),))\n        if self.autocommit:\n            self.commit()\n\n    def update(self, items=(), **kwds):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to update read-only SqliteDict')\n\n        try:\n            items = items.items()\n        except AttributeError:\n            pass\n        items = [(self.encode_key(k), self.encode(v)) for k, v in items]\n\n        UPDATE_ITEMS = 'REPLACE INTO \"%s\" (key, value) VALUES (?, ?)' % self.tablename\n        self.conn.executemany(UPDATE_ITEMS, items)\n        if kwds:\n            self.update(kwds)\n        if self.autocommit:\n            self.commit()\n\n    def __iter__(self):\n        return self.iterkeys()\n\n    def clear(self):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to clear read-only SqliteDict')\n\n        # avoid VACUUM, as it gives \"OperationalError: database schema has changed\"\n        CLEAR_ALL = 'DELETE FROM \"%s\";' % self.tablename\n        self.conn.commit()\n        self.conn.execute(CLEAR_ALL)\n        self.conn.commit()\n\n    @staticmethod\n    def get_tablenames(filename):\n        \"\"\"get the names of the tables in an sqlite db as a list\"\"\"\n        if not os.path.isfile(filename):\n            raise IOError('file %s does not exist' % (filename))\n        GET_TABLENAMES = 'SELECT name FROM sqlite_master WHERE type=\"table\"'\n        with sqlite3.connect(filename) as conn:\n            cursor = conn.execute(GET_TABLENAMES)\n            res = cursor.fetchall()\n\n        return [name[0] for name in res]\n\n    def commit(self, blocking=True):\n        \"\"\"\n        Persist all data to disk.\n\n        When `blocking` is False, the commit command is queued, but the data is\n        not guaranteed persisted (default implication when autocommit=True).\n        \"\"\"\n        if self.conn is not None:\n            self.conn.commit(blocking)\n    sync = commit\n\n    def close(self, do_log=True, force=False):\n        if do_log:\n            logger.debug(\"closing %s\" % self)\n        if hasattr(self, 'conn') and self.conn is not None:\n            if self.conn.autocommit and not force:\n                # typically calls to commit are non-blocking when autocommit is\n                # used.  However, we need to block on close() to ensure any\n                # awaiting exceptions are handled and that all data is\n                # persisted to disk before returning.\n                self.conn.commit(blocking=True)\n            self.conn.close(force=force)\n            self.conn = None\n        if self.in_temp:\n            try:\n                os.remove(self.filename)\n            except Exception:\n                pass\n\n    def terminate(self):\n        \"\"\"Delete the underlying database file. Use with care.\"\"\"\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to terminate read-only SqliteDict')\n\n        self.close()\n\n        if self.filename == ':memory:':\n            return\n\n        logger.info(\"deleting %s\" % self.filename)\n        try:\n            if os.path.isfile(self.filename):\n                os.remove(self.filename)\n        except (OSError, IOError):\n            logger.exception(\"failed to delete %s\" % (self.filename))\n\n    def __del__(self):\n        # like close(), but assume globals are gone by now (do not log!)\n        try:\n            self.close(do_log=False, force=True)\n        except Exception:\n            # prevent error log flood in case of multiple SqliteDicts\n            # closed after connection lost (exceptions are always ignored\n            # in __del__ method.\n            pass\n\n\nclass SqliteMultithread(threading.Thread):\n    \"\"\"\n    Wrap sqlite connection in a way that allows concurrent requests from multiple threads.\n\n    This is done by internally queueing the requests and processing them sequentially\n    in a separate thread (in the same order they arrived).\n\n    \"\"\"\n    def __init__(self, filename, autocommit, journal_mode, outer_stack=True):\n        super(SqliteMultithread, self).__init__()\n        self.filename = filename\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        # use request queue of unlimited size\n        self.reqs = Queue()\n        self.daemon = True\n        self._outer_stack = outer_stack\n        self.log = logging.getLogger('sqlitedict.SqliteMultithread')\n\n        #\n        # Parts of this object's state get accessed from different threads, so\n        # we use synchronization to avoid race conditions.  For example,\n        # .exception gets set inside the new daemon thread that we spawned, but\n        # gets read from the main thread.  This is particularly important\n        # during initialization: the Thread needs some time to actually start\n        # working, and until this happens, any calls to e.g.\n        # check_raise_error() will prematurely return None, meaning all is\n        # well.  If the that connection happens to fail, we'll never know about\n        # it, and instead wait for a result that never arrives (effectively,\n        # deadlocking).  Locking solves this problem by eliminating the race\n        # condition.\n        #\n        self._lock = threading.Lock()\n        self._lock.acquire()\n        self.exception = None\n\n        self.start()\n\n    def _connect(self):\n        \"\"\"Connect to the underlying database.\n\n        Raises an exception on failure.  Returns the connection and cursor on success.\n        \"\"\"\n        try:\n            if self.autocommit:\n                conn = sqlite3.connect(self.filename, isolation_level=None, check_same_thread=False)\n            else:\n                conn = sqlite3.connect(self.filename, check_same_thread=False)\n        except Exception:\n            self.log.exception(\"Failed to initialize connection for filename: %s\" % self.filename)\n            self.exception = sys.exc_info()\n            raise\n\n        try:\n            conn.execute('PRAGMA journal_mode = %s' % self.journal_mode)\n            conn.text_factory = str\n            cursor = conn.cursor()\n            conn.commit()\n            cursor.execute('PRAGMA synchronous=OFF')\n        except Exception:\n            self.log.exception(\"Failed to execute PRAGMA statements.\")\n            self.exception = sys.exc_info()\n            raise\n\n        return conn, cursor\n\n    def run(self):\n        #\n        # Nb. this is what actually runs inside the new daemon thread.\n        # self._lock is locked at this stage - see the initializer function.\n        #\n        try:\n            conn, cursor = self._connect()\n        finally:\n            self._lock.release()\n\n        res_ref = None\n        while True:\n            #\n            # req: an SQL command or one of the --magic-- commands we use internally\n            # arg: arguments for the command\n            # res_ref: a weak reference to the queue into which responses must be placed\n            # outer_stack: the outer stack, for producing more informative traces in case of error\n            #\n            req, arg, res_ref, outer_stack = self.reqs.get()\n\n            if req == _REQUEST_CLOSE:\n                assert res_ref, ('--close-- without return queue', res_ref)\n                break\n            elif req == _REQUEST_COMMIT:\n                conn.commit()\n                _put(res_ref, _RESPONSE_NO_MORE)\n            else:\n                try:\n                    cursor.execute(req, arg)\n                except Exception:\n                    with self._lock:\n                        self.exception = (e_type, e_value, e_tb) = sys.exc_info()\n\n                    inner_stack = traceback.extract_stack()\n\n                    # An exception occurred in our thread, but we may not\n                    # immediately able to throw it in our calling thread, if it has\n                    # no return `res` queue: log as level ERROR both the inner and\n                    # outer exception immediately.\n                    #\n                    # Any iteration of res.get() or any next call will detect the\n                    # inner exception and re-raise it in the calling Thread; though\n                    # it may be confusing to see an exception for an unrelated\n                    # statement, an ERROR log statement from the 'sqlitedict.*'\n                    # namespace contains the original outer stack location.\n                    self.log.error('Inner exception:')\n                    for item in traceback.format_list(inner_stack):\n                        self.log.error(item)\n                    self.log.error('')  # deliniate traceback & exception w/blank line\n                    for item in traceback.format_exception_only(e_type, e_value):\n                        self.log.error(item)\n\n                    self.log.error('')  # exception & outer stack w/blank line\n\n                    if self._outer_stack:\n                        self.log.error('Outer stack:')\n                        for item in traceback.format_list(outer_stack):\n                            self.log.error(item)\n                        self.log.error('Exception will be re-raised at next call.')\n                    else:\n                        self.log.error(\n                            'Unable to show the outer stack. Pass '\n                            'outer_stack=True when initializing the '\n                            'SqliteDict instance to show the outer stack.'\n                        )\n\n                if res_ref:\n                    for rec in cursor:\n                        if _put(res_ref, rec) == _PUT_REFERENT_DESTROYED:\n                            #\n                            # The queue we are sending responses to got garbage\n                            # collected.  Nobody is listening anymore, so we\n                            # stop sending responses.\n                            #\n                            break\n\n                    _put(res_ref, _RESPONSE_NO_MORE)\n\n                if self.autocommit:\n                    conn.commit()\n\n        self.log.debug('received: %s, send: --no more--', req)\n        conn.close()\n\n        _put(res_ref, _RESPONSE_NO_MORE)\n\n    def check_raise_error(self):\n        \"\"\"\n        Check for and raise exception for any previous sqlite query.\n\n        For the `execute*` family of method calls, such calls are non-blocking and any\n        exception raised in the thread cannot be handled by the calling Thread (usually\n        MainThread).  This method is called on `close`, and prior to any subsequent\n        calls to the `execute*` methods to check for and raise an exception in a\n        previous call to the MainThread.\n        \"\"\"\n        with self._lock:\n            if self.exception:\n                e_type, e_value, e_tb = self.exception\n\n                # clear self.exception, if the caller decides to handle such\n                # exception, we should not repeatedly re-raise it.\n                self.exception = None\n\n                self.log.error('An exception occurred from a previous statement, view '\n                               'the logging namespace \"sqlitedict\" for outer stack.')\n\n                # The third argument to raise is the traceback object, and it is\n                # substituted instead of the current location as the place where\n                # the exception occurred, this is so that when using debuggers such\n                # as `pdb', or simply evaluating the naturally raised traceback, we\n                # retain the original (inner) location of where the exception\n                # occurred.\n                reraise(e_type, e_value, e_tb)\n\n    def execute(self, req, arg=None, res=None):\n        \"\"\"\n        `execute` calls are non-blocking: just queue up the request and return immediately.\n\n        :param req: The request (an SQL command)\n        :param arg: Arguments to the SQL command\n        :param res: A queue in which to place responses as they become available\n        \"\"\"\n        self.check_raise_error()\n        stack = None\n\n        if self._outer_stack:\n            # NOTE: This might be a lot of information to pump into an input\n            # queue, affecting performance.  I've also seen earlier versions of\n            # jython take a severe performance impact for throwing exceptions\n            # so often.\n            stack = traceback.extract_stack()[:-1]\n\n        #\n        # We pass a weak reference to the response queue instead of a regular\n        # reference, because we want the queues to be garbage-collected\n        # more aggressively.\n        #\n        res_ref = None\n        if res:\n            res_ref = weakref.ref(res)\n\n        self.reqs.put((req, arg or tuple(), res_ref, stack))\n\n    def executemany(self, req, items):\n        for item in items:\n            self.execute(req, item)\n        self.check_raise_error()\n\n    def select(self, req, arg=None):\n        \"\"\"\n        Unlike sqlite's native select, this select doesn't handle iteration efficiently.\n\n        The result of `select` starts filling up with values as soon as the\n        request is dequeued, and although you can iterate over the result normally\n        (`for res in self.select(): ...`), the entire result will be in memory.\n        \"\"\"\n        res = Queue()  # results of the select will appear as items in this queue\n        self.execute(req, arg, res)\n        while True:\n            rec = res.get()\n            self.check_raise_error()\n            if rec == _RESPONSE_NO_MORE:\n                break\n            yield rec\n\n    def select_one(self, req, arg=None):\n        \"\"\"Return only the first row of the SELECT, or None if there are no matching rows.\"\"\"\n        try:\n            return next(iter(self.select(req, arg)))\n        except StopIteration:\n            return None\n\n    def commit(self, blocking=True):\n        if blocking:\n            # by default, we await completion of commit() unless\n            # blocking=False.  This ensures any available exceptions for any\n            # previous statement are thrown before returning, and that the\n            # data has actually persisted to disk!\n            self.select_one(_REQUEST_COMMIT)\n        else:\n            # otherwise, we fire and forget as usual.\n            self.execute(_REQUEST_COMMIT)\n\n    def close(self, force=False):\n        if force:\n            # If a SqliteDict is being killed or garbage-collected, then select_one()\n            # could hang forever because run() might already have exited and therefore\n            # can't process the request. Instead, push the close command to the requests\n            # queue directly. If run() is still alive, it will exit gracefully. If not,\n            # then there's nothing we can do anyway.\n            self.reqs.put((_REQUEST_CLOSE, None, weakref.ref(Queue()), None))\n        else:\n            # we abuse 'select' to \"iter\" over a \"--close--\" statement so that we\n            # can confirm the completion of close before joining the thread and\n            # returning (by semaphore '--no more--'\n            self.select_one(_REQUEST_CLOSE)\n            self.join()\n\n\n#\n# This is here for .github/workflows/release.yml\n#\nif __name__ == '__main__':\n    print(__version__)\n"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n\n\"\"\"\nRun with:\n\npython ./setup.py install\n\"\"\"\n\nimport os\nimport io\nimport subprocess\n\nimport setuptools.command.develop\nfrom setuptools import setup\n\n\ndef read(fname):\n    path = os.path.join(os.path.dirname(__file__), fname)\n    return io.open(path, encoding='utf8').read()\n\n\nclass SetupDevelop(setuptools.command.develop.develop):\n    \"\"\"Docstring is overwritten.\"\"\"\n\n    def run(self):\n        \"\"\"\n        Prepare environment for development.\n\n        - Ensures 'nose' and 'coverage.py' are installed for testing.\n        - Call super()'s run method.\n        \"\"\"\n        subprocess.check_call(('pip', 'install', 'nose', 'coverage'))\n\n        # Call super() (except develop is an old-style class, so we must call\n        # directly). The effect is that the development egg-link is installed.\n        setuptools.command.develop.develop.run(self)\n\n\nSetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n\n\nsetup(\n    name='sqlitedict',\n    version='2.1.0',\n    description='Persistent dict in Python, backed up by sqlite3 and pickle, multithread-safe.',\n    long_description=read('README.rst'),\n\n    py_modules=['sqlitedict'],\n\n    # there is a bug in python2.5, preventing distutils from using any non-ascii characters :(\n    # http://bugs.python.org/issue2562\n    author='Radim Rehurek, Victor R. Escobar, Andrey Usov, Prasanna Swaminathan, Jeff Quast',\n    author_email=\"me@radimrehurek.com\",\n    maintainer='Radim Rehurek',\n    maintainer_email='me@radimrehurek.com',\n\n    url='https://github.com/piskvorky/sqlitedict',\n    download_url='http://pypi.python.org/pypi/sqlitedict',\n\n    keywords='sqlite, persistent dict, multithreaded',\n\n    license='Apache 2.0',\n    platforms='any',\n\n    classifiers=[  # from http://pypi.python.org/pypi?%3Aaction=list_classifiers\n        'Development Status :: 5 - Production/Stable',\n        'Environment :: Console',\n        'Intended Audience :: Developers',\n        'License :: OSI Approved :: Apache Software License',\n        'Operating System :: OS Independent',\n        'Programming Language :: Python :: 3.7',\n        'Programming Language :: Python :: 3.8',\n        'Programming Language :: Python :: 3.9',\n        'Programming Language :: Python :: 3.10',\n        'Topic :: Database :: Front-Ends',\n    ],\n    cmdclass={'develop': SetupDevelop},\n)\n"
    },
    {
      "path": "piskvorky_sqlitedict/benchmarks/test_insert.py",
      "content": "import tempfile\n\nfrom sqlitedict import SqliteDict\n\n\ndef insert():\n    with tempfile.NamedTemporaryFile() as tmp:\n        for j in range(100):\n            with SqliteDict(tmp.name) as d:\n                d[\"tmp\"] = j\n                d.commit()\n\n\ndef test(benchmark):\n    benchmark(insert)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/autocommit.py",
      "content": "import sqlitedict\n\nd = sqlitedict.SqliteDict('tests/db/autocommit.sqlite', autocommit=True)\n\nfor i in range(1000):\n    d[i] = i\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_onimport.py",
      "content": "\"\"\"Test cases for on-import logic.\"\"\"\nimport unittest\nimport sys\n\n\nclass SqliteDict_cPickleImportTest(unittest.TestCase):\n    \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n    def setUp(self):\n        self.orig_meta_path = sys.meta_path\n        self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n\n        class FauxMissingImport(object):\n            def __init__(self, *args):\n                self.module_names = args\n\n            def find_module(self, fullname, path=None):\n                if fullname in self.module_names:\n                    return self\n                return None\n\n            def load_module(self, name):\n                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n\n        # ensure cPickle/pickle is not cached\n        sys.modules.pop('cPickle', None)\n        sys.modules.pop('pickle', None)\n\n        # add our custom importer\n        sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n\n    def tearDown(self):\n        sys.meta_path = self.orig_meta_path\n        if self.orig_sqlitedict:\n            sys.modules['sqlitedict'] = self.orig_sqlitedict\n\n    def test_cpickle_fallback_to_pickle(self):\n        # exercise,\n        sqlitedict = __import__(\"sqlitedict\")\n        # verify,\n        self.assertIn('pickle', sys.modules.keys())\n        self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\nimport unittest\nimport sqlitedict\n\nfrom sys import version_info\nmajor_version = version_info[0]\n\n\nclass TempSqliteDictTest(unittest.TestCase):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict()\n\n    def tearDown(self):\n        self.d.close()\n\n    def test_create_sqlitedict(self):\n        ''' test_create_sqlitedict\n        '''\n        self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n        self.assertEqual(dict(self.d), {})\n        self.assertEqual(list(self.d), [])\n        self.assertEqual(len(self.d), 0)\n\n    def test_assign_values(self):\n        ''' test_assign_values\n        '''\n        self.d['abc'] = 'edf'\n        self.assertEqual(self.d['abc'], 'edf')\n        self.assertEqual(len(self.d), 1)\n\n    def test_clear_data(self):\n        ''' test_clear_data\n        '''\n        self.d.update(a=1, b=2, c=3)\n        self.assertEqual(len(self.d), 3)\n        self.d.clear()\n        self.assertEqual(len(self.d), 0)\n\n    def test_manage_one_record(self):\n        ''' test_manage_one_record\n        '''\n        self.d['abc'] = 'rsvp' * 100\n        self.assertEqual(self.d['abc'], 'rsvp' * 100)\n        self.d['abc'] = 'lmno'\n        self.assertEqual(self.d['abc'], 'lmno')\n        self.assertEqual(len(self.d), 1)\n        del self.d['abc']\n        self.assertEqual(len(self.d), 0)\n        self.assertTrue(not self.d)\n\n    def test_manage_few_records(self):\n        ''' test_manage_few_records\n        '''\n        self.d['abc'] = 'lmno'\n        self.d['xyz'] = 'pdq'\n        self.assertEqual(len(self.d), 2)\n        if major_version == 2:\n            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n        self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n        self.assertEqual(list(self.d), ['abc', 'xyz'])\n\n    def test_update_records(self):\n        ''' test_update_records\n        '''\n        self.d.update([('v', 'w')], p='x', q='y', r='z')\n        self.assertEqual(len(self.d), 4)\n        # As far as I know dicts does not need to return\n        # the elements in a specified order (sort() is required )\n        self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n        self.assertEqual(sorted(list(self.d)), sorted(['q', 'p', 'r', 'v']))\n\n    def test_handling_errors(self):\n        ''' test_handling_errors\n        '''\n        def get_value(d, k):\n            return d[k]\n\n        def remove_nonexists(d, k):\n            del d[k]\n\n        with self.assertRaises(KeyError):\n            remove_nonexists(self.d, 'abc')\n        with self.assertRaises(KeyError):\n            get_value(self.d, 'abc')\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "import os\nimport sys\n\nimport sqlitedict\n\n\ndef test():\n    \"Verify autocommit just before program exits.\"\n    assert os.system('env PYTHONPATH=. %s tests/autocommit.py' % sys.executable) == 0\n    # The above script relies on the autocommit feature working correctly.\n    # Now, let's check if it actually worked.\n    d = sqlitedict.SqliteDict('tests/db/autocommit.sqlite')\n    for i in range(1000):\n        assert d[i] == i, \"actual: %s expected: %s\" % (d[i], i)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_core.py",
      "content": "# std imports\nimport json\nimport unittest\nimport tempfile\nimport os\nfrom unittest.mock import patch\n\n# local\nimport sqlitedict\nfrom sqlitedict import SqliteDict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass SqliteMiscTest(unittest.TestCase):\n\n    def test_with_statement(self):\n        \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n        with SqliteDict() as d:\n            self.assertTrue(isinstance(d, SqliteDict))\n            self.assertEqual(dict(d), {})\n            self.assertEqual(list(d), [])\n            self.assertEqual(len(d), 0)\n\n    def test_reopen_conn(self):\n        \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        db = SqliteDict(filename=fname)\n        with db:\n            db['key'] = 'value'\n            db.commit()\n        with db:\n            db['key'] = 'value'\n            db.commit()\n\n    def test_as_str(self):\n        \"\"\"Verify SqliteDict.__str__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__str__()\n        # test when db closed\n        db.close()\n        db.__str__()\n\n    def test_as_repr(self):\n        \"\"\"Verify SqliteDict.__repr__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__repr__()\n\n    def test_directory_notfound(self):\n        \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n        # given: a non-existent directory,\n        folder = tempfile.mkdtemp(prefix='sqlitedict-test')\n        os.rmdir(folder)\n        # exercise,\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=os.path.join(folder, 'nonexistent'))\n\n    def test_commit_nonblocking(self):\n        \"\"\"Coverage for non-blocking commit.\"\"\"\n        # given,\n        with SqliteDict(autocommit=True) as d:\n            # exercise: the implicit commit is nonblocking\n            d['key'] = 'value'\n            d.commit(blocking=False)\n\n    def test_cancel_iterate(self):\n        import time\n\n        class EndlessKeysIterator:\n            def __init__(self) -> None:\n                self.value = 0\n\n            def __iter__(self):\n                return self\n\n            def __next__(self):\n                self.value += 1\n                return [self.value]\n\n        with patch('sqlitedict.sqlite3') as mock_sqlite3:\n            ki = EndlessKeysIterator()\n            cursor = mock_sqlite3.connect().cursor()\n            cursor.__iter__.return_value = ki\n\n            with SqliteDict(autocommit=True) as d:\n                for i, k in enumerate(d.keys()):\n                    assert i + 1 == k\n                    if k > 100:\n                        break\n                assert ki.value > 101\n\n                # Release GIL, let background threads run.\n                # Don't use gc.collect because this is simulate user code.\n                time.sleep(0.01)\n\n                current = ki.value\n                time.sleep(1)\n                assert current == ki.value, 'Will not read more after iterate stop'\n\n\nclass NamedSqliteDictCreateOrReuseTest(TempSqliteDictTest):\n    \"\"\"Verify default flag='c', and flag='n' of SqliteDict().\"\"\"\n\n    def test_default_reuse_existing_flag_c(self):\n        \"\"\"Re-opening of a database does not destroy it.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        next_db = SqliteDict(filename=fname)\n        self.assertIn('key', next_db.keys())\n        self.assertEqual(next_db['key'], 'value')\n\n    def test_overwrite_using_flag_n(self):\n        \"\"\"Re-opening of a database with flag='c' destroys it all.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname, tablename='sometable')\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        # verify,\n        next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n        self.assertNotIn('key', next_db.keys())\n        self.assertNotIn('value', next_db.values())\n\n    def test_unrecognized_flag(self):\n\n        def build_with_bad_flag():\n            fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n            SqliteDict(filename=fname, flag='FOO')\n\n        with self.assertRaises(RuntimeError):\n            build_with_bad_flag()\n\n    def test_readonly(self):\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        readonly_db = SqliteDict(filename=fname, flag='r')\n        self.assertTrue(readonly_db['key'] == 'value')\n        self.assertTrue(readonly_db['key_two'] == 2)\n\n        def attempt_write():\n            readonly_db['key'] = ['new_value']\n\n        def attempt_update():\n            readonly_db.update(key='value2', key_two=2.1)\n\n        def attempt_delete():\n            del readonly_db['key']\n\n        def attempt_clear():\n            readonly_db.clear()\n\n        def attempt_terminate():\n            readonly_db.terminate()\n\n        attempt_funcs = [attempt_write, attempt_update, attempt_delete, attempt_clear, attempt_terminate]\n\n        for func in attempt_funcs:\n            with self.assertRaises(RuntimeError):\n                func()\n\n    def test_readonly_table(self):\n        \"\"\"\n        Read-only access on a non-existent tablename should raise RuntimeError,\n        and not create a new (empty) table.\n        \"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        dummy_tablename = 'table404'\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n\n        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n    def test_irregular_tablenames(self):\n        \"\"\"Irregular table names need to be quoted\"\"\"\n        def __test_irregular_tablenames(tablename):\n            filename = ':memory:'\n            db = SqliteDict(filename, tablename=tablename)\n            db['key'] = 'value'\n            db.commit()\n            self.assertEqual(db['key'], 'value')\n            db.close()\n\n        __test_irregular_tablenames('9nine')\n        __test_irregular_tablenames('outer space')\n        __test_irregular_tablenames('table with a \"quoted\" name')\n        __test_irregular_tablenames(\"table with a \\\"quoted \\xe1cute\\\" name\")\n\n    def test_overwrite_using_flag_w(self):\n        \"\"\"Re-opening of a database with flag='w' destroys only the target table.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db_1 = SqliteDict(filename=fname, tablename='one')\n        orig_db_1['key'] = 'value'\n        orig_db_1.commit()\n        orig_db_1.close()\n\n        orig_db_2 = SqliteDict(filename=fname, tablename='two')\n        orig_db_2['key'] = 'value'\n        orig_db_2.commit()\n        orig_db_2.close()\n\n        # verify, when re-opening table space 'one' with flag='2', we destroy\n        # its contents.  However, when re-opening table space 'two' with\n        # default flag='r', its contents remain.\n        next_db_1 = SqliteDict(filename=fname, tablename='one', flag='w')\n        self.assertNotIn('key', next_db_1.keys())\n\n        next_db_2 = SqliteDict(filename=fname, tablename='two')\n        self.assertIn('key', next_db_2.keys())\n\n\nclass SqliteDictTerminateTest(unittest.TestCase):\n\n    def test_terminate_instead_close(self):\n        ''' make terminate() instead of close()\n        '''\n        d = sqlitedict.open('tests/db/sqlitedict-terminate.sqlite')\n        d['abc'] = 'def'\n        d.commit()\n        self.assertEqual(d['abc'], 'def')\n        d.terminate()\n        self.assertFalse(os.path.isfile('tests/db/sqlitedict-terminate.sqlite'))\n\n\nclass SqliteDictTerminateFailTest(unittest.TestCase):\n    \"\"\"Provide Coverage for SqliteDict.terminate().\"\"\"\n\n    def setUp(self):\n        self.fname = norm_file('tests/db-permdenied/sqlitedict.sqlite')\n        self.db = SqliteDict(filename=self.fname)\n        os.chmod(self.fname, 0o000)\n        os.chmod(os.path.dirname(self.fname), 0o000)\n\n    def tearDown(self):\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def test_terminate_cannot_delete(self):\n        # exercise,\n        self.db.terminate()  # deletion failed, but no exception raised!\n\n        # verify,\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        self.assertTrue(os.path.exists(self.fname))\n\n\nclass SqliteDictJsonSerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-json/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test', encode=json.dumps, decode=json.loads\n        )\n\n    def tearDown(self):\n        self.db.close()\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def get_json(self, key):\n        return self.db.conn.select_one('SELECT value FROM test WHERE key = ?', (self.db.encode_key(key),))[0]\n\n    def test_int(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        assert self.get_json('test') == '-42'\n\n    def test_str(self):\n        test_str = u'Test \\u30c6\\u30b9\\u30c8'\n        self.db['test'] = test_str\n        assert self.db['test'] == test_str\n        assert self.get_json('test') == r'\"Test \\u30c6\\u30b9\\u30c8\"'\n\n    def test_bool(self):\n        self.db['test'] = False\n        assert self.db['test'] is False\n        assert self.get_json('test') == 'false'\n\n    def test_none(self):\n        self.db['test'] = None\n        assert self.db['test'] is None\n        assert self.get_json('test') == 'null'\n\n    def test_complex_struct(self):\n        test_value = {\n            'version': 2.5,\n            'items': ['one', 'two'],\n        }\n        self.db['test'] = test_value\n        assert self.db['test'] == test_value\n        assert self.get_json('test') == json.dumps(test_value)\n\n\nclass TablenamesTest(unittest.TestCase):\n    def tearDown(self):\n        for f in ('tablenames-test-1.sqlite', 'tablenames-test-2.sqlite'):\n            path = norm_file(os.path.join('tests/db', f))\n            if os.path.isfile(path):\n                os.unlink(path)\n\n    def test_tablenames_unnamed(self):\n        fname = norm_file('tests/db/tablenames-test-1.sqlite')\n        SqliteDict(fname)\n        self.assertEqual(SqliteDict.get_tablenames(fname), ['unnamed'])\n\n    def test_tablenams_named(self):\n        fname = norm_file('tests/db/tablenames-test-2.sqlite')\n        with SqliteDict(fname, tablename='table1'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1'])\n        with SqliteDict(fname, tablename='table2'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1', 'table2'])\n\n        tablenames = SqliteDict.get_tablenames('tests/db/tablenames-test-2.sqlite')\n        self.assertEqual(tablenames, ['table1', 'table2'])\n\n\nclass SqliteDictKeySerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-encode-key/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test',\n            encode_key=sqlitedict.encode_key, decode_key=sqlitedict.decode_key,\n        )\n\n    def test_nonstr_keys(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        self.db[(0, 1, 2)] = 17\n        assert self.db[(0, 1, 2)] == 17\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_named_db.py",
      "content": "import sqlitedict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass InMemorySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass NamedSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-def.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db)\n\n\nclass CreateNewSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-n-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass StartsWithEmptySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-w-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"w\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass SqliteDictAutocommitTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-autocommit.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/accessories.py",
      "content": "\"\"\"Accessories for test cases.\"\"\"\nimport os\n\n\ndef norm_file(fname):\n    \"\"\"Normalize test filename, creating a directory path to it if necessary\"\"\"\n    fname = os.path.abspath(fname)\n    dirname = os.path.dirname(fname)\n    if not os.path.exists(dirname):\n        os.makedirs(dirname)\n    return fname\n"
    },
    {
      "path": "piskvorky_sqlitedict/release/summarize_pr.py",
      "content": "#!/usr/bin/env python\nimport json\nimport sys\nimport urllib.request\n\n\ndef copy_to_clipboard(text):\n    try:\n        import pyperclip\n    except ImportError:\n        print('pyperclip <https://pypi.org/project/pyperclip/> is missing.', file=sys.stderr)\n        print('copy-paste the following text manually:', file=sys.stderr)\n        print('  ' + text, file=sys.stderr)\n    else:\n        pyperclip.copy(text)\n\n\nfor prid in sys.argv[1:]:\n    url = \"https://api.github.com/repos/RaRe-Technologies/sqlitedict/pulls/%s\" % prid\n    with urllib.request.urlopen(url) as fin:\n        prinfo = json.load(fin)\n\n    prinfo['user_login'] = prinfo['user']['login']\n    prinfo['user_html_url'] = prinfo['user']['html_url']\n    text = '- %(title)s (PR [#%(number)s](%(html_url)s), [@%(user_login)s](%(user_html_url)s))' % prinfo\n    print(text)\n"
    }
  ],
  "OriginCode": [
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n#\n# This code was inspired by:\n#  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n#  * http://code.activestate.com/recipes/526618/\n\n\"\"\"\nA lightweight wrapper around Python's sqlite3 database, with a dict-like interface\nand multi-thread access support::\n\n>>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n>>> mydict['some_key'] = any_picklable_object\n>>> print mydict['some_key']\n>>> print len(mydict) # etc... all dict functions work\n\nPickle is used internally to serialize the values. Keys are strings.\n\nIf you don't use autocommit (default is no autocommit for performance), then\ndon't forget to call `mydict.commit()` when done with a transaction.\n\n\"\"\"\n\nimport sqlite3\nimport os\nimport sys\nimport tempfile\nimport threading\nimport logging\nimport traceback\nfrom base64 import b64decode, b64encode\nimport weakref\n\n__version__ = '2.1.0'\n\n\ndef reraise(tp, value, tb=None):\n    if value is None:\n        value = tp()\n    if value.__traceback__ is not tb:\n        raise value.with_traceback(tb)\n    raise value\n\n\ntry:\n    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\nexcept ImportError:\n    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n\n# some Python 3 vs 2 imports\ntry:\n    from collections import UserDict as DictClass\nexcept ImportError:\n    from UserDict import DictMixin as DictClass\n\ntry:\n    from queue import Queue\nexcept ImportError:\n    from Queue import Queue\n\n\nlogger = logging.getLogger(__name__)\n\n#\n# There's a thread that holds the actual SQL connection (SqliteMultithread).\n# We communicate with this thread via queues (request and responses).\n# The requests can either be SQL commands or one of the \"special\" commands\n# below:\n#\n# _REQUEST_CLOSE: request that the SQL connection be closed\n# _REQUEST_COMMIT: request that any changes be committed to the DB\n#\n# Responses are either SQL records (e.g. results of a SELECT) or the magic\n# _RESPONSE_NO_MORE command, which indicates nothing else will ever be written\n# to the response queue.\n#\n_REQUEST_CLOSE = '--close--'\n_REQUEST_COMMIT = '--commit--'\n_RESPONSE_NO_MORE = '--no more--'\n\n#\n# We work with weak references for better memory efficiency.\n# Dereferencing, checking the referent queue still exists, and putting to it\n# is boring and repetitive, so we have a _put function to handle it for us.\n#\n_PUT_OK, _PUT_REFERENT_DESTROYED, _PUT_NOOP = 0, 1, 2\n\n\ndef _put(queue_reference, item):\n    if queue_reference is not None:\n        queue = queue_reference()\n        if queue is None:\n            #\n            # We got a reference to a queue, but that queue no longer exists\n            #\n            retval = _PUT_REFERENT_DESTROYED\n        else:\n            queue.put(item)\n            retval = _PUT_OK\n\n        del queue\n        return retval\n\n    #\n    # We didn't get a reference to a queue, so do nothing (no-op).\n    #\n    return _PUT_NOOP\n\n\ndef open(*args, **kwargs):\n    \"\"\"See documentation of the SqliteDict class.\"\"\"\n    return SqliteDict(*args, **kwargs)\n\n\ndef encode(obj):\n    \"\"\"Serialize an object using pickle to a binary format accepted by SQLite.\"\"\"\n    return sqlite3.Binary(dumps(obj, protocol=PICKLE_PROTOCOL))\n\n\ndef decode(obj):\n    \"\"\"Deserialize objects retrieved from SQLite.\"\"\"\n    return loads(bytes(obj))\n\n\ndef encode_key(key):\n    \"\"\"Serialize a key using pickle + base64 encoding to text accepted by SQLite.\"\"\"\n    return b64encode(dumps(key, protocol=PICKLE_PROTOCOL)).decode(\"ascii\")\n\n\ndef decode_key(key):\n    \"\"\"Deserialize a key retrieved from SQLite.\"\"\"\n    return loads(b64decode(key.encode(\"ascii\")))\n\n\ndef identity(obj):\n    \"\"\"Identity f(x) = x function for encoding/decoding.\"\"\"\n    return obj\n\n\nclass SqliteDict(DictClass):\n    VALID_FLAGS = ['c', 'r', 'w', 'n']\n\n    def __init__(self, filename=None, tablename='unnamed', flag='c',\n                 autocommit=False, journal_mode=\"DELETE\", encode=encode,\n                 decode=decode, encode_key=identity, decode_key=identity,\n                 timeout=5, outer_stack=True):\n        \"\"\"\n        Initialize a thread-safe sqlite-backed dictionary. The dictionary will\n        be a table `tablename` in database file `filename`. A single file (=database)\n        may contain multiple tables.\n\n        If no `filename` is given, a random file in temp will be used (and deleted\n        from temp once the dict is closed/deleted).\n\n        If you enable `autocommit`, changes will be committed after each operation\n        (more inefficient but safer). Otherwise, changes are committed on `self.commit()`,\n        `self.clear()` and `self.close()`.\n\n        Set `journal_mode` to 'OFF' if you're experiencing sqlite I/O problems\n        or if you need performance and don't care about crash-consistency.\n\n        Set `outer_stack` to False to disable the output of the outer exception\n        to the error logs.  This may improve the efficiency of sqlitedict\n        operation at the expense of a detailed exception trace.\n\n        The `flag` parameter. Exactly one of:\n          'c': default mode, open for read/write, creating the db/table if necessary.\n          'w': open for r/w, but drop `tablename` contents first (start with empty table)\n          'r': open as read-only\n          'n': create a new database (erasing any existing tables, not just `tablename`!).\n\n        The `encode` and `decode` parameters are used to customize how the values\n        are serialized and deserialized.\n        The `encode` parameter must be a function that takes a single Python\n        object and returns a serialized representation.\n        The `decode` function must be a function that takes the serialized\n        representation produced by `encode` and returns a deserialized Python\n        object.\n        The default is to use pickle.\n\n        The `timeout` defines the maximum time (in seconds) to wait for initial Thread startup.\n\n        \"\"\"\n        self.in_temp = filename is None\n        if self.in_temp:\n            fd, filename = tempfile.mkstemp(prefix='sqldict')\n            os.close(fd)\n\n        if flag not in SqliteDict.VALID_FLAGS:\n            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n        self.flag = flag\n\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n\n        dirname = os.path.dirname(filename)\n        if dirname:\n            if not os.path.exists(dirname):\n                raise RuntimeError('Error! The directory does not exist, %s' % dirname)\n\n        self.filename = filename\n\n        # Use standard SQL escaping of double quote characters in identifiers, by doubling them.\n        # See https://github.com/RaRe-Technologies/sqlitedict/pull/113\n        self.tablename = tablename.replace('\"', '\"\"')\n\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        self.encode = encode\n        self.decode = decode\n        self.encode_key = encode_key\n        self.decode_key = decode_key\n        self._outer_stack = outer_stack\n\n        logger.debug(\"opening Sqlite table %r in %r\" % (tablename, filename))\n        self.conn = self._new_conn()\n        if self.flag == 'r':\n            if self.tablename not in SqliteDict.get_tablenames(self.filename):\n                msg = 'Refusing to create a new table \"%s\" in read-only DB mode' % tablename\n                raise RuntimeError(msg)\n        else:\n            MAKE_TABLE = 'CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % self.tablename\n            self.conn.execute(MAKE_TABLE)\n            self.conn.commit()\n        if flag == 'w':\n            self.clear()\n\n    def _new_conn(self):\n        return SqliteMultithread(\n            self.filename,\n            autocommit=self.autocommit,\n            journal_mode=self.journal_mode,\n            outer_stack=self._outer_stack,\n        )\n\n    def __enter__(self):\n        if not hasattr(self, 'conn') or self.conn is None:\n            self.conn = self._new_conn()\n        return self\n\n    def __exit__(self, *exc_info):\n        self.close()\n\n    def __str__(self):\n        return \"SqliteDict(%s)\" % (self.filename)\n\n    def __repr__(self):\n        return str(self)  # no need of something complex\n\n    def __len__(self):\n        # `select count (*)` is super slow in sqlite (does a linear scan!!)\n        # As a result, len() is very slow too once the table size grows beyond trivial.\n        # We could keep the total count of rows ourselves, by means of triggers,\n        # but that seems too complicated and would slow down normal operation\n        # (insert/delete etc).\n        GET_LEN = 'SELECT COUNT(*) FROM \"%s\"' % self.tablename\n        rows = self.conn.select_one(GET_LEN)[0]\n        return rows if rows is not None else 0\n\n    def __bool__(self):\n        # No elements is False, otherwise True\n        GET_MAX = 'SELECT MAX(ROWID) FROM \"%s\"' % self.tablename\n        m = self.conn.select_one(GET_MAX)[0]\n        # Explicit better than implicit and bla bla\n        return True if m is not None else False\n\n    def iterkeys(self):\n        GET_KEYS = 'SELECT key FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key in self.conn.select(GET_KEYS):\n            yield self.decode_key(key[0])\n\n    def itervalues(self):\n        GET_VALUES = 'SELECT value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for value in self.conn.select(GET_VALUES):\n            yield self.decode(value[0])\n\n    def iteritems(self):\n        GET_ITEMS = 'SELECT key, value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key, value in self.conn.select(GET_ITEMS):\n            yield self.decode_key(key), self.decode(value)\n\n    def keys(self):\n        return self.iterkeys()\n\n    def values(self):\n        return self.itervalues()\n\n    def items(self):\n        return self.iteritems()\n\n    def __contains__(self, key):\n        HAS_ITEM = 'SELECT 1 FROM \"%s\" WHERE key = ?' % self.tablename\n        return self.conn.select_one(HAS_ITEM, (self.encode_key(key),)) is not None\n\n    def __getitem__(self, key):\n        GET_ITEM = 'SELECT value FROM \"%s\" WHERE key = ?' % self.tablename\n        item = self.conn.select_one(GET_ITEM, (self.encode_key(key),))\n        if item is None:\n            raise KeyError(key)\n        return self.decode(item[0])\n\n    def __setitem__(self, key, value):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to write to read-only SqliteDict')\n\n        ADD_ITEM = 'REPLACE INTO \"%s\" (key, value) VALUES (?,?)' % self.tablename\n        self.conn.execute(ADD_ITEM, (self.encode_key(key), self.encode(value)))\n        if self.autocommit:\n            self.commit()\n\n    def __delitem__(self, key):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to delete from read-only SqliteDict')\n\n        if key not in self:\n            raise KeyError(key)\n        DEL_ITEM = 'DELETE FROM \"%s\" WHERE key = ?' % self.tablename\n        self.conn.execute(DEL_ITEM, (self.encode_key(key),))\n        if self.autocommit:\n            self.commit()\n\n    def update(self, items=(), **kwds):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to update read-only SqliteDict')\n\n        try:\n            items = items.items()\n        except AttributeError:\n            pass\n        items = [(self.encode_key(k), self.encode(v)) for k, v in items]\n\n        UPDATE_ITEMS = 'REPLACE INTO \"%s\" (key, value) VALUES (?, ?)' % self.tablename\n        self.conn.executemany(UPDATE_ITEMS, items)\n        if kwds:\n            self.update(kwds)\n        if self.autocommit:\n            self.commit()\n\n    def __iter__(self):\n        return self.iterkeys()\n\n    def clear(self):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to clear read-only SqliteDict')\n\n        # avoid VACUUM, as it gives \"OperationalError: database schema has changed\"\n        CLEAR_ALL = 'DELETE FROM \"%s\";' % self.tablename\n        self.conn.commit()\n        self.conn.execute(CLEAR_ALL)\n        self.conn.commit()\n\n    @staticmethod\n    def get_tablenames(filename):\n        \"\"\"get the names of the tables in an sqlite db as a list\"\"\"\n        if not os.path.isfile(filename):\n            raise IOError('file %s does not exist' % (filename))\n        GET_TABLENAMES = 'SELECT name FROM sqlite_master WHERE type=\"table\"'\n        with sqlite3.connect(filename) as conn:\n            cursor = conn.execute(GET_TABLENAMES)\n            res = cursor.fetchall()\n\n        return [name[0] for name in res]\n\n    def commit(self, blocking=True):\n        \"\"\"\n        Persist all data to disk.\n\n        When `blocking` is False, the commit command is queued, but the data is\n        not guaranteed persisted (default implication when autocommit=True).\n        \"\"\"\n        if self.conn is not None:\n            self.conn.commit(blocking)\n    sync = commit\n\n    def close(self, do_log=True, force=False):\n        if do_log:\n            logger.debug(\"closing %s\" % self)\n        if hasattr(self, 'conn') and self.conn is not None:\n            if self.conn.autocommit and not force:\n                # typically calls to commit are non-blocking when autocommit is\n                # used.  However, we need to block on close() to ensure any\n                # awaiting exceptions are handled and that all data is\n                # persisted to disk before returning.\n                self.conn.commit(blocking=True)\n            self.conn.close(force=force)\n            self.conn = None\n        if self.in_temp:\n            try:\n                os.remove(self.filename)\n            except Exception:\n                pass\n\n    def terminate(self):\n        \"\"\"Delete the underlying database file. Use with care.\"\"\"\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to terminate read-only SqliteDict')\n\n        self.close()\n\n        if self.filename == ':memory:':\n            return\n\n        logger.info(\"deleting %s\" % self.filename)\n        try:\n            if os.path.isfile(self.filename):\n                os.remove(self.filename)\n        except (OSError, IOError):\n            logger.exception(\"failed to delete %s\" % (self.filename))\n\n    def __del__(self):\n        # like close(), but assume globals are gone by now (do not log!)\n        try:\n            self.close(do_log=False, force=True)\n        except Exception:\n            # prevent error log flood in case of multiple SqliteDicts\n            # closed after connection lost (exceptions are always ignored\n            # in __del__ method.\n            pass\n\n\nclass SqliteMultithread(threading.Thread):\n    \"\"\"\n    Wrap sqlite connection in a way that allows concurrent requests from multiple threads.\n\n    This is done by internally queueing the requests and processing them sequentially\n    in a separate thread (in the same order they arrived).\n\n    \"\"\"\n    def __init__(self, filename, autocommit, journal_mode, outer_stack=True):\n        super(SqliteMultithread, self).__init__()\n        self.filename = filename\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        # use request queue of unlimited size\n        self.reqs = Queue()\n        self.daemon = True\n        self._outer_stack = outer_stack\n        self.log = logging.getLogger('sqlitedict.SqliteMultithread')\n\n        #\n        # Parts of this object's state get accessed from different threads, so\n        # we use synchronization to avoid race conditions.  For example,\n        # .exception gets set inside the new daemon thread that we spawned, but\n        # gets read from the main thread.  This is particularly important\n        # during initialization: the Thread needs some time to actually start\n        # working, and until this happens, any calls to e.g.\n        # check_raise_error() will prematurely return None, meaning all is\n        # well.  If the that connection happens to fail, we'll never know about\n        # it, and instead wait for a result that never arrives (effectively,\n        # deadlocking).  Locking solves this problem by eliminating the race\n        # condition.\n        #\n        self._lock = threading.Lock()\n        self._lock.acquire()\n        self.exception = None\n\n        self.start()\n\n    def _connect(self):\n        \"\"\"Connect to the underlying database.\n\n        Raises an exception on failure.  Returns the connection and cursor on success.\n        \"\"\"\n        try:\n            if self.autocommit:\n                conn = sqlite3.connect(self.filename, isolation_level=None, check_same_thread=False)\n            else:\n                conn = sqlite3.connect(self.filename, check_same_thread=False)\n        except Exception:\n            self.log.exception(\"Failed to initialize connection for filename: %s\" % self.filename)\n            self.exception = sys.exc_info()\n            raise\n\n        try:\n            conn.execute('PRAGMA journal_mode = %s' % self.journal_mode)\n            conn.text_factory = str\n            cursor = conn.cursor()\n            conn.commit()\n            cursor.execute('PRAGMA synchronous=OFF')\n        except Exception:\n            self.log.exception(\"Failed to execute PRAGMA statements.\")\n            self.exception = sys.exc_info()\n            raise\n\n        return conn, cursor\n\n    def run(self):\n        #\n        # Nb. this is what actually runs inside the new daemon thread.\n        # self._lock is locked at this stage - see the initializer function.\n        #\n        try:\n            conn, cursor = self._connect()\n        finally:\n            self._lock.release()\n\n        res_ref = None\n        while True:\n            #\n            # req: an SQL command or one of the --magic-- commands we use internally\n            # arg: arguments for the command\n            # res_ref: a weak reference to the queue into which responses must be placed\n            # outer_stack: the outer stack, for producing more informative traces in case of error\n            #\n            req, arg, res_ref, outer_stack = self.reqs.get()\n\n            if req == _REQUEST_CLOSE:\n                assert res_ref, ('--close-- without return queue', res_ref)\n                break\n            elif req == _REQUEST_COMMIT:\n                conn.commit()\n                _put(res_ref, _RESPONSE_NO_MORE)\n            else:\n                try:\n                    cursor.execute(req, arg)\n                except Exception:\n                    with self._lock:\n                        self.exception = (e_type, e_value, e_tb) = sys.exc_info()\n\n                    inner_stack = traceback.extract_stack()\n\n                    # An exception occurred in our thread, but we may not\n                    # immediately able to throw it in our calling thread, if it has\n                    # no return `res` queue: log as level ERROR both the inner and\n                    # outer exception immediately.\n                    #\n                    # Any iteration of res.get() or any next call will detect the\n                    # inner exception and re-raise it in the calling Thread; though\n                    # it may be confusing to see an exception for an unrelated\n                    # statement, an ERROR log statement from the 'sqlitedict.*'\n                    # namespace contains the original outer stack location.\n                    self.log.error('Inner exception:')\n                    for item in traceback.format_list(inner_stack):\n                        self.log.error(item)\n                    self.log.error('')  # deliniate traceback & exception w/blank line\n                    for item in traceback.format_exception_only(e_type, e_value):\n                        self.log.error(item)\n\n                    self.log.error('')  # exception & outer stack w/blank line\n\n                    if self._outer_stack:\n                        self.log.error('Outer stack:')\n                        for item in traceback.format_list(outer_stack):\n                            self.log.error(item)\n                        self.log.error('Exception will be re-raised at next call.')\n                    else:\n                        self.log.error(\n                            'Unable to show the outer stack. Pass '\n                            'outer_stack=True when initializing the '\n                            'SqliteDict instance to show the outer stack.'\n                        )\n\n                if res_ref:\n                    for rec in cursor:\n                        if _put(res_ref, rec) == _PUT_REFERENT_DESTROYED:\n                            #\n                            # The queue we are sending responses to got garbage\n                            # collected.  Nobody is listening anymore, so we\n                            # stop sending responses.\n                            #\n                            break\n\n                    _put(res_ref, _RESPONSE_NO_MORE)\n\n                if self.autocommit:\n                    conn.commit()\n\n        self.log.debug('received: %s, send: --no more--', req)\n        conn.close()\n\n        _put(res_ref, _RESPONSE_NO_MORE)\n\n    def check_raise_error(self):\n        \"\"\"\n        Check for and raise exception for any previous sqlite query.\n\n        For the `execute*` family of method calls, such calls are non-blocking and any\n        exception raised in the thread cannot be handled by the calling Thread (usually\n        MainThread).  This method is called on `close`, and prior to any subsequent\n        calls to the `execute*` methods to check for and raise an exception in a\n        previous call to the MainThread.\n        \"\"\"\n        with self._lock:\n            if self.exception:\n                e_type, e_value, e_tb = self.exception\n\n                # clear self.exception, if the caller decides to handle such\n                # exception, we should not repeatedly re-raise it.\n                self.exception = None\n\n                self.log.error('An exception occurred from a previous statement, view '\n                               'the logging namespace \"sqlitedict\" for outer stack.')\n\n                # The third argument to raise is the traceback object, and it is\n                # substituted instead of the current location as the place where\n                # the exception occurred, this is so that when using debuggers such\n                # as `pdb', or simply evaluating the naturally raised traceback, we\n                # retain the original (inner) location of where the exception\n                # occurred.\n                reraise(e_type, e_value, e_tb)\n\n    def execute(self, req, arg=None, res=None):\n        \"\"\"\n        `execute` calls are non-blocking: just queue up the request and return immediately.\n\n        :param req: The request (an SQL command)\n        :param arg: Arguments to the SQL command\n        :param res: A queue in which to place responses as they become available\n        \"\"\"\n        self.check_raise_error()\n        stack = None\n\n        if self._outer_stack:\n            # NOTE: This might be a lot of information to pump into an input\n            # queue, affecting performance.  I've also seen earlier versions of\n            # jython take a severe performance impact for throwing exceptions\n            # so often.\n            stack = traceback.extract_stack()[:-1]\n\n        #\n        # We pass a weak reference to the response queue instead of a regular\n        # reference, because we want the queues to be garbage-collected\n        # more aggressively.\n        #\n        res_ref = None\n        if res:\n            res_ref = weakref.ref(res)\n\n        self.reqs.put((req, arg or tuple(), res_ref, stack))\n\n    def executemany(self, req, items):\n        for item in items:\n            self.execute(req, item)\n        self.check_raise_error()\n\n    def select(self, req, arg=None):\n        \"\"\"\n        Unlike sqlite's native select, this select doesn't handle iteration efficiently.\n\n        The result of `select` starts filling up with values as soon as the\n        request is dequeued, and although you can iterate over the result normally\n        (`for res in self.select(): ...`), the entire result will be in memory.\n        \"\"\"\n        res = Queue()  # results of the select will appear as items in this queue\n        self.execute(req, arg, res)\n        while True:\n            rec = res.get()\n            self.check_raise_error()\n            if rec == _RESPONSE_NO_MORE:\n                break\n            yield rec\n\n    def select_one(self, req, arg=None):\n        \"\"\"Return only the first row of the SELECT, or None if there are no matching rows.\"\"\"\n        try:\n            return next(iter(self.select(req, arg)))\n        except StopIteration:\n            return None\n\n    def commit(self, blocking=True):\n        if blocking:\n            # by default, we await completion of commit() unless\n            # blocking=False.  This ensures any available exceptions for any\n            # previous statement are thrown before returning, and that the\n            # data has actually persisted to disk!\n            self.select_one(_REQUEST_COMMIT)\n        else:\n            # otherwise, we fire and forget as usual.\n            self.execute(_REQUEST_COMMIT)\n\n    def close(self, force=False):\n        if force:\n            # If a SqliteDict is being killed or garbage-collected, then select_one()\n            # could hang forever because run() might already have exited and therefore\n            # can't process the request. Instead, push the close command to the requests\n            # queue directly. If run() is still alive, it will exit gracefully. If not,\n            # then there's nothing we can do anyway.\n            self.reqs.put((_REQUEST_CLOSE, None, weakref.ref(Queue()), None))\n        else:\n            # we abuse 'select' to \"iter\" over a \"--close--\" statement so that we\n            # can confirm the completion of close before joining the thread and\n            # returning (by semaphore '--no more--'\n            self.select_one(_REQUEST_CLOSE)\n            self.join()\n\n\n#\n# This is here for .github/workflows/release.yml\n#\nif __name__ == '__main__':\n    print(__version__)\n"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n\n\"\"\"\nRun with:\n\npython ./setup.py install\n\"\"\"\n\nimport os\nimport io\nimport subprocess\n\nimport setuptools.command.develop\nfrom setuptools import setup\n\n\ndef read(fname):\n    path = os.path.join(os.path.dirname(__file__), fname)\n    return io.open(path, encoding='utf8').read()\n\n\nclass SetupDevelop(setuptools.command.develop.develop):\n    \"\"\"Docstring is overwritten.\"\"\"\n\n    def run(self):\n        \"\"\"\n        Prepare environment for development.\n\n        - Ensures 'nose' and 'coverage.py' are installed for testing.\n        - Call super()'s run method.\n        \"\"\"\n        subprocess.check_call(('pip', 'install', 'nose', 'coverage'))\n\n        # Call super() (except develop is an old-style class, so we must call\n        # directly). The effect is that the development egg-link is installed.\n        setuptools.command.develop.develop.run(self)\n\n\nSetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n\n\nsetup(\n    name='sqlitedict',\n    version='2.1.0',\n    description='Persistent dict in Python, backed up by sqlite3 and pickle, multithread-safe.',\n    long_description=read('README.rst'),\n\n    py_modules=['sqlitedict'],\n\n    # there is a bug in python2.5, preventing distutils from using any non-ascii characters :(\n    # http://bugs.python.org/issue2562\n    author='Radim Rehurek, Victor R. Escobar, Andrey Usov, Prasanna Swaminathan, Jeff Quast',\n    author_email=\"me@radimrehurek.com\",\n    maintainer='Radim Rehurek',\n    maintainer_email='me@radimrehurek.com',\n\n    url='https://github.com/piskvorky/sqlitedict',\n    download_url='http://pypi.python.org/pypi/sqlitedict',\n\n    keywords='sqlite, persistent dict, multithreaded',\n\n    license='Apache 2.0',\n    platforms='any',\n\n    classifiers=[  # from http://pypi.python.org/pypi?%3Aaction=list_classifiers\n        'Development Status :: 5 - Production/Stable',\n        'Environment :: Console',\n        'Intended Audience :: Developers',\n        'License :: OSI Approved :: Apache Software License',\n        'Operating System :: OS Independent',\n        'Programming Language :: Python :: 3.7',\n        'Programming Language :: Python :: 3.8',\n        'Programming Language :: Python :: 3.9',\n        'Programming Language :: Python :: 3.10',\n        'Topic :: Database :: Front-Ends',\n    ],\n    cmdclass={'develop': SetupDevelop},\n)\n"
    },
    {
      "path": "piskvorky_sqlitedict/benchmarks/test_insert.py",
      "content": "import tempfile\n\nfrom sqlitedict import SqliteDict\n\n\ndef insert():\n    with tempfile.NamedTemporaryFile() as tmp:\n        for j in range(100):\n            with SqliteDict(tmp.name) as d:\n                d[\"tmp\"] = j\n                d.commit()\n\n\ndef test(benchmark):\n    benchmark(insert)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/autocommit.py",
      "content": "import sqlitedict\n\nd = sqlitedict.SqliteDict('tests/db/autocommit.sqlite', autocommit=True)\n\nfor i in range(1000):\n    d[i] = i\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_onimport.py",
      "content": "\"\"\"Test cases for on-import logic.\"\"\"\nimport unittest\nimport sys\n\n\nclass SqliteDict_cPickleImportTest(unittest.TestCase):\n    \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n    def setUp(self):\n        self.orig_meta_path = sys.meta_path\n        self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n\n        class FauxMissingImport(object):\n            def __init__(self, *args):\n                self.module_names = args\n\n            def find_module(self, fullname, path=None):\n                if fullname in self.module_names:\n                    return self\n                return None\n\n            def load_module(self, name):\n                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n\n        # ensure cPickle/pickle is not cached\n        sys.modules.pop('cPickle', None)\n        sys.modules.pop('pickle', None)\n\n        # add our custom importer\n        sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n\n    def tearDown(self):\n        sys.meta_path = self.orig_meta_path\n        if self.orig_sqlitedict:\n            sys.modules['sqlitedict'] = self.orig_sqlitedict\n\n    def test_cpickle_fallback_to_pickle(self):\n        # exercise,\n        sqlitedict = __import__(\"sqlitedict\")\n        # verify,\n        self.assertIn('pickle', sys.modules.keys())\n        self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\nimport unittest\nimport sqlitedict\n\nfrom sys import version_info\nmajor_version = version_info[0]\n\n\nclass TempSqliteDictTest(unittest.TestCase):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict()\n\n    def tearDown(self):\n        self.d.close()\n\n    def test_create_sqlitedict(self):\n        ''' test_create_sqlitedict\n        '''\n        self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n        self.assertEqual(dict(self.d), {})\n        self.assertEqual(list(self.d), [])\n        self.assertEqual(len(self.d), 0)\n\n    def test_assign_values(self):\n        ''' test_assign_values\n        '''\n        self.d['abc'] = 'edf'\n        self.assertEqual(self.d['abc'], 'edf')\n        self.assertEqual(len(self.d), 1)\n\n    def test_clear_data(self):\n        ''' test_clear_data\n        '''\n        self.d.update(a=1, b=2, c=3)\n        self.assertEqual(len(self.d), 3)\n        self.d.clear()\n        self.assertEqual(len(self.d), 0)\n\n    def test_manage_one_record(self):\n        ''' test_manage_one_record\n        '''\n        self.d['abc'] = 'rsvp' * 100\n        self.assertEqual(self.d['abc'], 'rsvp' * 100)\n        self.d['abc'] = 'lmno'\n        self.assertEqual(self.d['abc'], 'lmno')\n        self.assertEqual(len(self.d), 1)\n        del self.d['abc']\n        self.assertEqual(len(self.d), 0)\n        self.assertTrue(not self.d)\n\n    def test_manage_few_records(self):\n        ''' test_manage_few_records\n        '''\n        self.d['abc'] = 'lmno'\n        self.d['xyz'] = 'pdq'\n        self.assertEqual(len(self.d), 2)\n        if major_version == 2:\n            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n        self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n        self.assertEqual(list(self.d), ['abc', 'xyz'])\n\n    def test_update_records(self):\n        ''' test_update_records\n        '''\n        self.d.update([('v', 'w')], p='x', q='y', r='z')\n        self.assertEqual(len(self.d), 4)\n        # As far as I know dicts does not need to return\n        # the elements in a specified order (sort() is required )\n        self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n        self.assertEqual(sorted(list(self.d)), sorted(['q', 'p', 'r', 'v']))\n\n    def test_handling_errors(self):\n        ''' test_handling_errors\n        '''\n        def get_value(d, k):\n            return d[k]\n\n        def remove_nonexists(d, k):\n            del d[k]\n\n        with self.assertRaises(KeyError):\n            remove_nonexists(self.d, 'abc')\n        with self.assertRaises(KeyError):\n            get_value(self.d, 'abc')\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "import os\nimport sys\n\nimport sqlitedict\n\n\ndef test():\n    \"Verify autocommit just before program exits.\"\n    assert os.system('env PYTHONPATH=. %s tests/autocommit.py' % sys.executable) == 0\n    # The above script relies on the autocommit feature working correctly.\n    # Now, let's check if it actually worked.\n    d = sqlitedict.SqliteDict('tests/db/autocommit.sqlite')\n    for i in range(1000):\n        assert d[i] == i, \"actual: %s expected: %s\" % (d[i], i)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_core.py",
      "content": "# std imports\nimport json\nimport unittest\nimport tempfile\nimport os\nfrom unittest.mock import patch\n\n# local\nimport sqlitedict\nfrom sqlitedict import SqliteDict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass SqliteMiscTest(unittest.TestCase):\n\n    def test_with_statement(self):\n        \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n        with SqliteDict() as d:\n            self.assertTrue(isinstance(d, SqliteDict))\n            self.assertEqual(dict(d), {})\n            self.assertEqual(list(d), [])\n            self.assertEqual(len(d), 0)\n\n    def test_reopen_conn(self):\n        \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        db = SqliteDict(filename=fname)\n        with db:\n            db['key'] = 'value'\n            db.commit()\n        with db:\n            db['key'] = 'value'\n            db.commit()\n\n    def test_as_str(self):\n        \"\"\"Verify SqliteDict.__str__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__str__()\n        # test when db closed\n        db.close()\n        db.__str__()\n\n    def test_as_repr(self):\n        \"\"\"Verify SqliteDict.__repr__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__repr__()\n\n    def test_directory_notfound(self):\n        \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n        # given: a non-existent directory,\n        folder = tempfile.mkdtemp(prefix='sqlitedict-test')\n        os.rmdir(folder)\n        # exercise,\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=os.path.join(folder, 'nonexistent'))\n\n    def test_commit_nonblocking(self):\n        \"\"\"Coverage for non-blocking commit.\"\"\"\n        # given,\n        with SqliteDict(autocommit=True) as d:\n            # exercise: the implicit commit is nonblocking\n            d['key'] = 'value'\n            d.commit(blocking=False)\n\n    def test_cancel_iterate(self):\n        import time\n\n        class EndlessKeysIterator:\n            def __init__(self) -> None:\n                self.value = 0\n\n            def __iter__(self):\n                return self\n\n            def __next__(self):\n                self.value += 1\n                return [self.value]\n\n        with patch('sqlitedict.sqlite3') as mock_sqlite3:\n            ki = EndlessKeysIterator()\n            cursor = mock_sqlite3.connect().cursor()\n            cursor.__iter__.return_value = ki\n\n            with SqliteDict(autocommit=True) as d:\n                for i, k in enumerate(d.keys()):\n                    assert i + 1 == k\n                    if k > 100:\n                        break\n                assert ki.value > 101\n\n                # Release GIL, let background threads run.\n                # Don't use gc.collect because this is simulate user code.\n                time.sleep(0.01)\n\n                current = ki.value\n                time.sleep(1)\n                assert current == ki.value, 'Will not read more after iterate stop'\n\n\nclass NamedSqliteDictCreateOrReuseTest(TempSqliteDictTest):\n    \"\"\"Verify default flag='c', and flag='n' of SqliteDict().\"\"\"\n\n    def test_default_reuse_existing_flag_c(self):\n        \"\"\"Re-opening of a database does not destroy it.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        next_db = SqliteDict(filename=fname)\n        self.assertIn('key', next_db.keys())\n        self.assertEqual(next_db['key'], 'value')\n\n    def test_overwrite_using_flag_n(self):\n        \"\"\"Re-opening of a database with flag='c' destroys it all.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname, tablename='sometable')\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        # verify,\n        next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n        self.assertNotIn('key', next_db.keys())\n\n    def test_unrecognized_flag(self):\n\n        def build_with_bad_flag():\n            fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n            SqliteDict(filename=fname, flag='FOO')\n\n        with self.assertRaises(RuntimeError):\n            build_with_bad_flag()\n\n    def test_readonly(self):\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        readonly_db = SqliteDict(filename=fname, flag='r')\n        self.assertTrue(readonly_db['key'] == 'value')\n        self.assertTrue(readonly_db['key_two'] == 2)\n\n        def attempt_write():\n            readonly_db['key'] = ['new_value']\n\n        def attempt_update():\n            readonly_db.update(key='value2', key_two=2.1)\n\n        def attempt_delete():\n            del readonly_db['key']\n\n        def attempt_clear():\n            readonly_db.clear()\n\n        def attempt_terminate():\n            readonly_db.terminate()\n\n        attempt_funcs = [attempt_write, attempt_update, attempt_delete, attempt_clear, attempt_terminate]\n\n        for func in attempt_funcs:\n            with self.assertRaises(RuntimeError):\n                func()\n\n    def test_readonly_table(self):\n        \"\"\"\n        Read-only access on a non-existent tablename should raise RuntimeError,\n        and not create a new (empty) table.\n        \"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        dummy_tablename = 'table404'\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n\n        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n    def test_irregular_tablenames(self):\n        \"\"\"Irregular table names need to be quoted\"\"\"\n        def __test_irregular_tablenames(tablename):\n            filename = ':memory:'\n            db = SqliteDict(filename, tablename=tablename)\n            db['key'] = 'value'\n            db.commit()\n            self.assertEqual(db['key'], 'value')\n            db.close()\n\n        __test_irregular_tablenames('9nine')\n        __test_irregular_tablenames('outer space')\n        __test_irregular_tablenames('table with a \"quoted\" name')\n        __test_irregular_tablenames(\"table with a \\\"quoted \\xe1cute\\\" name\")\n\n    def test_overwrite_using_flag_w(self):\n        \"\"\"Re-opening of a database with flag='w' destroys only the target table.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db_1 = SqliteDict(filename=fname, tablename='one')\n        orig_db_1['key'] = 'value'\n        orig_db_1.commit()\n        orig_db_1.close()\n\n        orig_db_2 = SqliteDict(filename=fname, tablename='two')\n        orig_db_2['key'] = 'value'\n        orig_db_2.commit()\n        orig_db_2.close()\n\n        # verify, when re-opening table space 'one' with flag='2', we destroy\n        # its contents.  However, when re-opening table space 'two' with\n        # default flag='r', its contents remain.\n        next_db_1 = SqliteDict(filename=fname, tablename='one', flag='w')\n        self.assertNotIn('key', next_db_1.keys())\n\n        next_db_2 = SqliteDict(filename=fname, tablename='two')\n        self.assertIn('key', next_db_2.keys())\n\n\nclass SqliteDictTerminateTest(unittest.TestCase):\n\n    def test_terminate_instead_close(self):\n        ''' make terminate() instead of close()\n        '''\n        d = sqlitedict.open('tests/db/sqlitedict-terminate.sqlite')\n        d['abc'] = 'def'\n        d.commit()\n        self.assertEqual(d['abc'], 'def')\n        d.terminate()\n        self.assertFalse(os.path.isfile('tests/db/sqlitedict-terminate.sqlite'))\n\n\nclass SqliteDictTerminateFailTest(unittest.TestCase):\n    \"\"\"Provide Coverage for SqliteDict.terminate().\"\"\"\n\n    def setUp(self):\n        self.fname = norm_file('tests/db-permdenied/sqlitedict.sqlite')\n        self.db = SqliteDict(filename=self.fname)\n        os.chmod(self.fname, 0o000)\n        os.chmod(os.path.dirname(self.fname), 0o000)\n\n    def tearDown(self):\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def test_terminate_cannot_delete(self):\n        # exercise,\n        self.db.terminate()  # deletion failed, but no exception raised!\n\n        # verify,\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        self.assertTrue(os.path.exists(self.fname))\n\n\nclass SqliteDictJsonSerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-json/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test', encode=json.dumps, decode=json.loads\n        )\n\n    def tearDown(self):\n        self.db.close()\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def get_json(self, key):\n        return self.db.conn.select_one('SELECT value FROM test WHERE key = ?', (self.db.encode_key(key),))[0]\n\n    def test_int(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        assert self.get_json('test') == '-42'\n\n    def test_str(self):\n        test_str = u'Test \\u30c6\\u30b9\\u30c8'\n        self.db['test'] = test_str\n        assert self.db['test'] == test_str\n        assert self.get_json('test') == r'\"Test \\u30c6\\u30b9\\u30c8\"'\n\n    def test_bool(self):\n        self.db['test'] = False\n        assert self.db['test'] is False\n        assert self.get_json('test') == 'false'\n\n    def test_none(self):\n        self.db['test'] = None\n        assert self.db['test'] is None\n        assert self.get_json('test') == 'null'\n\n    def test_complex_struct(self):\n        test_value = {\n            'version': 2.5,\n            'items': ['one', 'two'],\n        }\n        self.db['test'] = test_value\n        assert self.db['test'] == test_value\n        assert self.get_json('test') == json.dumps(test_value)\n\n\nclass TablenamesTest(unittest.TestCase):\n    def tearDown(self):\n        for f in ('tablenames-test-1.sqlite', 'tablenames-test-2.sqlite'):\n            path = norm_file(os.path.join('tests/db', f))\n            if os.path.isfile(path):\n                os.unlink(path)\n\n    def test_tablenames_unnamed(self):\n        fname = norm_file('tests/db/tablenames-test-1.sqlite')\n        SqliteDict(fname)\n        self.assertEqual(SqliteDict.get_tablenames(fname), ['unnamed'])\n\n    def test_tablenams_named(self):\n        fname = norm_file('tests/db/tablenames-test-2.sqlite')\n        with SqliteDict(fname, tablename='table1'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1'])\n        with SqliteDict(fname, tablename='table2'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1', 'table2'])\n\n        tablenames = SqliteDict.get_tablenames('tests/db/tablenames-test-2.sqlite')\n        self.assertEqual(tablenames, ['table1', 'table2'])\n\n\nclass SqliteDictKeySerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-encode-key/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test',\n            encode_key=sqlitedict.encode_key, decode_key=sqlitedict.decode_key,\n        )\n\n    def test_nonstr_keys(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        self.db[(0, 1, 2)] = 17\n        assert self.db[(0, 1, 2)] == 17\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_named_db.py",
      "content": "import sqlitedict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass InMemorySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass NamedSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-def.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db)\n\n\nclass CreateNewSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-n-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass StartsWithEmptySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-w-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"w\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass SqliteDictAutocommitTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-autocommit.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/accessories.py",
      "content": "\"\"\"Accessories for test cases.\"\"\"\nimport os\n\n\ndef norm_file(fname):\n    \"\"\"Normalize test filename, creating a directory path to it if necessary\"\"\"\n    fname = os.path.abspath(fname)\n    dirname = os.path.dirname(fname)\n    if not os.path.exists(dirname):\n        os.makedirs(dirname)\n    return fname\n"
    },
    {
      "path": "piskvorky_sqlitedict/release/summarize_pr.py",
      "content": "#!/usr/bin/env python\nimport json\nimport sys\nimport urllib.request\n\n\ndef copy_to_clipboard(text):\n    try:\n        import pyperclip\n    except ImportError:\n        print('pyperclip <https://pypi.org/project/pyperclip/> is missing.', file=sys.stderr)\n        print('copy-paste the following text manually:', file=sys.stderr)\n        print('  ' + text, file=sys.stderr)\n    else:\n        pyperclip.copy(text)\n\n\nfor prid in sys.argv[1:]:\n    url = \"https://api.github.com/repos/RaRe-Technologies/sqlitedict/pulls/%s\" % prid\n    with urllib.request.urlopen(url) as fin:\n        prinfo = json.load(fin)\n\n    prinfo['user_login'] = prinfo['user']['login']\n    prinfo['user_html_url'] = prinfo['user']['html_url']\n    text = '- %(title)s (PR [#%(number)s](%(html_url)s), [@%(user_login)s](%(user_html_url)s))' % prinfo\n    print(text)\n"
    }
  ],
  "ErrorMessage": "======================================================================\nFAIL: test_readonly_table (test_core.NamedSqliteDictCreateOrReuseTest)\nRead-only access on a non-existent tablename should raise RuntimeError,\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 190, in test_readonly_table\n    self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\nAssertionError: False is not true\n\n----------------------------------------------------------------------\nRan 88 tests in 1.157s\n\nFAILED (failures=1)",
  "Patch": "--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -130,7 +130,6 @@\n         # verify,\n         next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n         self.assertNotIn('key', next_db.keys())\n-        self.assertNotIn('value', next_db.values())\n \n     def test_unrecognized_flag(self):\n \n@@ -187,7 +186,7 @@\n         orig_db.commit()\n         orig_db.close()\n \n-        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n+        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n \n         with self.assertRaises(RuntimeError):\n             SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n",
  "BuggyCodeLocation": [
    {
      "file": "piskvorky_sqlitedict/tests/test_core.py",
      "function": {
        "177": "test_readonly_table"
      },
      "content_all": {
        "187": "        orig_db.commit()\n",
        "188": "        orig_db.close()\n",
        "189": "\n",
        "190": "        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n",
        "191": "\n",
        "192": "        with self.assertRaises(RuntimeError):\n",
        "193": "            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n"
      },
      "content_change": {
        "190": "        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_core.py",
      "function": null,
      "content_all": {
        "133": "        self.assertNotIn('value', next_db.values())\n"
      },
      "content_change": {
        "133": "        self.assertNotIn('value', next_db.values())\n"
      }
    }
  ],
  "Issue": {
    "title": "Issues with Database Table Verification and Key-Value Existence Checks",
    "description": "There have been inconsistencies detected in the database table verification and key-value existence checks in the `SqliteDict` module. Specifically, when re-opening a database with the flag `'n'`, the expected behavior is that all existing tables should be erased, but it appears not to be the case.\n\nFurthermore, the checks for key existence performed through `next_db.keys()` and `next_db.values()` yield unexpected results. For example, a key may exist but the corresponding value check fails, leading to incorrect assumptions about the database state.\n\nAnother critical issue is observed in read-only table access, where unexpected tables are being listed even when access should be restricted, causing security and data integrity concerns.\n\nThese issues impact the reliability and correctness of database operations, warranting an urgent review and fix to ensure accurate database table management and key-value existence verifications.",
    "explanation": "### Summary of the Issue\n\nThe issue revolves around the `SqliteDict` module and relates to three main areas:\n1. **Database Re-opening with 'n' Flag**: When re-opening a database with the 'n' flag, the expected behavior is to erase all existing tables. However, inconsistencies were detected indicating this action does not always occur.\n2. **Key-Value Existence Checks**: During database operations, checks for key existence using `next_db.keys()` and `next_db.values()` yield unexpected results, with cases where a key appears to exist but the corresponding value check fails.\n3. **Read-Only Table Access**: In cases where database access should be read-only, unexpected tables can be listed, leading to security and data integrity concerns.\n\nThese issues collectively reduce the reliability and correctness of database operations, making it critical to review and fix them promptly.\n\n### Commit Content and Analysis\n\nTo address these issues, a commit was made with specific changes mainly targeting `tests/test_core.py`. Here's an overview of the modifications and their implications:\n\n1. **Cleanup During Database Re-opening**:\n   - When a database is re-opened with the 'n' flag, the tables should be wiped clean. The test cases were updated to ensure that tables and their contents are correctly erased during re-opening. This addresses inconsistencies with table erasure, ensuring the database behaves as expected by removing existing data upon re-opening.\n\n2. **Improved Key-Value Assertions**:\n   - The test case that previously checked both key and value existence now focuses on keys only, removing the redundant value assertion. By refining these checks, the tests more accurately reflect the database's internal state without confusing key existence with corresponding value checks. This likely resolves false negatives where a key existence check would pass, but the associated value check would fail.\n\n3. **Enhanced Read-Only Access Verification**:\n   - Adjustments were made to ensure that when a database is accessed in read-only mode, no unintended tables are listed. Tests now properly account for and verify that unexpected tables are not improperly accessed or displayed, addressing a significant security and data integrity concern.\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Database Re-opening Cleanup**:\n   - By ensuring that the test cases robustly check for complete table erasure during database re-opening with the 'n' flag, the commit guarantees the database starts afresh with no residual data. This precisely addresses the complaint of leftover tables when a database is expected to be fully reset.\n\n2. **Key-Value Existence Checks**:\n   - The update to the tests ensures they do not mistakenly interpret key existence and value presence as the same operation. This clearer delineation prevents the confusion that led to false assumptions about the database state. Operational reliability is enhanced by ensuring that keys and their corresponding values are verified appropriately and separately.\n\n3. **Read-Only Table Access**:\n   - The revised tests check more rigorously whether unintended tables appear during read-only access attempts. This fortifies the system against the inadvertent exposure of or interaction with tables that should stay concealed, thereby maintaining the integrity and security of the data.\n\n### Solution to the Issue\n\nThe issue was addressed by:\n- Ensuring the database correctly erases all tables upon re-opening with the 'n' flag.\n- Clarifying and accurately checking for key and value existence to prevent erroneous assumptions about database states.\n- Strengthening the validation of read-only access to keep unintended tables from appearing.\n\n### Conclusion\n\nThe commit effectively solves the highlighted issues by making precise and necessary adjustments to the test cases. These adjustments improve the reliability and security of the `SqliteDict` module, ensuring that database operations behave as expected, correctly manage key-value existence checks, and maintain data integrity, especially in read-only access scenarios."
  },
  "Explain": "### Summary of the Issue\n\nThe issue revolves around the `SqliteDict` module and relates to three main areas:\n1. **Database Re-opening with 'n' Flag**: When re-opening a database with the 'n' flag, the expected behavior is to erase all existing tables. However, inconsistencies were detected indicating this action does not always occur.\n2. **Key-Value Existence Checks**: During database operations, checks for key existence using `next_db.keys()` and `next_db.values()` yield unexpected results, with cases where a key appears to exist but the corresponding value check fails.\n3. **Read-Only Table Access**: In cases where database access should be read-only, unexpected tables can be listed, leading to security and data integrity concerns.\n\nThese issues collectively reduce the reliability and correctness of database operations, making it critical to review and fix them promptly.\n\n### Commit Content and Analysis\n\nTo address these issues, a commit was made with specific changes mainly targeting `tests/test_core.py`. Here's an overview of the modifications and their implications:\n\n1. **Cleanup During Database Re-opening**:\n   - When a database is re-opened with the 'n' flag, the tables should be wiped clean. The test cases were updated to ensure that tables and their contents are correctly erased during re-opening. This addresses inconsistencies with table erasure, ensuring the database behaves as expected by removing existing data upon re-opening.\n\n2. **Improved Key-Value Assertions**:\n   - The test case that previously checked both key and value existence now focuses on keys only, removing the redundant value assertion. By refining these checks, the tests more accurately reflect the database's internal state without confusing key existence with corresponding value checks. This likely resolves false negatives where a key existence check would pass, but the associated value check would fail.\n\n3. **Enhanced Read-Only Access Verification**:\n   - Adjustments were made to ensure that when a database is accessed in read-only mode, no unintended tables are listed. Tests now properly account for and verify that unexpected tables are not improperly accessed or displayed, addressing a significant security and data integrity concern.\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Database Re-opening Cleanup**:\n   - By ensuring that the test cases robustly check for complete table erasure during database re-opening with the 'n' flag, the commit guarantees the database starts afresh with no residual data. This precisely addresses the complaint of leftover tables when a database is expected to be fully reset.\n\n2. **Key-Value Existence Checks**:\n   - The update to the tests ensures they do not mistakenly interpret key existence and value presence as the same operation. This clearer delineation prevents the confusion that led to false assumptions about the database state. Operational reliability is enhanced by ensuring that keys and their corresponding values are verified appropriately and separately.\n\n3. **Read-Only Table Access**:\n   - The revised tests check more rigorously whether unintended tables appear during read-only access attempts. This fortifies the system against the inadvertent exposure of or interaction with tables that should stay concealed, thereby maintaining the integrity and security of the data.\n\n### Solution to the Issue\n\nThe issue was addressed by:\n- Ensuring the database correctly erases all tables upon re-opening with the 'n' flag.\n- Clarifying and accurately checking for key and value existence to prevent erroneous assumptions about database states.\n- Strengthening the validation of read-only access to keep unintended tables from appearing.\n\n### Conclusion\n\nThe commit effectively solves the highlighted issues by making precise and necessary adjustments to the test cases. These adjustments improve the reliability and security of the `SqliteDict` module, ensuring that database operations behave as expected, correctly manage key-value existence checks, and maintain data integrity, especially in read-only access scenarios.",
  "Source": "Human",
  "Token": 1265,
  "Command": [
    "python -m unittest discover -s tests"
  ],
  "FilteredCode": [
    {
      "path": "piskvorky_sqlitedict/tests/test_core.py",
      "content": "1 # std imports\n2 import json\n3 import unittest\n4 import tempfile\n5 import os\n6 from unittest.mock import patch\n7 \n8 # local\n9 import sqlitedict\n10 from sqlitedict import SqliteDict\n11 from test_temp_db import TempSqliteDictTest\n12 from accessories import norm_file\n13 \n14 \n15 class SqliteMiscTest(unittest.TestCase):\n16 \n17     def test_with_statement(self):\n18         \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n19         with SqliteDict() as d:\n20             self.assertTrue(isinstance(d, SqliteDict))\n21             self.assertEqual(dict(d), {})\n22             self.assertEqual(list(d), [])\n23             self.assertEqual(len(d), 0)\n24 \n25     def test_reopen_conn(self):\n26         \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n27         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n28         db = SqliteDict(filename=fname)\n29         with db:\n30             db['key'] = 'value'\n31             db.commit()\n32         with db:\n33             db['key'] = 'value'\n34             db.commit()\n35 \n36     def test_as_str(self):\n37         \"\"\"Verify SqliteDict.__str__().\"\"\"\n38         # given,\n39         db = SqliteDict()\n40         # exercise\n41         db.__str__()\n42         # test when db closed\n43         db.close()\n44         db.__str__()\n45 \n46     def test_as_repr(self):\n47         \"\"\"Verify SqliteDict.__repr__().\"\"\"\n48         # given,\n49         db = SqliteDict()\n50         # exercise\n51         db.__repr__()\n52 \n53     def test_directory_notfound(self):\n54         \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n55         # given: a non-existent director(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 #\n9 # This code was inspired by:\n10 #  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n11 #  * http://code.activestate.com/recipes/526618/\n12 \n13 \"\"\"\n14 A lightweight wrapper around Python's sqlite3 database, with a dict-like interface\n15 and multi-thread access support::\n16 \n17 >>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n18 >>> mydict['some_key'] = any_picklable_object\n19 >>> print mydict['some_key']\n20 >>> print len(mydict) # etc... all dict functions work\n21 \n22 Pickle is used internally to serialize the values. Keys are strings.\n23 \n24 If you don't use autocommit (default is no autocommit for performance), then\n25 don't forget to call `mydict.commit()` when done with a transaction.\n26 \n27 \"\"\"\n28 \n29 import sqlite3\n30 import os\n31 import sys\n32 import tempfile\n33 import threading\n34 import logging\n35 import traceback\n36 from base64 impo(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 \n9 \"\"\"\n10 Run with:\n11 \n12 python ./setup.py install\n13 \"\"\"\n14 \n15 import os\n16 import io\n17 import subprocess\n18 \n19 import setuptools.command.develop\n20 from setuptools import setup\n21 \n22 \n23 def read(fname):\n24     path = os.path.join(os.path.dirname(__file__), fname)\n25     return io.open(path, encoding='utf8').read()\n26 \n27 \n28 class SetupDevelop(setuptools.command.develop.develop):\n29     \"\"\"Docstring is overwritten.\"\"\"\n30 \n31     def run(self):\n32         \"\"\"\n33         Prepare environment for development.\n34 \n35         - Ensures 'nose' and 'coverage.py' are installed for testing.\n36         - Call s(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "1 import os\n2 import sys\n3 \n4 import sqlitedict\n5 \n6 \n7 def test():\n8     \"Verify autocommit just before program exits.\"\n9     assert os.system('env PYTHONPATH=. %s tests/autocommit.py' % sys.executable) == 0\n10     # The above script relies on the autocommit feature working correctly.\n11     # Now, let's check if it actually worked.\n12     d = sqlitedict.SqliteDict('tests/db/autocommit.sqlite')\n13     for i in range(1000):\n14         assert d[i] == i, \"actual: %s expected: %s\" % (d[i], i)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_onimport.py",
      "content": "1 \"\"\"Test cases for on-import logic.\"\"\"\n2 import unittest\n3 import sys\n4 \n5 \n6 class SqliteDict_cPickleImportTest(unittest.TestCase):\n7     \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n8     def setUp(self):\n9         self.orig_meta_path = sys.meta_path\n10         self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n11 \n12  (...truncated)"
    }
  ],
  "TokenAll": 13040,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Possible unhandled database integrity issues",
      "description": "The current tests in the file 'piskvorky_sqlitedict/tests/test_core.py' do not cover potential integrity issues, such as database corruption or concurrent write operations. It is advisable to add tests that simulate such scenarios to ensure robustness of the SqliteDict implementation.",
      "explanation": "### Summary of the Issue:\n\nThe main issue reported here is the lack of tests addressing potential database integrity issues in the `piskvorky_sqlitedict` project. Specifically, the current test suite, particularly in `test_core.py`, does not validate scenarios such as database corruption or concurrent write operations. To ensure the robustness of the `SqliteDict` implementation, it is essential to introduce tests that simulate these integrity challenges.\n\n### Detailed Content of the Commit:\n\nLet's consider a hypothetical commit that would address the reported issue. The content of the commit might include:\n\n1. **New Test Cases** to handle potential integrity issues:\n    - Simulating database corruption.\n    - Simulating concurrent write operations.\n    - Checking the behavior when accessing a database during a corruption/error state.\n  \n2. **Modifications to Existing Tests** to improve coverage and ensure new edge cases are considered.\n\nBelow is an example of how the commit might look:\n\n#### Modification to `piskvorky_sqlitedict/tests/test_core.py`:\n\n```python\n15 class SqliteMiscTest(unittest.TestCase):\n     .\n     .\n53     def test_directory_notfound(self):\n54         \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n55         # given: a non-existent directory\n56         self.assertRaises(RuntimeError, SqliteDict, 'non/existent/path/sqlitedict.sqlite')\n\n57     def test_database_corruption(self):\n58         \"\"\"Simulate database corruption and verify SqliteDict behavior.\"\"\"\n59         fname = norm_file('tests/db/sqlitedict-corrupt-test.sqlite')\n60         with open(fname, 'w') as f:\n61             f.write('CORRUPT DATABASE')  # simulate corruption by writing invalid data\n62         self.assertRaises(sqlite3.DatabaseError, SqliteDict, filename=fname)\n     \n63     def test_concurrent_writes(self):\n64         \"\"\"Simulate concurrent write operations.\"\"\"\n65         fname = norm_file('tests/db/sqlitedict-concurrent-test.sqlite')\n66         db1 = SqliteDict(filename=fname)\n67         db2 = SqliteDict(filename=fname)\n68         def write_data(db, key, value):\n69             db[key] = value\n70             db.commit()\n71         thread1 = threading.Thread(target=write_data, args=(db1, 'key1', 'value1'))\n72         thread2 = threading.Thread(target=write_data, args=(db2, 'key2', 'value2'))\n73         thread1.start()\n74         thread2.start()\n75         thread1.join()\n76         thread2.join()\n77         db1.commit()\n78         db2.commit()\n79         self.assertIn('key1', db1)  # verify key assigned by thread1\n80         self.assertIn('key2', db2)  # verify key assigned by thread2\n81         db1.close()\n82         db2.close()\n```\n\n### How the Commit Solves the Issue:\n\n1. **Simulating Database Corruption**:\n    - The new test `test_database_corruption` directly modifies the content of a SQLite file to simulate corruption. It ensures that the `SqliteDict` class can handle scenarios where the database becomes unreadable or corrupted.\n    - The test raises an expected `sqlite3.DatabaseError`, validating that the implementation correctly identifies and handles such exceptions.\n\n2. **Handling Concurrent Writes**:\n    - The `test_concurrent_writes` method tests the behavior of the `SqliteDict` under concurrent write operations. This is vital as multiple threads accessing a shared SQLite database can potentially lead to race conditions or database locks.\n    - Using Python’s `threading` module, two concurrent threads attempt to write to the same database file. After both threads complete their operations, the test checks whether each key written by the threads exists in the database, ensuring no data collision or corruption due to concurrency.\n\n### Explanation of the Solution:\n\n1. **Identifying Corruption Handling**:\n    - By adding a test that artificially corrupts the database file, we ensure that the system can identify and raise appropriate errors. This is crucial for applications relying heavily on database integrity.\n    - Handling such errors gracefully prevents undefined behavior or crashes in production environments.\n\n2. **Ensuring Thread Safety**:\n    - The concurrent writing test verifies that the `SqliteDict` can handle simultaneous write operations from different threads without data corruption or access issues. \n    - Thread 1 and Thread 2 interact with the same database instance concurrently, and the successful verification post-execution confirms that `SqliteDict` can manage concurrent access scenarios effectively.\n\nBy incorporating these changes, the commit effectively addresses the integrity scenarios that the original tests missed. This improvement enhances the robustness of the `SqliteDict` library, ensuring it remains reliable even under atypical and adverse conditions."
    },
    "issue_message": {
      "title": "Read-only access test failure: Non-existent tablename",
      "description": "The test_readonly_table in test_core.py fails because it expects a non-existent tablename to be in the list of tablenames retrieved by SqliteDict.get_tablenames(fname). This leads to an assertion failure as the dummy tablename is not found in the retrieved list.",
      "explanation": "### Summary of the Issue\n\nThe issue occurs in a test case named `test_readonly_table` within `test_core.py`. The test is failing because it expects a non-existent tablename (`dummy_tablename`) to be included in the list of tablenames retrieved by the method `SqliteDict.get_tablenames(fname)`. Since the dummy table is not found in the list, an assertion fails.\n\n### Key Problem\n\n- **Expectation vs Reality:** The test expects a dummy/non-existent tablename to appear in the list, which logically should not be the case. This suggests a misunderstanding or a logical flaw within either the test case or the underlying functionality.\n\n### Detailing the Commit\n\nWithout the commit message or details, let's propose a likely effective commit and explain:\n\n1. **Test Correction:** It's probable that the test is faulty, expecting a non-existent table to be found. A correct test should verify that an existing table name is in the list instead.\n2. **Handling Non-Existent Tables:** We could also ensure that the `get_tablenames` method appropriately handles both existent and non-existent tables, making the failure informative and handling edge cases.\n\nHere’s a plausible commit to address this issue:\n\n#### Commit Message:\n```\nFix test_readonly_table to accurately check for non-existent tablename in test_core.py\n\nThe current test incorrectly expects a non-existent tablename to appear in the list of table names. This commit modifies the test to validate the correct behavior, ensuring accurate assertions.\n\nAdditionally, ensures SqliteDict.get_tablenames appropriately handles and signals non-existent table scenarios.\n```\n\n#### Changes in `test_core.py`:\n```python\n15 class SqliteMiscTest(unittest.TestCase):\n# ...\n\n189     def test_readonly_table(self):\n190         fname = 'tests/db/sqlitedict-readonly-test.sqlite'\n191         dummy_tablename = 'dummy_table'\n192         db = SqliteDict(filename=fname)\n193         with db:\n194             db['key'] = 'value'\n195             db.commit()\n\n197         tablenames = SqliteDict.get_tablenames(fname)\n198         # Correcting the test expectation to verify dummy tablename should NOT exist\n199         self.assertFalse(dummy_tablename in tablenames, \"Non-existent tablename was found in tablename list.\")\n200         self.assertTrue('data' in tablenames, \"Expected tablename 'data' not found in tablename list.\")\n```\n\nIn combination with an update in `sqlitedict.py` if necessary:\n\n#### Changes in `sqlitedict.py`:\n```python\n# ...\n28 import sqlite3\n34 import logging\n\n# Add or modify get_tablenames function if needed\nclass SqliteDict:\n    # ...\n    \n    @staticmethod\n    def get_tablenames(filename):\n        conn = sqlite3.connect(filename)\n        cursor = conn.cursor()\n        try:\n            cursor.execute(\"SELECT name FROM sqlite_master WHERE type='table';\")\n            return [row[0] for row in cursor.fetchall()]\n        finally:\n            conn.close()\n```\n\n### Explanation of the Fix\n\n1. **Test Update:**\n   - The modified test now correctly asserts that `dummy_tablename` should not be in the list of tablenames.\n   - It also asserts that an expected table, such as 'data', which is more likely to exist, should be in the list.\n   \n2. **Code Correctness:**\n   - If `get_tablenames` did not already exist or was flawed, it ensures to correctly fetch tablenames from the database.\n   - It iterates through the results of a query that lists table names in the database and properly closes the connection, maintaining resource management best practices.\n\n### Conclusion\n\nThe issue is rooted in a logical mismatch within the test, where it incorrectly expects a non-existent table to be in the list. The commit corrects this misunderstanding by updating the test to ensure logical checks, and confirms the healthy execution of `get_tablenames`. This aligns the test behavior with realistic expectations, stabilizing the assertion logic and improving test accuracy."
    },
    "issue_ground": {
      "title": "Issues with Database Table Verification and Key-Value Existence Checks",
      "description": "There have been inconsistencies detected in the database table verification and key-value existence checks in the `SqliteDict` module. Specifically, when re-opening a database with the flag `'n'`, the expected behavior is that all existing tables should be erased, but it appears not to be the case.\n\nFurthermore, the checks for key existence performed through `next_db.keys()` and `next_db.values()` yield unexpected results. For example, a key may exist but the corresponding value check fails, leading to incorrect assumptions about the database state.\n\nAnother critical issue is observed in read-only table access, where unexpected tables are being listed even when access should be restricted, causing security and data integrity concerns.\n\nThese issues impact the reliability and correctness of database operations, warranting an urgent review and fix to ensure accurate database table management and key-value existence verifications.",
      "explanation": "### Issue Summary\n\nThe reported issue concerns inconsistent behaviors in the `SqliteDict` module when handling database table verification and key-value existence checks. Specifically:\n1. When reopening a database with the flag `'n'`, existing tables are not erased as expected.\n2. The `next_db.keys()` and `next_db.values()` methods return inconsistent results regarding key existence.\n3. In read-only table access, unexpected tables are being listed, posing security and data integrity concerns.\n\nThese inconsistencies impact the reliability and correctness of database operations, particularly affecting table management and key-value verification.\n\n### Inspecting the Relevant Code\n\nLooking at the provided code snippets, here are some key areas to focus on:\n\n1. **Test Files:**\n    - `test_core.py`: Contains tests verifying various functionalities of `SqliteDict`.\n    - `test_autocommit.py`: Validates the autocommit feature.\n    - `test_onimport.py`: Checks for behavior on import, especially around pickle vs. cPickle imports.\n    \n2. **Main Implementation:**\n    - `sqlitedict.py`: The main module containing the `SqliteDict` class with its methods and functionalities.\n\n### Identifying the Problem\n\nBased on the error message specifically:\n\n```\nFAIL: test_readonly_table (test_core.NamedSqliteDictCreateOrReuseTest)\nRead-only access on a non-existent tablename should raise RuntimeError,\n```\n\nAn assertion fails because `dummy_tablename` was not found in the list of table names from `SqliteDict.get_tablenames(fname)`.\n\n### Understanding the Fix\n\nWithout the actual commit, a plausible fix would involve:\n\n**1. Properly handling the 'n' flag to ensure tables are correctly erased.**\n   Code to ensure tables are dropped when the database is reopened with the 'n' flag could include:\n   ```python\n   def __handle_new_flag(self):\n       if self.flag == 'n':\n           cursor = self.conn.cursor()\n           cursor.execute('DROP TABLE IF EXISTS your_table_name')\n           self.conn.commit()\n   ```\n\n**2. Ensuring key-value checks are consistent in `next_db.keys()` and `next_db.values()`.**\n   Updated methods could ensure values are synchronized with keys:\n   ```python\n   def keys(self):\n       cursor = self.conn.cursor()\n       cursor.execute('SELECT key FROM your_table_name')\n       return [row[0] for row in cursor.fetchall()]\n   \n   def values(self):\n       cursor = self.conn.cursor()\n       cursor.execute('SELECT value FROM your_table_name')\n       return [row[0] for row in cursor.fetchall()]\n   ```\n\n**3. Properly managing readonly access and restricting table listings appropriately.**\n   Modifying table listing logic to respect read-only restrictions:\n   ```python\n   def get_tablenames(self, read_only=False):\n       cursor = self.conn.cursor()\n       if read_only:\n           cursor.execute(\"PRAGMA read_only_list;\")\n       else:\n           cursor.execute(\"PRAGMA table_list;\")\n       return [row[1] for row in cursor.fetchall()]\n   ```\n\n### Commit Explanation\n\n1. **Commit Details:**\n   ```plaintext\n   Commit Message: Fix inconsistencies in database operations for SqliteDict\n   \n   - Ensure tables are erased when reopening with the 'n' flag.\n   - Synchronize key-value presence checks in `keys` and `values`.\n   - Restrict access to tables based on read-only mode.\n   ```\n   2. **Changes Made:**\n      - Added a method to handle the 'n' flag and drop tables.\n      - Updated `keys` and `values` methods to synchronize key-value checks.\n      - Modified `get_tablenames` to respect read-only access and restrict results appropriately.\n\n### Explanation of the Solution\n\n**Erasing Tables with 'n' Flag:**\nBy adding logic to drop tables when the database is reopened with the 'n' flag, we ensure that the database starts fresh, matching the intended behavior. This helps avoid leftover data that confuses subsequent operations.\n\n**Consistent Key-Value Existence Checks:**\nBy ensuring that `keys` and `values` directly query the database, we synchronize their views of the data, thus avoiding scenarios where a key exists but its value is not correctly reflected.\n\n**Restricting Read-Only Access:**\nBy modifying the logic to list tables based on read-only restrictions, we ensure that tables not intended to be listed in read-only mode are hidden. This enhances security and data integrity by preventing unauthorized access.\n\nIn summary, these changes comprehensively address the core issues, ensuring correct database state management, consistent key-value checks, and appropriate access control."
    },
    "issue_ground_truth": {
      "title": "Issues with Database Table Verification and Key-Value Existence Checks",
      "description": "There have been inconsistencies detected in the database table verification and key-value existence checks in the `SqliteDict` module. Specifically, when re-opening a database with the flag `'n'`, the expected behavior is that all existing tables should be erased, but it appears not to be the case.\n\nFurthermore, the checks for key existence performed through `next_db.keys()` and `next_db.values()` yield unexpected results. For example, a key may exist but the corresponding value check fails, leading to incorrect assumptions about the database state.\n\nAnother critical issue is observed in read-only table access, where unexpected tables are being listed even when access should be restricted, causing security and data integrity concerns.\n\nThese issues impact the reliability and correctness of database operations, warranting an urgent review and fix to ensure accurate database table management and key-value existence verifications.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `SqliteDict` module and relates to three main areas:\n1. **Database Re-opening with 'n' Flag**: When re-opening a database with the 'n' flag, the expected behavior is to erase all existing tables. However, inconsistencies were detected indicating this action does not always occur.\n2. **Key-Value Existence Checks**: During database operations, checks for key existence using `next_db.keys()` and `next_db.values()` yield unexpected results, with cases where a key appears to exist but the corresponding value check fails.\n3. **Read-Only Table Access**: In cases where database access should be read-only, unexpected tables can be listed, leading to security and data integrity concerns.\n\nThese issues collectively reduce the reliability and correctness of database operations, making it critical to review and fix them promptly.\n\n### Commit Content and Analysis\n\nTo address these issues, a commit was made with specific changes mainly targeting `tests/test_core.py`. Here's an overview of the modifications and their implications:\n\n1. **Cleanup During Database Re-opening**:\n   - When a database is re-opened with the 'n' flag, the tables should be wiped clean. The test cases were updated to ensure that tables and their contents are correctly erased during re-opening. This addresses inconsistencies with table erasure, ensuring the database behaves as expected by removing existing data upon re-opening.\n\n2. **Improved Key-Value Assertions**:\n   - The test case that previously checked both key and value existence now focuses on keys only, removing the redundant value assertion. By refining these checks, the tests more accurately reflect the database's internal state without confusing key existence with corresponding value checks. This likely resolves false negatives where a key existence check would pass, but the associated value check would fail.\n\n3. **Enhanced Read-Only Access Verification**:\n   - Adjustments were made to ensure that when a database is accessed in read-only mode, no unintended tables are listed. Tests now properly account for and verify that unexpected tables are not improperly accessed or displayed, addressing a significant security and data integrity concern.\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Database Re-opening Cleanup**:\n   - By ensuring that the test cases robustly check for complete table erasure during database re-opening with the 'n' flag, the commit guarantees the database starts afresh with no residual data. This precisely addresses the complaint of leftover tables when a database is expected to be fully reset.\n\n2. **Key-Value Existence Checks**:\n   - The update to the tests ensures they do not mistakenly interpret key existence and value presence as the same operation. This clearer delineation prevents the confusion that led to false assumptions about the database state. Operational reliability is enhanced by ensuring that keys and their corresponding values are verified appropriately and separately.\n\n3. **Read-Only Table Access**:\n   - The revised tests check more rigorously whether unintended tables appear during read-only access attempts. This fortifies the system against the inadvertent exposure of or interaction with tables that should stay concealed, thereby maintaining the integrity and security of the data.\n\n### Solution to the Issue\n\nThe issue was addressed by:\n- Ensuring the database correctly erases all tables upon re-opening with the 'n' flag.\n- Clarifying and accurately checking for key and value existence to prevent erroneous assumptions about database states.\n- Strengthening the validation of read-only access to keep unintended tables from appearing.\n\n### Conclusion\n\nThe commit effectively solves the highlighted issues by making precise and necessary adjustments to the test cases. These adjustments improve the reliability and security of the `SqliteDict` module, ensuring that database operations behave as expected, correctly manage key-value existence checks, and maintain data integrity, especially in read-only access scenarios."
    },
    "location_origin": [
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "15": "SqliteMiscTest"
        },
        "content_all": {
          "54": "53     def test_directory_notfound(self):\n54         \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n55         # given: a non-existent directory\n56         self.assertRaises(RuntimeError, SqliteDict, 'non/existent/path/sqlitedict.sqlite')\n",
          "63": "57     def test_database_corruption(self):\n58         \"\"\"Simulate database corruption and verify SqliteDict behavior.\"\"\"\n59         fname = norm_file('tests/db/sqlitedict-corrupt-test.sqlite')\n60         with open(fname, 'w') as f:\n61             f.write('CORRUPT DATABASE')  # simulate corruption by writing invalid data\n62         self.assertRaises(sqlite3.DatabaseError, SqliteDict, filename=fname)\n",
          "81": "63     def test_concurrent_writes(self):\n64         \"\"\"Simulate concurrent write operations.\"\"\"\n65         fname = norm_file('tests/db/sqlitedict-concurrent-test.sqlite')\n66         db1 = SqliteDict(filename=fname)\n67         db2 = SqliteDict(filename=fname)\n68         def write_data(db, key, value):\n69             db[key] = value\n70             db.commit()\n71         thread1 = threading.Thread(target=write_data, args=(db1, 'key1', 'value1'))\n72         thread2 = threading.Thread(target=write_data, args=(db2, 'key2', 'value2'))\n73         thread1.start()\n74         thread2.start()\n75         thread1.join()\n76         thread2.join()\n77         db1.commit()\n78         db2.commit()\n79         self.assertIn('key1', db1)  # verify key assigned by thread1\n80         self.assertIn('key2', db2)  # verify key assigned by thread2\n81         db1.close()\n82         db2.close()\n"
        },
        "content_change": {
          "57": "57     def test_database_corruption(self):\n58         \"\"\"Simulate database corruption and verify SqliteDict behavior.\"\"\"\n59         fname = norm_file('tests/db/sqlitedict-corrupt-test.sqlite')\n60         with open(fname, 'w') as f:\n61             f.write('CORRUPT DATABASE')  # simulate corruption by writing invalid data\n62         self.assertRaises(sqlite3.DatabaseError, SqliteDict, filename=fname)\n",
          "63": "63     def test_concurrent_writes(self):\n64         \"\"\"Simulate concurrent write operations.\"\"\"\n65         fname = norm_file('tests/db/sqlitedict-concurrent-test.sqlite')\n66         db1 = SqliteDict(filename=fname)\n67         db2 = SqliteDict(filename=fname)\n68         def write_data(db, key, value):\n69             db[key] = value\n70             db.commit()\n71         thread1 = threading.Thread(target=write_data, args=(db1, 'key1', 'value1'))\n72         thread2 = threading.Thread(target=write_data, args=(db2, 'key2', 'value2'))\n73         thread1.start()\n74         thread2.start()\n75         thread1.join()\n76         thread2.join()\n77         db1.commit()\n78         db2.commit()\n79         self.assertIn('key1', db1)  # verify key assigned by thread1\n80         self.assertIn('key2', db2)  # verify key assigned by thread2\n81         db1.close()\n82         db2.close()\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "189": "test_readonly_table"
        },
        "content_all": {
          "186": "     @staticmethod\n",
          "187": "     def setUpClass(cls):\n",
          "188": "         cls.test_readonly_table = 'test_readonly_table'\n",
          "189": "     def test_readonly_table(self):\n",
          "190": "         fname = 'tests/db/sqlitedict-readonly-test.sqlite'\n",
          "191": "         dummy_tablename = 'dummy_table'\n",
          "192": "         db = SqliteDict(filename=fname)\n",
          "193": "         with db:\n",
          "194": "             db['key'] = 'value'\n",
          "195": "             db.commit()\n",
          "196": "         tablenames = SqliteDict.get_tablenames(fname)\n",
          "197": "         self.assertTrue(dummy_tablename in tablenames)\n",
          "198": "         self.assertTrue('data' in tablenames)\n",
          "199": "\n",
          "200": "\n"
        },
        "content_change": {
          "197": "         self.assertFalse(dummy_tablename in tablenames, \"Non-existent tablename was found in tablename list.\")\n",
          "198": "         self.assertTrue('data' in tablenames, \"Expected tablename 'data' not found in tablename list.\")\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "28": "SqliteDict"
        },
        "content_all": {
          "24": "\n",
          "25": " '''\n",
          "26": " A lightweight wrapper around Python's sqlite3 database, with a dict-like interface\n",
          "27": " and multi-thread access support::\n",
          "28": " import sqlite3\n",
          "29": " '''\n",
          "30": "\n",
          "31": " import sqlite3\n",
          "32": " import os\n",
          "33": " import sys\n",
          "34": " import tempfile\n",
          "35": " import threading\n",
          "36": " import logging\n",
          "37": " import traceback\n",
          "38": "\n",
          "39": " class SqliteDict:\n",
          "40": " def __init__(self, filename):\n",
          "41": "\n",
          "42": "     @staticmethod\n",
          "43": "     def get_tablenames(filename):\n",
          "44": "         conn = sqlite3.connect(filename)\n",
          "45": "         cursor = conn.cursor()\n",
          "46": "         try:\n",
          "47": "             cursor.execute(\"SELECT name FROM sqlite_master WHERE type='table';\")\n",
          "48": "             return [row[0] for row in cursor.fetchall()]\n",
          "49": "         finally:\n",
          "50": "             conn.close()\n"
        },
        "content_change": {
          "42": "     '''\n",
          "43": "     def get_tablenames(filename):\n",
          "44": "         conn = sqlite3.connect(filename)\n",
          "45": "         cursor = conn.cursor()\n",
          "46": "         try:\n",
          "47": "             cursor.execute(\"SELECT name FROM sqlite_master WHERE type='table';\")\n",
          "48": "             return [row[0] for row in cursor.fetchall()]\n",
          "49": "         finally:\n",
          "50": "             conn.close()\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "398": "clear"
        },
        "content_all": {
          "395": "    def clear(self):",
          "396": "        \"\"\"Clear all items from the dictionary.\"\"\"",
          "397": "        with self.conn:",
          "398": "            self.conn.cursor().execute('DELETE FROM %s' % self.tablename)",
          "399": "        self.conn.commit()",
          "400": "",
          "401": "    def __contains__(self, key):"
        },
        "content_change": {
          "398": "            self.conn.cursor().execute('DELETE FROM %s' % self.tablename)"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "433": "_execute_on_tables"
        },
        "content_all": {
          "430": "            yield key, value",
          "431": "",
          "432": "",
          "433": "    def _execute_on_tables(self, query, params=()):",
          "434": "        cur = self.conn.execute(query, params)",
          "435": "        return [row for row in cur]",
          "436": "",
          "437": "    def _initialize_table(self, tablename):"
        },
        "content_change": {
          "434": "        cur = self.conn.execute(query, params)"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "455": "get_tablenames"
        },
        "content_all": {
          "452": "        c.execute(sql.strip())",
          "453": "        self.conn.commit()",
          "454": "",
          "455": "    def get_tablenames(self, readonly=False):",
          "456": "        if readonly:",
          "457": "            cursor = self.conn.execute(\"PRAGMA read_only_list;\")",
          "458": "        else:",
          "459": "            cursor = self.conn.execute(\"PRAGMA table_list;\")",
          "460": "        return [row[1] for row in cursor.fetchall()]",
          "461": "",
          "462": "    def commit(self):"
        },
        "content_change": {
          "456": "        if readonly:",
          "457": "            cursor = self.conn.execute(\"PRAGMA read_only_list;\")",
          "458": "        else:",
          "459": "            cursor = self.conn.execute(\"PRAGMA table_list;\")"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "76": "test_readonly_access"
        },
        "content_all": {
          "72": "        c.execute(sql)",
          "73": "        self.conn.commit()",
          "74": "",
          "75": "    def test_readonly_access(self):",
          "76": "        \"\"\"Verify readonly access behaves correctly.\"\"\"",
          "77": "        dummy_tablename = 'readonly_table_test'",
          "78": "        with SqliteDict('tests/db/sqlitedict-override-test.sqlite', tablename=dummy_tablename, flag='c', readonly=True) as db:",
          "79": "            self.assertRaises(RuntimeError, db.__setitem__, 'key', 'value')",
          "80": "        self.assertNotIn(dummy_tablename, SqliteDict.get_tablenames('tests/db/sqlitedict-override-test.sqlite', readonly=True))"
        },
        "content_change": {
          "78": "        with SqliteDict('tests/db/sqlitedict-override-test.sqlite', tablename=dummy_tablename, flag='c', readonly=True) as db:"
        }
      }
    ],
    "location_ground_exp": [
      {
        "location": "```json\n[\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"58\": \"__init__\"\n        },\n        \"content_all\": {\n            \"55\": \"    def __init__(self, filename=':memory:', tablename='unnamed', flag='c', autocommit=False, journal_mode='DELETE'):\\n\",\n            \"56\": \"        self.filename = filename\\n\",\n            \"57\": \"        self.tablename = tablename\\n\",\n            \"58\": \"        self.flag = flag\\n\",\n            \"59\": \"        self.journal_mode = journal_mode\\n\",\n            \"60\": \"\\n\",\n            \"61\": \"        # Connect to database\\n\",\n            \"62\": \"        self.conn = self._new_conn()\\n\"\n        },\n        \"content_change\": {\n            \"60\": \"        if self.flag == 'n':\\n\",\n            \"61\": \"            self._erase_existing_tables()\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"355\": \"keys\"\n        },\n        \"content_all\": {\n            \"352\": \"    def keys(self):\\n\",\n            \"353\": \"        self._execute('BEGIN')\\n\",\n            \"354\": \"        try:\\n\",\n            \"355\": \"            for key in self.conn.execute(f'SELECT key FROM \\\"{self.tablename}\\\"'):\\n\",\n            \"356\": \"                yield key[0]\\n\",\n            \"357\": \"        finally:\\n\",\n            \"358\": \"            self._execute('COMMIT')\\n\"\n        },\n        \"content_change\": {\n            \"355\": \"            query = f'SELECT key FROM \\\"{self.tablename}\\\"'\\n\",\n            \"356\": \"            for key in self.conn.execute(query):\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"364\": \"values\"\n        },\n        \"content_all\": {\n            \"361\": \"    def values(self):\\n\",\n            \"362\": \"        self._execute('BEGIN')\\n\",\n            \"363\": \"        try:\\n\",\n            \"364\": \"            for value in self.conn.execute(f'SELECT value FROM \\\"{self.tablename}\\\"'):\\n\",\n            \"365\": \"                yield self._decode(value[0])\\n\",\n            \"366\": \"        finally:\\n\",\n            \"367\": \"            self._execute('COMMIT')\\n\"\n        },\n        \"content_change\": {\n            \"364\": \"            query = f'SELECT value FROM \\\"{self.tablename}\\\"'\\n\",\n            \"365\": \"            for value in self.conn.execute(query):\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"217\": \"_new_conn\"\n        },\n        \"content_all\": {\n            \"214\": \"    def _new_conn(self):\\n\",\n            \"215\": \"        conn = sqlite3.connect(self.filename, timeout=60, check_same_thread=False)\\n\",\n            \"216\": \"        self._set_text_factory(conn)\\n\",\n            \"217\": \"        conn.execute(f'PRAGMA journal_mode={self.journal_mode}')\\n\",\n            \"218\": \"        conn.execute('PRAGMA synchronous=FULL')\\n\",\n            \"219\": \"        return conn\\n\"\n        },\n        \"content_change\": {\n            \"217\": \"        if self.flag == 'c':\\n\",\n            \"218\": \"            conn.execute(f'PRAGMA journal_mode={self.journal_mode}')\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_core.py\",\n        \"function\": {\n            \"25\": \"test_reopen_conn\"\n        },\n        \"content_all\": {\n            \"22\": \"\\n\",\n            \"23\": \"    def test_reopen_conn(self):\\n\",\n            \"24\": \"        \\\"\\\"\\\"Verify using a contextmanager that a connection can be reopened.\\\"\\\"\\\"\\n\",\n            \"25\": \"        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\\n\",\n            \"26\": \"        db = SqliteDict(filename=fname)\\n\",\n            \"27\": \"        with db:\\n\",\n            \"28\": \"            db['key'] = 'value'\\n\",\n            \"29\": \"            db.commit()\\n\",\n            \"30\": \"        with db:\\n\"\n        },\n        \"content_change\": {\n            \"26\": \"        db = SqliteDict(filename=fname, flag='n')\\n\"\n        }\n"
      }
    ],
    "location_ground_truth": [
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "177": "test_readonly_table"
        },
        "content_all": {
          "187": "        orig_db.commit()\n",
          "188": "        orig_db.close()\n",
          "189": "\n",
          "190": "        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n",
          "191": "\n",
          "192": "        with self.assertRaises(RuntimeError):\n",
          "193": "            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n"
        },
        "content_change": {
          "190": "        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": null,
        "content_all": {
          "133": "        self.assertNotIn('value', next_db.values())\n"
        },
        "content_change": {
          "133": "        self.assertNotIn('value', next_db.values())\n"
        }
      }
    ],
    "patch_i": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -4,6 +4,7 @@\n import unittest\n import tempfile\n import os\n+import threading\n from unittest.mock import patch\n \n # local\n@@ -52,4 +53,45 @@\n         self.assertRaises(RuntimeError, SqliteDict, 'non/existent/path/sqlitedict.sqlite')\n+\n+    def test_database_corruption(self):\n+        \"\"\"Simulate database corruption and verify SqliteDict behavior.\"\"\"\n+        fname = norm_file('tests/db/sqlitedict-corrupt-test.sqlite')\n+        with open(fname, 'w') as f:\n+            f.write('CORRUPT DATABASE')  # simulate corruption by writing invalid data\n+        with self.assertRaises(sqlite3.DatabaseError):\n+            with SqliteDict(filename=fname) as corrupt_db:\n+                pass\n+\n+    def test_concurrent_writes(self):\n+        \"\"\"Simulate concurrent write operations.\"\"\"\n+        fname = norm_file('tests/db/sqlitedict-concurrent-test.sqlite')\n+        db1 = SqliteDict(filename=fname)\n+        db2 = SqliteDict(filename=fname)\n+        def write_data(db, key, value):\n+            db[key] = value\n+            db.commit()\n+        thread1 = threading.Thread(target=write_data, args=(db1, 'key1', 'value1'))\n+        thread2 = threading.Thread(target=write_data, args=(db2, 'key2', 'value2'))\n+        thread1.start()\n+        thread2.start()\n+        thread1.join()\n+        thread2.join()\n+        db1.commit()\n+        db2.commit()\n+        self.assertIn('key1', db1)  # verify key assigned by thread1\n+        self.assertIn('key2', db2)  # verify key assigned by thread2\n+        db1.close()\n+        db2.close()\n+\n",
    "patch_im": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -185,7 +185,17 @@\n         self.assertEqual(list(d), [])\n         self.assertEqual(len(d), 0)\n \n-    def test_readonly_table(self):\n+    def test_readonly_table(self):\n+        fname = 'tests/db/sqlitedict-readonly-test.sqlite'\n+        dummy_tablename = 'dummy_table'\n+        db = SqliteDict(filename=fname)\n+        with db:\n+            db['key'] = 'value'\n+            db.commit()\n+\n+        tablenames = SqliteDict.get_tablenames(fname)\n+        # Correcting the test expectation to verify dummy tablename should NOT exist\n+        self.assertFalse(dummy_tablename in tablenames, \"Non-existent tablename was found in tablename list.\")\n+        self.assertTrue('data' in tablenames, \"Expected tablename 'data' not found in tablename list.\")\n \n class SqliteMiscTest(unittest.TestCase):\n \n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -35,6 +35,17 @@\n from collections.abc import MutableMapping\n import pickle as pickle\n\n+class SqliteDict:\n+\n+    @staticmethod\n+    def get_tablenames(filename):\n+        conn = sqlite3.connect(filename)\n+        cursor = conn.cursor()\n+        try:\n+            cursor.execute(\"SELECT name FROM sqlite_master WHERE type='table';\")\n+            return [row[0] for row in cursor.fetchall()]\n+        finally:\n+            conn.close()\n+\n def _decode_key(serializer, key):\n     if serializer == 'json':\n         return key.decode('utf-8')\n",
    "patch_il": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -1,5 +1,10 @@\n # std imports\n import json\n+import sqlite3\n+import threading\n+\n+# additional imports for new test cases\n+from accessories import norm_file\n \n import unittest\n import tempfile\n@@ -51,4 +56,37 @@ class SqliteMiscTest(unittest.TestCase):\n         # given: a non-existent directory\n         self.assertRaises(RuntimeError, SqliteDict, 'non/existent/path/sqlitedict.sqlite')\n \n+    def test_database_corruption(self):\n+        \"\"\"Simulate database corruption and verify SqliteDict behavior.\"\"\"\n+        fname = norm_file('tests/db/sqlitedict-corrupt-test.sqlite')\n+        with open(fname, 'w') as f:\n+            f.write('CORRUPT DATABASE')  # simulate corruption by writing invalid data\n+        self.assertRaises(sqlite3.DatabaseError, SqliteDict, filename=fname)\n+\n+    def test_concurrent_writes(self):\n+        \"\"\"Simulate concurrent write operations.\"\"\"\n+        fname = norm_file('tests/db/sqlitedict-concurrent-test.sqlite')\n+        db1 = SqliteDict(filename=fname)\n+        db2 = SqliteDict(filename=fname)\n+        def write_data(db, key, value):\n+            db[key] = value\n+            db.commit()\n+        thread1 = threading.Thread(target=write_data, args=(db1, 'key1', 'value1'))\n+        thread2 = threading.Thread(target=write_data, args=(db2, 'key2', 'value2'))\n+        thread1.start()\n+        thread2.start()\n+        thread1.join()\n+        thread2.join()\n+        db1.commit()\n+        db2.commit()\n+        self.assertIn('key1', db1)  # verify key assigned by thread1\n+        self.assertIn('key2', db2)  # verify key assigned by thread2\n+        db1.close()\n+        db2.close()\n+\n \n if __name__ == \"__main__\":\n     unittest.main()\n",
    "patch_iml": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -186,14 +186,19 @@\n     @staticmethod\n     def setUpClass(cls):\n         cls.test_readonly_table = 'test_readonly_table'\n     def test_readonly_table(self):\n         fname = 'tests/db/sqlitedict-readonly-test.sqlite'\n         dummy_tablename = 'dummy_table'\n         db = SqliteDict(filename=fname)\n         with db:\n             db['key'] = 'value'\n             db.commit()\n         tablenames = SqliteDict.get_tablenames(fname)\n-        self.assertTrue(dummy_tablename in tablenames)\n-        self.assertTrue('data' in tablenames)\n+        # Correcting the test expectation to verify dummy tablename should NOT exist\n+        self.assertFalse(dummy_tablename in tablenames, \"Non-existent tablename was found in tablename list.\")\n+        self.assertTrue('data' in tablenames, \"Expected tablename 'data' not found in tablename list.\")\n+\n",
    "patch_ground": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -100,6 +100,12 @@\n         self.filename = filename\n         if flag == 'n':  # New empty database\n             flag = 'c'\n+            self._erase_existing_tables()\n\n         WAL_MODE = os.environ.get('SQLITEDICT_WAL_MODE', \"2\") == \"1\"\n\n+    def _erase_existing_tables(self):\n+        cursor = self.conn.cursor()\n+        cursor.execute(\"SELECT name FROM sqlite_master WHERE type='table';\")\n+        tables = cursor.fetchall()\n+        for table in tables:\n+            cursor.execute(f\"DROP TABLE IF EXISTS {table[0]}\")\n+        self.conn.commit()\n",
    "patch_ground_location": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -395,7 +395,12 @@\n     def clear(self):\n         \"\"\"Clear all items from the dictionary.\"\"\"\n         with self.conn:\n-            self.conn.cursor().execute('DELETE FROM %s' % self.tablename)\n+            self.conn.cursor().execute(f'DELETE FROM \"{self.tablename}\"')\n         self.conn.commit()\n\n+    def _handle_new_flag(self):\n+        \"\"\"Handle 'n' flag to clear all tables for a fresh start.\"\"\"\n+        with self.conn:\n+            self.conn.execute('DROP TABLE IF EXISTS \"%s\"' % self.tablename)\n+        self.conn.commit()\n+\n     def __contains__(self, key):\n         \"\"\"Return true if key is in the dictionary, else false.\"\"\"\n         if self.autocommit:\n@@ -433,7 +438,7 @@\n             yield key, value\n\n     def _execute_on_tables(self, query, params=()):\n-        cur = self.conn.execute(query, params)\n+        cur = self.conn.execute(query, params)\n         return [row for row in cur]\n\n     def _initialize_table(self, tablename):\n@@ -455,10 +460,10 @@\n     def get_tablenames(self, readonly=False):\n         if readonly:\n             cursor = self.conn.execute(\"PRAGMA read_only_list;\")\n         else:\n             cursor = self.conn.execute(\"PRAGMA table_list;\")\n         return [row[1] for row in cursor.fetchall()]\n\n@@ -480,9 +485,25 @@\n         if tablename is None:\n             tablename = self.tablename\n         if self.flag == 'n':\n-            self.clear()\n+            self._handle_new_flag()\n         self.conn.execute('CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % tablename)\n         self.conn.commit()\n+\n\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -75,7 +75,8 @@\n     def test_readonly_access(self):\n         \"\"\"Verify readonly access behaves correctly.\"\"\"\n         dummy_tablename = 'readonly_table_test'\n         with SqliteDict('tests/db/sqlitedict-override-test.sqlite', tablename=dummy_tablename, flag='c', readonly=True) as db:\n             self.assertRaises(RuntimeError, db.__setitem__, 'key', 'value')\n         self.assertNotIn(dummy_tablename, SqliteDict.get_tablenames('tests/db/sqlitedict-override-test.sqlite', readonly=True))\n",
    "patch_ground_exp": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -60,6 +60,7 @@ def __init__(self, filename=':memory:', tablename='unnamed', flag='c', autoc\n         self.conn = self._new_conn()\n \n         if self.flag == 'n':\n+            self._erase_existing_tables()\n \n     def keys(self):\n         self._execute('BEGIN')\n@@ -61,6 +62,7 @@ def keys(self):\n         try:\n-            for key in self.conn.execute(f'SELECT key FROM \"{self.tablename}\"'):\n+            query = f'SELECT key FROM \"{self.tablename}\"'\n+            for key in self.conn.execute(query):\n                 yield key[0]\n         finally:\n             self._execute('COMMIT')\n@@ -62,6 +64,7 @@ def values(self):\n         try:\n-            for value in self.conn.execute(f'SELECT value FROM \"{self.tablename}\"'):\n+            query = f'SELECT value FROM \"{self.tablename}\"'\n+            for value in self.conn.execute(query):\n                 yield self._decode(value[0])\n         finally:\n             self._execute('COMMIT')\n@@ -217,7 +220,8 @@ def _new_conn(self):\n         conn.execute('PRAGMA synchronous=FULL')\n         return conn\n \n \n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -25,7 +25,7 @@ def test_reopen_conn(self):\n         \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n-        db = SqliteDict(filename=fname)\n+        db = SqliteDict(filename=fname, flag='n')\n         with db:\n             db['key'] = 'value'\n             db.commit()\n",
    "patch_ground_all": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -128,8 +128,10 @@\n         next_db['key'] = 'new_value'\n         next_db.commit()\n         next_db.close()\n+        # Ensure the old value does not exist in the new reopened database with 'n' flag\n         self.assertNotIn('value', next_db.values())\n         self.assertIn('key', next_db.keys())\n+        self.assertEqual(next_db['key'], 'new_value')\n\n     def test_keys(self):\n         \"\"\"Verify keys() method.\"\"\"\n@@ -177,7 +179,10 @@\n         self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n         with self.assertRaises(RuntimeError):\n-            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n+            readonly_db = SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n+            self.assertIn(dummy_tablename, readonly_db.keys())\n+            self.assertNotIn('unexpected_tablename', readonly_db.keys())\n+            readonly_db.close()\n\n     def test_directory_notfound(self):\n         \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n",
    "patch_ground_truth": "--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -130,7 +130,6 @@\n         # verify,\n         next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n         self.assertNotIn('key', next_db.keys())\n-        self.assertNotIn('value', next_db.values())\n \n     def test_unrecognized_flag(self):\n \n@@ -187,7 +186,7 @@\n         orig_db.commit()\n         orig_db.close()\n \n-        self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\n+        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n \n         with self.assertRaises(RuntimeError):\n             SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n",
    "message": "======================================================================\nFAIL: test_readonly_table (test_core.NamedSqliteDictCreateOrReuseTest)\nRead-only access on a non-existent tablename should raise RuntimeError,\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 190, in test_readonly_table\n    self.assertTrue(dummy_tablename in SqliteDict.get_tablenames(fname))\nAssertionError: False is not true\n\n----------------------------------------------------------------------\nRan 88 tests in 1.157s\n\nFAILED (failures=1)",
    "CodeBase": [
      {
        "path": "piskvorky_sqlitedict/tests/test_core.py",
        "content": "1 # std imports\n2 import json\n3 import unittest\n4 import tempfile\n5 import os\n6 from unittest.mock import patch\n7 \n8 # local\n9 import sqlitedict\n10 from sqlitedict import SqliteDict\n11 from test_temp_db import TempSqliteDictTest\n12 from accessories import norm_file\n13 \n14 \n15 class SqliteMiscTest(unittest.TestCase):\n16 \n17     def test_with_statement(self):\n18         \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n19         with SqliteDict() as d:\n20             self.assertTrue(isinstance(d, SqliteDict))\n21             self.assertEqual(dict(d), {})\n22             self.assertEqual(list(d), [])\n23             self.assertEqual(len(d), 0)\n24 \n25     def test_reopen_conn(self):\n26         \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n27         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n28         db = SqliteDict(filename=fname)\n29         with db:\n30             db['key'] = 'value'\n31             db.commit()\n32         with db:\n33             db['key'] = 'value'\n34             db.commit()\n35 \n36     def test_as_str(self):\n37         \"\"\"Verify SqliteDict.__str__().\"\"\"\n38         # given,\n39         db = SqliteDict()\n40         # exercise\n41         db.__str__()\n42         # test when db closed\n43         db.close()\n44         db.__str__()\n45 \n46     def test_as_repr(self):\n47         \"\"\"Verify SqliteDict.__repr__().\"\"\"\n48         # given,\n49         db = SqliteDict()\n50         # exercise\n51         db.__repr__()\n52 \n53     def test_directory_notfound(self):\n54         \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n55         # given: a non-existent director(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/sqlitedict.py",
        "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 #\n9 # This code was inspired by:\n10 #  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n11 #  * http://code.activestate.com/recipes/526618/\n12 \n13 \"\"\"\n14 A lightweight wrapper around Python's sqlite3 database, with a dict-like interface\n15 and multi-thread access support::\n16 \n17 >>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n18 >>> mydict['some_key'] = any_picklable_object\n19 >>> print mydict['some_key']\n20 >>> print len(mydict) # etc... all dict functions work\n21 \n22 Pickle is used internally to serialize the values. Keys are strings.\n23 \n24 If you don't use autocommit (default is no autocommit for performance), then\n25 don't forget to call `mydict.commit()` when done with a transaction.\n26 \n27 \"\"\"\n28 \n29 import sqlite3\n30 import os\n31 import sys\n32 import tempfile\n33 import threading\n34 import logging\n35 import traceback\n36 from base64 impo(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/setup.py",
        "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 \n9 \"\"\"\n10 Run with:\n11 \n12 python ./setup.py install\n13 \"\"\"\n14 \n15 import os\n16 import io\n17 import subprocess\n18 \n19 import setuptools.command.develop\n20 from setuptools import setup\n21 \n22 \n23 def read(fname):\n24     path = os.path.join(os.path.dirname(__file__), fname)\n25     return io.open(path, encoding='utf8').read()\n26 \n27 \n28 class SetupDevelop(setuptools.command.develop.develop):\n29     \"\"\"Docstring is overwritten.\"\"\"\n30 \n31     def run(self):\n32         \"\"\"\n33         Prepare environment for development.\n34 \n35         - Ensures 'nose' and 'coverage.py' are installed for testing.\n36         - Call s(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
        "content": "1 import os\n2 import sys\n3 \n4 import sqlitedict\n5 \n6 \n7 def test():\n8     \"Verify autocommit just before program exits.\"\n9     assert os.system('env PYTHONPATH=. %s tests/autocommit.py' % sys.executable) == 0\n10     # The above script relies on the autocommit feature working correctly.\n11     # Now, let's check if it actually worked.\n12     d = sqlitedict.SqliteDict('tests/db/autocommit.sqlite')\n13     for i in range(1000):\n14         assert d[i] == i, \"actual: %s expected: %s\" % (d[i], i)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_onimport.py",
        "content": "1 \"\"\"Test cases for on-import logic.\"\"\"\n2 import unittest\n3 import sys\n4 \n5 \n6 class SqliteDict_cPickleImportTest(unittest.TestCase):\n7     \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n8     def setUp(self):\n9         self.orig_meta_path = sys.meta_path\n10         self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n11 \n12  (...truncated)"
      }
    ],
    "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 4,
      "Description": 5,
      "Reproducibility": 3,
      "Relevance": 6,
      "Explanation": 6,
      "Overall": 5
    },
    "issue_message": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 6,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 7,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_ground_truth": {
      "title": "Issues with Database Table Verification and Key-Value Existence Checks",
      "description": "There have been inconsistencies detected in the database table verification and key-value existence checks in the `SqliteDict` module. Specifically, when re-opening a database with the flag `'n'`, the expected behavior is that all existing tables should be erased, but it appears not to be the case.\n\nFurthermore, the checks for key existence performed through `next_db.keys()` and `next_db.values()` yield unexpected results. For example, a key may exist but the corresponding value check fails, leading to incorrect assumptions about the database state.\n\nAnother critical issue is observed in read-only table access, where unexpected tables are being listed even when access should be restricted, causing security and data integrity concerns.\n\nThese issues impact the reliability and correctness of database operations, warranting an urgent review and fix to ensure accurate database table management and key-value existence verifications.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `SqliteDict` module and relates to three main areas:\n1. **Database Re-opening with 'n' Flag**: When re-opening a database with the 'n' flag, the expected behavior is to erase all existing tables. However, inconsistencies were detected indicating this action does not always occur.\n2. **Key-Value Existence Checks**: During database operations, checks for key existence using `next_db.keys()` and `next_db.values()` yield unexpected results, with cases where a key appears to exist but the corresponding value check fails.\n3. **Read-Only Table Access**: In cases where database access should be read-only, unexpected tables can be listed, leading to security and data integrity concerns.\n\nThese issues collectively reduce the reliability and correctness of database operations, making it critical to review and fix them promptly.\n\n### Commit Content and Analysis\n\nTo address these issues, a commit was made with specific changes mainly targeting `tests/test_core.py`. Here's an overview of the modifications and their implications:\n\n1. **Cleanup During Database Re-opening**:\n   - When a database is re-opened with the 'n' flag, the tables should be wiped clean. The test cases were updated to ensure that tables and their contents are correctly erased during re-opening. This addresses inconsistencies with table erasure, ensuring the database behaves as expected by removing existing data upon re-opening.\n\n2. **Improved Key-Value Assertions**:\n   - The test case that previously checked both key and value existence now focuses on keys only, removing the redundant value assertion. By refining these checks, the tests more accurately reflect the database's internal state without confusing key existence with corresponding value checks. This likely resolves false negatives where a key existence check would pass, but the associated value check would fail.\n\n3. **Enhanced Read-Only Access Verification**:\n   - Adjustments were made to ensure that when a database is accessed in read-only mode, no unintended tables are listed. Tests now properly account for and verify that unexpected tables are not improperly accessed or displayed, addressing a significant security and data integrity concern.\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Database Re-opening Cleanup**:\n   - By ensuring that the test cases robustly check for complete table erasure during database re-opening with the 'n' flag, the commit guarantees the database starts afresh with no residual data. This precisely addresses the complaint of leftover tables when a database is expected to be fully reset.\n\n2. **Key-Value Existence Checks**:\n   - The update to the tests ensures they do not mistakenly interpret key existence and value presence as the same operation. This clearer delineation prevents the confusion that led to false assumptions about the database state. Operational reliability is enhanced by ensuring that keys and their corresponding values are verified appropriately and separately.\n\n3. **Read-Only Table Access**:\n   - The revised tests check more rigorously whether unintended tables appear during read-only access attempts. This fortifies the system against the inadvertent exposure of or interaction with tables that should stay concealed, thereby maintaining the integrity and security of the data.\n\n### Solution to the Issue\n\nThe issue was addressed by:\n- Ensuring the database correctly erases all tables upon re-opening with the 'n' flag.\n- Clarifying and accurately checking for key and value existence to prevent erroneous assumptions about database states.\n- Strengthening the validation of read-only access to keep unintended tables from appearing.\n\n### Conclusion\n\nThe commit effectively solves the highlighted issues by making precise and necessary adjustments to the test cases. These adjustments improve the reliability and security of the `SqliteDict` module, ensuring that database operations behave as expected, correctly manage key-value existence checks, and maintain data integrity, especially in read-only access scenarios."
    }
  }
}