{
  "RepoName": "https://github.com/piskvorky/sqlitedict.git",
  "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "undefined methods",
  "BuggyCode": [
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n#\n# This code was inspired by:\n#  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n#  * http://code.activestate.com/recipes/526618/\n\n\"\"\"\nA lightweight wrapper around Python's sqlite3 database, with a dict-like interface\nand multi-thread access support::\n\n>>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n>>> mydict['some_key'] = any_picklable_object\n>>> print mydict['some_key']\n>>> print len(mydict) # etc... all dict functions work\n\nPickle is used internally to serialize the values. Keys are strings.\n\nIf you don't use autocommit (default is no autocommit for performance), then\ndon't forget to call `mydict.commit()` when done with a transaction.\n\n\"\"\"\n\nimport sqlite3\nimport os\nimport sys\nimport tempfile\nimport threading\nimport logging\nimport traceback\nfrom base64 import b64decode, b64encode\nimport weakref\n\n__version__ = '2.1.0'\n\n\ndef reraise(tp, value, tb=None):\n    if value is None:\n        value = tp()\n    if value.__traceback__ is not tb:\n        raise value.with_traceback(tb)\n    raise value\n\n\ntry:\n    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\nexcept ImportError:\n    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n\n# some Python 3 vs 2 imports\ntry:\n    from collections import UserDict as DictClass\nexcept ImportError:\n    from UserDict import DictMixin as DictClass\n\ntry:\n    from queue import Queue\nexcept ImportError:\n    from Queue import Queue\n\n\nlogger = logging.getLogger(__name__)\n\n#\n# There's a thread that holds the actual SQL connection (SqliteMultithread).\n# We communicate with this thread via queues (request and responses).\n# The requests can either be SQL commands or one of the \"special\" commands\n# below:\n#\n# _REQUEST_CLOSE: request that the SQL connection be closed\n# _REQUEST_COMMIT: request that any changes be committed to the DB\n#\n# Responses are either SQL records (e.g. results of a SELECT) or the magic\n# _RESPONSE_NO_MORE command, which indicates nothing else will ever be written\n# to the response queue.\n#\n_REQUEST_CLOSE = '--close--'\n_REQUEST_COMMIT = '--commit--'\n_RESPONSE_NO_MORE = '--no more--'\n\n#\n# We work with weak references for better memory efficiency.\n# Dereferencing, checking the referent queue still exists, and putting to it\n# is boring and repetitive, so we have a _put function to handle it for us.\n#\n_PUT_OK, _PUT_REFERENT_DESTROYED, _PUT_NOOP = 0, 1, 2\n\n\ndef _put(queue_reference, item):\n    if queue_reference is not None:\n        queue = queue_reference()\n        if queue is None:\n            #\n            # We got a reference to a queue, but that queue no longer exists\n            #\n            retval = _PUT_REFERENT_DESTROYED\n        else:\n            queue.put(item)\n            retval = _PUT_OK\n\n        del queue\n        return retval\n\n    #\n    # We didn't get a reference to a queue, so do nothing (no-op).\n    #\n    return _PUT_NOOP\n\n\ndef open(*args, **kwargs):\n    \"\"\"See documentation of the SqliteDict class.\"\"\"\n    return SqliteDict(*args, **kwargs)\n\n\ndef encode(obj):\n    \"\"\"Serialize an object using pickle to a binary format accepted by SQLite.\"\"\"\n    return sqlite3.Binary(dumps(obj, protocol=PICKLE_PROTOCOL))\n\n\ndef decode(obj):\n    \"\"\"Deserialize objects retrieved from SQLite.\"\"\"\n    return loads(bytes(obj))\n\n\ndef encode_key(key):\n    \"\"\"Serialize a key using pickle + base64 encoding to text accepted by SQLite.\"\"\"\n    return b64encode(dumps(key, protocol=PICKLE_PROTOCOL)).decode(\"ascii\")\n\n\ndef decode_key(key):\n    \"\"\"Deserialize a key retrieved from SQLite.\"\"\"\n    return loads(b64decode(key.encode(\"ascii\")))\n\n\ndef identity(obj):\n    \"\"\"Identity f(x) = x function for encoding/decoding.\"\"\"\n    return obj\n\n\nclass SqliteDict(DictClass):\n    VALID_FLAGS = ['c', 'r', 'w', 'n']\n\n    def __init__(self, filename=None, tablename='unnamed', flag='c',\n                 autocommit=False, journal_mode=\"DELETE\", encode=encode,\n                 decode=decode, encode_key=identity, decode_key=identity,\n                 timeout=5, outer_stack=True):\n        \"\"\"\n        Initialize a thread-safe sqlite-backed dictionary. The dictionary will\n        be a table `tablename` in database file `filename`. A single file (=database)\n        may contain multiple tables.\n\n        If no `filename` is given, a random file in temp will be used (and deleted\n        from temp once the dict is closed/deleted).\n\n        If you enable `autocommit`, changes will be committed after each operation\n        (more inefficient but safer). Otherwise, changes are committed on `self.commit()`,\n        `self.clear()` and `self.close()`.\n\n        Set `journal_mode` to 'OFF' if you're experiencing sqlite I/O problems\n        or if you need performance and don't care about crash-consistency.\n\n        Set `outer_stack` to False to disable the output of the outer exception\n        to the error logs.  This may improve the efficiency of sqlitedict\n        operation at the expense of a detailed exception trace.\n\n        The `flag` parameter. Exactly one of:\n          'c': default mode, open for read/write, creating the db/table if necessary.\n          'w': open for r/w, but drop `tablename` contents first (start with empty table)\n          'r': open as read-only\n          'n': create a new database (erasing any existing tables, not just `tablename`!).\n\n        The `encode` and `decode` parameters are used to customize how the values\n        are serialized and deserialized.\n        The `encode` parameter must be a function that takes a single Python\n        object and returns a serialized representation.\n        The `decode` function must be a function that takes the serialized\n        representation produced by `encode` and returns a deserialized Python\n        object.\n        The default is to use pickle.\n\n        The `timeout` defines the maximum time (in seconds) to wait for initial Thread startup.\n\n        \"\"\"\n        self.in_temp = filename is None\n        if self.in_temp:\n            fd, filename = tempfile.mkstemp(prefix='sqldict')\n            os.close(fd)\n\n        if flag not in SqliteDict.VALID_FLAGS:\n            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n        self.flag = flag\n\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n\n        dirname = os.path.dirname(filename)\n        if dirname:\n            if not os.path.exists(dirname):\n                raise RuntimeError('Error! The directory does not exist, %s' % dirname)\n\n        self.filename = filename\n\n        # Use standard SQL escaping of double quote characters in identifiers, by doubling them.\n        # See https://github.com/RaRe-Technologies/sqlitedict/pull/113\n        self.tablename = tablename.replace('\"', '\"\"')\n\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        self.encode = encode\n        self.decode = decode\n        self.encode_key = encode_key\n        self.decode_key = decode_key\n        self._outer_stack = outer_stack\n\n        logger.debug(\"opening Sqlite table %r in %r\" % (tablename, filename))\n        self.conn = self._new_conn()\n        if self.flag == 'r':\n            if self.tablename not in SqliteDict.get_tablenames(self.filename):\n                msg = 'Refusing to create a new table \"%s\" in read-only DB mode' % tablename\n                raise RuntimeError(msg)\n        else:\n            MAKE_TABLE = 'CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % self.tablename\n            self.conn.execute(MAKE_TABLE)\n            self.conn.commit()\n        if flag == 'w':\n            self.clear()\n\n    def _new_conn(self):\n        return SqliteMultithread(\n            self.filename,\n            autocommit=self.autocommit,\n            journal_mode=self.journal_mode,\n            outer_stack=self._outer_stack,\n        )\n\n    def __enter__(self):\n        if not hasattr(self, 'conn') or self.conn is None:\n            self.conn = self._new_conn()\n        return self\n\n    def __exit__(self, *exc_info):\n        self.close()\n\n    def __str__(self):\n        return \"SqliteDict(%s)\" % (self.filename)\n\n    def __repr__(self):\n        return str(self)  # no need of something complex\n\n    def __len__(self):\n        # `select count (*)` is super slow in sqlite (does a linear scan!!)\n        # As a result, len() is very slow too once the table size grows beyond trivial.\n        # We could keep the total count of rows ourselves, by means of triggers,\n        # but that seems too complicated and would slow down normal operation\n        # (insert/delete etc).\n        GET_LEN = 'SELECT COUNT(*) FROM \"%s\"' % self.tablename\n        rows = self.conn.select_one(GET_LEN)[0]\n        return rows if rows is not None else 0\n\n    def __bool__(self):\n        # No elements is False, otherwise True\n        GET_MAX = 'SELECT MAX(ROWID) FROM \"%s\"' % self.tablename\n        m = self.conn.select_one(GET_MAX)[0]\n        # Explicit better than implicit and bla bla\n        return True if m is not None else False\n\n    def iterkeys(self):\n        GET_KEYS = 'SELECT key FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key in self.conn.select(GET_KEYS):\n            yield self.decode_key(key[0])\n\n    def itervalues(self):\n        GET_VALUES = 'SELECT value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for value in self.conn.select(GET_VALUES):\n            yield self.decode(value[0])\n\n    def iteritems(self):\n        GET_ITEMS = 'SELECT key, value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key, value in self.conn.select(GET_ITEMS):\n            yield self.decode_key(key), self.decode(value)\n\n    def keys(self):\n        return self.iterkeys()\n\n    def values(self):\n        return self.itervalues()\n\n    def items(self):\n        return self.iteritems()\n\n    def __contains__(self, key):\n        HAS_ITEM = 'SELECT 1 FROM \"%s\" WHERE key = ?' % self.tablename\n        return self.conn.select_one(HAS_ITEM, (self.encode_key(key),)) is not None\n\n    def __getitem__(self, key):\n        GET_ITEM = 'SELECT value FROM \"%s\" WHERE key = ?' % self.tablename\n        item = self.conn.select_one(GET_ITEM, (self.encode_key(key),))\n        if item is None:\n            raise KeyError(key)\n        return self.decode(item[0])\n\n    def __setitem__(self, key, value):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to write to read-only SqliteDict')\n\n        ADD_ITEM = 'REPLACE INTO \"%s\" (key, value) VALUES (?,?)' % self.tablename\n        self.conn.execute(ADD_ITEM, (self.encode_key(key), self.encode(value)))\n        if self.autocommit:\n            self.commit()\n\n    def __delitem__(self, key):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to delete from read-only SqliteDict')\n\n        if key not in self:\n            raise KeyError(key)\n        DEL_ITEM = 'DELETE FROM \"%s\" WHERE key = ?' % self.tablename\n        self.conn.execute(DEL_ITEM, (self.encode_key(key),))\n        if self.autocommit:\n            self.commit()\n\n    def update(self, items=(), **kwds):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to update read-only SqliteDict')\n\n        try:\n            items = items.items()\n        except AttributeError:\n            pass\n        items = [(self.encode_key(k), self.encode(v)) for k, v in items]\n\n        UPDATE_ITEMS = 'REPLACE INTO \"%s\" (key, value) VALUES (?, ?)' % self.tablename\n        self.conn.executemany(UPDATE_ITEMS, items)\n        if kwds:\n            self.update(kwds)\n        if self.autocommit:\n            self.commit()\n\n    def __iter__(self):\n        return self.iterkeys()\n\n    def clear(self):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to clear read-only SqliteDict')\n\n        # avoid VACUUM, as it gives \"OperationalError: database schema has changed\"\n        CLEAR_ALL = 'DELETE FROM \"%s\";' % self.tablename\n        self.conn.commit()\n        self.conn.execute(CLEAR_ALL)\n        self.conn.commit()\n\n    @staticmethod\n    def get_tablenames(filename):\n        \"\"\"get the names of the tables in an sqlite db as a list\"\"\"\n        if not os.path.isfile(filename):\n            raise IOError('file %s does not exist' % (filename))\n        GET_TABLENAMES = 'SELECT name FROM sqlite_master WHERE type=\"table\"'\n        with sqlite3.connect(filename) as conn:\n            cursor = conn.execute(GET_TABLENAMES)\n            res = cursor.fetchall()\n\n        return [name[0] for name in res]\n\n    def commit(self, blocking=True):\n        \"\"\"\n        Persist all data to disk.\n\n        When `blocking` is False, the commit command is queued, but the data is\n        not guaranteed persisted (default implication when autocommit=True).\n        \"\"\"\n        if self.conn is not None:\n            self.conn.commit(blocking)\n    sync = commit\n\n    def close(self, do_log=True, force=False):\n        if do_log:\n            logger.debug(\"closing %s\" % self)\n        if hasattr(self, 'conn') and self.conn is not None:\n            if self.conn.autocommit and not force:\n                # typically calls to commit are non-blocking when autocommit is\n                # used.  However, we need to block on close() to ensure any\n                # awaiting exceptions are handled and that all data is\n                # persisted to disk before returning.\n                self.conn.commit(blocking=True)\n            self.conn.close(force=force)\n            self.conn = None\n        if self.in_temp:\n            try:\n                os.remove(self.filename)\n            except Exception:\n                pass\n\n    def terminate(self):\n        \"\"\"Delete the underlying database file. Use with care.\"\"\"\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to terminate read-only SqliteDict')\n\n        self.close()\n\n        if self.filename == ':memory:':\n            return\n\n        logger.info(\"deleting %s\" % self.filename)\n        try:\n            if os.path.isfile(self.filename):\n                os.remove(self.filename)\n        except (OSError, IOError):\n            logger.exception(\"failed to delete %s\" % (self.filename))\n\n    def __del__(self):\n        # like close(), but assume globals are gone by now (do not log!)\n        try:\n            self.close(do_log=False, force=True)\n        except Exception:\n            # prevent error log flood in case of multiple SqliteDicts\n            # closed after connection lost (exceptions are always ignored\n            # in __del__ method.\n            pass\n\n\nclass SqliteMultithread(threading.Thread):\n    \"\"\"\n    Wrap sqlite connection in a way that allows concurrent requests from multiple threads.\n\n    This is done by internally queueing the requests and processing them sequentially\n    in a separate thread (in the same order they arrived).\n\n    \"\"\"\n    def __init__(self, filename, autocommit, journal_mode, outer_stack=True):\n        super(SqliteMultithread, self).__init__()\n        self.filename = filename\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        # use request queue of unlimited size\n        self.reqs = Queue()\n        self.daemon = True\n        self._outer_stack = outer_stack\n        self.log = logging.getLogger('sqlitedict.SqliteMultithread')\n\n        #\n        # Parts of this object's state get accessed from different threads, so\n        # we use synchronization to avoid race conditions.  For example,\n        # .exception gets set inside the new daemon thread that we spawned, but\n        # gets read from the main thread.  This is particularly important\n        # during initialization: the Thread needs some time to actually start\n        # working, and until this happens, any calls to e.g.\n        # check_raise_error() will prematurely return None, meaning all is\n        # well.  If the that connection happens to fail, we'll never know about\n        # it, and instead wait for a result that never arrives (effectively,\n        # deadlocking).  Locking solves this problem by eliminating the race\n        # condition.\n        #\n        self._lock = threading.Lock()\n        self._lock.acquire()\n        self.exception = None\n\n        self.start()\n\n    def _connect(self):\n        \"\"\"Connect to the underlying database.\n\n        Raises an exception on failure.  Returns the connection and cursor on success.\n        \"\"\"\n        try:\n            if self.autocommit:\n                conn = sqlite3.connect(self.filename, isolation_level=None, check_same_thread=False)\n            else:\n                conn = sqlite3.connect(self.filename, check_same_thread=False)\n        except Exception:\n            self.log.exception(\"Failed to initialize connection for filename: %s\" % self.filename)\n            self.exception = sys.exc_info()\n            raise\n\n        try:\n            conn.execute('PRAGMA journal_mode = %s' % self.journal_mode)\n            conn.text_factory = str\n            cursor = conn.cursor()\n            conn.commit()\n            cursor.execute('PRAGMA synchronous=OFF')\n        except Exception:\n            self.log.exception(\"Failed to execute PRAGMA statements.\")\n            self.exception = sys.exc_info()\n            raise\n\n        return conn, cursor\n\n    def run(self):\n        #\n        # Nb. this is what actually runs inside the new daemon thread.\n        # self._lock is locked at this stage - see the initializer function.\n        #\n        try:\n            conn, cursor = self._connect()\n        finally:\n            self._lock.release()\n\n        res_ref = None\n        while True:\n            #\n            # req: an SQL command or one of the --magic-- commands we use internally\n            # arg: arguments for the command\n            # res_ref: a weak reference to the queue into which responses must be placed\n            # outer_stack: the outer stack, for producing more informative traces in case of error\n            #\n            req, arg, res_ref, outer_stack = self.reqs.get()\n\n            if req == _REQUEST_CLOSE:\n                assert res_ref, ('--close-- without return queue', res_ref)\n                break\n            elif req == _REQUEST_COMMIT:\n                conn.commit()\n                _put(res_ref, _RESPONSE_NO_MORE)\n            else:\n                try:\n                    cursor.execute(req, arg)\n                except Exception:\n                    with self._lock:\n                        self.exception = (e_type, e_value, e_tb) = sys.exc_info()\n\n                    inner_stack = traceback.extract_stack()\n\n                    # An exception occurred in our thread, but we may not\n                    # immediately able to throw it in our calling thread, if it has\n                    # no return `res` queue: log as level ERROR both the inner and\n                    # outer exception immediately.\n                    #\n                    # Any iteration of res.get() or any next call will detect the\n                    # inner exception and re-raise it in the calling Thread; though\n                    # it may be confusing to see an exception for an unrelated\n                    # statement, an ERROR log statement from the 'sqlitedict.*'\n                    # namespace contains the original outer stack location.\n                    self.log.error('Inner exception:')\n                    for item in traceback.format_list(inner_stack):\n                        self.log.error(item)\n                    self.log.error('')  # deliniate traceback & exception w/blank line\n                    for item in traceback.format_exception_only(e_type, e_value):\n                        self.log.error(item)\n\n                    self.log.error('')  # exception & outer stack w/blank line\n\n                    if self._outer_stack:\n                        self.log.error('Outer stack:')\n                        for item in traceback.format_list(outer_stack):\n                            self.log.error(item)\n                        self.log.error('Exception will be re-raised at next call.')\n                    else:\n                        self.log.error(\n                            'Unable to show the outer stack. Pass '\n                            'outer_stack=True when initializing the '\n                            'SqliteDict instance to show the outer stack.'\n                        )\n\n                if res_ref:\n                    for rec in cursor:\n                        if _put(res_ref, rec) == _PUT_REFERENT_DESTROYED:\n                            #\n                            # The queue we are sending responses to got garbage\n                            # collected.  Nobody is listening anymore, so we\n                            # stop sending responses.\n                            #\n                            break\n\n                    _put(res_ref, _RESPONSE_NO_MORE)\n\n                if self.autocommit:\n                    conn.commit()\n\n        self.log.debug('received: %s, send: --no more--', req)\n        conn.close()\n\n        _put(res_ref, _RESPONSE_NO_MORE)\n\n    def check_raise_error(self):\n        \"\"\"\n        Check for and raise exception for any previous sqlite query.\n\n        For the `execute*` family of method calls, such calls are non-blocking and any\n        exception raised in the thread cannot be handled by the calling Thread (usually\n        MainThread).  This method is called on `close`, and prior to any subsequent\n        calls to the `execute*` methods to check for and raise an exception in a\n        previous call to the MainThread.\n        \"\"\"\n        with self._lock:\n            if self.exception:\n                e_type, e_value, e_tb = self.exception\n\n                # clear self.exception, if the caller decides to handle such\n                # exception, we should not repeatedly re-raise it.\n                self.exception = None\n\n                self.log.error('An exception occurred from a previous statement, view '\n                               'the logging namespace \"sqlitedict\" for outer stack.')\n\n                # The third argument to raise is the traceback object, and it is\n                # substituted instead of the current location as the place where\n                # the exception occurred, this is so that when using debuggers such\n                # as `pdb', or simply evaluating the naturally raised traceback, we\n                # retain the original (inner) location of where the exception\n                # occurred.\n                reraise(e_type, e_value, e_tb)\n\n    def execute(self, req, arg=None, res=None):\n        \"\"\"\n        `execute` calls are non-blocking: just queue up the request and return immediately.\n\n        :param req: The request (an SQL command)\n        :param arg: Arguments to the SQL command\n        :param res: A queue in which to place responses as they become available\n        \"\"\"\n        self.check_raise_error()\n        stack = None\n\n        if self._outer_stack:\n            # NOTE: This might be a lot of information to pump into an input\n            # queue, affecting performance.  I've also seen earlier versions of\n            # jython take a severe performance impact for throwing exceptions\n            # so often.\n            stack = traceback.extract_stack()[:-1]\n\n        #\n        # We pass a weak reference to the response queue instead of a regular\n        # reference, because we want the queues to be garbage-collected\n        # more aggressively.\n        #\n        res_ref = None\n        if res:\n            res_ref = weakref.ref(res)\n\n        self.reqs.put((req, arg or tuple(), res_ref, stack))\n\n    def executemany(self, req, items):\n        for item in items:\n            self.execute(req, item)\n        self.check_raise_error()\n\n    def select(self, req, arg=None):\n        \"\"\"\n        Unlike sqlite's native select, this select doesn't handle iteration efficiently.\n\n        The result of `select` starts filling up with values as soon as the\n        request is dequeued, and although you can iterate over the result normally\n        (`for res in self.select(): ...`), the entire result will be in memory.\n        \"\"\"\n        res = Queue()  # results of the select will appear as items in this queue\n        self.execute(req, arg, res)\n        while True:\n            rec = res.get()\n            self.check_raise_error()\n            if rec == _RESPONSE_NO_MORE:\n                break\n            yield rec\n\n    def select_one(self, req, arg=None):\n        \"\"\"Return only the first row of the SELECT, or None if there are no matching rows.\"\"\"\n        try:\n            return next(iter(self.select(req, arg)))\n        except StopIteration:\n            return None\n\n    def commit(self, blocking=True):\n        if blocking:\n            # by default, we await completion of commit() unless\n            # blocking=False.  This ensures any available exceptions for any\n            # previous statement are thrown before returning, and that the\n            # data has actually persisted to disk!\n            self.select_one(_REQUEST_COMMIT)\n        else:\n            # otherwise, we fire and forget as usual.\n            self.execute(_REQUEST_COMMIT)\n\n    def close(self, force=False):\n        if force:\n            # If a SqliteDict is being killed or garbage-collected, then select_one()\n            # could hang forever because run() might already have exited and therefore\n            # can't process the request. Instead, push the close command to the requests\n            # queue directly. If run() is still alive, it will exit gracefully. If not,\n            # then there's nothing we can do anyway.\n            self.reqs.put((_REQUEST_CLOSE, None, weakref.ref(Queue()), None))\n        else:\n            # we abuse 'select' to \"iter\" over a \"--close--\" statement so that we\n            # can confirm the completion of close before joining the thread and\n            # returning (by semaphore '--no more--'\n            self.select_one(_REQUEST_CLOSE)\n            self.join()\n\n\n#\n# This is here for .github/workflows/release.yml\n#\nif __name__ == '__main__':\n    print(__version__)\n"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n\n\"\"\"\nRun with:\n\npython ./setup.py install\n\"\"\"\n\nimport os\nimport io\nimport subprocess\n\nimport setuptools.command.develop\nfrom setuptools import setup\n\n\ndef read(fname):\n    path = os.path.join(os.path.dirname(__file__), fname)\n    return io.open(path, encoding='utf8').read()\n\n\nclass SetupDevelop(setuptools.command.develop.develop):\n    \"\"\"Docstring is overwritten.\"\"\"\n\n    def run(self):\n        \"\"\"\n        Prepare environment for development.\n\n        - Ensures 'nose' and 'coverage.py' are installed for testing.\n        - Call super()'s run method.\n        \"\"\"\n        subprocess.check_call(('pip', 'install', 'nose', 'coverage'))\n\n        # Call super() (except develop is an old-style class, so we must call\n        # directly). The effect is that the development egg-link is installed.\n        setuptools.command.develop.develop.run(self)\n\n\nSetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n\n\nsetup(\n    name='sqlitedict',\n    version='2.1.0',\n    description='Persistent dict in Python, backed up by sqlite3 and pickle, multithread-safe.',\n    long_description=read('README.rst'),\n\n    py_modules=['sqlitedict'],\n\n    # there is a bug in python2.5, preventing distutils from using any non-ascii characters :(\n    # http://bugs.python.org/issue2562\n    author='Radim Rehurek, Victor R. Escobar, Andrey Usov, Prasanna Swaminathan, Jeff Quast',\n    author_email=\"me@radimrehurek.com\",\n    maintainer='Radim Rehurek',\n    maintainer_email='me@radimrehurek.com',\n\n    url='https://github.com/piskvorky/sqlitedict',\n    download_url='http://pypi.python.org/pypi/sqlitedict',\n\n    keywords='sqlite, persistent dict, multithreaded',\n\n    license='Apache 2.0',\n    platforms='any',\n\n    classifiers=[  # from http://pypi.python.org/pypi?%3Aaction=list_classifiers\n        'Development Status :: 5 - Production/Stable',\n        'Environment :: Console',\n        'Intended Audience :: Developers',\n        'License :: OSI Approved :: Apache Software License',\n        'Operating System :: OS Independent',\n        'Programming Language :: Python :: 3.7',\n        'Programming Language :: Python :: 3.8',\n        'Programming Language :: Python :: 3.9',\n        'Programming Language :: Python :: 3.10',\n        'Topic :: Database :: Front-Ends',\n    ],\n    cmdclass={'develop': SetupDevelop},\n)\n"
    },
    {
      "path": "piskvorky_sqlitedict/benchmarks/test_insert.py",
      "content": "import tempfile\n\nfrom sqlitedict import SqliteDict\n\n\ndef insert():\n    with tempfile.NamedTemporaryFile() as tmp:\n        for j in range(100):\n            with SqliteDict(tmp.name) as d:\n                d[\"tmp\"] = j\n                d.commit()\n\n\ndef test(benchmark):\n    benchmark(insert)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/autocommit.py",
      "content": "import sqlitedict\n\nd = sqlitedict.SqliteDict('tests/db/autocommit.sqlite', autocommit=True)\n\nfor i in range(1000):\n    d[i] = i\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_onimport.py",
      "content": "\"\"\"Test cases for on-import logic.\"\"\"\nimport unittest\nimport sys\n\n\nclass SqliteDict_cPickleImportTest(unittest.TestCase):\n    \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n    def setUp(self):\n        self.orig_meta_path = sys.meta_path\n        self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n\n        class FauxMissingImport(object):\n            def __init__(self, *args):\n                self.module_names = args\n\n            def find_module(self, fullname, path=None):\n                if fullname in self.module_names:\n                    return self\n                return None\n\n            def load_module(self, name):\n                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n\n        # ensure cPickle/pickle is not cached\n        sys.modules.pop('cPickle', None)\n        sys.modules.pop('pickle', None)\n\n        # add our custom importer\n        sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n\n    def tearDown(self):\n        sys.meta_path = self.orig_meta_path\n        if self.orig_sqlitedict:\n            sys.modules['sqlitedict'] = self.orig_sqlitedict\n\n    def test_cpickle_fallback_to_pickle(self):\n        # exercise,\n        sqlitedict = __import__(\"sqlitedict\")\n        # verify,\n        self.assertIn('pickle', sys.modules.keys())\n        self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\nimport unittest\nimport sqlitedict\n\nfrom sys import version_info\nmajor_version = version_info[0]\n\n\nclass TempSqliteDictTest(unittest.TestCase):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict()\n\n    def tearDown(self):\n        self.d.close()\n\n    def test_create_sqlitedict(self):\n        ''' test_create_sqlitedict\n        '''\n        self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n        self.assertEqual(dict(self.d), {})\n        self.assertEqual(list(self.d), [])\n        self.assertEqual(self.d.size(), 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_create_sqlitedict (test_core.NamedSqliteDictCreateOrReuseTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\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_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.CreateNewSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.InMemorySqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.NamedSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.SqliteDictAutocommitTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.StartsWithEmptySqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n----------------------------------------------------------------------\nRan 88 tests in 1.148s\n\nFAILED (errors=10)",
  "Patch": "--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -24,7 +24,7 @@\n         self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n         self.assertEqual(dict(self.d), {})\n         self.assertEqual(list(self.d), [])\n-        self.assertEqual(self.d.size(), 0)\n+        self.assertEqual(len(self.d), 0)\n \n     def test_assign_values(self):\n         ''' test_assign_values\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_temp_db.py",
      "function": null,
      "content_all": {
        "27": "        self.assertEqual(self.d.size(), 0)\n"
      },
      "content_change": {
        "27": "        self.assertEqual(self.d.size(), 0)\n"
      }
    }
  ],
  "Issue": {
    "title": "Inconsistent Method Usage and Redundant Test in SqliteDict",
    "description": "There are a couple of issues with the current implementation and tests of `SqliteDict`.\n\n1. **Inconsistent Method Usage**: In `test_temp_db.py`, the `size()` method is inconsistently used to check the size of the `SqliteDict` object instead of the built-in `__len__()` method (i.e., using `len(self.d)`). This inconsistency can cause confusion for the developers and maintainers, as it deviates from the common Python practices and the rest of the codebase.\n\n2. **Redundant Test**: In `test_core.py`, the test checks for the absence of a specific key after reopening a database with a new flag. This test redundantly verifies that a specific value is not in the database, which is unnecessary after confirming the key's absence. The extra check does not add any additional value but makes the test cases unnecessarily verbose and harder to maintain.\n\n**Impact**: The inconsistency in method usage can lead to confusing code and potential misuse of the `SqliteDict` class methods. The redundant test makes the codebase less clean and harder to understand.\n\n**Potential Solution**: Refactor the code to ensure consistent use of methods and remove redundant checks from the test cases.",
    "explanation": "### Summary of the Issue\n\nThe issue highlighted revolves around inconsistencies and redundancies in the testing code for the `SqliteDict` class. The main points are:\n\n1. **Inconsistent Method Usage**: In the `test_temp_db.py` file, the size of a `SqliteDict` object is being checked using the `size()` method instead of the more standard Python approach of using `len(self.d)`. This inconsistency deviates from the rest of the codebase and Pythonic practices.\n   \n2. **Redundant Test**: In the `test_core.py` file, a specific test checks if reopening a database with a new flag results in the absence of a key. This redundancy in checking for both the key and its value adds verbosity and maintenance complexity to the codebase without providing any additional value.\n\n### Analysis of the Commit\n\nThe commit addresses these two issues by making the following changes:\n\n1. **Consistent Method Usage**: It refactors the `test_temp_db.py` file by replacing the `size()` method calls with the more standard `len()` function to check the size of the `SqliteDict` object. This makes the code more consistent and aligned with Pythonic conventions.\n   \n2. **Removing Redundancy**: It edits the `test_core.py` file to remove the extra check for the value associated with a key after verifying the absence of the key. This reduces verbosity and simplifies the test logic, making it easier to understand and maintain.\n\n### Explanation of the Issue and Solution\n\n#### Inconsistent Method Usage\n\n**Cause**: Inconsistent usage of methods to check the size of the `SqliteDict` object could confuse developers. While `size()` is a valid method, it’s not as intuitive or widely recognized as the built-in `len()` function, which is a standard approach in Python for obtaining the size of a collection.\n\n**Solution**: The commit updates the test cases to use `len(self.d)` instead of `self.d.size()`. This ensures that all instances in the codebase comply with common Python practices, making the code easier to read and understand for all developers familiar with Python's conventions.\n\n#### Redundant Test\n\n**Cause**: A test case in `test_core.py` redundantly checks both the absence of the key and its corresponding value in the database. Since it’s already established that the key is absent, further checking for its value is unnecessary and does not contribute any additional meaningful information.\n\n**Solution**: The redundant check for the value in the test case was removed. This streamlines the test, making it more concise and focused solely on verifying the behavior of reopening a database with a new flag. Simplified tests are easier to maintain and less prone to errors.\n\n### Explanation of How the Commit Solves the Issues\n\nThe commit solves the issues by applying the following changes:\n\n1. **Refactoring for Consistency**: By changing the method for checking the size from `size()` to `len()` in `test_temp_db.py`, it ensures that the approach aligns with standard Python practices. This makes the codebase more uniform and intuitive, reducing the cognitive load on developers who are maintaining or extending the tests.\n\n2. **Simplifying the Test Case**: By eliminating the redundant check in `test_core.py`, the commit makes the test cases cleaner and more focused. This not only reduces the verbosity but also enhances the clarity and reliability of the tests. Simplified tests can help in avoiding potential misunderstandings and reduce the overhead in test maintenance.\n\n### Conclusion\n\nThe detailed analysis of the issue and the explanation of the commit show that the changes primarily aim to enhance code consistency and reduce unnecessary complexity in tests. These improvements make the codebase adhere better to Pythonic practices and simplify test logic, ultimately leading to easier maintenance and better clarity in the code."
  },
  "Explain": "### Summary of the Issue\n\nThe issue highlighted revolves around inconsistencies and redundancies in the testing code for the `SqliteDict` class. The main points are:\n\n1. **Inconsistent Method Usage**: In the `test_temp_db.py` file, the size of a `SqliteDict` object is being checked using the `size()` method instead of the more standard Python approach of using `len(self.d)`. This inconsistency deviates from the rest of the codebase and Pythonic practices.\n   \n2. **Redundant Test**: In the `test_core.py` file, a specific test checks if reopening a database with a new flag results in the absence of a key. This redundancy in checking for both the key and its value adds verbosity and maintenance complexity to the codebase without providing any additional value.\n\n### Analysis of the Commit\n\nThe commit addresses these two issues by making the following changes:\n\n1. **Consistent Method Usage**: It refactors the `test_temp_db.py` file by replacing the `size()` method calls with the more standard `len()` function to check the size of the `SqliteDict` object. This makes the code more consistent and aligned with Pythonic conventions.\n   \n2. **Removing Redundancy**: It edits the `test_core.py` file to remove the extra check for the value associated with a key after verifying the absence of the key. This reduces verbosity and simplifies the test logic, making it easier to understand and maintain.\n\n### Explanation of the Issue and Solution\n\n#### Inconsistent Method Usage\n\n**Cause**: Inconsistent usage of methods to check the size of the `SqliteDict` object could confuse developers. While `size()` is a valid method, it’s not as intuitive or widely recognized as the built-in `len()` function, which is a standard approach in Python for obtaining the size of a collection.\n\n**Solution**: The commit updates the test cases to use `len(self.d)` instead of `self.d.size()`. This ensures that all instances in the codebase comply with common Python practices, making the code easier to read and understand for all developers familiar with Python's conventions.\n\n#### Redundant Test\n\n**Cause**: A test case in `test_core.py` redundantly checks both the absence of the key and its corresponding value in the database. Since it’s already established that the key is absent, further checking for its value is unnecessary and does not contribute any additional meaningful information.\n\n**Solution**: The redundant check for the value in the test case was removed. This streamlines the test, making it more concise and focused solely on verifying the behavior of reopening a database with a new flag. Simplified tests are easier to maintain and less prone to errors.\n\n### Explanation of How the Commit Solves the Issues\n\nThe commit solves the issues by applying the following changes:\n\n1. **Refactoring for Consistency**: By changing the method for checking the size from `size()` to `len()` in `test_temp_db.py`, it ensures that the approach aligns with standard Python practices. This makes the codebase more uniform and intuitive, reducing the cognitive load on developers who are maintaining or extending the tests.\n\n2. **Simplifying the Test Case**: By eliminating the redundant check in `test_core.py`, the commit makes the test cases cleaner and more focused. This not only reduces the verbosity but also enhances the clarity and reliability of the tests. Simplified tests can help in avoiding potential misunderstandings and reduce the overhead in test maintenance.\n\n### Conclusion\n\nThe detailed analysis of the issue and the explanation of the commit show that the changes primarily aim to enhance code consistency and reduce unnecessary complexity in tests. These improvements make the codebase adhere better to Pythonic practices and simplify test logic, ultimately leading to easier maintenance and better clarity in the code.",
  "Source": "Human",
  "Token": 1155,
  "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/setup.py",
      "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 \n9 \"\"\"\n10 Run with:\n11 \n12 python ./setup.py install\n13 \"\"\"\n14 \n15 import os\n16 import io\n17 import subprocess\n18 \n19 import setuptools.command.develop\n20 from setuptools import setup\n21 \n22 \n23 def read(fname):\n24     path = os.path.join(os.path.dirname(__file__), fname)\n25     return io.open(path, encoding='utf8').read()\n26 \n27 \n28 class SetupDevelop(setuptools.command.develop.develop):\n29     \"\"\"Docstring is overwritten.\"\"\"\n30 \n31     def run(self):\n32         \"\"\"\n33         Prepare environment for development.\n34 \n35         - Ensures 'nose' and 'coverage.py' are installed for testing.\n36         - Call (...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_temp_db.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 import unittest\n7 import sqlitedict\n8 \n9 from sys import version_info\n10 major_version = version_info[0]\n11 \n12 \n13 class TempSqliteDictTest(unittest.TestCase):\n14 \n15     def setUp(self):\n16         self.d = sqlitedict.SqliteDict()\n17 \n18     def tearDown(self):\n19         self.d.close()\n20 \n21     def test_create_sqlitedict(self):\n22         ''' test_create_sqlitedict\n23         '''\n24         self(...truncated)"
    }
  ],
  "TokenAll": 13040,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Missing test case verification",
      "description": "The test case 'test_reopen_conn' in `piskvorky_sqlitedict/tests/test_core.py` doesn't verify if the value of the key remains consistent after reopening the connection. Please add assertions to check the value of `db['key']` after reopening the database to ensure data integrity.",
      "explanation": "Let's first summarize the issue, analyze the current code, then detail the required commit to solve the issue.\n\n### Summary of the Issue\n\nThe reported issue concerns the `test_reopen_conn` test case in `piskvorky_sqlitedict/tests/test_core.py`. The problem is that this test case does not check if the value of the key remains the same after reopening the database connection, which is essential for ensuring data integrity.\n\n### Current Code Analysis\n\nLooking at the current implementation of the `test_reopen_conn` function in `piskvorky_sqlitedict/tests/test_core.py`:\n```python\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()\n```\nThe test case only sets a value (`'value'`) for the key (`'key'`) twice and commits the changes, but it does not verify if the key-value pair is still consistent after reopening the database connection.\n\n### How the Commit Solves the Issue\n\nThe fix involves adding assertions to check the value of the key after reopening the connection. Here's the revised `test_reopen_conn` function:\n\n```python\n25     def test_reopen_conn(self):\n26         \"\"\"Verify using a contextmanager that a connection can be reopened and the value is consistent.\"\"\"\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         \n33         db2 = SqliteDict(filename=fname)\n34         with db2:\n35             self.assertEqual(db2['key'], 'value')\n36             db2['key'] = 'new_value'\n37             db2.commit()\n38         \n39         db3 = SqliteDict(filename=fname)\n40         with db3:\n41             self.assertEqual(db3['key'], 'new_value')\n```\n### Explanation\n\n1. **Setup the Database (`db`)**:\n   - The test first writes `'value'` to the key `'key'` and commits the transaction (line 30-32).\n\n2. **Reopen the Database (`db2`) and Verify Initial Value**:\n   - The database is reopened in a new `SqliteDict` instance (`db2`) (line 33).\n   - The value of `'key'` is asserted to be `'value'` to ensure it was correctly written and persisted in the previous step (line 35).\n\n3. **Update the Value and Commit**:\n   - The value of `'key'` is updated to `'new_value'` and committed (line 36-37).\n\n4. **Reopen the Database (`db3`) and Verify Updated Value**:\n   - The database is reopened in another new `SqliteDict` instance (`db3`) (line 39).\n   - The value of `'key'` is asserted to be `'new_value'`, confirming the key retains the updated value after reopening the connection (line 41).\n\n### How the Commit Solves the Issue\n\nThe commit enhances the `test_reopen_conn` function by adding necessary assertions to verify data integrity across multiple reopenings of the database. Specifically, it:\n\n- Confirms the value persists after the connection is reopened (assertion in line 35).\n- Validates the new value remains consistent after updating and reopening the connection again (assertion in line 41).\n\nThis ensures the `SqliteDict` class correctly retains key-value pairs through connection reopenings, addressing the issue of missing verification and enhancing the reliability of the data persistence mechanism.\n\n### Summary\n\nTo conclude, the issue was due to the lack of verification in the `test_reopen_conn` function regarding the persistence of key-value pairs after reopening the database connection. The commit solves this by including assertions that check the values after each reopening, ensuring the integrity of the data."
    },
    "issue_message": {
      "title": "Missing Method 'size' in SqliteDict Class",
      "description": "The 'SqliteDict' object has no attribute 'size', causing an AttributeError in the test 'test_create_sqlitedict'. Please implement the 'size' method in the 'SqliteDict' class or update the test to use an existing method.",
      "explanation": "### Issue Summary\n\nThe primary issue is an `AttributeError` caused by the absence of a `size` method in the `SqliteDict` class. This error pops up during the execution of the test case `test_create_sqlitedict` in multiple test files. Specifically, the error message states: `'SqliteDict' object has no attribute 'size'`.\n\n### Commit Content\n\nTo fix the issue, we need to introduce the missing `size` method in the `SqliteDict` class. Below is a proposed commit to add this missing method:\n\n```python\n# Add this method to the SqliteDict class in piskvorky_sqlitedict/sqlitedict.py\n\nclass SqliteDict:\n    # existing methods ...\n        \n    def size(self):\n        \"\"\"\n        Return the number of items in the SqliteDict.\n        \"\"\"\n        with self.conn:\n            cursor = self.conn.execute(\"SELECT COUNT(*) FROM key_value\")\n            result = cursor.fetchone()\n            return result[0] if result else 0\n```\n\n### Detailed Explanation\n\n1. **Summary of the Issue**\n    - The `SqliteDict` class lacks a `size` method, which causes multiple test cases to fail when they try to call `self.d.size()`.\n    - The test cases were written assuming that a method named `size` exists and returns the number of entries in the dictionary.\n\n2. **Content of the Commit**\n    - **Method Definition**: The new `size` method will be defined under the `SqliteDict` class.\n    - **Functionality**: This method will:\n      - Open a connection to the SQLite database.\n      - Execute a SQL query that counts the number of rows in the `key_value` table (the table which stores our dictionary's data).\n      - Return the count of these rows, which effectively represents the number of items in the `SqliteDict`.\n\n3. **Explanation of the Solution**\n    - **Adding the `size` Method**: By defining a `size` method, we ensure that the `SqliteDict` class can respond to queries about its size. This directly addresses the AttributeError.\n    - **SQL Query**: The `SELECT COUNT(*) FROM key_value` SQL query is a standard SQLite command for counting the number of rows in a table. The `key_value` table is presumed to store the dictionary entries (keys and their associated pickled values).\n    - **Database Connection**: Using `with self.conn` ensures that the database connection is properly managed and that the query is executed within a safe context.\n    - **Fetching the Result**: `cursor.fetchone()` retrieves the result of the query. If the query succeeds, `result[0]` gives the count. If the query fails or returns nothing, the method returns `0`.\n    - **Returning the Count**: This count will satisfy the tests by providing the expected number of entries (initially zero when the dictionary is created).\n\n4. **Benefits of the Solution**\n    - **Test Suite Integration**: With this method in place, any test case checking the size of a `SqliteDict` instance will pass, as the `size` method provides a valid way to get this information.\n    - **Compliance with Dictionary-like Interface**: Adding this method aligns with the documentation string at the start of `sqlitedict.py`, which mentions that the class should have a dictionary-like interface. A typical dictionary has a method to get its size (`len(d)` or a similar custom method).\n    - **Improved Usability**: This addition makes the `SqliteDict` class more intuitive to users who might expect such a method in a dictionary-like data structure.\n\n### Conclusion\n\nThe proposed commit addresses the issue by adding a `size` method to the `SqliteDict` class. This method uses a straightforward SQL query to count the number of entries in the database, ensuring the tests can run successfully and the dictionary-like interface of the `SqliteDict` class is enhanced. This change resolves the `AttributeError` and aligns the implementation with user expectations and existing test cases."
    },
    "issue_ground": {
      "title": "Inconsistent Method Usage and Redundant Test in SqliteDict",
      "description": "There are a couple of issues with the current implementation and tests of `SqliteDict`.\n\n1. **Inconsistent Method Usage**: In `test_temp_db.py`, the `size()` method is inconsistently used to check the size of the `SqliteDict` object instead of the built-in `__len__()` method (i.e., using `len(self.d)`). This inconsistency can cause confusion for the developers and maintainers, as it deviates from the common Python practices and the rest of the codebase.\n\n2. **Redundant Test**: In `test_core.py`, the test checks for the absence of a specific key after reopening a database with a new flag. This test redundantly verifies that a specific value is not in the database, which is unnecessary after confirming the key's absence. The extra check does not add any additional value but makes the test cases unnecessarily verbose and harder to maintain.\n\n**Impact**: The inconsistency in method usage can lead to confusing code and potential misuse of the `SqliteDict` class methods. The redundant test makes the codebase less clean and harder to understand.\n\n**Potential Solution**: Refactor the code to ensure consistent use of methods and remove redundant checks from the test cases.",
      "explanation": "### Issue Summary\n\nThe issue reported concerns two primary problems in the `SqliteDict` implementation and its corresponding tests:\n\n1. **Inconsistent Method Usage**: In `test_temp_db.py`, the `size()` method is used to check the size of the `SqliteDict` object instead of the `__len__()` method (i.e., using `len(self.d)`). This inconsistency leads to errors because the `SqliteDict` class does not have a `size()` method.\n2. **Redundant Test**: In `test_core.py`, the absence of a specific key is redundantly verified after reopening a database, which is unnecessary and makes the tests verbose.\n\nThese issues create confusion for developers and maintainers, and they also lead to errors during test execution.\n\n### Commit Details\n\nLet's analyze the commit that addresses these problems:\n\n```diff\ndiff --git a/piskvorky_sqlitedict/tests/test_temp_db.py b/piskvorky_sqlitedict/tests/test_temp_db.py\nindex 1d0aa59..f53d25e 100644\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -24,7 +24,7 @@ class TempSqliteDictTest(unittest.TestCase):\n         ''' test_create_sqlitedict\n         '''\n         self.d['key'] = 'value'\n-        self.assertEqual(self.d.size(), 1)\n+        self.assertEqual(len(self.d), 1)\n         self.d.pop('key')\n         self.assertEqual(len(self.d), 0)\n```\n\n### Detailed Explanation\n\n1. **Inconsistent Method Usage**:\n   - **Cause**: The `SqliteDict` class in `sqlitedict.py` does not define a `size()` method, which leads to an `AttributeError` whenever `self.d.size()` is called in the tests.\n   - **Solution**: Replace the usage of `self.d.size()` with the built-in Python method `len(self.d)`, which is supported by `SqliteDict` as it overrides the `__len__()` method.\n   - **Explanation**: The commit changes the line `self.assertEqual(self.d.size(), 1)` to `self.assertEqual(len(self.d), 1)`. This aligns the test method with the typical Python practice of using `len()` to get the size of a collection. It also ensures consistency with other parts of the codebase that correctly use `len()`.\n\n2. **Redundant Test**:\n   - **Cause**: The test case in `test_core.py` verifies the absence of a specific key after reopening a database, which is redundant given the prior checks.\n   - **Solution**: Remove the redundant test case that verifies the non-existence of a specific value after the key has been confirmed to be absent.\n   - **Explanation**: While the commit doesn't explicitly show modifications related to all redundant checks, addressing redundant tests typically involves:\n     - Identifying tests that do not add extra value (e.g., checking for a value after confirming the key's absence).\n     - Removing or refactoring such tests to make the codebase cleaner and more maintainable.\n\n### How the Commit Resolves the Issues\n\n- **For the Inconsistent Method Usage**:\n  - The test `test_create_sqlitedict` in `test_temp_db.py` is modified to use `len(self.d)` instead of `self.d.size()`. This rectifies the AttributeError caused due to the non-existent `size()` method and aligns the test with Python's built-in functionality.\n- **For the Redundant Test**:\n  - Although not shown in the provided diff, removing redundant test cases (as described in the issue summary) reduces unnecessary checks and simplifies the test suite. This makes the tests easier to understand and maintain while ensuring they remain focused and relevant.\n\nBy making these adjustments, the commit enhances the consistency and clarity of the `SqliteDict` implementation and its tests, preventing potential confusion and errors during development and maintenance."
    },
    "issue_ground_truth": {
      "title": "Inconsistent Method Usage and Redundant Test in SqliteDict",
      "description": "There are a couple of issues with the current implementation and tests of `SqliteDict`.\n\n1. **Inconsistent Method Usage**: In `test_temp_db.py`, the `size()` method is inconsistently used to check the size of the `SqliteDict` object instead of the built-in `__len__()` method (i.e., using `len(self.d)`). This inconsistency can cause confusion for the developers and maintainers, as it deviates from the common Python practices and the rest of the codebase.\n\n2. **Redundant Test**: In `test_core.py`, the test checks for the absence of a specific key after reopening a database with a new flag. This test redundantly verifies that a specific value is not in the database, which is unnecessary after confirming the key's absence. The extra check does not add any additional value but makes the test cases unnecessarily verbose and harder to maintain.\n\n**Impact**: The inconsistency in method usage can lead to confusing code and potential misuse of the `SqliteDict` class methods. The redundant test makes the codebase less clean and harder to understand.\n\n**Potential Solution**: Refactor the code to ensure consistent use of methods and remove redundant checks from the test cases.",
      "explanation": "### Summary of the Issue\n\nThe issue highlighted revolves around inconsistencies and redundancies in the testing code for the `SqliteDict` class. The main points are:\n\n1. **Inconsistent Method Usage**: In the `test_temp_db.py` file, the size of a `SqliteDict` object is being checked using the `size()` method instead of the more standard Python approach of using `len(self.d)`. This inconsistency deviates from the rest of the codebase and Pythonic practices.\n   \n2. **Redundant Test**: In the `test_core.py` file, a specific test checks if reopening a database with a new flag results in the absence of a key. This redundancy in checking for both the key and its value adds verbosity and maintenance complexity to the codebase without providing any additional value.\n\n### Analysis of the Commit\n\nThe commit addresses these two issues by making the following changes:\n\n1. **Consistent Method Usage**: It refactors the `test_temp_db.py` file by replacing the `size()` method calls with the more standard `len()` function to check the size of the `SqliteDict` object. This makes the code more consistent and aligned with Pythonic conventions.\n   \n2. **Removing Redundancy**: It edits the `test_core.py` file to remove the extra check for the value associated with a key after verifying the absence of the key. This reduces verbosity and simplifies the test logic, making it easier to understand and maintain.\n\n### Explanation of the Issue and Solution\n\n#### Inconsistent Method Usage\n\n**Cause**: Inconsistent usage of methods to check the size of the `SqliteDict` object could confuse developers. While `size()` is a valid method, it’s not as intuitive or widely recognized as the built-in `len()` function, which is a standard approach in Python for obtaining the size of a collection.\n\n**Solution**: The commit updates the test cases to use `len(self.d)` instead of `self.d.size()`. This ensures that all instances in the codebase comply with common Python practices, making the code easier to read and understand for all developers familiar with Python's conventions.\n\n#### Redundant Test\n\n**Cause**: A test case in `test_core.py` redundantly checks both the absence of the key and its corresponding value in the database. Since it’s already established that the key is absent, further checking for its value is unnecessary and does not contribute any additional meaningful information.\n\n**Solution**: The redundant check for the value in the test case was removed. This streamlines the test, making it more concise and focused solely on verifying the behavior of reopening a database with a new flag. Simplified tests are easier to maintain and less prone to errors.\n\n### Explanation of How the Commit Solves the Issues\n\nThe commit solves the issues by applying the following changes:\n\n1. **Refactoring for Consistency**: By changing the method for checking the size from `size()` to `len()` in `test_temp_db.py`, it ensures that the approach aligns with standard Python practices. This makes the codebase more uniform and intuitive, reducing the cognitive load on developers who are maintaining or extending the tests.\n\n2. **Simplifying the Test Case**: By eliminating the redundant check in `test_core.py`, the commit makes the test cases cleaner and more focused. This not only reduces the verbosity but also enhances the clarity and reliability of the tests. Simplified tests can help in avoiding potential misunderstandings and reduce the overhead in test maintenance.\n\n### Conclusion\n\nThe detailed analysis of the issue and the explanation of the commit show that the changes primarily aim to enhance code consistency and reduce unnecessary complexity in tests. These improvements make the codebase adhere better to Pythonic practices and simplify test logic, ultimately leading to easier maintenance and better clarity in the code."
    },
    "location_origin": [
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "25": "test_reopen_conn"
        },
        "content_all": {
          "22": "             self.assertEqual(len(d), 0)\n",
          "23": " \n",
          "24": "     def test_reopen_conn(self):\n",
          "25": "         \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n",
          "26": "         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n",
          "27": "         db = SqliteDict(filename=fname)\n",
          "28": "         with db:\n",
          "29": "             db['key'] = 'value'\n",
          "30": "             db.commit()\n",
          "31": "         with db:\n",
          "32": "             db['key'] = 'value'\n",
          "33": "             db.commit()\n",
          "34": " \n",
          "35": "     def test_as_str(self):\n"
        },
        "content_change": {
          "24": "     def test_reopen_conn(self):\n",
          "25": "         \"\"\"Verify using a contextmanager that a connection can be reopened and the value is consistent.\"\"\"\n",
          "26": "         fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n",
          "27": "         db = SqliteDict(filename=fname)\n",
          "28": "         with db:\n",
          "29": "             db['key'] = 'value'\n",
          "30": "             db.commit()\n",
          "31": "         \n",
          "32": "         db2 = SqliteDict(filename=fname)\n",
          "33": "         with db2:\n",
          "34": "             self.assertEqual(db2['key'], 'value')\n",
          "35": "             db2['key'] = 'new_value'\n",
          "36": "             db2.commit()\n",
          "37": "         \n",
          "38": "         db3 = SqliteDict(filename=fname)\n",
          "39": "         with db3:\n",
          "40": "             self.assertEqual(db3['key'], 'new_value')\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "100": "SqliteDict.size"
        },
        "content_all": {
          "97": "\n",
          "98": "    # Additional methods\n",
          "99": "\n",
          "100": "    def size(self):",
          "101": "        \"\"\"",
          "102": "        Return the number of items in the SqliteDict.",
          "103": "        \"\"\"",
          "104": "        with self.conn:",
          "105": "            cursor = self.conn.execute(\"SELECT COUNT(*) FROM key_value\")",
          "106": "            result = cursor.fetchone()",
          "107": "            return result[0] if result else 0\n"
        },
        "content_change": {
          "100": "    def size(self):",
          "101": "        \"\"\"",
          "102": "        Return the number of items in the SqliteDict.",
          "103": "        \"\"\"",
          "104": "        with self.conn:",
          "105": "            cursor = self.conn.execute(\"SELECT COUNT(*) FROM key_value\")",
          "106": "            result = cursor.fetchone()",
          "107": "            return result[0] if result else 0\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "21": "test_create_sqlitedict"
        },
        "content_all": {
          "20": "        self.d.close()",
          "21": "    def test_create_sqlitedict(self):",
          "22": "        ''' test_create_sqlitedict",
          "23": "        '''",
          "24": "        self.d['key'] = 'value'",
          "25": "        self.assertEqual(self.d.size(), 1)",
          "26": "        self.d.pop('key')",
          "27": "        self.assertEqual(len(self.d), 0)"
        },
        "content_change": {
          "25": "        self.assertEqual(len(self.d), 1)"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "25": "test_reopen_conn"
        },
        "content_all": {
          "24": "        self.assertEqual(len(d), 0)",
          "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": "        self.assertNotIn('key', db)",
          "36": "        with db:",
          "37": "            self.assertNotIn('value', db.values())"
        },
        "content_change": {
          "35": "        self.assertNotIn('key', db)"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "21": "test_create_sqlitedict"
        },
        "content_all": {
          "18": "    def tearDown(self):\n",
          "19": "        self.d.close()\n",
          "20": "\n",
          "21": "    def test_create_sqlitedict(self):\n",
          "22": "        ''' test_create_sqlitedict\n",
          "23": "        '''\n",
          "24": "        self.assertEqual(self.d.size(), 0)\n",
          "25": "        self.d['test'] = 'test'\n",
          "26": "        self.assertEqual(self.d.size(), 1)\n",
          "27": "        del self.d['test']\n",
          "28": "        self.assertEqual(self.d.size(), 0)\n"
        },
        "content_change": {
          "24": "        self.assertEqual(len(self.d), 0)\n",
          "26": "        self.assertEqual(len(self.d), 1)\n",
          "28": "        self.assertEqual(len(self.d), 0)\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "25": "test_reopen_conn"
        },
        "content_all": {
          "24": "\n",
          "25": "    def test_reopen_conn(self):\n",
          "26": "        \"\"\"Verify using a context manager that a connection can be reopened.\"\"\"\n",
          "27": "        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n",
          "28": "        db = SqliteDict(filename=fname)\n",
          "29": "        with db:\n",
          "30": "            db['key'] = 'value'\n",
          "31": "            db.commit()\n",
          "32": "        with db:\n",
          "33": "            db['key'] = 'value'\n",
          "34": "            db.commit()\n",
          "35": "        with db:\n",
          "36": "            self.assertNotIn('key', db)  # Checking if 'key' is absent\n",
          "37": "            # self.assertNotIn('value', db.values())  # Redundant check to be removed\n",
          "38": "\n"
        },
        "content_change": {
          "37": "            # self.assertNotIn('value', db.values())  # Redundant check to be removed\n"
        }
      }
    ],
    "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_temp_db.py",
        "function": null,
        "content_all": {
          "27": "        self.assertEqual(self.d.size(), 0)\n"
        },
        "content_change": {
          "27": "        self.assertEqual(self.d.size(), 0)\n"
        }
      }
    ],
    "patch_i": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -25,12 +25,19 @@\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+        db2 = SqliteDict(filename=fname)\n+        with db2:\n+            self.assertEqual(db2['key'], 'value')\n+            db2['key'] = 'new_value'\n+            db2.commit()\n+\n+        db3 = SqliteDict(filename=fname)\n+        with db3:\n+            self.assertEqual(db3['key'], 'new_value')\n",
    "patch_im": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -50,6 +50,7 @@ except ImportError:\n         raise value.with_traceback(tb)\n     raise value\n \n+\n try:\n     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n except ImportError:\n@@ -529,6 +530,16 @@ class SqliteDict(collections.MutableMapping):\n         self.conn.close()\n         self.conn = None\n \n+    def size(self):\n+        \"\"\"\n+        Return the number of items in the SqliteDict.\n+        \"\"\"\n+        with self.conn:\n+            cursor = self.conn.execute(\"SELECT COUNT(*) FROM key_value\")\n+            result = cursor.fetchone()\n+            return result[0] if result else 0\n+\n+\n class SqliteDictDemo(object):\n     def __init__(self, dbfilename):\n         self.dbfilename = dbfilename\n",
    "patch_il": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -22,14 +22,23 @@\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+        \"\"\"Verify using a contextmanager that a connection can be reopened and the value is consistent.\"\"\"\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+        db2 = SqliteDict(filename=fname)\n+        with db2:\n+            self.assertEqual(db2['key'], 'value')\n+            db2['key'] = 'new_value'\n+            db2.commit()\n+        \n+        db3 = SqliteDict(filename=fname)\n+        with db3:\n+            self.assertEqual(db3['key'], 'new_value')\n \n     def test_as_str(self):\n         \"\"\"Verify SqliteDict.__str__().\"\"\"\n",
    "patch_iml": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -97,6 +97,8 @@ class SqliteDict(object):\n \n     # Additional methods\n \n+    def size(self):\n+        \"\"\"Return the number of items in the SqliteDict.\"\"\"\n+        with self.conn:\n+            cursor = self.conn.execute(\"SELECT COUNT(*) FROM key_value\")\n+            result = cursor.fetchone()\n+            return result[0] if result else 0\n",
    "patch_ground": "\n--- piskvorky_sqlitedict/tests/test_temp_db.py\n+++ piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -21,7 +21,7 @@\n         ''' test_create_sqlitedict\n         '''\n         self.d['key'] = 'value'\n-        self.assertEqual(self.d.size(), 1)\n+        self.assertEqual(len(self.d), 1)\n         self.d.pop('key')\n         self.assertEqual(len(self.d), 0)\n \n--- piskvorky_sqlitedict/tests/test_core.py\n+++ piskvorky_sqlitedict/tests/test_core.py\n@@ -33,11 +33,6 @@\n             db['key'] = 'new_value'\n             db.commit()\n         self.assertNotIn('non_existent_key', db)\n-        # Redundant check, just checking key's existence is sufficient.\n-        with self.assertRaises(KeyError):\n-            _ = db['non_existent_key']\n-        with self.assertRaises(KeyError):\n-            self.assertIsNone(db.get('non_existent_key'))\n \n     def test_as_str(self):\n         \"\"\"Verify SqliteDict.__str__().\"\"\"\n",
    "patch_ground_location": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -24,7 +24,7 @@ class TempSqliteDictTest(unittest.TestCase):\n         ''' test_create_sqlitedict\n         '''\n         self.d['key'] = 'value'\n-        self.assertEqual(self.d.size(), 1)\n+        self.assertEqual(len(self.d), 1)\n         self.d.pop('key')\n         self.assertEqual(len(self.d), 0)\n\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -34,8 +34,6 @@ class SqliteMiscTest(unittest.TestCase):\n             db.commit()\n         self.assertNotIn('key', db)\n-        with db:\n-            self.assertNotIn('value', db.values())\n",
    "patch_ground_exp": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -23,9 +23,9 @@\n         ''' test_create_sqlitedict\n         '''\n         self.assertEqual(self.d.size(), 0)\n-        self.assertEqual(self.d.size(), 0)\n+        self.assertEqual(len(self.d), 0)\n         self.d['test'] = 'test'\n-        self.assertEqual(self.d.size(), 1)\n+        self.assertEqual(len(self.d), 1)\n         del self.d['test']\n-        self.assertEqual(self.d.size(), 0)\n+        self.assertEqual(len(self.d), 0)\n\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -35,6 +35,7 @@\n             db.commit()\n         with db:\n             db['key'] = 'value'\n             db.commit()\n         with db:\n             self.assertNotIn('key', db)  # Checking if 'key' is absent\n-            # self.assertNotIn('value', db.values())  # Redundant check to be removed\n+            # self.assertNotIn('value', db.values())  # Redundant check removed\n\n",
    "patch_ground_all": "\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -133,7 +133,6 @@\n         self.assertNotIn('value', next_db.values())\n \n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -27,7 +27,7 @@\n         self.assertEqual(self.d.size(), 0)\n+        self.assertEqual(len(self.d), 0)\n",
    "patch_ground_truth": "--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -24,7 +24,7 @@\n         self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n         self.assertEqual(dict(self.d), {})\n         self.assertEqual(list(self.d), [])\n-        self.assertEqual(self.d.size(), 0)\n+        self.assertEqual(len(self.d), 0)\n \n     def test_assign_values(self):\n         ''' test_assign_values\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_create_sqlitedict (test_core.NamedSqliteDictCreateOrReuseTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\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_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.CreateNewSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.InMemorySqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.NamedSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.SqliteDictAutocommitTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.StartsWithEmptySqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n======================================================================\nERROR: test_create_sqlitedict (test_temp_db.TempSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 27, in test_create_sqlitedict\n    self.assertEqual(self.d.size(), 0)\nAttributeError: 'SqliteDict' object has no attribute 'size'\n\n----------------------------------------------------------------------\nRan 88 tests in 1.148s\n\nFAILED (errors=10)",
    "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/setup.py",
        "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the terms and conditions\n5 # from the Apache License, Version 2.0\n6 #\n7 # http://opensource.org/licenses/apache2.0.php\n8 \n9 \"\"\"\n10 Run with:\n11 \n12 python ./setup.py install\n13 \"\"\"\n14 \n15 import os\n16 import io\n17 import subprocess\n18 \n19 import setuptools.command.develop\n20 from setuptools import setup\n21 \n22 \n23 def read(fname):\n24     path = os.path.join(os.path.dirname(__file__), fname)\n25     return io.open(path, encoding='utf8').read()\n26 \n27 \n28 class SetupDevelop(setuptools.command.develop.develop):\n29     \"\"\"Docstring is overwritten.\"\"\"\n30 \n31     def run(self):\n32         \"\"\"\n33         Prepare environment for development.\n34 \n35         - Ensures 'nose' and 'coverage.py' are installed for testing.\n36         - Call (...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_temp_db.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 import unittest\n7 import sqlitedict\n8 \n9 from sys import version_info\n10 major_version = version_info[0]\n11 \n12 \n13 class TempSqliteDictTest(unittest.TestCase):\n14 \n15     def setUp(self):\n16         self.d = sqlitedict.SqliteDict()\n17 \n18     def tearDown(self):\n19         self.d.close()\n20 \n21     def test_create_sqlitedict(self):\n22         ''' test_create_sqlitedict\n23         '''\n24         self(...truncated)"
      }
    ],
    "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 6,
      "Description": 7,
      "Reproducibility": 4,
      "Relevance": 7,
      "Explanation": 9,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 4,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 9,
      "Description": 7,
      "Reproducibility": 6,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_ground_truth": {
      "title": "Inconsistent Method Usage and Redundant Test in SqliteDict",
      "description": "There are a couple of issues with the current implementation and tests of `SqliteDict`.\n\n1. **Inconsistent Method Usage**: In `test_temp_db.py`, the `size()` method is inconsistently used to check the size of the `SqliteDict` object instead of the built-in `__len__()` method (i.e., using `len(self.d)`). This inconsistency can cause confusion for the developers and maintainers, as it deviates from the common Python practices and the rest of the codebase.\n\n2. **Redundant Test**: In `test_core.py`, the test checks for the absence of a specific key after reopening a database with a new flag. This test redundantly verifies that a specific value is not in the database, which is unnecessary after confirming the key's absence. The extra check does not add any additional value but makes the test cases unnecessarily verbose and harder to maintain.\n\n**Impact**: The inconsistency in method usage can lead to confusing code and potential misuse of the `SqliteDict` class methods. The redundant test makes the codebase less clean and harder to understand.\n\n**Potential Solution**: Refactor the code to ensure consistent use of methods and remove redundant checks from the test cases.",
      "explanation": "### Summary of the Issue\n\nThe issue highlighted revolves around inconsistencies and redundancies in the testing code for the `SqliteDict` class. The main points are:\n\n1. **Inconsistent Method Usage**: In the `test_temp_db.py` file, the size of a `SqliteDict` object is being checked using the `size()` method instead of the more standard Python approach of using `len(self.d)`. This inconsistency deviates from the rest of the codebase and Pythonic practices.\n   \n2. **Redundant Test**: In the `test_core.py` file, a specific test checks if reopening a database with a new flag results in the absence of a key. This redundancy in checking for both the key and its value adds verbosity and maintenance complexity to the codebase without providing any additional value.\n\n### Analysis of the Commit\n\nThe commit addresses these two issues by making the following changes:\n\n1. **Consistent Method Usage**: It refactors the `test_temp_db.py` file by replacing the `size()` method calls with the more standard `len()` function to check the size of the `SqliteDict` object. This makes the code more consistent and aligned with Pythonic conventions.\n   \n2. **Removing Redundancy**: It edits the `test_core.py` file to remove the extra check for the value associated with a key after verifying the absence of the key. This reduces verbosity and simplifies the test logic, making it easier to understand and maintain.\n\n### Explanation of the Issue and Solution\n\n#### Inconsistent Method Usage\n\n**Cause**: Inconsistent usage of methods to check the size of the `SqliteDict` object could confuse developers. While `size()` is a valid method, it’s not as intuitive or widely recognized as the built-in `len()` function, which is a standard approach in Python for obtaining the size of a collection.\n\n**Solution**: The commit updates the test cases to use `len(self.d)` instead of `self.d.size()`. This ensures that all instances in the codebase comply with common Python practices, making the code easier to read and understand for all developers familiar with Python's conventions.\n\n#### Redundant Test\n\n**Cause**: A test case in `test_core.py` redundantly checks both the absence of the key and its corresponding value in the database. Since it’s already established that the key is absent, further checking for its value is unnecessary and does not contribute any additional meaningful information.\n\n**Solution**: The redundant check for the value in the test case was removed. This streamlines the test, making it more concise and focused solely on verifying the behavior of reopening a database with a new flag. Simplified tests are easier to maintain and less prone to errors.\n\n### Explanation of How the Commit Solves the Issues\n\nThe commit solves the issues by applying the following changes:\n\n1. **Refactoring for Consistency**: By changing the method for checking the size from `size()` to `len()` in `test_temp_db.py`, it ensures that the approach aligns with standard Python practices. This makes the codebase more uniform and intuitive, reducing the cognitive load on developers who are maintaining or extending the tests.\n\n2. **Simplifying the Test Case**: By eliminating the redundant check in `test_core.py`, the commit makes the test cases cleaner and more focused. This not only reduces the verbosity but also enhances the clarity and reliability of the tests. Simplified tests can help in avoiding potential misunderstandings and reduce the overhead in test maintenance.\n\n### Conclusion\n\nThe detailed analysis of the issue and the explanation of the commit show that the changes primarily aim to enhance code consistency and reduce unnecessary complexity in tests. These improvements make the codebase adhere better to Pythonic practices and simplify test logic, ultimately leading to easier maintenance and better clarity in the code."
    }
  }
}