{
  "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                return self\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.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"
    }
  ],
  "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": "======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 616, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 606, in check_raise_error\n    reraise(e_type, e_value, e_tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 46, in reraise\n    raise value.with_traceback(tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 474, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nERROR: test_cpickle_fallback_to_pickle (test_onimport.SqliteDict_cPickleImportTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_onimport.py\", line 36, in test_cpickle_fallback_to_pickle\n    sqlitedict = __import__(\"sqlitedict\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_onimport.py\", line 20, in load_module\n    raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\nImportError: No module named sqlitedict (FauxMissingImport)\n\n----------------------------------------------------------------------\nRan 88 tests in 1.129s\n\nFAILED (errors=2)",
  "Patch": "--- a/piskvorky_sqlitedict/tests/test_onimport.py\n+++ b/piskvorky_sqlitedict/tests/test_onimport.py\n@@ -14,7 +14,9 @@\n                 self.module_names = args\n \n             def find_module(self, fullname, path=None):\n-                return self\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--- 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",
  "BuggyCodeLocation": [
    {
      "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"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_onimport.py",
      "function": {
        "8": "setUp"
      },
      "content_all": {
        "14": "                self.module_names = args\n",
        "15": "\n",
        "16": "            def find_module(self, fullname, path=None):\n",
        "17": "                return self\n",
        "18": "\n",
        "19": "            def load_module(self, name):\n",
        "20": "                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n"
      },
      "content_change": {
        "17": "                return self\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_onimport.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Bug in FauxMissingImport Handling and Incorrect Assertion in SqliteDict Tests",
    "description": "### Description\n\nThere are two issues observed in the current implementation:\n1. **FauxMissingImport Handling Issue**: When importing the `cPickle` module or the fallback `pickle` module, the logic does not correctly identify whether the module is missing. Specifically, the `find_module` method always returns a module finder object, even if the module is not in the specified list of module names.\n2. **Incorrect Assertion in SqliteDict Tests**: The test case mistakenly asserts the presence of a value in `next_db.values()`, which is not yielding the expected results.\n\n### Impact\n\n1. **FauxMissingImport Handling Issue**: This could lead to false positives when detecting missing modules, causing unnecessary ImportErrors and unexpected behavior during module loading.\n2. **Incorrect Assertion in SqliteDict Tests**: This results in test failures that prevent the successful validation and deployment of the SqliteDict library.\n\n### Steps to Reproduce\n\n1. **FauxMissingImport Handling Issue**:\n   - Simulate the absence of the `cPickle` module using a custom module finder object.\n   - Attempt to import the `sqlitedict` module.\n   - The import should fail with `ImportError`, but it incorrectly returns the custom finder object.\n\n2. **Incorrect Assertion in SqliteDict Tests**:\n   - Run the test suite for SqliteDict.\n   - Observe the specific test case where it checks for values in `next_db.values()` after re-opening the database.\n   - The test should fail due to the incorrect assertion.\n\n### Expected Behavior\n\n1. The `find_module` method should return None if the module is not in the specified list of module names.\n2. The test case should correctly assert the expected values or revise the checking mechanism in the `next_db` after re-opening.\n\n### Suggested Fixes\nWhile direct modification guidance isn't provided here, the necessary changes involve refining the logic for module import detection and correcting the test assertions to align with the expected behavior.\n\nThese modifications will ensure accurate module handling and reliable test validation, leading to enhanced robustness and correctness of the SqliteDict library.",
    "explanation": "### Summary of the Issue\n\nThe issue report highlights two primary problems within the `sqlitedict` library:\n\n1. **FauxMissingImport Handling Issue**: The mechanism responsible for detecting whether the `cPickle` module or its fallback `pickle` module is present fails to properly identify the absence of these modules. This results in the function always returning a module finder object regardless of whether the module is actually present.\n   \n2. **Incorrect Assertion in SqliteDict Tests**: A specific test case in the SqliteDict tests makes an incorrect assertion by checking for a value in `next_db.values()`, leading to unexpected test failures.\n\n### Commit Overview\n\nTo address these issues, a recent commit made the following changes:\n\n1. For the **FauxMissingImport Handling Issue**, the commit updates the logic to ensure the `find_module` method only returns the custom module finder object if the module is indeed part of the specified list of module names. If the module name is not present in the list, it returns `None`.\n\n2. For the **Incorrect Assertion in SqliteDict Tests**, the commit fixes the test case by removing the incorrect assertion related to checking values in `next_db.values()`.\n\n### Explanation of the Issues and Resolutions\n\n#### FauxMissingImport Handling Issue\n\n**Cause of the Issue:**\nThe `find_module` method within the `FauxMissingImport` class was designed to emulate the behavior of a module finder object. The logic was too simplistic and always returned an object, even if the module was not found. This resulted in false positives, where the absence of a module like `cPickle` would not be correctly identified, leading to unnecessary `ImportError` exceptions or other unexpected behaviors during the module import process.\n\n**Solution Explanation:**\nThe commit refined the `find_module` function to first check if the module's name is in its list of expected module names. Only if the module name matches one of the intended names (like `cPickle`), it returns the module finder object. Otherwise, it returns `None`, indicating that the module wasn't found. This accurately emulates the behavior of a missing module and ensures that the correct fallback mechanisms are activated (e.g., falling back to `pickle` if `cPickle` is not found).\n\n#### Incorrect Assertion in SqliteDict Tests\n\n**Cause of the Issue:**\nA test case within the `test_core.py` file performed an incorrect assertion by checking for a specific value within `next_db.values()`. This caused test failures when the expected value was not found, preventing the successful validation and deployment of the library.\n\n**Solution Explanation:**\nThe commit addressed this by removing the erroneous assertion. This change ensures that the test case only checks relevant conditions—such as the presence of keys—without overreaching into conditions that are incorrect or unreliable. By eliminating the faulty assertion, the tests now align with the expected, valid behavior of the `SqliteDict` class.\n\n### Detailed Explanation of the Fixes\n\n1. **Better Module Import Detection**:\n    - The corrected `find_module` function ensures only relevant module names are handled by returning the module finder object conditionally.\n    - This improves the robustness and reliability of the faux module import mechanism, reducing false positives and misdirected exceptions.\n\n2. **Improved Test Assertions**:\n    - By removing the incorrect assertion, the tests now focus on verifying key existence and other relevant conditions.\n    - This prevents test failures triggered by invalid assumptions and ensures accurate testing of the `SqliteDict` functionalities.\n\n### Conclusion\n\nThe developer's commit effectively addresses both highlighted issues. The adjustment in the module finding logic ensures accurate detection and handling of missing modules, reducing unexpected behavior. Simultaneously, the correction in the test suite ensures reliable and valid validation of the project's functionalities. These changes enhance the overall stability and correctness of the `sqlitedict` library, fostering a more robust software development lifecycle."
  },
  "Explain": "### Summary of the Issue\n\nThe issue report highlights two primary problems within the `sqlitedict` library:\n\n1. **FauxMissingImport Handling Issue**: The mechanism responsible for detecting whether the `cPickle` module or its fallback `pickle` module is present fails to properly identify the absence of these modules. This results in the function always returning a module finder object regardless of whether the module is actually present.\n   \n2. **Incorrect Assertion in SqliteDict Tests**: A specific test case in the SqliteDict tests makes an incorrect assertion by checking for a value in `next_db.values()`, leading to unexpected test failures.\n\n### Commit Overview\n\nTo address these issues, a recent commit made the following changes:\n\n1. For the **FauxMissingImport Handling Issue**, the commit updates the logic to ensure the `find_module` method only returns the custom module finder object if the module is indeed part of the specified list of module names. If the module name is not present in the list, it returns `None`.\n\n2. For the **Incorrect Assertion in SqliteDict Tests**, the commit fixes the test case by removing the incorrect assertion related to checking values in `next_db.values()`.\n\n### Explanation of the Issues and Resolutions\n\n#### FauxMissingImport Handling Issue\n\n**Cause of the Issue:**\nThe `find_module` method within the `FauxMissingImport` class was designed to emulate the behavior of a module finder object. The logic was too simplistic and always returned an object, even if the module was not found. This resulted in false positives, where the absence of a module like `cPickle` would not be correctly identified, leading to unnecessary `ImportError` exceptions or other unexpected behaviors during the module import process.\n\n**Solution Explanation:**\nThe commit refined the `find_module` function to first check if the module's name is in its list of expected module names. Only if the module name matches one of the intended names (like `cPickle`), it returns the module finder object. Otherwise, it returns `None`, indicating that the module wasn't found. This accurately emulates the behavior of a missing module and ensures that the correct fallback mechanisms are activated (e.g., falling back to `pickle` if `cPickle` is not found).\n\n#### Incorrect Assertion in SqliteDict Tests\n\n**Cause of the Issue:**\nA test case within the `test_core.py` file performed an incorrect assertion by checking for a specific value within `next_db.values()`. This caused test failures when the expected value was not found, preventing the successful validation and deployment of the library.\n\n**Solution Explanation:**\nThe commit addressed this by removing the erroneous assertion. This change ensures that the test case only checks relevant conditions—such as the presence of keys—without overreaching into conditions that are incorrect or unreliable. By eliminating the faulty assertion, the tests now align with the expected, valid behavior of the `SqliteDict` class.\n\n### Detailed Explanation of the Fixes\n\n1. **Better Module Import Detection**:\n    - The corrected `find_module` function ensures only relevant module names are handled by returning the module finder object conditionally.\n    - This improves the robustness and reliability of the faux module import mechanism, reducing false positives and misdirected exceptions.\n\n2. **Improved Test Assertions**:\n    - By removing the incorrect assertion, the tests now focus on verifying key existence and other relevant conditions.\n    - This prevents test failures triggered by invalid assumptions and ensures accurate testing of the `SqliteDict` functionalities.\n\n### Conclusion\n\nThe developer's commit effectively addresses both highlighted issues. The adjustment in the module finding logic ensures accurate detection and handling of missing modules, reducing unexpected behavior. Simultaneously, the correction in the test suite ensures reliable and valid validation of the project's functionalities. These changes enhance the overall stability and correctness of the `sqlitedict` library, fostering a more robust software development lifecycle.",
  "Source": "Human",
  "Token": 1359,
  "Command": [
    "python -m unittest discover -s tests"
  ],
  "FilteredCode": [
    {
      "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 import b64decode, b64encode\n37 import weakref\n38 \n39 __version__ = '2.1.0'\n40 \n41 \n42 def reraise(tp, value, tb=None):\n43     if value is None:\n44         value = tp()\n45     if value.__traceback__ is not tb:\n46         raise value.with_traceback(tb)\n47     raise value\n48 \n49 \n50 try:\n51     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n52 except ImportError:\n53     from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n54 \n55 # some Python 3 vs 2 im(...truncated)"
    },
    {
      "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__().(...truncated)"
    },
    {
      "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         class FauxMissingImport(object):\n13             def __init__(self, *args):\n14                 self.module_names = args\n15 \n16             def find_module(self, fullname, path=None):\n17                 return self\n18 \n19             def load_module(self, name):\n20                 raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n21 \n22         # ensure cPickle/pickle is not cached\n23         sys.modules.pop('cPickle', None)\n24         sys.modules.pop('pickle', None)\n25 \n26         # add our custom importer\n27         sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n28 \n29     def tearDown(self):\n30         sys.meta_path = self.orig_meta_path\n31         if self.orig_sqlitedict:\n32             sys.modules['sqlitedict'] = self.orig_sqlitedict\n33 \n34     def test_cpickle_fallback_to_pickle(self):\n35         # exercise,\n36         sqlitedict = __import__(\"sqlitedict\")\n37         # verify,\n38         self.assertIn('pickle', sys.modules.keys())\n39         self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_named_db.py",
      "content": "1 import sqlitedict\n2 from test_temp_db import TempSqliteDictTest\n3 from accessories import norm_file\n4 \n5 \n6 class InMemorySqliteDictTest(TempSqliteDictTest):\n7 \n8     def setUp(self):\n9         self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n10 \n11     def tearDown(self):\n12         self.d.terminate()\n13 \n14 \n15 class NamedSqliteDictTest(TempSqliteDictTest):\n16 \n17    (...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 ./s(...truncated)"
    }
  ],
  "TokenAll": 13027,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Potential Context Manager Misuse",
      "description": "In piskvorky_sqlitedict/tests/test_core.py, the test_reopen_conn method reopens the same SqliteDict instance within nested context managers without reinitialization. This can lead to unexpected behavior or resource locking issues. Consider initializing a new SqliteDict instance for each with block to avoid potential problems.",
      "explanation": "### Issue Summary\n\nThe provided issue reports a potential misuse of context managers in the test file `piskvorky_sqlitedict/tests/test_core.py`, specifically within the `test_reopen_conn` method. The method attempts to reopen an existing `SqliteDict` instance inside nested context managers without reinitializing it. This misuse can lead to unexpected behavior such as resource locking issues or other concurrency-related troubles.\n\n### Reviewing the `test_reopen_conn` Method\n\nHere is the problematic method extracted from the code:\n```python\ndef test_reopen_conn(self):\n    \"\"\"Verify using a context manager 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\nIn this code, the same `db` instance is used in two separate `with` context managers. However, the issue arises because the `db` instance is not reinitialized between these context managers, which may cause unexpected behavior or lock the database resources.\n\n### Commit to Resolve the Issue\n\n**Commit Details:**\n```diff\ndef test_reopen_conn(self):\n    \"\"\"Verify using a context manager that a connection can be reopened.\"\"\"\n    fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n    - db = SqliteDict(filename=fname)\n    with SqliteDict(filename=fname) as db:\n        db['key'] = 'value'\n        db.commit()\n    with SqliteDict(filename=fname) as db:\n        db['key'] = 'value'\n        db.commit()\n```\n\n### How the Commit Solves the Problem\n\n1. **Reinitialization:** The modified code creates a new `SqliteDict` instance within each `with` block. This ensures that any resources used by the SqliteDict in the previous context are properly released before a new instance is created.\n2. **Context Management:** By ensuring that the `SqliteDict` instances are properly managed within their respective context managers, we avoid any possible resource locking or unexpected behavior resulting from multiple context managers operating on a single instance.\n3. **Isolation:** Each `with` block now operates independently, and the lifecycle of each `SqliteDict` instance is correctly managed. This improves the reliability and isolation of the test cases, ensuring that each context operates in its own sandbox.\n\n### Detailed Explanation\n\n- **Old Implementation:** \n  - The same `db` instance is reused in the two `with` blocks. When exiting the first `with` block, the context manager's `__exit__` method is called, which is supposed to clean up resources. However, if you re-enter with the same instance, the state might not be as expected and resource locks might still be in place.\n  - Keeping the same instance can lead to connections not being properly closed, causing database resource locks that the second `with` block cannot resolve.\n\n- **New Implementation:** \n  - The code within each `with` statement initializes a new `SqliteDict` instance: `with SqliteDict(filename=fname) as db:`.\n  - Each block creates a fresh instance of `SqliteDict` which means a new connection to the SQLite database is made in each context. This avoids any interference or locking issues and ensures clean resource management.\n  - The instances are correctly closed and cleaned up at the end of each `with` block. This makes each database operation isolated and much safer in the context of accessing resources.\n\nBy making these changes, the potential misuse of context managers is circumvented, leading to a more robust and reliable test that can safely handle database connections. This ensures that resource locking issues and other unexpected behaviors are avoided."
    },
    "issue_message": {
      "title": "Database connection error in SqliteDict initialization",
      "description": "The test test_terminate_cannot_delete in test_core.py fails due to an inability to open the database file. The error originates from SqliteDict.__init__ and sqlite3.connect calls in sqlitedict.py. Ensure that the database file path is correct and accessible, and check for any file system permissions issues.",
      "explanation": "### Issue Summary\n\nThe issue titled \"Database connection error in SqliteDict initialization\" points out a problem where the `test_terminate_cannot_delete` test case fails because it cannot open the database file. The error occurs within the `SqliteDict.__init__` method and specifically at the `sqlite3.connect` call, suggesting potential issues with the database file path or file system permissions.\n\n### Detailed Analysis\n\n#### `sqlitedict.py`\n\nThe provided segment of `sqlitedict.py` shows the initial context but does not include the core `SqliteDict` class or its `__init__` method. However, we know from the error that the method `_connect` is causing the issue:\n\n```python\n474 conn = sqlite3.connect(self.filename, check_same_thread=False)\n```\n\n#### `test_core.py`\n\nRelevant parts of the `test_core.py` snippet include:\n\n```python\n254 self.db = SqliteDict(filename=self.fname)\n```\n\nThis line initializes `SqliteDict` with a specified `filename`, but it fails due to an invalid or inaccessible file path.\n\n### Error Message Details\n\nThe important part of the error trace includes:\n\n```python\nsqlite3.OperationalError: unable to open database file\n```\n\nThis indicates either:\n- `self.filename` is incorrect, e.g., directory does not exist.\n- File system permissions prevent access.\n\n### Potential Causes\n\n1. **Incorrect File Path**: The path to the database file might be incorrect or the directories leading to the file might not exist.\n2. **File Permissions**: The process running the test might not have the required permissions to access or create the file at the specified location.\n\n### Suggested Solution\n\nEnsure the file path exists and is accessible:\n- Create necessary directories.\n- Ensure file write permissions are set.\n\n### Commit and Explanation\n\nSuppose we have a commit titled \"Fix database connection issue by ensuring directory creation and handling file permissions.\"\n\n#### Commit Content\n\n```python\ndef _connect(self):\n    try:\n        # Ensure directory exists\n        os.makedirs(os.path.dirname(self.filename), exist_ok=True)\n    except OSError as e:\n        # Handle directory creation failure\n        logging.error(\"Failed to create directory for database file: %s\", e)\n        raise\n\n    try:\n        conn = sqlite3.connect(self.filename, check_same_thread=False)\n    except sqlite3.OperationalError as e:\n        logging.error(\"Unable to open database file %s: %s\", self.filename, e)\n        raise\n\n    return conn\n```\n\n#### Explanation\n\n1. **Directory Creation (`os.makedirs`)**:\n    - Before attempting the SQLite connection, ensure the directories for the filepath exist. `os.makedirs(os.path.dirname(self.filename), exist_ok=True)` ensures that any necessary directories are created. This handles cases where intermediate directories might not exist.\n    - The `exist_ok=True` parameter prevents raising an error if the directory already exists.\n\n2. **Connection Attempt with Error Handling**:\n    - Attempt to connect to the database file using `sqlite3.connect`.\n    - If an `sqlite3.OperationalError` is raised, log an appropriate error message and re-raise the exception. This helps in debugging by providing clearer insights into why the connection might be failing.\n\nOverall, this ensures that the necessary file paths are in place before attempting to access the SQLite database file, thereby mitigating the cause of the described issue."
    },
    "issue_ground": {
      "title": "Bug in FauxMissingImport Handling and Incorrect Assertion in SqliteDict Tests",
      "description": "### Description\n\nThere are two issues observed in the current implementation:\n1. **FauxMissingImport Handling Issue**: When importing the `cPickle` module or the fallback `pickle` module, the logic does not correctly identify whether the module is missing. Specifically, the `find_module` method always returns a module finder object, even if the module is not in the specified list of module names.\n2. **Incorrect Assertion in SqliteDict Tests**: The test case mistakenly asserts the presence of a value in `next_db.values()`, which is not yielding the expected results.\n\n### Impact\n\n1. **FauxMissingImport Handling Issue**: This could lead to false positives when detecting missing modules, causing unnecessary ImportErrors and unexpected behavior during module loading.\n2. **Incorrect Assertion in SqliteDict Tests**: This results in test failures that prevent the successful validation and deployment of the SqliteDict library.\n\n### Steps to Reproduce\n\n1. **FauxMissingImport Handling Issue**:\n   - Simulate the absence of the `cPickle` module using a custom module finder object.\n   - Attempt to import the `sqlitedict` module.\n   - The import should fail with `ImportError`, but it incorrectly returns the custom finder object.\n\n2. **Incorrect Assertion in SqliteDict Tests**:\n   - Run the test suite for SqliteDict.\n   - Observe the specific test case where it checks for values in `next_db.values()` after re-opening the database.\n   - The test should fail due to the incorrect assertion.\n\n### Expected Behavior\n\n1. The `find_module` method should return None if the module is not in the specified list of module names.\n2. The test case should correctly assert the expected values or revise the checking mechanism in the `next_db` after re-opening.\n\n### Suggested Fixes\nWhile direct modification guidance isn't provided here, the necessary changes involve refining the logic for module import detection and correcting the test assertions to align with the expected behavior.\n\nThese modifications will ensure accurate module handling and reliable test validation, leading to enhanced robustness and correctness of the SqliteDict library.",
      "explanation": "### Issue Summary:\nThe provided information highlights two specific issues with the `sqlitedict` library:\n1. **FauxMissingImport Handling Issue**: The current method used to handle module imports (`find_module` method) does not correctly identify whether the module is missing. It always returns a module finder object, leading to false positives for missing modules.\n2. **Incorrect Assertion in SqliteDict Tests**: In the test suite for `SqliteDict`, there is an incorrect assertion where the test checks for a value in `next_db.values()`, yielding unexpected results and causing test failures.\n\n### Commit Content:\nTo address these issues, let's break down the necessary changes:\n\n1. **Fixing FauxMissingImport Handling**:\n    - **Original Code:**\n      ```python\n      def find_module(self, fullname, path=None):\n          return self\n      ```\n      The method `find_module` in the `FauxMissingImport` class returns `self` unconditionally, which means it always indicates that the module was found even if it is not present.\n\n    - **Revised Code:**\n      ```python\n      def find_module(self, fullname, path=None):\n          if fullname in self.module_names:\n              return self\n          return None\n      ```\n      This modification ensures that `find_module` only returns the finder object if the module name is in `self.module_names`. Otherwise, it correctly returns `None`, indicating the module is not found.\n\n2. **Correcting the Test Assertion in SqliteDict Tests**:\n    - **Original Code:**\n      ```python\n      self.assertIn('expected_value', next_db.values())\n      ```\n      The original test might be asserting the existence of a value directly in the dictionary values, but this is failing due to how values are hashed or stored.\n\n    - **Revised Code:**\n      Instead, you may want to iterate and check the values:\n      ```python\n      expected_values = ['expected_value']\n      self.assertListEqual(sorted(next_db.values()), sorted(expected_values))\n      ```\n      This ensures that the test correctly checks for the expected values in `next_db`.\n\n### Cause of the Issue:\n1. **FauxMissingImport Handling Issue**: The primary cause is that the `find_module` method does not correctly indicate when a module is missing, leading to false positives. This causes the import mechanisms to behave unexpectedly and can result in an `ImportError` if re-imported assumptions are inaccurate.\n  \n2. **Incorrect Assertion in SqliteDict Tests**: The test suite assumes that checking for direct values would work seamlessly, but the method does not properly handle the complexity of the underlying data structures in `SqliteDict`.\n\n### How the Commit Solves the Issue:\n1. **Fixing FauxMissingImport Handling**:\n    - The revised `find_module` method now properly checks if the requested module name is one of those intended to be simulated as missing. This change directly resolves the problem by ensuring it only returns the finder object for the modules that are being faked as missing, and returns `None` for others, preventing false positives and unnecessary import errors.\n  \n    - **Example Outcome**: When the `sqlitedict` library attempts to import `cPickle` and it is simulated as missing, the fallback to `pickle` happens correctly without false positives, ensuring smooth operation.\n\n2. **Correcting the Test Assertion in SqliteDict Tests**:\n    - By changing the assertion to compare sorted lists of values, the test now correctly validates the state of `next_db` after re-opening. This ensures the tests are aligned with the expected results, removing the test failures.\n  \n    - **Example Outcome**: When the test suite is run, it now correctly verifies the stored values in `SqliteDict`, making the library's validation reliable.\n\n### Detail of the Commit Changes:\n1. **FauxMissingImport Class:**\n    ```python\n    def find_module(self, fullname, path=None):\n        if fullname in self.module_names:\n            return self\n        return None\n    ```\n\n2. **Test Correction:**\n    ```python\n    expected_values = ['expected_value']\n    self.assertListEqual(sorted(next_db.values()), sorted(expected_values))\n    ```\n\nIn essence, the commit ensures better handling of module imports and corrects the tests to reflect intended operations, providing enhanced robustness and reliability for the `sqlitedict` library."
    },
    "issue_ground_truth": {
      "title": "Bug in FauxMissingImport Handling and Incorrect Assertion in SqliteDict Tests",
      "description": "### Description\n\nThere are two issues observed in the current implementation:\n1. **FauxMissingImport Handling Issue**: When importing the `cPickle` module or the fallback `pickle` module, the logic does not correctly identify whether the module is missing. Specifically, the `find_module` method always returns a module finder object, even if the module is not in the specified list of module names.\n2. **Incorrect Assertion in SqliteDict Tests**: The test case mistakenly asserts the presence of a value in `next_db.values()`, which is not yielding the expected results.\n\n### Impact\n\n1. **FauxMissingImport Handling Issue**: This could lead to false positives when detecting missing modules, causing unnecessary ImportErrors and unexpected behavior during module loading.\n2. **Incorrect Assertion in SqliteDict Tests**: This results in test failures that prevent the successful validation and deployment of the SqliteDict library.\n\n### Steps to Reproduce\n\n1. **FauxMissingImport Handling Issue**:\n   - Simulate the absence of the `cPickle` module using a custom module finder object.\n   - Attempt to import the `sqlitedict` module.\n   - The import should fail with `ImportError`, but it incorrectly returns the custom finder object.\n\n2. **Incorrect Assertion in SqliteDict Tests**:\n   - Run the test suite for SqliteDict.\n   - Observe the specific test case where it checks for values in `next_db.values()` after re-opening the database.\n   - The test should fail due to the incorrect assertion.\n\n### Expected Behavior\n\n1. The `find_module` method should return None if the module is not in the specified list of module names.\n2. The test case should correctly assert the expected values or revise the checking mechanism in the `next_db` after re-opening.\n\n### Suggested Fixes\nWhile direct modification guidance isn't provided here, the necessary changes involve refining the logic for module import detection and correcting the test assertions to align with the expected behavior.\n\nThese modifications will ensure accurate module handling and reliable test validation, leading to enhanced robustness and correctness of the SqliteDict library.",
      "explanation": "### Summary of the Issue\n\nThe issue report highlights two primary problems within the `sqlitedict` library:\n\n1. **FauxMissingImport Handling Issue**: The mechanism responsible for detecting whether the `cPickle` module or its fallback `pickle` module is present fails to properly identify the absence of these modules. This results in the function always returning a module finder object regardless of whether the module is actually present.\n   \n2. **Incorrect Assertion in SqliteDict Tests**: A specific test case in the SqliteDict tests makes an incorrect assertion by checking for a value in `next_db.values()`, leading to unexpected test failures.\n\n### Commit Overview\n\nTo address these issues, a recent commit made the following changes:\n\n1. For the **FauxMissingImport Handling Issue**, the commit updates the logic to ensure the `find_module` method only returns the custom module finder object if the module is indeed part of the specified list of module names. If the module name is not present in the list, it returns `None`.\n\n2. For the **Incorrect Assertion in SqliteDict Tests**, the commit fixes the test case by removing the incorrect assertion related to checking values in `next_db.values()`.\n\n### Explanation of the Issues and Resolutions\n\n#### FauxMissingImport Handling Issue\n\n**Cause of the Issue:**\nThe `find_module` method within the `FauxMissingImport` class was designed to emulate the behavior of a module finder object. The logic was too simplistic and always returned an object, even if the module was not found. This resulted in false positives, where the absence of a module like `cPickle` would not be correctly identified, leading to unnecessary `ImportError` exceptions or other unexpected behaviors during the module import process.\n\n**Solution Explanation:**\nThe commit refined the `find_module` function to first check if the module's name is in its list of expected module names. Only if the module name matches one of the intended names (like `cPickle`), it returns the module finder object. Otherwise, it returns `None`, indicating that the module wasn't found. This accurately emulates the behavior of a missing module and ensures that the correct fallback mechanisms are activated (e.g., falling back to `pickle` if `cPickle` is not found).\n\n#### Incorrect Assertion in SqliteDict Tests\n\n**Cause of the Issue:**\nA test case within the `test_core.py` file performed an incorrect assertion by checking for a specific value within `next_db.values()`. This caused test failures when the expected value was not found, preventing the successful validation and deployment of the library.\n\n**Solution Explanation:**\nThe commit addressed this by removing the erroneous assertion. This change ensures that the test case only checks relevant conditions—such as the presence of keys—without overreaching into conditions that are incorrect or unreliable. By eliminating the faulty assertion, the tests now align with the expected, valid behavior of the `SqliteDict` class.\n\n### Detailed Explanation of the Fixes\n\n1. **Better Module Import Detection**:\n    - The corrected `find_module` function ensures only relevant module names are handled by returning the module finder object conditionally.\n    - This improves the robustness and reliability of the faux module import mechanism, reducing false positives and misdirected exceptions.\n\n2. **Improved Test Assertions**:\n    - By removing the incorrect assertion, the tests now focus on verifying key existence and other relevant conditions.\n    - This prevents test failures triggered by invalid assumptions and ensures accurate testing of the `SqliteDict` functionalities.\n\n### Conclusion\n\nThe developer's commit effectively addresses both highlighted issues. The adjustment in the module finding logic ensures accurate detection and handling of missing modules, reducing unexpected behavior. Simultaneously, the correction in the test suite ensures reliable and valid validation of the project's functionalities. These changes enhance the overall stability and correctness of the `sqlitedict` library, fostering a more robust software development lifecycle."
    },
    "location_origin": [
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "25": "test_reopen_conn"
        },
        "content_all": {
          "22": "    def test_reopen_conn(self):",
          "23": "        \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"",
          "24": "        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')",
          "25": "        db = SqliteDict(filename=fname)",
          "26": "        with db:",
          "27": "            db['key'] = 'value'",
          "28": "            db.commit()",
          "29": "        with db:",
          "30": "            db['key'] = 'value'",
          "31": "            db.commit()"
        },
        "content_change": {
          "25": "        with SqliteDict(filename=fname) as db:",
          "26": "            db['key'] = 'value'",
          "27": "            db.commit()",
          "28": "        with SqliteDict(filename=fname) as db:",
          "29": "            db['key'] = 'value'",
          "30": "            db.commit()"
        }
      }
    ],
    "location_message": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "229": "__init__"
        },
        "content_all": {
          "226": "        self.autocommit = autocommit\n",
          "227": "        self.timeout = timeout\n",
          "228": "        self.journal_mode = journal_mode\n",
          "229": "        self.conn = self._connect()\n",
          "230": "        self.conn.execute(MAKE_TABLE)\n",
          "231": "        self.conn.commit()\n",
          "232": "        self.tablename = tablename\n"
        },
        "content_change": {
          "229": "        self.conn = self._connect()"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "474": "_connect"
        },
        "content_all": {
          "470": "        logging.debug(\"Connecting to %s\" % (self.filename,))\n",
          "471": "        assert self.filename\n",
          "472": "\n",
          "473": "        try:\n",
          "474": "            conn = sqlite3.connect(self.filename, check_same_thread=False)\n",
          "475": "        except sqlite3.OperationalError as e:\n",
          "476": "            logging.error(\"Unable to open database file %s: %s\", self.filename, e)\n",
          "477": "            raise\n"
        },
        "content_change": {
          "474": "            conn = sqlite3.connect(self.filename, check_same_thread=False)"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "254": "setUp"
        },
        "content_all": {
          "251": "        self.fname = norm_file('tests/db/sqlitedict-terminate-fail.sqlite')\n",
          "252": "        self._os_unlink_noraise(self.fname)\n",
          "253": "        self.dbpath = norm_file('tests/db')\n",
          "254": "        os.makedirs(self.dbpath, exist_ok=True)\n",
          "255": "        self.db = SqliteDict(filename=self.fname)\n",
          "256": "\n",
          "257": "    def tearDown(self):\n",
          "258": "        self.db.terminate()\n"
        },
        "content_change": {
          "254": "        os.makedirs(self.dbpath, exist_ok=True)"
        }
      }
    ],
    "location_ground": [
      {
        "file": "piskvorky_sqlitedict/tests/test_onimport.py",
        "function": {
          "12": "FauxMissingImport.__init__"
        },
        "content_all": {
          "9": "        self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n",
          "10": "\n",
          "11": "        class FauxMissingImport(object):\n",
          "12": "            def __init__(self, *args):\n",
          "13": "                self.module_names = args\n",
          "14": "\n",
          "15": "            def find_module(self, fullname, path=None):\n",
          "16": "                return self\n",
          "17": "\n",
          "18": "            def load_module(self, name):\n",
          "19": "                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n",
          "20": "\n",
          "21": "        # ensure cPickle/pickle is not cached\n"
        },
        "content_change": {
          "15": "            def find_module(self, fullname, path=None):\n",
          "16": "                if fullname in self.module_names:\n",
          "17": "                    return self\n",
          "18": "                return None\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_named_db.py",
        "function": {
          "6": "InMemorySqliteDictTest.tearDown"
        },
        "content_all": {
          "10": "         self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n",
          "11": "\n",
          "12": "     def tearDown(self):\n",
          "13": "         self.d.terminate()\n",
          "14": "\n",
          "15": " \n",
          "16": " class NamedSqliteDictTest(TempSqliteDictTest):\n",
          "17": "\n",
          "18": "     def setUp(self):\n",
          "19": "         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n",
          "20": "         self.d = sqlitedict.SqliteDict(filename=fname, autocommit=True)\n",
          "21": "\n",
          "22": "     def tearDown(self):\n",
          "23": "         self.d.terminate()\n",
          "24": "\n",
          "25": "     def test_data_persistence(self):\n"
        },
        "content_change": {
          "13": "         self.d.terminate()\n",
          "19": "         expected_values = ['expected_value']\n",
          "20": "         self.assertListEqual(sorted(next_db.values()), sorted(expected_values))\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "50": "(top-level)"
        },
        "content_all": {
          "48": "",
          "49": "try:",
          "50": "    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL",
          "51": "except ImportError:",
          "52": "    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL",
          "53": ""
        },
        "content_change": {
          "50": "    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "25": "test_reopen_conn"
        },
        "content_all": {
          "24": "",
          "25": "    def test_reopen_conn(self):",
          "26": "        \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"",
          "27": "        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')",
          "28": "        db = SqliteDict(filename=fname)",
          "29": "        with db:",
          "30": "            db['key'] = 'value'",
          "31": "            db.commit()",
          "32": "        with db:",
          "33": "            db['key'] = 'value'",
          "34": "            db.commit()",
          "35": ""
        },
        "content_change": {
          "34": "            db.commit()"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_onimport.py",
        "function": {
          "16": "SqliteDict_cPickleImportTest"
        },
        "content_all": {
          "14": "             self.module_names = args",
          "15": "",
          "16": "             def find_module(self, fullname, path=None):",
          "17": "                 return self",
          "18": "",
          "19": "            def load_module(self, name):",
          "20": "                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))",
          "21": "",
          "22": "        # ensure cPickle/pickle is not cached",
          "23": "        sys.modules.pop('cPickle', None)",
          "24": "        sys.modules.pop('pickle', None)",
          "25": ""
        },
        "content_change": {
          "17": "                 return self"
        }
      }
    ],
    "location_ground_truth": [
      {
        "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"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_onimport.py",
        "function": {
          "8": "setUp"
        },
        "content_all": {
          "14": "                self.module_names = args\n",
          "15": "\n",
          "16": "            def find_module(self, fullname, path=None):\n",
          "17": "                return self\n",
          "18": "\n",
          "19": "            def load_module(self, name):\n",
          "20": "                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n"
        },
        "content_change": {
          "17": "                return self\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_onimport.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -25,10 +25,10 @@\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+        with SqliteDict(filename=fname) as db:\n             db['key'] = 'value'\n             db.commit()\n-        with db:\n+        with SqliteDict(filename=fname) as db:\n             db['key'] = 'value'\n             db.commit()\n \n",
    "patch_im": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -467,7 +467,17 @@\n             self.conn.text_factory = bytes\n \n     def _connect(self):\n-        conn = sqlite3.connect(self.filename, check_same_thread=False)\n+        try:\n+            # Ensure the directory exists\n+            os.makedirs(os.path.dirname(self.filename), exist_ok=True)\n+        except OSError as e:\n+            # Handle directory creation failure\n+            logging.error(\"Failed to create directory for database file: %s\", e)\n+            raise\n+\n+        try:\n+            conn = sqlite3.connect(self.filename, check_same_thread=False)\n+        except sqlite3.OperationalError as e:\n+            logging.error(\"Unable to open database file %s: %s\", self.filename, e)\n+            raise\n \n         if self.journal_mode is not None:\n             conn.execute('PRAGMA journal_mode = {}'.format(self.journal_mode))\n",
    "patch_il": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -25,8 +25,9 @@\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+        with SqliteDict(filename=fname) as db:\n             db['key'] = 'value'\n             db.commit()\n-        with db:\n+        with SqliteDict(filename=fname) as db:\n             db['key'] = 'value'\n             db.commit()\n",
    "patch_iml": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -229,6 +229,13 @@\n         self.autocommit = autocommit\n         self.timeout = timeout\n         self.journal_mode = journal_mode\n+        \n+        # Ensure directory exists\n+        try:\n+            os.makedirs(os.path.dirname(self.filename), exist_ok=True)\n+        except OSError as e:\n+            logging.error(\"Failed to create directory for database file: %s\", e)\n+            raise\n\n         self.conn = self._connect()\n         self.conn.execute(MAKE_TABLE)\n@@ -470,6 +477,13 @@\n     logging.debug(\"Connecting to %s\" % (self.filename,))\n     assert self.filename\n\n+    # Ensure directory exists\n+    try:\n+        os.makedirs(os.path.dirname(self.filename), exist_ok=True)\n+    except OSError as e:\n+        logging.error(\"Failed to create directory for database file: %s\", e)\n+        raise\n+\n     try:\n         conn = sqlite3.connect(self.filename, check_same_thread=False)\n     except sqlite3.OperationalError as e:\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -251,6 +251,7 @@\n         self.fname = norm_file('tests/db/sqlitedict-terminate-fail.sqlite')\n         self._os_unlink_noraise(self.fname)\n         self.dbpath = norm_file('tests/db')\n+        os.makedirs(self.dbpath, exist_ok=True)\n         self.db = SqliteDict(filename=self.fname)\n\n     def tearDown(self):\n",
    "patch_ground": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -50,7 +50,7 @@\n # some Python 3 vs 2 interoperability shims\n try:\n     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n-except ImportError:\n+except ModuleNotFoundError:\n     from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n \n # define a custom exception with more informative error messages\n--- a/piskvorky_sqlitedict/tests/test_onimport.py\n+++ b/piskvorky_sqlitedict/tests/test_onimport.py\n@@ -16,7 +16,9 @@\n \n             def find_module(self, fullname, path=None):\n                 if fullname in self.module_names:\n-                    return self\n+                    return None\n+                return None\n-\n+            return self\n             \n             def load_module(self, name):\n                 raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -33,7 +33,8 @@\n             db.commit()\n \n         with db:\n-            self.assertIn('value', next_db.values())\n+            expected_values = ['value']\n+            self.assertListEqual(sorted(next_db.values()), sorted(expected_values))\n \n     def test_with_statement(self):\n         \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n",
    "patch_ground_location": "\n--- a/piskvorky_sqlitedict/tests/test_onimport.py\n+++ b/piskvorky_sqlitedict/tests/test_onimport.py\n@@ -16,7 +16,9 @@\n \n             def find_module(self, fullname, path=None):\n-                return self\n+                if fullname in self.module_names:\n+                    return self\n+                return None\n \n --- a/piskvorky_sqlitedict/tests/test_named_db.py\n+++ b/piskvorky_sqlitedict/tests/test_named_db.py\n@@ -22,7 +22,9 @@\n \n     def test_data_persistence(self):\n         with self.d as db:\n             db['key'] = 'expected_value'\n         self.assertEqual(list(db.items()), [('key', 'expected_value')])\n-        self.assertIn('expected_value', db.values())\n+        expected_values = ['expected_value']\n+        with sqlitedict.SqliteDict(filename=db.filename) as next_db:\n+            self.assertListEqual(sorted(next_db.values()), sorted(expected_values))\n",
    "patch_ground_exp": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -48,6 +48,11 @@\n \n try:\n     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n-except ImportError:\n+except ImportError as e:\n+    import sys\n+    faux_import_error = sys.modules.get('cPickle') is None\n+    if not faux_import_error:\n+        raise e\n     from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n \n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -24,7 +24,9 @@\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+        fname = norm_file(':memory:')\n         db = SqliteDict(filename=fname)\n-        with db:\n+        with db as db1:\n             db1['key'] = 'value'\n             db1.commit()\n-        with db:\n+        with db as db2:\n+            self.assertEqual(db2['key'], 'value')\n+            self.assertIn('key', db2)\n             db2.commit()\n\n--- a/piskvorky_sqlitedict/tests/test_onimport.py\n+++ b/piskvorky_sqlitedict/tests/test_onimport.py\n@@ -16,7 +16,7 @@\n             def find_module(self, fullname, path=None):\n-                return self\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",
    "patch_ground_all": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -50,7 +50,9 @@ try:\n \n         class FauxMissingImport(object):\n             def __init__(self, *args):\n-                self.module_names = args\n+                self.module_names = set(args)  # Store as a set for quick lookup\n+\n+            def find_spec(self, fullname, path=None, target=None):  # Updated for Python 3.4+\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--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -130,7 +130,6 @@ class SqliteMiscTest(unittest.TestCase):\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             next_db = SqliteDict(filename=fname)\n             self.assertNotIn('value', next_db.values())  # Incorrect assertion removed\n",
    "patch_ground_truth": "--- a/piskvorky_sqlitedict/tests/test_onimport.py\n+++ b/piskvorky_sqlitedict/tests/test_onimport.py\n@@ -14,7 +14,9 @@\n                 self.module_names = args\n \n             def find_module(self, fullname, path=None):\n-                return self\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--- 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",
    "message": "======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 616, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 606, in check_raise_error\n    reraise(e_type, e_value, e_tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 46, in reraise\n    raise value.with_traceback(tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 474, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nERROR: test_cpickle_fallback_to_pickle (test_onimport.SqliteDict_cPickleImportTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_onimport.py\", line 36, in test_cpickle_fallback_to_pickle\n    sqlitedict = __import__(\"sqlitedict\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_onimport.py\", line 20, in load_module\n    raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\nImportError: No module named sqlitedict (FauxMissingImport)\n\n----------------------------------------------------------------------\nRan 88 tests in 1.129s\n\nFAILED (errors=2)",
    "CodeBase": [
      {
        "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 import b64decode, b64encode\n37 import weakref\n38 \n39 __version__ = '2.1.0'\n40 \n41 \n42 def reraise(tp, value, tb=None):\n43     if value is None:\n44         value = tp()\n45     if value.__traceback__ is not tb:\n46         raise value.with_traceback(tb)\n47     raise value\n48 \n49 \n50 try:\n51     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n52 except ImportError:\n53     from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n54 \n55 # some Python 3 vs 2 im(...truncated)"
      },
      {
        "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__().(...truncated)"
      },
      {
        "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         class FauxMissingImport(object):\n13             def __init__(self, *args):\n14                 self.module_names = args\n15 \n16             def find_module(self, fullname, path=None):\n17                 return self\n18 \n19             def load_module(self, name):\n20                 raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n21 \n22         # ensure cPickle/pickle is not cached\n23         sys.modules.pop('cPickle', None)\n24         sys.modules.pop('pickle', None)\n25 \n26         # add our custom importer\n27         sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n28 \n29     def tearDown(self):\n30         sys.meta_path = self.orig_meta_path\n31         if self.orig_sqlitedict:\n32             sys.modules['sqlitedict'] = self.orig_sqlitedict\n33 \n34     def test_cpickle_fallback_to_pickle(self):\n35         # exercise,\n36         sqlitedict = __import__(\"sqlitedict\")\n37         # verify,\n38         self.assertIn('pickle', sys.modules.keys())\n39         self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_named_db.py",
        "content": "1 import sqlitedict\n2 from test_temp_db import TempSqliteDictTest\n3 from accessories import norm_file\n4 \n5 \n6 class InMemorySqliteDictTest(TempSqliteDictTest):\n7 \n8     def setUp(self):\n9         self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n10 \n11     def tearDown(self):\n12         self.d.terminate()\n13 \n14 \n15 class NamedSqliteDictTest(TempSqliteDictTest):\n16 \n17    (...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 ./s(...truncated)"
      }
    ],
    "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 7,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 5,
      "Relevance": 6,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Bug in FauxMissingImport Handling and Incorrect Assertion in SqliteDict Tests",
      "description": "### Description\n\nThere are two issues observed in the current implementation:\n1. **FauxMissingImport Handling Issue**: When importing the `cPickle` module or the fallback `pickle` module, the logic does not correctly identify whether the module is missing. Specifically, the `find_module` method always returns a module finder object, even if the module is not in the specified list of module names.\n2. **Incorrect Assertion in SqliteDict Tests**: The test case mistakenly asserts the presence of a value in `next_db.values()`, which is not yielding the expected results.\n\n### Impact\n\n1. **FauxMissingImport Handling Issue**: This could lead to false positives when detecting missing modules, causing unnecessary ImportErrors and unexpected behavior during module loading.\n2. **Incorrect Assertion in SqliteDict Tests**: This results in test failures that prevent the successful validation and deployment of the SqliteDict library.\n\n### Steps to Reproduce\n\n1. **FauxMissingImport Handling Issue**:\n   - Simulate the absence of the `cPickle` module using a custom module finder object.\n   - Attempt to import the `sqlitedict` module.\n   - The import should fail with `ImportError`, but it incorrectly returns the custom finder object.\n\n2. **Incorrect Assertion in SqliteDict Tests**:\n   - Run the test suite for SqliteDict.\n   - Observe the specific test case where it checks for values in `next_db.values()` after re-opening the database.\n   - The test should fail due to the incorrect assertion.\n\n### Expected Behavior\n\n1. The `find_module` method should return None if the module is not in the specified list of module names.\n2. The test case should correctly assert the expected values or revise the checking mechanism in the `next_db` after re-opening.\n\n### Suggested Fixes\nWhile direct modification guidance isn't provided here, the necessary changes involve refining the logic for module import detection and correcting the test assertions to align with the expected behavior.\n\nThese modifications will ensure accurate module handling and reliable test validation, leading to enhanced robustness and correctness of the SqliteDict library.",
      "explanation": "### Summary of the Issue\n\nThe issue report highlights two primary problems within the `sqlitedict` library:\n\n1. **FauxMissingImport Handling Issue**: The mechanism responsible for detecting whether the `cPickle` module or its fallback `pickle` module is present fails to properly identify the absence of these modules. This results in the function always returning a module finder object regardless of whether the module is actually present.\n   \n2. **Incorrect Assertion in SqliteDict Tests**: A specific test case in the SqliteDict tests makes an incorrect assertion by checking for a value in `next_db.values()`, leading to unexpected test failures.\n\n### Commit Overview\n\nTo address these issues, a recent commit made the following changes:\n\n1. For the **FauxMissingImport Handling Issue**, the commit updates the logic to ensure the `find_module` method only returns the custom module finder object if the module is indeed part of the specified list of module names. If the module name is not present in the list, it returns `None`.\n\n2. For the **Incorrect Assertion in SqliteDict Tests**, the commit fixes the test case by removing the incorrect assertion related to checking values in `next_db.values()`.\n\n### Explanation of the Issues and Resolutions\n\n#### FauxMissingImport Handling Issue\n\n**Cause of the Issue:**\nThe `find_module` method within the `FauxMissingImport` class was designed to emulate the behavior of a module finder object. The logic was too simplistic and always returned an object, even if the module was not found. This resulted in false positives, where the absence of a module like `cPickle` would not be correctly identified, leading to unnecessary `ImportError` exceptions or other unexpected behaviors during the module import process.\n\n**Solution Explanation:**\nThe commit refined the `find_module` function to first check if the module's name is in its list of expected module names. Only if the module name matches one of the intended names (like `cPickle`), it returns the module finder object. Otherwise, it returns `None`, indicating that the module wasn't found. This accurately emulates the behavior of a missing module and ensures that the correct fallback mechanisms are activated (e.g., falling back to `pickle` if `cPickle` is not found).\n\n#### Incorrect Assertion in SqliteDict Tests\n\n**Cause of the Issue:**\nA test case within the `test_core.py` file performed an incorrect assertion by checking for a specific value within `next_db.values()`. This caused test failures when the expected value was not found, preventing the successful validation and deployment of the library.\n\n**Solution Explanation:**\nThe commit addressed this by removing the erroneous assertion. This change ensures that the test case only checks relevant conditions—such as the presence of keys—without overreaching into conditions that are incorrect or unreliable. By eliminating the faulty assertion, the tests now align with the expected, valid behavior of the `SqliteDict` class.\n\n### Detailed Explanation of the Fixes\n\n1. **Better Module Import Detection**:\n    - The corrected `find_module` function ensures only relevant module names are handled by returning the module finder object conditionally.\n    - This improves the robustness and reliability of the faux module import mechanism, reducing false positives and misdirected exceptions.\n\n2. **Improved Test Assertions**:\n    - By removing the incorrect assertion, the tests now focus on verifying key existence and other relevant conditions.\n    - This prevents test failures triggered by invalid assumptions and ensures accurate testing of the `SqliteDict` functionalities.\n\n### Conclusion\n\nThe developer's commit effectively addresses both highlighted issues. The adjustment in the module finding logic ensures accurate detection and handling of missing modules, reducing unexpected behavior. Simultaneously, the correction in the test suite ensures reliable and valid validation of the project's functionalities. These changes enhance the overall stability and correctness of the `sqlitedict` library, fostering a more robust software development lifecycle."
    }
  }
}