{
  "RepoName": "https://github.com/piskvorky/sqlitedict.git",
  "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12",
  "Time": "",
  "Difficulty": "Easy",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n#\n# This code was inspired by:\n#  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n#  * http://code.activestate.com/recipes/526618/\n\n\"\"\"\nA lightweight wrapper around Python's sqlite3 database, with a dict-like interface\nand multi-thread access support::\n\n>>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n>>> mydict['some_key'] = any_picklable_object\n>>> print mydict['some_key']\n>>> print len(mydict) # etc... all dict functions work\n\nPickle is used internally to serialize the values. Keys are strings.\n\nIf you don't use autocommit (default is no autocommit for performance), then\ndon't forget to call `mydict.commit()` when done with a transaction.\n\n\"\"\"\n\nimport sqlite3\nimport os\nimport sys\nimport tempfile\nimport threading\nimport logging\nimport traceback\nfrom base64 import b64decode, b64encode\nimport weakref\n\n__version__ = '2.1.0'\n\n\ndef reraise(tp, value, tb=None):\n    if value is None:\n        value = tp()\n    if value.__traceback__ is not tb:\n        raise value.with_traceback(tb)\n    raise value\n\n\ntry:\n    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\nexcept ImportError:\n    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n\n# some Python 3 vs 2 imports\ntry:\n    from collections import UserDict as DictClass\nexcept ImportError:\n    from UserDict import DictMixin as DictClass\n\ntry:\n    from queue import Queue\nexcept ImportError:\n    from Queue import Queue\n\n\nlogger = logging.getLogger(__name__)\n\n#\n# There's a thread that holds the actual SQL connection (SqliteMultithread).\n# We communicate with this thread via queues (request and responses).\n# The requests can either be SQL commands or one of the \"special\" commands\n# below:\n#\n# _REQUEST_CLOSE: request that the SQL connection be closed\n# _REQUEST_COMMIT: request that any changes be committed to the DB\n#\n# Responses are either SQL records (e.g. results of a SELECT) or the magic\n# _RESPONSE_NO_MORE command, which indicates nothing else will ever be written\n# to the response queue.\n#\n_REQUEST_CLOSE = '--close--'\n_REQUEST_COMMIT = '--commit--'\n_RESPONSE_NO_MORE = '--no more--'\n\n#\n# We work with weak references for better memory efficiency.\n# Dereferencing, checking the referent queue still exists, and putting to it\n# is boring and repetitive, so we have a _put function to handle it for us.\n#\n_PUT_OK, _PUT_REFERENT_DESTROYED, _PUT_NOOP = 0, 1, 2\n\n\ndef _put(queue_reference, item):\n    if queue_reference is not None:\n        queue = queue_reference()\n        if queue is None:\n            #\n            # We got a reference to a queue, but that queue no longer exists\n            #\n            retval = _PUT_REFERENT_DESTROYED\n        else:\n            queue.put(item)\n            retval = _PUT_OK\n\n        del queue\n        return retval\n\n    #\n    # We didn't get a reference to a queue, so do nothing (no-op).\n    #\n    return _PUT_NOOP\n\n\ndef open(*args, **kwargs):\n    \"\"\"See documentation of the SqliteDict class.\"\"\"\n    return SqliteDict(*args, **kwargs)\n\n\ndef encode(obj):\n    \"\"\"Serialize an object using pickle to a binary format accepted by SQLite.\"\"\"\n    return sqlite3.Binary(dumps(obj, protocol=PICKLE_PROTOCOL))\n\n\ndef decode(obj):\n    \"\"\"Deserialize objects retrieved from SQLite.\"\"\"\n    return loads(bytes(obj))\n\n\ndef encode_key(key):\n    \"\"\"Serialize a key using pickle + base64 encoding to text accepted by SQLite.\"\"\"\n    return b64encode(dumps(key, protocol=PICKLE_PROTOCOL)).decode(\"ascii\")\n\n\ndef decode_key(key):\n    \"\"\"Deserialize a key retrieved from SQLite.\"\"\"\n    return loads(b64decode(key.encode(\"ascii\")))\n\n\ndef identity(obj):\n    \"\"\"Identity f(x) = x function for encoding/decoding.\"\"\"\n    return obj\n\n\nclass SqliteDict(DictClass):\n    VALID_FLAGS = ['c', 'r', 'w', 'n']\n\n    def __init__(self, filename=None, tablename='unnamed', flag='c',\n                 autocommit=False, journal_mode=\"DELETE\", encode=encode,\n                 decode=decode, encode_key=identity, decode_key=identity,\n                 timeout=5, outer_stack=True):\n        \"\"\"\n        Initialize a thread-safe sqlite-backed dictionary. The dictionary will\n        be a table `tablename` in database file `filename`. A single file (=database)\n        may contain multiple tables.\n\n        If no `filename` is given, a random file in temp will be used (and deleted\n        from temp once the dict is closed/deleted).\n\n        If you enable `autocommit`, changes will be committed after each operation\n        (more inefficient but safer). Otherwise, changes are committed on `self.commit()`,\n        `self.clear()` and `self.close()`.\n\n        Set `journal_mode` to 'OFF' if you're experiencing sqlite I/O problems\n        or if you need performance and don't care about crash-consistency.\n\n        Set `outer_stack` to False to disable the output of the outer exception\n        to the error logs.  This may improve the efficiency of sqlitedict\n        operation at the expense of a detailed exception trace.\n\n        The `flag` parameter. Exactly one of:\n          'c': default mode, open for read/write, creating the db/table if necessary.\n          'w': open for r/w, but drop `tablename` contents first (start with empty table)\n          'r': open as read-only\n          'n': create a new database (erasing any existing tables, not just `tablename`!).\n\n        The `encode` and `decode` parameters are used to customize how the values\n        are serialized and deserialized.\n        The `encode` parameter must be a function that takes a single Python\n        object and returns a serialized representation.\n        The `decode` function must be a function that takes the serialized\n        representation produced by `encode` and returns a deserialized Python\n        object.\n        The default is to use pickle.\n\n        The `timeout` defines the maximum time (in seconds) to wait for initial Thread startup.\n\n        \"\"\"\n        self.in_temp = filename is None\n        if self.in_temp:\n            fd, filename = tempfile.mkstemp(prefix='sqldict')\n            os.close(fd)\n\n        if flag not in SqliteDict.VALID_FLAGS:\n            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n        self.flag = flag\n\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n\n        dirname = os.path.dirname(filename)\n        if dirname:\n            if not os.path.exists(dirname):\n                raise RuntimeError('Error! The directory does not exist, %s' % dirname)\n\n        self.filename = filename\n\n        # Use standard SQL escaping of double quote characters in identifiers, by doubling them.\n        # See https://github.com/RaRe-Technologies/sqlitedict/pull/113\n        self.tablename = tablename.replace('\"', '\"\"')\n\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        self.encode = encode\n        self.decode = decode\n        self.encode_key = encode_key\n        self.decode_key = decode_key\n        self._outer_stack = outer_stack\n\n        logger.debug(\"opening Sqlite table %r in %r\" % (tablename, filename))\n        self.conn = self._new_conn()\n        if self.flag == 'r':\n            if self.tablename not in SqliteDict.get_tablenames(self.filename):\n                msg = 'Refusing to create a new table \"%s\" in read-only DB mode' % tablename\n                raise RuntimeError(msg)\n        else:\n            MAKE_TABLE = 'CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % self.tablename\n            self.conn.execute(MAKE_TABLE)\n            self.conn.commit()\n        if flag == 'w':\n            self.clear()\n\n    def _new_conn(self):\n        return SqliteMultithread(\n            self.filename,\n            autocommit=self.autocommit,\n            journal_mode=self.journal_mode,\n            outer_stack=self._outer_stack,\n        )\n\n    def __enter__(self):\n        if not hasattr(self, 'conn') or self.conn is None:\n            self.conn = self._new_conn()\n        return self\n\n    def __exit__(self, *exc_info):\n        self.close()\n\n    def __str__(self):\n        return \"SqliteDict(%s)\" % (self.filename)\n\n    def __repr__(self):\n        return str(self)  # no need of something complex\n\n    def __len__(self):\n        # `select count (*)` is super slow in sqlite (does a linear scan!!)\n        # As a result, len() is very slow too once the table size grows beyond trivial.\n        # We could keep the total count of rows ourselves, by means of triggers,\n        # but that seems too complicated and would slow down normal operation\n        # (insert/delete etc).\n        GET_LEN = 'SELECT COUNT(*) FROM \"%s\"' % self.tablename\n        rows = self.conn.select_one(GET_LEN)[0]\n        return rows if rows is not None else 0\n\n    def __bool__(self):\n        # No elements is False, otherwise True\n        GET_MAX = 'SELECT MAX(ROWID) FROM \"%s\"' % self.tablename\n        m = self.conn.select_one(GET_MAX)[0]\n        # Explicit better than implicit and bla bla\n        return True if m is not None else False\n\n    def iterkeys(self):\n        GET_KEYS = 'SELECT key FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key in self.conn.select(GET_KEYS):\n            yield self.decode_key(key[0])\n\n    def itervalues(self):\n        GET_VALUES = 'SELECT value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for value in self.conn.select(GET_VALUES):\n            yield self.decode(value[0])\n\n    def iteritems(self):\n        GET_ITEMS = 'SELECT key, value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key, value in self.conn.select(GET_ITEMS):\n            yield self.decode_key(key), self.decode(value)\n\n    def keys(self):\n        return self.iterkeys()\n\n    def values(self):\n        return self.itervalues()\n\n    def items(self):\n        return self.iteritems()\n\n    def __contains__(self, key):\n        HAS_ITEM = 'SELECT 1 FROM \"%s\" WHERE key = ?' % self.tablename\n        return self.conn.select_one(HAS_ITEM, (self.encode_key(key),)) is not None\n\n    def __getitem__(self, key):\n        GET_ITEM = 'SELECT value FROM \"%s\" WHERE key = ?' % self.tablename\n        item = self.conn.select_one(GET_ITEM, (self.encode_key(key),))\n        if item is None:\n            raise KeyError(key)\n        return self.decode(item[0])\n\n    def __setitem__(self, key, value):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to write to read-only SqliteDict')\n\n        ADD_ITEM = 'REPLACE INTO \"%s\" (key, value) VALUES (?,?)' % self.tablename\n        self.conn.execute(ADD_ITEM, (self.encode_key(key), self.encode(value)))\n        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\n# SetupDevelop.__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'] = 100\n        self.assertEqual(self.d['abc'], 100)\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(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        return text\n\n\nfor prid in sys.argv[1:]:\n    url = \"https://api.github.com/repos/RaRe-Technologies/sqlitedict/pulls/%s\" % prid\n    with urllib.request.urlopen(url) as fin:\n        prinfo = json.load(fin)\n\n    prinfo['user_login'] = prinfo['user']['login']\n    prinfo['user_html_url'] = prinfo['user']['html_url']\n    text = '- %(title)s (PR [#%(number)s](%(html_url)s), [@%(user_login)s](%(user_html_url)s))' % prinfo\n    print(text)\n"
    }
  ],
  "OriginCode": [
    {
      "path": "piskvorky_sqlitedict/sqlitedict.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n#\n# This code was inspired by:\n#  * http://code.activestate.com/recipes/576638-draft-for-an-sqlite3-based-dbm/\n#  * http://code.activestate.com/recipes/526618/\n\n\"\"\"\nA lightweight wrapper around Python's sqlite3 database, with a dict-like interface\nand multi-thread access support::\n\n>>> mydict = SqliteDict('some.db', autocommit=True) # the mapping will be persisted to file `some.db`\n>>> mydict['some_key'] = any_picklable_object\n>>> print mydict['some_key']\n>>> print len(mydict) # etc... all dict functions work\n\nPickle is used internally to serialize the values. Keys are strings.\n\nIf you don't use autocommit (default is no autocommit for performance), then\ndon't forget to call `mydict.commit()` when done with a transaction.\n\n\"\"\"\n\nimport sqlite3\nimport os\nimport sys\nimport tempfile\nimport threading\nimport logging\nimport traceback\nfrom base64 import b64decode, b64encode\nimport weakref\n\n__version__ = '2.1.0'\n\n\ndef reraise(tp, value, tb=None):\n    if value is None:\n        value = tp()\n    if value.__traceback__ is not tb:\n        raise value.with_traceback(tb)\n    raise value\n\n\ntry:\n    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\nexcept ImportError:\n    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n\n# some Python 3 vs 2 imports\ntry:\n    from collections import UserDict as DictClass\nexcept ImportError:\n    from UserDict import DictMixin as DictClass\n\ntry:\n    from queue import Queue\nexcept ImportError:\n    from Queue import Queue\n\n\nlogger = logging.getLogger(__name__)\n\n#\n# There's a thread that holds the actual SQL connection (SqliteMultithread).\n# We communicate with this thread via queues (request and responses).\n# The requests can either be SQL commands or one of the \"special\" commands\n# below:\n#\n# _REQUEST_CLOSE: request that the SQL connection be closed\n# _REQUEST_COMMIT: request that any changes be committed to the DB\n#\n# Responses are either SQL records (e.g. results of a SELECT) or the magic\n# _RESPONSE_NO_MORE command, which indicates nothing else will ever be written\n# to the response queue.\n#\n_REQUEST_CLOSE = '--close--'\n_REQUEST_COMMIT = '--commit--'\n_RESPONSE_NO_MORE = '--no more--'\n\n#\n# We work with weak references for better memory efficiency.\n# Dereferencing, checking the referent queue still exists, and putting to it\n# is boring and repetitive, so we have a _put function to handle it for us.\n#\n_PUT_OK, _PUT_REFERENT_DESTROYED, _PUT_NOOP = 0, 1, 2\n\n\ndef _put(queue_reference, item):\n    if queue_reference is not None:\n        queue = queue_reference()\n        if queue is None:\n            #\n            # We got a reference to a queue, but that queue no longer exists\n            #\n            retval = _PUT_REFERENT_DESTROYED\n        else:\n            queue.put(item)\n            retval = _PUT_OK\n\n        del queue\n        return retval\n\n    #\n    # We didn't get a reference to a queue, so do nothing (no-op).\n    #\n    return _PUT_NOOP\n\n\ndef open(*args, **kwargs):\n    \"\"\"See documentation of the SqliteDict class.\"\"\"\n    return SqliteDict(*args, **kwargs)\n\n\ndef encode(obj):\n    \"\"\"Serialize an object using pickle to a binary format accepted by SQLite.\"\"\"\n    return sqlite3.Binary(dumps(obj, protocol=PICKLE_PROTOCOL))\n\n\ndef decode(obj):\n    \"\"\"Deserialize objects retrieved from SQLite.\"\"\"\n    return loads(bytes(obj))\n\n\ndef encode_key(key):\n    \"\"\"Serialize a key using pickle + base64 encoding to text accepted by SQLite.\"\"\"\n    return b64encode(dumps(key, protocol=PICKLE_PROTOCOL)).decode(\"ascii\")\n\n\ndef decode_key(key):\n    \"\"\"Deserialize a key retrieved from SQLite.\"\"\"\n    return loads(b64decode(key.encode(\"ascii\")))\n\n\ndef identity(obj):\n    \"\"\"Identity f(x) = x function for encoding/decoding.\"\"\"\n    return obj\n\n\nclass SqliteDict(DictClass):\n    VALID_FLAGS = ['c', 'r', 'w', 'n']\n\n    def __init__(self, filename=None, tablename='unnamed', flag='c',\n                 autocommit=False, journal_mode=\"DELETE\", encode=encode,\n                 decode=decode, encode_key=identity, decode_key=identity,\n                 timeout=5, outer_stack=True):\n        \"\"\"\n        Initialize a thread-safe sqlite-backed dictionary. The dictionary will\n        be a table `tablename` in database file `filename`. A single file (=database)\n        may contain multiple tables.\n\n        If no `filename` is given, a random file in temp will be used (and deleted\n        from temp once the dict is closed/deleted).\n\n        If you enable `autocommit`, changes will be committed after each operation\n        (more inefficient but safer). Otherwise, changes are committed on `self.commit()`,\n        `self.clear()` and `self.close()`.\n\n        Set `journal_mode` to 'OFF' if you're experiencing sqlite I/O problems\n        or if you need performance and don't care about crash-consistency.\n\n        Set `outer_stack` to False to disable the output of the outer exception\n        to the error logs.  This may improve the efficiency of sqlitedict\n        operation at the expense of a detailed exception trace.\n\n        The `flag` parameter. Exactly one of:\n          'c': default mode, open for read/write, creating the db/table if necessary.\n          'w': open for r/w, but drop `tablename` contents first (start with empty table)\n          'r': open as read-only\n          'n': create a new database (erasing any existing tables, not just `tablename`!).\n\n        The `encode` and `decode` parameters are used to customize how the values\n        are serialized and deserialized.\n        The `encode` parameter must be a function that takes a single Python\n        object and returns a serialized representation.\n        The `decode` function must be a function that takes the serialized\n        representation produced by `encode` and returns a deserialized Python\n        object.\n        The default is to use pickle.\n\n        The `timeout` defines the maximum time (in seconds) to wait for initial Thread startup.\n\n        \"\"\"\n        self.in_temp = filename is None\n        if self.in_temp:\n            fd, filename = tempfile.mkstemp(prefix='sqldict')\n            os.close(fd)\n\n        if flag not in SqliteDict.VALID_FLAGS:\n            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n        self.flag = flag\n\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n\n        dirname = os.path.dirname(filename)\n        if dirname:\n            if not os.path.exists(dirname):\n                raise RuntimeError('Error! The directory does not exist, %s' % dirname)\n\n        self.filename = filename\n\n        # Use standard SQL escaping of double quote characters in identifiers, by doubling them.\n        # See https://github.com/RaRe-Technologies/sqlitedict/pull/113\n        self.tablename = tablename.replace('\"', '\"\"')\n\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        self.encode = encode\n        self.decode = decode\n        self.encode_key = encode_key\n        self.decode_key = decode_key\n        self._outer_stack = outer_stack\n\n        logger.debug(\"opening Sqlite table %r in %r\" % (tablename, filename))\n        self.conn = self._new_conn()\n        if self.flag == 'r':\n            if self.tablename not in SqliteDict.get_tablenames(self.filename):\n                msg = 'Refusing to create a new table \"%s\" in read-only DB mode' % tablename\n                raise RuntimeError(msg)\n        else:\n            MAKE_TABLE = 'CREATE TABLE IF NOT EXISTS \"%s\" (key TEXT PRIMARY KEY, value BLOB)' % self.tablename\n            self.conn.execute(MAKE_TABLE)\n            self.conn.commit()\n        if flag == 'w':\n            self.clear()\n\n    def _new_conn(self):\n        return SqliteMultithread(\n            self.filename,\n            autocommit=self.autocommit,\n            journal_mode=self.journal_mode,\n            outer_stack=self._outer_stack,\n        )\n\n    def __enter__(self):\n        if not hasattr(self, 'conn') or self.conn is None:\n            self.conn = self._new_conn()\n        return self\n\n    def __exit__(self, *exc_info):\n        self.close()\n\n    def __str__(self):\n        return \"SqliteDict(%s)\" % (self.filename)\n\n    def __repr__(self):\n        return str(self)  # no need of something complex\n\n    def __len__(self):\n        # `select count (*)` is super slow in sqlite (does a linear scan!!)\n        # As a result, len() is very slow too once the table size grows beyond trivial.\n        # We could keep the total count of rows ourselves, by means of triggers,\n        # but that seems too complicated and would slow down normal operation\n        # (insert/delete etc).\n        GET_LEN = 'SELECT COUNT(*) FROM \"%s\"' % self.tablename\n        rows = self.conn.select_one(GET_LEN)[0]\n        return rows if rows is not None else 0\n\n    def __bool__(self):\n        # No elements is False, otherwise True\n        GET_MAX = 'SELECT MAX(ROWID) FROM \"%s\"' % self.tablename\n        m = self.conn.select_one(GET_MAX)[0]\n        # Explicit better than implicit and bla bla\n        return True if m is not None else False\n\n    def iterkeys(self):\n        GET_KEYS = 'SELECT key FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key in self.conn.select(GET_KEYS):\n            yield self.decode_key(key[0])\n\n    def itervalues(self):\n        GET_VALUES = 'SELECT value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for value in self.conn.select(GET_VALUES):\n            yield self.decode(value[0])\n\n    def iteritems(self):\n        GET_ITEMS = 'SELECT key, value FROM \"%s\" ORDER BY rowid' % self.tablename\n        for key, value in self.conn.select(GET_ITEMS):\n            yield self.decode_key(key), self.decode(value)\n\n    def keys(self):\n        return self.iterkeys()\n\n    def values(self):\n        return self.itervalues()\n\n    def items(self):\n        return self.iteritems()\n\n    def __contains__(self, key):\n        HAS_ITEM = 'SELECT 1 FROM \"%s\" WHERE key = ?' % self.tablename\n        return self.conn.select_one(HAS_ITEM, (self.encode_key(key),)) is not None\n\n    def __getitem__(self, key):\n        GET_ITEM = 'SELECT value FROM \"%s\" WHERE key = ?' % self.tablename\n        item = self.conn.select_one(GET_ITEM, (self.encode_key(key),))\n        if item is None:\n            raise KeyError(key)\n        return self.decode(item[0])\n\n    def __setitem__(self, key, value):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to write to read-only SqliteDict')\n\n        ADD_ITEM = 'REPLACE INTO \"%s\" (key, value) VALUES (?,?)' % self.tablename\n        self.conn.execute(ADD_ITEM, (self.encode_key(key), self.encode(value)))\n        if self.autocommit:\n            self.commit()\n\n    def __delitem__(self, key):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to delete from read-only SqliteDict')\n\n        if key not in self:\n            raise KeyError(key)\n        DEL_ITEM = 'DELETE FROM \"%s\" WHERE key = ?' % self.tablename\n        self.conn.execute(DEL_ITEM, (self.encode_key(key),))\n        if self.autocommit:\n            self.commit()\n\n    def update(self, items=(), **kwds):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to update read-only SqliteDict')\n\n        try:\n            items = items.items()\n        except AttributeError:\n            pass\n        items = [(self.encode_key(k), self.encode(v)) for k, v in items]\n\n        UPDATE_ITEMS = 'REPLACE INTO \"%s\" (key, value) VALUES (?, ?)' % self.tablename\n        self.conn.executemany(UPDATE_ITEMS, items)\n        if kwds:\n            self.update(kwds)\n        if self.autocommit:\n            self.commit()\n\n    def __iter__(self):\n        return self.iterkeys()\n\n    def clear(self):\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to clear read-only SqliteDict')\n\n        # avoid VACUUM, as it gives \"OperationalError: database schema has changed\"\n        CLEAR_ALL = 'DELETE FROM \"%s\";' % self.tablename\n        self.conn.commit()\n        self.conn.execute(CLEAR_ALL)\n        self.conn.commit()\n\n    @staticmethod\n    def get_tablenames(filename):\n        \"\"\"get the names of the tables in an sqlite db as a list\"\"\"\n        if not os.path.isfile(filename):\n            raise IOError('file %s does not exist' % (filename))\n        GET_TABLENAMES = 'SELECT name FROM sqlite_master WHERE type=\"table\"'\n        with sqlite3.connect(filename) as conn:\n            cursor = conn.execute(GET_TABLENAMES)\n            res = cursor.fetchall()\n\n        return [name[0] for name in res]\n\n    def commit(self, blocking=True):\n        \"\"\"\n        Persist all data to disk.\n\n        When `blocking` is False, the commit command is queued, but the data is\n        not guaranteed persisted (default implication when autocommit=True).\n        \"\"\"\n        if self.conn is not None:\n            self.conn.commit(blocking)\n    sync = commit\n\n    def close(self, do_log=True, force=False):\n        if do_log:\n            logger.debug(\"closing %s\" % self)\n        if hasattr(self, 'conn') and self.conn is not None:\n            if self.conn.autocommit and not force:\n                # typically calls to commit are non-blocking when autocommit is\n                # used.  However, we need to block on close() to ensure any\n                # awaiting exceptions are handled and that all data is\n                # persisted to disk before returning.\n                self.conn.commit(blocking=True)\n            self.conn.close(force=force)\n            self.conn = None\n        if self.in_temp:\n            try:\n                os.remove(self.filename)\n            except Exception:\n                pass\n\n    def terminate(self):\n        \"\"\"Delete the underlying database file. Use with care.\"\"\"\n        if self.flag == 'r':\n            raise RuntimeError('Refusing to terminate read-only SqliteDict')\n\n        self.close()\n\n        if self.filename == ':memory:':\n            return\n\n        logger.info(\"deleting %s\" % self.filename)\n        try:\n            if os.path.isfile(self.filename):\n                os.remove(self.filename)\n        except (OSError, IOError):\n            logger.exception(\"failed to delete %s\" % (self.filename))\n\n    def __del__(self):\n        # like close(), but assume globals are gone by now (do not log!)\n        try:\n            self.close(do_log=False, force=True)\n        except Exception:\n            # prevent error log flood in case of multiple SqliteDicts\n            # closed after connection lost (exceptions are always ignored\n            # in __del__ method.\n            pass\n\n\nclass SqliteMultithread(threading.Thread):\n    \"\"\"\n    Wrap sqlite connection in a way that allows concurrent requests from multiple threads.\n\n    This is done by internally queueing the requests and processing them sequentially\n    in a separate thread (in the same order they arrived).\n\n    \"\"\"\n    def __init__(self, filename, autocommit, journal_mode, outer_stack=True):\n        super(SqliteMultithread, self).__init__()\n        self.filename = filename\n        self.autocommit = autocommit\n        self.journal_mode = journal_mode\n        # use request queue of unlimited size\n        self.reqs = Queue()\n        self.daemon = True\n        self._outer_stack = outer_stack\n        self.log = logging.getLogger('sqlitedict.SqliteMultithread')\n\n        #\n        # Parts of this object's state get accessed from different threads, so\n        # we use synchronization to avoid race conditions.  For example,\n        # .exception gets set inside the new daemon thread that we spawned, but\n        # gets read from the main thread.  This is particularly important\n        # during initialization: the Thread needs some time to actually start\n        # working, and until this happens, any calls to e.g.\n        # check_raise_error() will prematurely return None, meaning all is\n        # well.  If the that connection happens to fail, we'll never know about\n        # it, and instead wait for a result that never arrives (effectively,\n        # deadlocking).  Locking solves this problem by eliminating the race\n        # condition.\n        #\n        self._lock = threading.Lock()\n        self._lock.acquire()\n        self.exception = None\n\n        self.start()\n\n    def _connect(self):\n        \"\"\"Connect to the underlying database.\n\n        Raises an exception on failure.  Returns the connection and cursor on success.\n        \"\"\"\n        try:\n            if self.autocommit:\n                conn = sqlite3.connect(self.filename, isolation_level=None, check_same_thread=False)\n            else:\n                conn = sqlite3.connect(self.filename, check_same_thread=False)\n        except Exception:\n            self.log.exception(\"Failed to initialize connection for filename: %s\" % self.filename)\n            self.exception = sys.exc_info()\n            raise\n\n        try:\n            conn.execute('PRAGMA journal_mode = %s' % self.journal_mode)\n            conn.text_factory = str\n            cursor = conn.cursor()\n            conn.commit()\n            cursor.execute('PRAGMA synchronous=OFF')\n        except Exception:\n            self.log.exception(\"Failed to execute PRAGMA statements.\")\n            self.exception = sys.exc_info()\n            raise\n\n        return conn, cursor\n\n    def run(self):\n        #\n        # Nb. this is what actually runs inside the new daemon thread.\n        # self._lock is locked at this stage - see the initializer function.\n        #\n        try:\n            conn, cursor = self._connect()\n        finally:\n            self._lock.release()\n\n        res_ref = None\n        while True:\n            #\n            # req: an SQL command or one of the --magic-- commands we use internally\n            # arg: arguments for the command\n            # res_ref: a weak reference to the queue into which responses must be placed\n            # outer_stack: the outer stack, for producing more informative traces in case of error\n            #\n            req, arg, res_ref, outer_stack = self.reqs.get()\n\n            if req == _REQUEST_CLOSE:\n                assert res_ref, ('--close-- without return queue', res_ref)\n                break\n            elif req == _REQUEST_COMMIT:\n                conn.commit()\n                _put(res_ref, _RESPONSE_NO_MORE)\n            else:\n                try:\n                    cursor.execute(req, arg)\n                except Exception:\n                    with self._lock:\n                        self.exception = (e_type, e_value, e_tb) = sys.exc_info()\n\n                    inner_stack = traceback.extract_stack()\n\n                    # An exception occurred in our thread, but we may not\n                    # immediately able to throw it in our calling thread, if it has\n                    # no return `res` queue: log as level ERROR both the inner and\n                    # outer exception immediately.\n                    #\n                    # Any iteration of res.get() or any next call will detect the\n                    # inner exception and re-raise it in the calling Thread; though\n                    # it may be confusing to see an exception for an unrelated\n                    # statement, an ERROR log statement from the 'sqlitedict.*'\n                    # namespace contains the original outer stack location.\n                    self.log.error('Inner exception:')\n                    for item in traceback.format_list(inner_stack):\n                        self.log.error(item)\n                    self.log.error('')  # deliniate traceback & exception w/blank line\n                    for item in traceback.format_exception_only(e_type, e_value):\n                        self.log.error(item)\n\n                    self.log.error('')  # exception & outer stack w/blank line\n\n                    if self._outer_stack:\n                        self.log.error('Outer stack:')\n                        for item in traceback.format_list(outer_stack):\n                            self.log.error(item)\n                        self.log.error('Exception will be re-raised at next call.')\n                    else:\n                        self.log.error(\n                            'Unable to show the outer stack. Pass '\n                            'outer_stack=True when initializing the '\n                            'SqliteDict instance to show the outer stack.'\n                        )\n\n                if res_ref:\n                    for rec in cursor:\n                        if _put(res_ref, rec) == _PUT_REFERENT_DESTROYED:\n                            #\n                            # The queue we are sending responses to got garbage\n                            # collected.  Nobody is listening anymore, so we\n                            # stop sending responses.\n                            #\n                            break\n\n                    _put(res_ref, _RESPONSE_NO_MORE)\n\n                if self.autocommit:\n                    conn.commit()\n\n        self.log.debug('received: %s, send: --no more--', req)\n        conn.close()\n\n        _put(res_ref, _RESPONSE_NO_MORE)\n\n    def check_raise_error(self):\n        \"\"\"\n        Check for and raise exception for any previous sqlite query.\n\n        For the `execute*` family of method calls, such calls are non-blocking and any\n        exception raised in the thread cannot be handled by the calling Thread (usually\n        MainThread).  This method is called on `close`, and prior to any subsequent\n        calls to the `execute*` methods to check for and raise an exception in a\n        previous call to the MainThread.\n        \"\"\"\n        with self._lock:\n            if self.exception:\n                e_type, e_value, e_tb = self.exception\n\n                # clear self.exception, if the caller decides to handle such\n                # exception, we should not repeatedly re-raise it.\n                self.exception = None\n\n                self.log.error('An exception occurred from a previous statement, view '\n                               'the logging namespace \"sqlitedict\" for outer stack.')\n\n                # The third argument to raise is the traceback object, and it is\n                # substituted instead of the current location as the place where\n                # the exception occurred, this is so that when using debuggers such\n                # as `pdb', or simply evaluating the naturally raised traceback, we\n                # retain the original (inner) location of where the exception\n                # occurred.\n                reraise(e_type, e_value, e_tb)\n\n    def execute(self, req, arg=None, res=None):\n        \"\"\"\n        `execute` calls are non-blocking: just queue up the request and return immediately.\n\n        :param req: The request (an SQL command)\n        :param arg: Arguments to the SQL command\n        :param res: A queue in which to place responses as they become available\n        \"\"\"\n        self.check_raise_error()\n        stack = None\n\n        if self._outer_stack:\n            # NOTE: This might be a lot of information to pump into an input\n            # queue, affecting performance.  I've also seen earlier versions of\n            # jython take a severe performance impact for throwing exceptions\n            # so often.\n            stack = traceback.extract_stack()[:-1]\n\n        #\n        # We pass a weak reference to the response queue instead of a regular\n        # reference, because we want the queues to be garbage-collected\n        # more aggressively.\n        #\n        res_ref = None\n        if res:\n            res_ref = weakref.ref(res)\n\n        self.reqs.put((req, arg or tuple(), res_ref, stack))\n\n    def executemany(self, req, items):\n        for item in items:\n            self.execute(req, item)\n        self.check_raise_error()\n\n    def select(self, req, arg=None):\n        \"\"\"\n        Unlike sqlite's native select, this select doesn't handle iteration efficiently.\n\n        The result of `select` starts filling up with values as soon as the\n        request is dequeued, and although you can iterate over the result normally\n        (`for res in self.select(): ...`), the entire result will be in memory.\n        \"\"\"\n        res = Queue()  # results of the select will appear as items in this queue\n        self.execute(req, arg, res)\n        while True:\n            rec = res.get()\n            self.check_raise_error()\n            if rec == _RESPONSE_NO_MORE:\n                break\n            yield rec\n\n    def select_one(self, req, arg=None):\n        \"\"\"Return only the first row of the SELECT, or None if there are no matching rows.\"\"\"\n        try:\n            return next(iter(self.select(req, arg)))\n        except StopIteration:\n            return None\n\n    def commit(self, blocking=True):\n        if blocking:\n            # by default, we await completion of commit() unless\n            # blocking=False.  This ensures any available exceptions for any\n            # previous statement are thrown before returning, and that the\n            # data has actually persisted to disk!\n            self.select_one(_REQUEST_COMMIT)\n        else:\n            # otherwise, we fire and forget as usual.\n            self.execute(_REQUEST_COMMIT)\n\n    def close(self, force=False):\n        if force:\n            # If a SqliteDict is being killed or garbage-collected, then select_one()\n            # could hang forever because run() might already have exited and therefore\n            # can't process the request. Instead, push the close command to the requests\n            # queue directly. If run() is still alive, it will exit gracefully. If not,\n            # then there's nothing we can do anyway.\n            self.reqs.put((_REQUEST_CLOSE, None, weakref.ref(Queue()), None))\n        else:\n            # we abuse 'select' to \"iter\" over a \"--close--\" statement so that we\n            # can confirm the completion of close before joining the thread and\n            # returning (by semaphore '--no more--'\n            self.select_one(_REQUEST_CLOSE)\n            self.join()\n\n\n#\n# This is here for .github/workflows/release.yml\n#\nif __name__ == '__main__':\n    print(__version__)\n"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\n#\n# http://opensource.org/licenses/apache2.0.php\n\n\"\"\"\nRun with:\n\npython ./setup.py install\n\"\"\"\n\nimport os\nimport io\nimport subprocess\n\nimport setuptools.command.develop\nfrom setuptools import setup\n\n\ndef read(fname):\n    path = os.path.join(os.path.dirname(__file__), fname)\n    return io.open(path, encoding='utf8').read()\n\n\nclass SetupDevelop(setuptools.command.develop.develop):\n    \"\"\"Docstring is overwritten.\"\"\"\n\n    def run(self):\n        \"\"\"\n        Prepare environment for development.\n\n        - Ensures 'nose' and 'coverage.py' are installed for testing.\n        - Call super()'s run method.\n        \"\"\"\n        subprocess.check_call(('pip', 'install', 'nose', 'coverage'))\n\n        # Call super() (except develop is an old-style class, so we must call\n        # directly). The effect is that the development egg-link is installed.\n        setuptools.command.develop.develop.run(self)\n\n\nSetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n\n\nsetup(\n    name='sqlitedict',\n    version='2.1.0',\n    description='Persistent dict in Python, backed up by sqlite3 and pickle, multithread-safe.',\n    long_description=read('README.rst'),\n\n    py_modules=['sqlitedict'],\n\n    # there is a bug in python2.5, preventing distutils from using any non-ascii characters :(\n    # http://bugs.python.org/issue2562\n    author='Radim Rehurek, Victor R. Escobar, Andrey Usov, Prasanna Swaminathan, Jeff Quast',\n    author_email=\"me@radimrehurek.com\",\n    maintainer='Radim Rehurek',\n    maintainer_email='me@radimrehurek.com',\n\n    url='https://github.com/piskvorky/sqlitedict',\n    download_url='http://pypi.python.org/pypi/sqlitedict',\n\n    keywords='sqlite, persistent dict, multithreaded',\n\n    license='Apache 2.0',\n    platforms='any',\n\n    classifiers=[  # from http://pypi.python.org/pypi?%3Aaction=list_classifiers\n        'Development Status :: 5 - Production/Stable',\n        'Environment :: Console',\n        'Intended Audience :: Developers',\n        'License :: OSI Approved :: Apache Software License',\n        'Operating System :: OS Independent',\n        'Programming Language :: Python :: 3.7',\n        'Programming Language :: Python :: 3.8',\n        'Programming Language :: Python :: 3.9',\n        'Programming Language :: Python :: 3.10',\n        'Topic :: Database :: Front-Ends',\n    ],\n    cmdclass={'develop': SetupDevelop},\n)\n"
    },
    {
      "path": "piskvorky_sqlitedict/benchmarks/test_insert.py",
      "content": "import tempfile\n\nfrom sqlitedict import SqliteDict\n\n\ndef insert():\n    with tempfile.NamedTemporaryFile() as tmp:\n        for j in range(100):\n            with SqliteDict(tmp.name) as d:\n                d[\"tmp\"] = j\n                d.commit()\n\n\ndef test(benchmark):\n    benchmark(insert)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/autocommit.py",
      "content": "import sqlitedict\n\nd = sqlitedict.SqliteDict('tests/db/autocommit.sqlite', autocommit=True)\n\nfor i in range(1000):\n    d[i] = i\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_onimport.py",
      "content": "\"\"\"Test cases for on-import logic.\"\"\"\nimport unittest\nimport sys\n\n\nclass SqliteDict_cPickleImportTest(unittest.TestCase):\n    \"\"\"Verify fallback to 'pickle' module when 'cPickle' is not found.\"\"\"\n    def setUp(self):\n        self.orig_meta_path = sys.meta_path\n        self.orig_sqlitedict = sys.modules.pop('sqlitedict', None)\n\n        class FauxMissingImport(object):\n            def __init__(self, *args):\n                self.module_names = args\n\n            def find_module(self, fullname, path=None):\n                if fullname in self.module_names:\n                    return self\n                return None\n\n            def load_module(self, name):\n                raise ImportError(\"No module named %s (FauxMissingImport)\" % (name,))\n\n        # ensure cPickle/pickle is not cached\n        sys.modules.pop('cPickle', None)\n        sys.modules.pop('pickle', None)\n\n        # add our custom importer\n        sys.meta_path.insert(0, FauxMissingImport('cPickle'))\n\n    def tearDown(self):\n        sys.meta_path = self.orig_meta_path\n        if self.orig_sqlitedict:\n            sys.modules['sqlitedict'] = self.orig_sqlitedict\n\n    def test_cpickle_fallback_to_pickle(self):\n        # exercise,\n        sqlitedict = __import__(\"sqlitedict\")\n        # verify,\n        self.assertIn('pickle', sys.modules.keys())\n        self.assertIs(sqlitedict.dumps, sys.modules['pickle'].dumps)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "content": "#!/usr/bin/env python\n# -*- coding: utf-8 -*-\n#\n# This code is distributed under the terms and conditions\n# from the Apache License, Version 2.0\nimport unittest\nimport sqlitedict\n\nfrom sys import version_info\nmajor_version = version_info[0]\n\n\nclass TempSqliteDictTest(unittest.TestCase):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict()\n\n    def tearDown(self):\n        self.d.close()\n\n    def test_create_sqlitedict(self):\n        ''' test_create_sqlitedict\n        '''\n        self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n        self.assertEqual(dict(self.d), {})\n        self.assertEqual(list(self.d), [])\n        self.assertEqual(len(self.d), 0)\n\n    def test_assign_values(self):\n        ''' test_assign_values\n        '''\n        self.d['abc'] = 'edf'\n        self.assertEqual(self.d['abc'], 'edf')\n        self.assertEqual(len(self.d), 1)\n\n    def test_clear_data(self):\n        ''' test_clear_data\n        '''\n        self.d.update(a=1, b=2, c=3)\n        self.assertEqual(len(self.d), 3)\n        self.d.clear()\n        self.assertEqual(len(self.d), 0)\n\n    def test_manage_one_record(self):\n        ''' test_manage_one_record\n        '''\n        self.d['abc'] = 'rsvp' * 100\n        self.assertEqual(self.d['abc'], 'rsvp' * 100)\n        self.d['abc'] = 'lmno'\n        self.assertEqual(self.d['abc'], 'lmno')\n        self.assertEqual(len(self.d), 1)\n        del self.d['abc']\n        self.assertEqual(len(self.d), 0)\n        self.assertTrue(not self.d)\n\n    def test_manage_few_records(self):\n        ''' test_manage_few_records\n        '''\n        self.d['abc'] = 'lmno'\n        self.d['xyz'] = 'pdq'\n        self.assertEqual(len(self.d), 2)\n        if major_version == 2:\n            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n        self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n        self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n        self.assertEqual(list(self.d), ['abc', 'xyz'])\n\n    def test_update_records(self):\n        ''' test_update_records\n        '''\n        self.d.update([('v', 'w')], p='x', q='y', r='z')\n        self.assertEqual(len(self.d), 4)\n        # As far as I know dicts does not need to return\n        # the elements in a specified order (sort() is required )\n        self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n        self.assertEqual(sorted(list(self.d)), sorted(['q', 'p', 'r', 'v']))\n\n    def test_handling_errors(self):\n        ''' test_handling_errors\n        '''\n        def get_value(d, k):\n            return d[k]\n\n        def remove_nonexists(d, k):\n            del d[k]\n\n        with self.assertRaises(KeyError):\n            remove_nonexists(self.d, 'abc')\n        with self.assertRaises(KeyError):\n            get_value(self.d, 'abc')\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "import os\nimport sys\n\nimport sqlitedict\n\n\ndef test():\n    \"Verify autocommit just before program exits.\"\n    assert os.system('env PYTHONPATH=. %s tests/autocommit.py' % sys.executable) == 0\n    # The above script relies on the autocommit feature working correctly.\n    # Now, let's check if it actually worked.\n    d = sqlitedict.SqliteDict('tests/db/autocommit.sqlite')\n    for i in range(1000):\n        assert d[i] == i, \"actual: %s expected: %s\" % (d[i], i)\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_core.py",
      "content": "# std imports\nimport json\nimport unittest\nimport tempfile\nimport os\nfrom unittest.mock import patch\n\n# local\nimport sqlitedict\nfrom sqlitedict import SqliteDict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass SqliteMiscTest(unittest.TestCase):\n\n    def test_with_statement(self):\n        \"\"\"Verify using sqlitedict as a contextmanager . \"\"\"\n        with SqliteDict() as d:\n            self.assertTrue(isinstance(d, SqliteDict))\n            self.assertEqual(dict(d), {})\n            self.assertEqual(list(d), [])\n            self.assertEqual(len(d), 0)\n\n    def test_reopen_conn(self):\n        \"\"\"Verify using a contextmanager that a connection can be reopened.\"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        db = SqliteDict(filename=fname)\n        with db:\n            db['key'] = 'value'\n            db.commit()\n        with db:\n            db['key'] = 'value'\n            db.commit()\n\n    def test_as_str(self):\n        \"\"\"Verify SqliteDict.__str__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__str__()\n        # test when db closed\n        db.close()\n        db.__str__()\n\n    def test_as_repr(self):\n        \"\"\"Verify SqliteDict.__repr__().\"\"\"\n        # given,\n        db = SqliteDict()\n        # exercise\n        db.__repr__()\n\n    def test_directory_notfound(self):\n        \"\"\"Verify RuntimeError: directory does not exist.\"\"\"\n        # given: a non-existent directory,\n        folder = tempfile.mkdtemp(prefix='sqlitedict-test')\n        os.rmdir(folder)\n        # exercise,\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=os.path.join(folder, 'nonexistent'))\n\n    def test_commit_nonblocking(self):\n        \"\"\"Coverage for non-blocking commit.\"\"\"\n        # given,\n        with SqliteDict(autocommit=True) as d:\n            # exercise: the implicit commit is nonblocking\n            d['key'] = 'value'\n            d.commit(blocking=False)\n\n    def test_cancel_iterate(self):\n        import time\n\n        class EndlessKeysIterator:\n            def __init__(self) -> None:\n                self.value = 0\n\n            def __iter__(self):\n                return self\n\n            def __next__(self):\n                self.value += 1\n                return [self.value]\n\n        with patch('sqlitedict.sqlite3') as mock_sqlite3:\n            ki = EndlessKeysIterator()\n            cursor = mock_sqlite3.connect().cursor()\n            cursor.__iter__.return_value = ki\n\n            with SqliteDict(autocommit=True) as d:\n                for i, k in enumerate(d.keys()):\n                    assert i + 1 == k\n                    if k > 100:\n                        break\n                assert ki.value > 101\n\n                # Release GIL, let background threads run.\n                # Don't use gc.collect because this is simulate user code.\n                time.sleep(0.01)\n\n                current = ki.value\n                time.sleep(1)\n                assert current == ki.value, 'Will not read more after iterate stop'\n\n\nclass NamedSqliteDictCreateOrReuseTest(TempSqliteDictTest):\n    \"\"\"Verify default flag='c', and flag='n' of SqliteDict().\"\"\"\n\n    def test_default_reuse_existing_flag_c(self):\n        \"\"\"Re-opening of a database does not destroy it.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        next_db = SqliteDict(filename=fname)\n        self.assertIn('key', next_db.keys())\n        self.assertEqual(next_db['key'], 'value')\n\n    def test_overwrite_using_flag_n(self):\n        \"\"\"Re-opening of a database with flag='c' destroys it all.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname, tablename='sometable')\n        orig_db['key'] = 'value'\n        orig_db.commit()\n        orig_db.close()\n\n        # verify,\n        next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n        self.assertNotIn('key', next_db.keys())\n\n    def test_unrecognized_flag(self):\n\n        def build_with_bad_flag():\n            fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n            SqliteDict(filename=fname, flag='FOO')\n\n        with self.assertRaises(RuntimeError):\n            build_with_bad_flag()\n\n    def test_readonly(self):\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        readonly_db = SqliteDict(filename=fname, flag='r')\n        self.assertTrue(readonly_db['key'] == 'value')\n        self.assertTrue(readonly_db['key_two'] == 2)\n\n        def attempt_write():\n            readonly_db['key'] = ['new_value']\n\n        def attempt_update():\n            readonly_db.update(key='value2', key_two=2.1)\n\n        def attempt_delete():\n            del readonly_db['key']\n\n        def attempt_clear():\n            readonly_db.clear()\n\n        def attempt_terminate():\n            readonly_db.terminate()\n\n        attempt_funcs = [attempt_write, attempt_update, attempt_delete, attempt_clear, attempt_terminate]\n\n        for func in attempt_funcs:\n            with self.assertRaises(RuntimeError):\n                func()\n\n    def test_readonly_table(self):\n        \"\"\"\n        Read-only access on a non-existent tablename should raise RuntimeError,\n        and not create a new (empty) table.\n        \"\"\"\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        dummy_tablename = 'table404'\n        orig_db = SqliteDict(filename=fname)\n        orig_db['key'] = 'value'\n        orig_db['key_two'] = 2\n        orig_db.commit()\n        orig_db.close()\n\n        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n        with self.assertRaises(RuntimeError):\n            SqliteDict(filename=fname, tablename=dummy_tablename, flag='r')\n\n        self.assertFalse(dummy_tablename in SqliteDict.get_tablenames(fname))\n\n    def test_irregular_tablenames(self):\n        \"\"\"Irregular table names need to be quoted\"\"\"\n        def __test_irregular_tablenames(tablename):\n            filename = ':memory:'\n            db = SqliteDict(filename, tablename=tablename)\n            db['key'] = 'value'\n            db.commit()\n            self.assertEqual(db['key'], 'value')\n            db.close()\n\n        __test_irregular_tablenames('9nine')\n        __test_irregular_tablenames('outer space')\n        __test_irregular_tablenames('table with a \"quoted\" name')\n        __test_irregular_tablenames(\"table with a \\\"quoted \\xe1cute\\\" name\")\n\n    def test_overwrite_using_flag_w(self):\n        \"\"\"Re-opening of a database with flag='w' destroys only the target table.\"\"\"\n        # given,\n        fname = norm_file('tests/db/sqlitedict-override-test.sqlite')\n        orig_db_1 = SqliteDict(filename=fname, tablename='one')\n        orig_db_1['key'] = 'value'\n        orig_db_1.commit()\n        orig_db_1.close()\n\n        orig_db_2 = SqliteDict(filename=fname, tablename='two')\n        orig_db_2['key'] = 'value'\n        orig_db_2.commit()\n        orig_db_2.close()\n\n        # verify, when re-opening table space 'one' with flag='2', we destroy\n        # its contents.  However, when re-opening table space 'two' with\n        # default flag='r', its contents remain.\n        next_db_1 = SqliteDict(filename=fname, tablename='one', flag='w')\n        self.assertNotIn('key', next_db_1.keys())\n\n        next_db_2 = SqliteDict(filename=fname, tablename='two')\n        self.assertIn('key', next_db_2.keys())\n\n\nclass SqliteDictTerminateTest(unittest.TestCase):\n\n    def test_terminate_instead_close(self):\n        ''' make terminate() instead of close()\n        '''\n        d = sqlitedict.open('tests/db/sqlitedict-terminate.sqlite')\n        d['abc'] = 'def'\n        d.commit()\n        self.assertEqual(d['abc'], 'def')\n        d.terminate()\n        self.assertFalse(os.path.isfile('tests/db/sqlitedict-terminate.sqlite'))\n\n\nclass SqliteDictTerminateFailTest(unittest.TestCase):\n    \"\"\"Provide Coverage for SqliteDict.terminate().\"\"\"\n\n    def setUp(self):\n        self.fname = norm_file('tests/db-permdenied/sqlitedict.sqlite')\n        self.db = SqliteDict(filename=self.fname)\n        os.chmod(self.fname, 0o000)\n        os.chmod(os.path.dirname(self.fname), 0o000)\n\n    def tearDown(self):\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def test_terminate_cannot_delete(self):\n        # exercise,\n        self.db.terminate()  # deletion failed, but no exception raised!\n\n        # verify,\n        os.chmod(os.path.dirname(self.fname), 0o700)\n        os.chmod(self.fname, 0o600)\n        self.assertTrue(os.path.exists(self.fname))\n\n\nclass SqliteDictJsonSerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-json/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test', encode=json.dumps, decode=json.loads\n        )\n\n    def tearDown(self):\n        self.db.close()\n        os.unlink(self.fname)\n        os.rmdir(os.path.dirname(self.fname))\n\n    def get_json(self, key):\n        return self.db.conn.select_one('SELECT value FROM test WHERE key = ?', (self.db.encode_key(key),))[0]\n\n    def test_int(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        assert self.get_json('test') == '-42'\n\n    def test_str(self):\n        test_str = u'Test \\u30c6\\u30b9\\u30c8'\n        self.db['test'] = test_str\n        assert self.db['test'] == test_str\n        assert self.get_json('test') == r'\"Test \\u30c6\\u30b9\\u30c8\"'\n\n    def test_bool(self):\n        self.db['test'] = False\n        assert self.db['test'] is False\n        assert self.get_json('test') == 'false'\n\n    def test_none(self):\n        self.db['test'] = None\n        assert self.db['test'] is None\n        assert self.get_json('test') == 'null'\n\n    def test_complex_struct(self):\n        test_value = {\n            'version': 2.5,\n            'items': ['one', 'two'],\n        }\n        self.db['test'] = test_value\n        assert self.db['test'] == test_value\n        assert self.get_json('test') == json.dumps(test_value)\n\n\nclass TablenamesTest(unittest.TestCase):\n    def tearDown(self):\n        for f in ('tablenames-test-1.sqlite', 'tablenames-test-2.sqlite'):\n            path = norm_file(os.path.join('tests/db', f))\n            if os.path.isfile(path):\n                os.unlink(path)\n\n    def test_tablenames_unnamed(self):\n        fname = norm_file('tests/db/tablenames-test-1.sqlite')\n        SqliteDict(fname)\n        self.assertEqual(SqliteDict.get_tablenames(fname), ['unnamed'])\n\n    def test_tablenams_named(self):\n        fname = norm_file('tests/db/tablenames-test-2.sqlite')\n        with SqliteDict(fname, tablename='table1'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1'])\n        with SqliteDict(fname, tablename='table2'):\n            self.assertEqual(SqliteDict.get_tablenames(fname), ['table1', 'table2'])\n\n        tablenames = SqliteDict.get_tablenames('tests/db/tablenames-test-2.sqlite')\n        self.assertEqual(tablenames, ['table1', 'table2'])\n\n\nclass SqliteDictKeySerializationTest(unittest.TestCase):\n    def setUp(self):\n        self.fname = norm_file('tests/db-encode-key/sqlitedict.sqlite')\n        self.db = SqliteDict(\n            filename=self.fname, tablename='test',\n            encode_key=sqlitedict.encode_key, decode_key=sqlitedict.decode_key,\n        )\n\n    def test_nonstr_keys(self):\n        self.db['test'] = -42\n        assert self.db['test'] == -42\n        self.db[(0, 1, 2)] = 17\n        assert self.db[(0, 1, 2)] == 17\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_named_db.py",
      "content": "import sqlitedict\nfrom test_temp_db import TempSqliteDictTest\nfrom accessories import norm_file\n\n\nclass InMemorySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        self.d = sqlitedict.SqliteDict(filename=':memory:', autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass NamedSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-def.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db)\n\n\nclass CreateNewSqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-n-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass StartsWithEmptySqliteDictTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-with-w-flag.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, flag=\"w\")\n\n    def tearDown(self):\n        self.d.terminate()\n\n\nclass SqliteDictAutocommitTest(TempSqliteDictTest):\n\n    def setUp(self):\n        db = norm_file('tests/db/sqlitedict-autocommit.sqlite')\n        self.d = sqlitedict.SqliteDict(filename=db, autocommit=True)\n\n    def tearDown(self):\n        self.d.terminate()\n"
    },
    {
      "path": "piskvorky_sqlitedict/tests/accessories.py",
      "content": "\"\"\"Accessories for test cases.\"\"\"\nimport os\n\n\ndef norm_file(fname):\n    \"\"\"Normalize test filename, creating a directory path to it if necessary\"\"\"\n    fname = os.path.abspath(fname)\n    dirname = os.path.dirname(fname)\n    if not os.path.exists(dirname):\n        os.makedirs(dirname)\n    return fname\n"
    },
    {
      "path": "piskvorky_sqlitedict/release/summarize_pr.py",
      "content": "#!/usr/bin/env python\nimport json\nimport sys\nimport urllib.request\n\n\ndef copy_to_clipboard(text):\n    try:\n        import pyperclip\n    except ImportError:\n        print('pyperclip <https://pypi.org/project/pyperclip/> is missing.', file=sys.stderr)\n        print('copy-paste the following text manually:', file=sys.stderr)\n        print('  ' + text, file=sys.stderr)\n    else:\n        pyperclip.copy(text)\n\n\nfor prid in sys.argv[1:]:\n    url = \"https://api.github.com/repos/RaRe-Technologies/sqlitedict/pulls/%s\" % prid\n    with urllib.request.urlopen(url) as fin:\n        prinfo = json.load(fin)\n\n    prinfo['user_login'] = prinfo['user']['login']\n    prinfo['user_html_url'] = prinfo['user']['html_url']\n    text = '- %(title)s (PR [#%(number)s](%(html_url)s), [@%(user_login)s](%(user_html_url)s))' % prinfo\n    print(text)\n"
    }
  ],
  "ErrorMessage": "======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 615, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 605, 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 473, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nFAIL: test_clear_data (test_named_db.NamedSqliteDictTest)\ntest_clear_data\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 40, in test_clear_data\n    self.assertEqual(len(self.d), 3)\nAssertionError: 4 != 3\n\n======================================================================\nFAIL: 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 25, in test_create_sqlitedict\n    self.assertEqual(dict(self.d), {})\nAssertionError: {'abc': 'edf'} != {}\n- {'abc': 'edf'}\n+ {}\n\n======================================================================\nFAIL: test_handling_errors (test_named_db.NamedSqliteDictTest)\ntest_handling_errors\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 89, in test_handling_errors\n    remove_nonexists(self.d, 'abc')\nAssertionError: KeyError not raised\n\n======================================================================\nFAIL: test_manage_one_record (test_named_db.NamedSqliteDictTest)\ntest_manage_one_record\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 51, in test_manage_one_record\n    self.assertEqual(len(self.d), 1)\nAssertionError: 2 != 1\n\n======================================================================\nFAIL: test_update_records (test_named_db.NamedSqliteDictTest)\ntest_update_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 73, in test_update_records\n    self.assertEqual(len(self.d), 4)\nAssertionError: 6 != 4\n\n----------------------------------------------------------------------\nRan 88 tests in 1.156s\n\nFAILED (failures=5, errors=1)",
  "Patch": "--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -311,7 +311,8 @@\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-        self.commit()\n+        if self.autocommit:\n+            self.commit()\n \n     def __delitem__(self, key):\n         if self.flag == 'r':\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -42,7 +42,7 @@\n         setuptools.command.develop.develop.run(self)\n \n \n-# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n+SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n \n \n setup(\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -46,8 +46,8 @@\n         '''\n         self.d['abc'] = 'rsvp' * 100\n         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n-        self.d['abc'] = 100\n-        self.assertEqual(self.d['abc'], 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@@ -74,7 +74,7 @@\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(self.d), sorted(['q', 'p', 'r', 'v']))\n+        self.assertEqual(sorted(list(self.d)), sorted(['q', 'p', 'r', 'v']))\n \n     def test_handling_errors(self):\n         ''' test_handling_errors\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--- a/piskvorky_sqlitedict/release/summarize_pr.py\n+++ b/piskvorky_sqlitedict/release/summarize_pr.py\n@@ -13,7 +13,6 @@\n         print('  ' + text, file=sys.stderr)\n     else:\n         pyperclip.copy(text)\n-        return text\n \n \n for prid in sys.argv[1:]:\n",
  "BuggyCodeLocation": [
    {
      "file": "piskvorky_sqlitedict/release/summarize_pr.py",
      "function": null,
      "content_all": {
        "16": "        return text\n",
        "17": "\n",
        "18": "\n",
        "19": "for prid in sys.argv[1:]:\n"
      },
      "content_change": {
        "16": "        return text\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/setup.py",
      "function": null,
      "content_all": {
        "42": "        setuptools.command.develop.develop.run(self)\n",
        "43": "\n",
        "44": "\n",
        "45": "# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n",
        "46": "\n",
        "47": "\n",
        "48": "setup(\n"
      },
      "content_change": {
        "45": "# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/sqlitedict.py",
      "function": null,
      "content_all": {
        "314": "        self.commit()\n"
      },
      "content_change": {
        "314": "        self.commit()\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_core.py",
      "function": null,
      "content_all": {
        "133": "        self.assertNotIn('value', next_db.values())\n"
      },
      "content_change": {
        "133": "        self.assertNotIn('value', next_db.values())\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "function": {
        "44": "test_manage_one_record"
      },
      "content_all": {
        "46": "        '''\n",
        "47": "        self.d['abc'] = 'rsvp' * 100\n",
        "48": "        self.assertEqual(self.d['abc'], 'rsvp' * 100)\n",
        "49": "        self.d['abc'] = 100\n",
        "50": "        self.assertEqual(self.d['abc'], 100)\n",
        "51": "        self.assertEqual(len(self.d), 1)\n",
        "52": "        del self.d['abc']\n",
        "53": "        self.assertEqual(len(self.d), 0)\n"
      },
      "content_change": {
        "49": "        self.d['abc'] = 100\n",
        "50": "        self.assertEqual(self.d['abc'], 100)\n"
      }
    },
    {
      "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
      "function": null,
      "content_all": {
        "77": "        self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n"
      },
      "content_change": {
        "77": "        self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n"
      }
    }
  ],
  "Issue": {
    "title": "Autocommit Feature Not Working Correctly and Minor Bugs in Test Cases",
    "description": "Users have reported an issue where the `autocommit` feature of the SqliteDict class does not work as expected. Specifically, changes are not being committed as they should be when `autocommit` is enabled. This has potential data integrity issues as users rely on autocommit to ensure their data is saved without manually calling the commit function.\n\nAdditionally, there are minor bugs in the test cases for key value assignments and list sorting when retrieving keys. These issues do not immediately cause failures but may lead to data inconsistency in certain edge cases.\n\nThe issue manifests in the following ways:\n1. Assigning values to the SqliteDict does not trigger an automatic commit when `autocommit` is enabled.\n2. Test cases, particularly `test_manage_one_record` and `test_manage_few_records`, have inconsistencies in value assignments and checks.\n3. The `__delitem__` method should check for `autocommit` to ensure deletions are committed when the feature is enabled.\n4. The setup script does not correctly set the docstring for `SetupDevelop` class, leading to potential confusion and incomplete documentation.\n5. The `summarize_pr.py` script might return an unnecessary value that should be handled or removed to prevent runtime errors or unintended behavior.\n\nSteps to Reproduce:\n1. Initialize a SqliteDict with `autocommit=True`.\n2. Attempt to assign a value and see if it gets automatically committed without manually calling commit.\n3. Run the provided test cases and observe any inconsistencies or failures.\n\nExpected Behavior:\n1. Values should be automatically committed upon assignment when `autocommit` is enabled.\n2. Test cases should correctly handle value assignments and retrievals without inconsistencies.\n3. Deletions should also trigger a commit if `autocommit` is enabled.\n4. The setup script should correctly set the docstring for clarity.\n5. The `summarize_pr.py` script should not return unnecessary values.\n\nImpact:\nThis issue impacts the reliability and data integrity of applications relying on SqliteDict for persistent storage and could lead to undetected bugs in test cases and scripts.",
    "explanation": "### Summary of the Issue:\n\nThe issue addresses several problems in the `SqliteDict` class and its related test cases:\n1. **Autocommit Feature**: The `autocommit` feature of `SqliteDict` does not work as expected. When `autocommit` is enabled, changes are not being automatically committed, potentially leading to data integrity issues.\n2. **Test Case Bugs**: Minor bugs exist in the test cases for key-value assignments and key retrieval, which could lead to data inconsistencies.\n3. **`__delitem__` Method**: This method does not check the `autocommit` flag, causing deletion operations not to be committed automatically.\n4. **Setup Script**: The setup script fails to set the docstring for the `SetupDevelop` class correctly.\n5. **Utility Script `summarize_pr.py`**: The script has a functionality where it returns an unnecessary value, potentially leading to runtime errors.\n\n### Commit Content and Explanation:\n\nThe commit introduces changes to various parts of the codebase to address each of the identified issues:\n\n1. **Autocommit Implementation**: \n   - The commit adds a conditional check to ensure that commits are only triggered if the `autocommit` attribute is set to `True`. This change ensures that whenever a value is assigned or deleted, the changes are committed automatically, preventing potential data integrity issues.\n   \n2. **Test Case Fixes**:\n   - The commit updates the test cases to ensure that key-value assignments and key retrievals handle `SqliteDict` entries correctly. Specifically:\n     - Correcting the assigned values and their assertions to be consistent.\n     - Adjusting the key sorting procedure to convert dictionary keys to a list before sorting, ensuring it works as intended across different Python versions.\n   \n3. **`__delitem__` Method Adjustment**:\n   - The `__delitem__` method was modified to check for `autocommit` and perform a commit after deletion if `autocommit` is enabled. This ensures automatic data persistence upon deletion operations.\n\n4. **Setup Script Docstring**:\n   - The docstring for the `SetupDevelop` class is correctly set, ensuring that the documentation is comprehensive and aids in understanding the setup process.\n   \n5. **Removal of Unnecessary Return Value in Utility Script**:\n   - The commit removes the unnecessary return value from the `summarize_pr.py` script. By doing this, potential runtime errors and unintended behaviors are avoided, enhancing the script's robustness and reliability.\n\n### Solution Explanation:\n\nThe solution provided in the commit effectively addresses the issues:\n\n- **Autocommit Feature**: The conditional commit logic ensures that changes are immediately committed to the database when `autocommit` is enabled. This change significantly enhances data integrity by ensuring that each modification is persisted without needing explicit commits from the user.\n\n- **Test Case Reliability**: By fixing inconsistencies in the test cases, especially around key-value assignments and retrievals, the commit ensures the test suite accurately reflects the intended functionality of `SqliteDict`. This reduces the chance of undetected bugs and guarantees that the class works correctly in various scenarios.\n\n- **Deletion and Autocommit Integration**: The modification to the `__delitem__` method to respect the `autocommit` flag ensures that detrimental changes (like deletions) are also automatically persisted. This is crucial for data consistency and integrity, especially in applications relying heavily on frequent deletions.\n\n- **Improved Documentation in Setup Script**: Setting the correct docstring in the setup script helps developers better understand the setup process, leading to fewer misunderstandings and smoother environment preparation.\n\n- **Utility Script Enhancement**: By removing the unnecessary return value, the commit improves the stability of the `summarize_pr.py` script, preventing errors that may arise from handling unintended return values.\n\nOverall, the commit systematically addresses each aspect of the reported issue with targeted changes, ensuring the `SqliteDict` works as expected and improving the overall robustness and reliability of the codebase."
  },
  "Explain": "### Summary of the Issue:\n\nThe issue addresses several problems in the `SqliteDict` class and its related test cases:\n1. **Autocommit Feature**: The `autocommit` feature of `SqliteDict` does not work as expected. When `autocommit` is enabled, changes are not being automatically committed, potentially leading to data integrity issues.\n2. **Test Case Bugs**: Minor bugs exist in the test cases for key-value assignments and key retrieval, which could lead to data inconsistencies.\n3. **`__delitem__` Method**: This method does not check the `autocommit` flag, causing deletion operations not to be committed automatically.\n4. **Setup Script**: The setup script fails to set the docstring for the `SetupDevelop` class correctly.\n5. **Utility Script `summarize_pr.py`**: The script has a functionality where it returns an unnecessary value, potentially leading to runtime errors.\n\n### Commit Content and Explanation:\n\nThe commit introduces changes to various parts of the codebase to address each of the identified issues:\n\n1. **Autocommit Implementation**: \n   - The commit adds a conditional check to ensure that commits are only triggered if the `autocommit` attribute is set to `True`. This change ensures that whenever a value is assigned or deleted, the changes are committed automatically, preventing potential data integrity issues.\n   \n2. **Test Case Fixes**:\n   - The commit updates the test cases to ensure that key-value assignments and key retrievals handle `SqliteDict` entries correctly. Specifically:\n     - Correcting the assigned values and their assertions to be consistent.\n     - Adjusting the key sorting procedure to convert dictionary keys to a list before sorting, ensuring it works as intended across different Python versions.\n   \n3. **`__delitem__` Method Adjustment**:\n   - The `__delitem__` method was modified to check for `autocommit` and perform a commit after deletion if `autocommit` is enabled. This ensures automatic data persistence upon deletion operations.\n\n4. **Setup Script Docstring**:\n   - The docstring for the `SetupDevelop` class is correctly set, ensuring that the documentation is comprehensive and aids in understanding the setup process.\n   \n5. **Removal of Unnecessary Return Value in Utility Script**:\n   - The commit removes the unnecessary return value from the `summarize_pr.py` script. By doing this, potential runtime errors and unintended behaviors are avoided, enhancing the script's robustness and reliability.\n\n### Solution Explanation:\n\nThe solution provided in the commit effectively addresses the issues:\n\n- **Autocommit Feature**: The conditional commit logic ensures that changes are immediately committed to the database when `autocommit` is enabled. This change significantly enhances data integrity by ensuring that each modification is persisted without needing explicit commits from the user.\n\n- **Test Case Reliability**: By fixing inconsistencies in the test cases, especially around key-value assignments and retrievals, the commit ensures the test suite accurately reflects the intended functionality of `SqliteDict`. This reduces the chance of undetected bugs and guarantees that the class works correctly in various scenarios.\n\n- **Deletion and Autocommit Integration**: The modification to the `__delitem__` method to respect the `autocommit` flag ensures that detrimental changes (like deletions) are also automatically persisted. This is crucial for data consistency and integrity, especially in applications relying heavily on frequent deletions.\n\n- **Improved Documentation in Setup Script**: Setting the correct docstring in the setup script helps developers better understand the setup process, leading to fewer misunderstandings and smoother environment preparation.\n\n- **Utility Script Enhancement**: By removing the unnecessary return value, the commit improves the stability of the `summarize_pr.py` script, preventing errors that may arise from handling unintended return values.\n\nOverall, the commit systematically addresses each aspect of the reported issue with targeted changes, ensuring the `SqliteDict` works as expected and improving the overall robustness and reliability of the codebase.",
  "Source": "Human",
  "Token": 1493,
  "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_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.assertIsInstance(self.d, sqlitedict.SqliteDict)\n25         self.assertEqual(dict(self.d), {})\n26         self.assertEqual(list(self.d), [])\n27         self.assertEqual(len(self.d), 0)\n28 \n29     def test_assign_values(self):\n30         ''' test_assign_values\n31         '''\n32         self.d['abc'] = 'edf'\n33         self.assertEqual(self.d['abc'], 'edf')\n34         self.assertEqual(len(self.d), 1)\n35 \n36     def test_clear_data(self):\n37         ''' test_clear_data\n38         '''\n39         self.d.update(a=1, b=2, c=3)\n40         self.assertEqual(len(self.d), 3)\n41         self.d.clear()\n42         self.assertEqual(len(self.d), 0)\n43 \n44     def test_manage_one_record(self):\n45         ''' test_manage_one_record\n46         '''\n47         self.d['abc'] = 'rsvp' * 100\n48         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n49         self.d['abc'] = 100\n50         self.assertEqual(self.d['abc'], 100)\n51         self.assertEqual(len(self.d), 1)\n52         del self.d['abc']\n53         self.assertEqual(len(self.d), 0)\n54         self.assertTrue(not self.d)\n55 \n56     def test_manage_few_records(self):\n57         ''' test_manage_few_records\n58         '''\n59         self.d['abc'] = 'lmno'\n60         self.d['xyz'] = 'pdq'\n61         self.assertEqual(len(self.d), 2)\n62         if major_version == 2:\n63             self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n64         self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n65         self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n66         self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n67         self.assertEqual(list(self.d), ['abc', 'xyz'])\n68 \n69     def test_update_records(self):\n70         ''' test_update_records\n71         '''\n72         self.d.update([('v', 'w')], p='x', q='y', r='z')\n73         self.assertEqual(len(self.d), 4)\n74         # As far as I know dicts does not need to return\n75         # the elements in a specified order (sort() is required )\n76         self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n77         self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n78 \n79     def test_handling_errors(self):\n80         ''' test_handling_errors\n81         '''\n82         def get_value(d, k):\n83             return d[k]\n84 \n85         def remove_nonexists(d, k):\n86             del d[k]\n87 \n88         with self.assertRaises(KeyError):\n89             remove_nonexists(self.d, 'abc')\n90         with self.assertRaises(KeyError):\n91             get_value(self.d, 'abc')"
    },
    {
      "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 im(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "1 import os\n2 import sys\n3 \n4 import sqlitedict\n5 \n6 \n7 def test():\n8     \"Verify autocommit just be(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/setup.py",
      "content": "1 #!/usr/bin/env python\n2 # -*- c(...truncated)"
    }
  ],
  "TokenAll": 13032,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Easy",
    "issue_origin": {
      "title": "Inconsistent method usage for Python versions",
      "description": "The test method test_manage_few_records in file piskvorky_sqlitedict/tests/test_temp_db.py uses self.d.iteritems() for Python 2 but does not use self.d.items() for both Python 2 and 3 versions, which could lead to incorrect behavior in Python 2 as iteritems() does not exist in Python 3. Ensure compatibility for both versions by properly guarding the output with 'six' or checking the Python version.",
      "explanation": "### Issue Summary\n\nThe issue is related to the usage of dictionary methods in the `test_manage_few_records` method within a test file. The test method is written with Python 2’s `iteritems()` method for dictionaries, which is not available in Python 3. By using `self.d.iteritems()` in Python 2, the code does not ensure compatibility across both Python 2 and Python 3, potentially leading to errors or incorrect behavior in Python 3. The suggested solution is to use the `six` library or a version check to handle compatibility issues.\n\n### Commit Content\n\nBased on the given issue, a likely commit to address this problem might involve:\n\n1. Importing the `six` library, which provides utility functions for writing code that is compatible with both Python 2 and 3.\n2. Modifying the `test_manage_few_records` method to use `six.iteritems()` for compatibility.\n   \nThe commit would likely look something like this:\n\n```python\n6 import unittest\n7 import sqlitedict\n8 import six  # Added this import\n  \n9 from sys import version_info\n10 major_version = version_info[0]\n\n56     def test_manage_few_records(self):\n57         ''' test_manage_few_records\n58         '''\n59         self.d['abc'] = 'lmno'\n60         self.d['xyz'] = 'pdq'\n61         self.assertEqual(len(self.d), 2)\n62         if major_version == 2:\n63             self.assertEqual(list(six.iteritems(self.d)), [('abc', 'lmno'), ('xyz', 'pdq')])  # Changed usage to six.iteritems\n64         self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n65         self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n66         self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n67         self.assertEqual(list(self.d), ['abc', 'xyz'])\n```\n\n### Explanation of How the Commit Solves The Issue\n\nHere’s the detailed explanation of how this commit addresses the issue:\n\n1. **Importing the `six` Library**:\n   - The `six` library is specifically designed to aid in writing code that is compatible with both Python 2 and Python 3. By importing `six`, the code gains access to its compatibility functions, like `six.iteritems()`.\n\n2. **Changing `iteritems()` to `six.iteritems()`**:\n   - The original code uses `self.d.iteritems()` when running in Python 2, which is invalid in Python 3 as `iteritems()` was removed in this version.\n   - By replacing `self.d.iteritems()` with `six.iteritems(self.d)`, the code utilizes a compatibility layer that works correctly in both Python 2 and 3. `six.iteritems()` will call `iteritems()` in Python 2 and `items()` (the equivalent in Python 3) when running in Python 3.\n\n3. **Guarding the Code with a Version Check**:\n   - The version check (`if major_version == 2`) ensures that this specific block of code (`six.iteritems(self.d)` check) is only run in Python 2, since `self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])` already accounts for Python 3.\n\n### Benefits of the Solution\n\nThis approach ensures the following:\n\n- **Cross-Version Compatibility**: By using `six.iteritems()`, the code can seamlessly handle dictionary iterations in both Python 2 and Python 3.\n- **Minimal Code Changes**: Only the problematic part of the test method is altered. The rest of the method remains unchanged and functions correctly.\n- **Maintained Functionality**: The intended assertions and functionality of the test are preserved, ensuring that it accurately verifies the behavior of `sqlitedict`.\n\nThus, this commit efficiently resolves the compatibility issue, ensuring that the same test code works across different Python versions without raising method-related errors."
    },
    "issue_message": {
      "title": "Temporary SqliteDict not properly cleared in tests",
      "description": "Several tests in the \"piskvorky_sqlitedict/tests/test_temp_db.py\" file fail due to leftover data from previous tests. For example, `test_create_sqlitedict` fails because the SqliteDict contains the entry {'abc': 'edf'}. This indicates that `tearDown` methods do not effectively clear or reset the environment. Ensure `tearDown` methods and possibly `setUp` methods properly clear the database to guarantee isolated and independent tests.",
      "explanation": "### Summary of the Issue\n\nThe issue at hand is related to temporary `SqliteDict` instances not being properly cleared or reset between tests in the `piskvorky_sqlitedict/tests/test_temp_db.py` file. This results in leftover data from previous tests affecting the outcomes of subsequent tests, leading to several test failures. Specific symptoms include unexpected entries in the `SqliteDict`, incorrect counts of entries, and tests for non-existent keys failing because the keys actually exist due to leftovers.\n\n### Cause of the Issue\n\nThe primary cause of this issue is that the `tearDown` method in the test class `TempSqliteDictTest` merely closes the `SqliteDict` instance without thoroughly ensuring that the database is reset or the file is properly deleted. Consequently, leftover data remains, contaminating subsequent tests.\n\n### Solution\n\nTo properly address this issue, we need to ensure that the test setup and teardown mechanisms completely clean up any remnants of the database. This involves not just closing the `SqliteDict` instance, but also deleting the temporary database file created during the tests.\n\n### Suggested Commit\n\nHere is a proposed change to the `test_temp_db.py` file to fix the issue:\n\n```python\n13 class TempSqliteDictTest(unittest.TestCase):\n14 \n15     def setUp(self):\n16         self.db_file = tempfile.NamedTemporaryFile(delete=False)\n17         self.d = sqlitedict.SqliteDict(self.db_file.name)\n18 \n19     def tearDown(self):\n20         self.d.close()\n21         os.remove(self.db_file.name)\n\n22\n23     def test_create_sqlitedict(self):\n24         ''' test_create_sqlitedict\n25         '''\n26         self.assertIsInstance(self.d, sqlitedict.SqliteDict)\n27         self.assertEqual(dict(self.d), {})\n28         self.assertEqual(list(self.d), [])\n29         self.assertEqual(len(self.d), 0)\n\n30\n31     def test_assign_values(self):\n32         ''' test_assign_values\n33         '''\n34         self.d['abc'] = 'edf'\n35         self.assertEqual(self.d['abc'], 'edf')\n36         self.assertEqual(len(self.d), 1)\n\n37\n38     def test_clear_data(self):\n39         ''' test_clear_data\n40         '''\n41         self.d.update(a=1, b=2, c=3)\n42         self.assertEqual(len(self.d), 3)\n43         self.d.clear()\n44         self.assertEqual(len(self.d), 0)\n\n45\n46     def test_manage_one_record(self):\n47         ''' test_manage_one_record\n48         '''\n49         self.d['abc'] = 'rsvp' * 100\n50         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n51         self.d['abc'] = 100\n52         self.assertEqual(self.d['abc'], 100)\n53         self.assertEqual(len(self.d), 1)\n54         del self.d['abc']\n55         self.assertEqual(len(self.d), 0)\n56         self.assertTrue(not self.d)\n\n57     \n58     def test_manage_few_records(self):\n59         ''' test_manage_few_records\n60         '''\n61         self.d['abc'] = 'lmno'\n62         self.d['xyz'] = 'pdq'\n63         self.assertEqual(len(self.d), 2)\n64         if major_version == 2:\n65             self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n66         self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n67         self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n68         self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n69         self.assertEqual(list(self.d), ['abc', 'xyz'])\n\n70\n71     def test_update_records(self):\n72         ''' test_update_records\n73         '''\n74         self.d.update([('v', 'w')], p='x', q='y', r='z')\n75         self.assertEqual(len(self.d), 4)\n76         self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n77         self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n\n78\n79     def test_handling_errors(self):\n80         ''' test_handling_errors\n81         '''\n82         def get_value(d, k):\n83             return d[k]\n\n84\n85         def remove_nonexists(d, k):\n86             del d[k]\n\n87\n88         with self.assertRaises(KeyError):\n89             remove_nonexists(self.d, 'abc')\n90         with self.assertRaises(KeyError):\n91             get_value(self.d, 'abc')\n```\n\n### Explanation of the Changes"
    },
    "issue_ground": {
      "title": "Autocommit Feature Not Working Correctly and Minor Bugs in Test Cases",
      "description": "Users have reported an issue where the `autocommit` feature of the SqliteDict class does not work as expected. Specifically, changes are not being committed as they should be when `autocommit` is enabled. This has potential data integrity issues as users rely on autocommit to ensure their data is saved without manually calling the commit function.\n\nAdditionally, there are minor bugs in the test cases for key value assignments and list sorting when retrieving keys. These issues do not immediately cause failures but may lead to data inconsistency in certain edge cases.\n\nThe issue manifests in the following ways:\n1. Assigning values to the SqliteDict does not trigger an automatic commit when `autocommit` is enabled.\n2. Test cases, particularly `test_manage_one_record` and `test_manage_few_records`, have inconsistencies in value assignments and checks.\n3. The `__delitem__` method should check for `autocommit` to ensure deletions are committed when the feature is enabled.\n4. The setup script does not correctly set the docstring for `SetupDevelop` class, leading to potential confusion and incomplete documentation.\n5. The `summarize_pr.py` script might return an unnecessary value that should be handled or removed to prevent runtime errors or unintended behavior.\n\nSteps to Reproduce:\n1. Initialize a SqliteDict with `autocommit=True`.\n2. Attempt to assign a value and see if it gets automatically committed without manually calling commit.\n3. Run the provided test cases and observe any inconsistencies or failures.\n\nExpected Behavior:\n1. Values should be automatically committed upon assignment when `autocommit` is enabled.\n2. Test cases should correctly handle value assignments and retrievals without inconsistencies.\n3. Deletions should also trigger a commit if `autocommit` is enabled.\n4. The setup script should correctly set the docstring for clarity.\n5. The `summarize_pr.py` script should not return unnecessary values.\n\nImpact:\nThis issue impacts the reliability and data integrity of applications relying on SqliteDict for persistent storage and could lead to undetected bugs in test cases and scripts.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `autocommit` feature of the `SqliteDict` class, which isn't functioning as intended. When `autocommit` is enabled, changes should be automatically committed, ensuring data integrity without needing to manually call the commit function. Besides, there are minor bugs in the test cases related to value assignments and list sorting. Moreover, the `__delitem__` method is missing the `autocommit` check to ensure deletions are committed, the setup script's docstring isn't set correctly, and there's a potential issue with the `summarize_pr.py` script returning an unnecessary value.\n\n### Details of the Commit\nGiven that the specific commit isn't provided, we'll need to make educated assumptions about the necessary changes. Based on the issue, the commit likely contains:\n\n1. **Changes to SqliteDict to Ensure Autocommit Behavior:**\n    - Modifying assignment operations to trigger auto-commit.\n    - Updating the `__delitem__` method to check for and perform an auto-commit after deletion.\n   \n2. **Changes to the Test Cases:**\n    - Ensuring that the test cases reflect the correct handling of autocommit and correctly test the functionalities without inconsistencies.\n    - Fixing errors in test assertions to match the expected outcomes.\n   \n3. **Updates to the Setup Script and summarize_pr.py Script:**\n    - Correctly setting the docstring for the `SetupDevelop` class.\n    - Handling or removing unnecessary return values in `summarize_pr.py` to prevent runtime errors.\n\n### Explanation of the Solution\n\n#### 1. Ensuring Autocommit in SqliteDict\n\nTo resolve the autocommit issue, the `SqliteDict` class needs to be modified so that any changes (insertion, update, deletion) are automatically committed if `autocommit` is enabled.\n\n```python\n# Inside the __setitem__ method\ndef __setitem__(self, key, value):\n    ...\n    self.conn.execute(\"INSERT OR REPLACE INTO ...\", (key, serialized_value))\n    if self.autocommit:\n        self.commit()\n\n# Inside the __delitem__ method\ndef __delitem__(self, key):\n    ...\n    self.conn.execute(\"DELETE FROM ... WHERE key=?\", (key,))\n    if self.autocommit:\n        self.commit()\n```\n\n#### 2. Fixing Test Cases\n\nThe test cases need to be fixed to ensure they correctly reflect and verify the autocommit behavior. Here are a few examples based on the issues mentioned:\n\n```python\n# Fixing test_clear_data\ndef test_clear_data(self):\n    self.d.update(a=1, b=2, c=3)\n    self.assertEqual(len(self.d), 3)  # Original assertion\n    self.d.clear()\n    self.assertEqual(len(self.d), 0)\n\n# Fixing test_manage_one_record\ndef test_manage_one_record(self):\n    self.d['abc'] = 'rsvp' * 100\n    self.assertEqual(self.d['abc'], 'rsvp' * 100)\n    self.d['abc'] = 100\n    self.assertEqual(self.d['abc'], 100)\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)  # Ensure the dict is empty\n```\n\n#### 3. Updating the Setup Script and summarize_pr.py\n\n```python\n# Correctly setting the docstring for the SetupDevelop class\nclass SetupDevelop:\n    \"\"\"\n    SetupDevelop is used to provide a custom installation for development purposes.\n    \"\"\"\n    ...\n\n# Removing or handling the unnecessary return value in summarize_pr.py\ndef summarize_pr_function(...):\n    ...\n    # Remove or handle the unnecessary return value here\n    return None  # If the return value is not required\n```\n\n### Explanation of how the Commit Solves the Issue\n\n1. **Autocommit**:\n    - By adding `self.commit()` under each modification method (e.g., `__setitem__`, `__delitem__`), the `SqliteDict` ensures that every change is committed immediately if `autocommit` is enabled. This guarantees data integrity as users expect.\n\n2. **Test Cases**:\n    - Fixing the test cases ensures they accurately reflect the correct behavior of the class. This prevents false negatives and ensures that the `autocommit` functionality, among others, is thoroughly verified.\n\n3. **Docstring and summarize_pr.py**:\n    - Correctly setting the docstring improves clarity and documentation, helping future developers understand the setup script better.\n    - Removing unnecessary return values from `summarize_pr.py` helps prevent runtime errors or unintended behavior, making the script more robust and reliable.\n\nOverall, the commit ensures that the `SqliteDict` behaves as expected with `autocommit` enabled, test cases validate the desired functionality accurately"
    },
    "issue_ground_truth": {
      "title": "Autocommit Feature Not Working Correctly and Minor Bugs in Test Cases",
      "description": "Users have reported an issue where the `autocommit` feature of the SqliteDict class does not work as expected. Specifically, changes are not being committed as they should be when `autocommit` is enabled. This has potential data integrity issues as users rely on autocommit to ensure their data is saved without manually calling the commit function.\n\nAdditionally, there are minor bugs in the test cases for key value assignments and list sorting when retrieving keys. These issues do not immediately cause failures but may lead to data inconsistency in certain edge cases.\n\nThe issue manifests in the following ways:\n1. Assigning values to the SqliteDict does not trigger an automatic commit when `autocommit` is enabled.\n2. Test cases, particularly `test_manage_one_record` and `test_manage_few_records`, have inconsistencies in value assignments and checks.\n3. The `__delitem__` method should check for `autocommit` to ensure deletions are committed when the feature is enabled.\n4. The setup script does not correctly set the docstring for `SetupDevelop` class, leading to potential confusion and incomplete documentation.\n5. The `summarize_pr.py` script might return an unnecessary value that should be handled or removed to prevent runtime errors or unintended behavior.\n\nSteps to Reproduce:\n1. Initialize a SqliteDict with `autocommit=True`.\n2. Attempt to assign a value and see if it gets automatically committed without manually calling commit.\n3. Run the provided test cases and observe any inconsistencies or failures.\n\nExpected Behavior:\n1. Values should be automatically committed upon assignment when `autocommit` is enabled.\n2. Test cases should correctly handle value assignments and retrievals without inconsistencies.\n3. Deletions should also trigger a commit if `autocommit` is enabled.\n4. The setup script should correctly set the docstring for clarity.\n5. The `summarize_pr.py` script should not return unnecessary values.\n\nImpact:\nThis issue impacts the reliability and data integrity of applications relying on SqliteDict for persistent storage and could lead to undetected bugs in test cases and scripts.",
      "explanation": "### Summary of the Issue:\n\nThe issue addresses several problems in the `SqliteDict` class and its related test cases:\n1. **Autocommit Feature**: The `autocommit` feature of `SqliteDict` does not work as expected. When `autocommit` is enabled, changes are not being automatically committed, potentially leading to data integrity issues.\n2. **Test Case Bugs**: Minor bugs exist in the test cases for key-value assignments and key retrieval, which could lead to data inconsistencies.\n3. **`__delitem__` Method**: This method does not check the `autocommit` flag, causing deletion operations not to be committed automatically.\n4. **Setup Script**: The setup script fails to set the docstring for the `SetupDevelop` class correctly.\n5. **Utility Script `summarize_pr.py`**: The script has a functionality where it returns an unnecessary value, potentially leading to runtime errors.\n\n### Commit Content and Explanation:\n\nThe commit introduces changes to various parts of the codebase to address each of the identified issues:\n\n1. **Autocommit Implementation**: \n   - The commit adds a conditional check to ensure that commits are only triggered if the `autocommit` attribute is set to `True`. This change ensures that whenever a value is assigned or deleted, the changes are committed automatically, preventing potential data integrity issues.\n   \n2. **Test Case Fixes**:\n   - The commit updates the test cases to ensure that key-value assignments and key retrievals handle `SqliteDict` entries correctly. Specifically:\n     - Correcting the assigned values and their assertions to be consistent.\n     - Adjusting the key sorting procedure to convert dictionary keys to a list before sorting, ensuring it works as intended across different Python versions.\n   \n3. **`__delitem__` Method Adjustment**:\n   - The `__delitem__` method was modified to check for `autocommit` and perform a commit after deletion if `autocommit` is enabled. This ensures automatic data persistence upon deletion operations.\n\n4. **Setup Script Docstring**:\n   - The docstring for the `SetupDevelop` class is correctly set, ensuring that the documentation is comprehensive and aids in understanding the setup process.\n   \n5. **Removal of Unnecessary Return Value in Utility Script**:\n   - The commit removes the unnecessary return value from the `summarize_pr.py` script. By doing this, potential runtime errors and unintended behaviors are avoided, enhancing the script's robustness and reliability.\n\n### Solution Explanation:\n\nThe solution provided in the commit effectively addresses the issues:\n\n- **Autocommit Feature**: The conditional commit logic ensures that changes are immediately committed to the database when `autocommit` is enabled. This change significantly enhances data integrity by ensuring that each modification is persisted without needing explicit commits from the user.\n\n- **Test Case Reliability**: By fixing inconsistencies in the test cases, especially around key-value assignments and retrievals, the commit ensures the test suite accurately reflects the intended functionality of `SqliteDict`. This reduces the chance of undetected bugs and guarantees that the class works correctly in various scenarios.\n\n- **Deletion and Autocommit Integration**: The modification to the `__delitem__` method to respect the `autocommit` flag ensures that detrimental changes (like deletions) are also automatically persisted. This is crucial for data consistency and integrity, especially in applications relying heavily on frequent deletions.\n\n- **Improved Documentation in Setup Script**: Setting the correct docstring in the setup script helps developers better understand the setup process, leading to fewer misunderstandings and smoother environment preparation.\n\n- **Utility Script Enhancement**: By removing the unnecessary return value, the commit improves the stability of the `summarize_pr.py` script, preventing errors that may arise from handling unintended return values.\n\nOverall, the commit systematically addresses each aspect of the reported issue with targeted changes, ensuring the `SqliteDict` works as expected and improving the overall robustness and reliability of the codebase."
    },
    "location_origin": [
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "56": "test_manage_few_records"
        },
        "content_all": {
          "53": "        self.assertEqual(len(self.d), 0)\n",
          "54": "        self.assertTrue(not self.d)\n",
          "55": " \n",
          "56": "    def test_manage_few_records(self):\n",
          "57": "        ''' test_manage_few_records\n",
          "58": "        '''\n",
          "59": "        self.d['abc'] = 'lmno'\n",
          "60": "        self.d['xyz'] = 'pdq'\n",
          "61": "        self.assertEqual(len(self.d), 2)\n",
          "62": "        if major_version == 2:\n",
          "63": "            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n",
          "64": "        self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n",
          "65": "        self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n",
          "66": "        self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n",
          "67": "        self.assertEqual(list(self.d), ['abc', 'xyz'])\n",
          "68": " \n",
          "69": "    def test_update_records(self):\n",
          "70": "        ''' test_update_records\n",
          "71": "        '''\n"
        },
        "content_change": {
          "63": "            self.assertEqual(list(six.iteritems(self.d)), [('abc', 'lmno'), ('xyz', 'pdq')])\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "10": "module-level"
        },
        "content_all": {
          "7": "import sqlitedict\n",
          "8": " \n",
          "9": "from sys import version_info\n",
          "10": "major_version = version_info[0]\n",
          "11": " \n",
          "12": " \n",
          "13": "class TempSqliteDictTest(unittest.TestCase):\n",
          "14": " \n",
          "15": "    def setUp(self):\n",
          "16": "        self.d = sqlitedict.SqliteDict()\n",
          "17": " \n",
          "18": "    def tearDown(self):\n",
          "19": "        self.d.close()\n",
          "20": " \n",
          "21": "    def test_create_sqlitedict(self):\n",
          "22": "        ''' test_create_sqlitedict\n"
        },
        "content_change": {
          "7": "import sqlitedict\n",
          "8": "import six\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "13": "TempSqliteDictTest"
        },
        "content_all": {
          "12": "\n",
          "13": "class TempSqliteDictTest(unittest.TestCase):\n",
          "14": "\n",
          "15": "    def setUp(self):\n",
          "16": "        self.d = sqlitedict.SqliteDict()\n",
          "17": "\n",
          "18": "    def tearDown(self):\n",
          "19": "        self.d.close()\n",
          "20": "\n",
          "21": "    def test_create_sqlitedict(self):\n",
          "22": "        ''' test_create_sqlitedict\n"
        },
        "content_change": {
          "15": "    def setUp(self):\n",
          "16": "        self.db_file = tempfile.NamedTemporaryFile(delete=False)\n",
          "17": "        self.d = sqlitedict.SqliteDict(self.db_file.name)\n",
          "18": "    def tearDown(self):\n",
          "19": "        self.d.close()\n",
          "20": "        os.remove(self.db_file.name)\n"
        }
      }
    ],
    "location_ground": [
      {
        "location": "```json\n[\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"571\": \"__setitem__\"\n        },\n        \"content_all\": {\n            \"568\": \"\\n\",\n            \"569\": \"    def __setitem__(self, key, value):\\n\",\n            \"570\": \"        self.conn.execute(\\\"INSERT OR REPLACE INTO some_table (key, value) VALUES (?, ?)\\\", (key, value))\\n\",\n            \"571\": \"        if self.autocommit:\\n\",\n            \"572\": \"            self.commit()\\n\",\n            \"573\": \"\\n\",\n            \"574\": \"    def __delitem__(self, key):\\n\"\n        },\n        \"content_change\": {\n            \"569\": \"    def __setitem__(self, key, value):\\n\",\n            \"570\": \"        self.conn.execute(\\\"INSERT OR REPLACE INTO some_table (key, value) VALUES (?, ?)\\\", (key, value))\\n\",\n            \"571\": \"        if self.autocommit:\\n\",\n            \"572\": \"            self.commit()\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"574\": \"__delitem__\"\n        },\n        \"content_all\": {\n            \"572\": \"            self.commit()\\n\",\n            \"573\": \"\\n\",\n            \"574\": \"    def __delitem__(self, key):\\n\",\n            \"575\": \"        self.conn.execute(\\\"DELETE FROM some_table WHERE key=?\\\", (key,))\\n\",\n            \"576\": \"        if self.autocommit:\\n\",\n            \"577\": \"            self.commit()\\n\",\n            \"578\": \"\\n\",\n            \"579\": \"    def __init__(self, ... ): # other methods\\n\"\n        },\n        \"content_change\": {\n            \"574\": \"    def __delitem__(self, key):\\n\",\n            \"575\": \"        self.conn.execute(\\\"DELETE FROM some_table WHERE key=?\\\", (key,))\\n\",\n            \"576\": \"        if self.autocommit:\\n\",\n            \"577\": \"            self.commit()\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"36\": \"test_clear_data\"\n        },\n        \"content_all\": {\n            \"35\": \"\\n\",\n            \"36\": \"    def test_clear_data(self):\\n\",\n            \"37\": \"        ''' test_clear_data\\n\",\n            \"38\": \"        '''\\n\",\n            \"39\": \"        self.d.update(a=1, b=2, c=3)\\n\",\n            \"40\": \"        self.assertEqual(len(self.d), 3)\\n\",\n            \"41\": \"        self.d.clear()\\n\",\n            \"42\": \"        self.assertEqual(len(self.d), 0)\\n\",\n            \"43\": \"        self.assertTrue(not self.d)  # Ensure the dict is empty\\n\",\n            \"44\": \"\\n\"\n        },\n        \"content_change\": {\n            \"43\": \"        self.assertTrue(not self.d)  # Ensure the dict is empty\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"44\": \"test_manage_one_record\"\n        },\n        \"content_all\": {\n            \"48\": \"        self.d['abc'] = 'rsvp' * 100\\n\",\n            \"49\": \"        self.assertEqual(self.d['abc'], 'rsvp' * 100)\\n\",\n            \"50\": \"        self.d['abc'] = 100\\n\",\n            \"51\": \"        self.assertEqual(self.d['abc'], 100)\\n\",\n            \"52\": \"        self.assertEqual(len(self.d), 1)\\n\",\n            \"53\": \"        del self.d['abc']\\n\",\n            \"54\": \"        self.assertTrue(not self.d)  # Ensure the dict is empty\\n\"\n        },\n        \"content_change\": {\n            \"50\": \"        self.d['abc'] = 100\\n\",\n            \"51\": \"        self.assertEqual(self.d['abc'], 100)\\n\",\n            \"52\": \"        self.assertEqual(len(self.d), 1)\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"56\": \"test_manage_few_records\"\n        },\n        \"content_all\": {\n            \"60\": \"        self.d['abc'] = 'lmno'\\n\",\n            \"61\": \"        self.d['xyz'] ="
      }
    ],
    "location_ground_exp": [
      {
        "location": "```json\n[\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"189\": \"__setitem__\"\n        },\n        \"content_all\": {\n            \"186\": \"\\n\",\n            \"187\": \"    def __setitem__(self, key, value):\\n\",\n            \"188\": \"        with self.conn:\\n\",\n            \"189\": \"            pickled_value = dumps((value,), PICKLE_PROTOCOL)\\n\",\n            \"190\": \"            self.conn.execute('INSERT OR REPLACE INTO kv (key, value) VALUES (?, ?)', (key, pickled_value))\\n\",\n            \"191\": \"            if self.autocommit:\\n\",\n            \"192\": \"                self.conn.commit()\\n\",\n            \"193\": \"\\n\",\n            \"194\": \"    def __delitem__(self, key):\\n\"\n        },\n        \"content_change\": {\n            \"192\": \"            if self.autocommit:\\n\",\n            \"193\": \"                self.conn.commit()\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"194\": \"__delitem__\"\n        },\n        \"content_all\": {\n            \"191\": \"            if self.autocommit:\\n\",\n            \"192\": \"                self.conn.commit()\\n\",\n            \"193\": \"\\n\",\n            \"194\": \"    def __delitem__(self, key):\\n\",\n            \"195\": \"        with self.conn:\\n\",\n            \"196\": \"            cur = self.conn.execute('DELETE FROM kv WHERE key = ?', (key,))\\n\",\n            \"197\": \"            if cur.rowcount == 0:\\n\",\n            \"198\": \"                raise KeyError(key)\\n\",\n            \"199\": \"            if self.autocommit:\\n\",\n            \"200\": \"                self.conn.commit()\\n\",\n            \"201\": \"\\n\",\n            \"202\": \"    def __contains__(self, key):\\n\"\n        },\n        \"content_change\": {\n            \"199\": \"            if self.autocommit:\\n\",\n            \"200\": \"                self.conn.commit()\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"29\": \"test_assign_values\"\n        },\n        \"content_all\": {\n            \"26\": \"        self.assertEqual(list(self.d), [])\\n\",\n            \"27\": \"        self.assertEqual(len(self.d), 0)\\n\",\n            \"28\": \" \\n\",\n            \"29\": \"    def test_assign_values(self):\\n\",\n            \"30\": \"        ''' test_assign_values\\n\",\n            \"31\": \"        '''\\n\",\n            \"32\": \"        self.d['abc'] = 'edf'\\n\",\n            \"33\": \"        self.assertEqual(self.d['abc'], 'edf')\\n\",\n            \"34\": \"        self.assertEqual(len(self.d), 1)\\n\",\n            \"35\": \"        self.d.commit()  # Ensure commit for autocommit=False\\n\",\n            \"36\": \"\\n\",\n            \"37\": \"    def test_clear_data(self):\\n\"\n        },\n        \"content_change\": {\n            \"35\": \"        self.d.commit()  # Ensure commit for autocommit=False\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"44\": \"test_manage_one_record\"\n        },\n        \"content_all\": {\n            \"41\": \"        self.assertEqual(len(self.d), 0)\\n\",\n            \"42\": \" \\n\",\n            \"43\": \"     def test_clear_data(self):\",\n            \"44\": \"     ''' test_clear_data\\n\",\n            \"45\": \"     '''\\n\",\n            \"46\": \"     self.d.update(a=1, b=2, c=3)\\n\",\n            \"47\": \"     self.assertEqual(len(self.d), 3)\\n\",\n            \"48\": \"     self.d.clear()\\n\",\n            \"49\": \"     self.assertEqual(len(self.d), 0)\\n\",\n            \"50\": \" \\n\",\n            \"51\": \"     def test_manage_one_record(self):\\n\",\n            \"52\": \"     ''' test_manage_one_record\\n\",\n            \"53\": \"     '''\\n\",\n            \"54\": \"     self.d['abc'] = 'rsvp' * 100\\n\",\n            \"55\": \"     self.assertEqual(self.d['abc'], 'rsvp' * 100)\\n\",\n            \"56\": \"     self.d['abc'] = 100\\n\",\n            \"57\": \"     self.assertEqual"
      }
    ],
    "location_ground_truth": [
      {
        "file": "piskvorky_sqlitedict/release/summarize_pr.py",
        "function": null,
        "content_all": {
          "16": "        return text\n",
          "17": "\n",
          "18": "\n",
          "19": "for prid in sys.argv[1:]:\n"
        },
        "content_change": {
          "16": "        return text\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/setup.py",
        "function": null,
        "content_all": {
          "42": "        setuptools.command.develop.develop.run(self)\n",
          "43": "\n",
          "44": "\n",
          "45": "# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n",
          "46": "\n",
          "47": "\n",
          "48": "setup(\n"
        },
        "content_change": {
          "45": "# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": null,
        "content_all": {
          "314": "        self.commit()\n"
        },
        "content_change": {
          "314": "        self.commit()\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": null,
        "content_all": {
          "133": "        self.assertNotIn('value', next_db.values())\n"
        },
        "content_change": {
          "133": "        self.assertNotIn('value', next_db.values())\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "44": "test_manage_one_record"
        },
        "content_all": {
          "46": "        '''\n",
          "47": "        self.d['abc'] = 'rsvp' * 100\n",
          "48": "        self.assertEqual(self.d['abc'], 'rsvp' * 100)\n",
          "49": "        self.d['abc'] = 100\n",
          "50": "        self.assertEqual(self.d['abc'], 100)\n",
          "51": "        self.assertEqual(len(self.d), 1)\n",
          "52": "        del self.d['abc']\n",
          "53": "        self.assertEqual(len(self.d), 0)\n"
        },
        "content_change": {
          "49": "        self.d['abc'] = 100\n",
          "50": "        self.assertEqual(self.d['abc'], 100)\n"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": null,
        "content_all": {
          "77": "        self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n"
        },
        "content_change": {
          "77": "        self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n"
        }
      }
    ],
    "patch_i": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -7,6 +7,7 @@ import sqlitedict\n+import six\n from sys import version_info\n major_version = version_info[0]\n \n@@ -62,7 +63,7 @@ class TempSqliteDictTest(unittest.TestCase):\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(six.iteritems(self.d)), [('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",
    "patch_im": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -15,10 +15,11 @@\n     def setUp(self):\n-        self.d = sqlitedict.SqliteDict()\n+        self.db_file = tempfile.NamedTemporaryFile(delete=False)\n+        self.d = sqlitedict.SqliteDict(self.db_file.name)\n\n     def tearDown(self):\n         self.d.close()\n+        os.remove(self.db_file.name)\n \n     def test_create_sqlitedict(self):\n         ''' test_create_sqlitedict\n",
    "patch_il": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -7,6 +7,7 @@\n import sqlitedict\n \n from sys import version_info\n+import six\n major_version = version_info[0]\n \n \n@@ -62,7 +63,7 @@\n         if major_version == 2:\n-            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n+            self.assertEqual(list(six.iteritems(self.d)), [('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\n",
    "patch_iml": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -12,7 +12,9 @@\n \n class TempSqliteDictTest(unittest.TestCase):\n \n-    def setUp(self):\n-        self.d = sqlitedict.SqliteDict()\n+    def setUp(self):\n+        self.db_file = tempfile.NamedTemporaryFile(delete=False)\n+        self.d = sqlitedict.SqliteDict(self.db_file.name)\n \n     def tearDown(self):\n         self.d.close()\n+        os.remove(self.db_file.name)\n",
    "patch_ground": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -173,6 +173,10 @@\n         self.conn.execute(\"INSERT OR REPLACE INTO {0} (key, value) VALUES (?, ?)\".format(self.tablename),\n                           (key, bytes(dumps(value, protocol=PICKLE_PROTOCOL))))\n         if self.autocommit:\n+            self.commit()\n+\n+    def __delitem__(self, key):\n+        with self.conn:\n             self.commit()\n \n     def __iter__(self):\n@@ -316,6 +320,12 @@\n         return len(self)\n \n     def __delitem__(self, key):\n+        if key not in self:\n+            raise KeyError(key)\n+        self.conn.execute(\"DELETE FROM {} WHERE key = ?\".format(self.tablename), (key,))\n+        if self.autocommit:\n+            self.commit()\n+\n         del self.conn.execute(\"DELETE FROM {0} WHERE key = ?\".format(self.tablename)).fetchone()\n \n     def commit(self):\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -36,10 +36,10 @@\n         self.assertEqual(len(self.d), 3)\n         self.d.clear()\n         self.assertEqual(len(self.d), 0)\n-        self.d['abc'] = 'edf'\n-        self.assertEqual(self.d['abc'], 'edf')\n+        self.d['xyz'] = 'pdq'\n+        self.d['abc'] = 'lmno'\n         self.assertEqual(len(self.d), 2)\n-        self.assertEqual(self.d['abc'], 100)\n+        self.assertEqual(self.d['xyz'], 'pdq')\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@@ -67,7 +67,7 @@\n             self.assertEqual(list(self.d), ['abc', 'xyz'])\n \n     def test_update_records(self):\n-        self.d.update([('v', 'w')], p='x', q='y', r='z')\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@@ -85,7 +85,7 @@\n \n     def test_handling_errors(self):\n         ''' test_handling_errors\n         '''\n         def get_value(d, k):\n             return d[k]\n \n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -1,6 +1,6 @@\n #!/usr/bin/env python\n # -*- coding: utf-8 -*-\n \"\"\"\n SetupDevelop is used to provide a custom installation for development purposes.\n \"\"\"\n from setuptools import setup, find_packages\n \n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -1,10 +1,8 @@\n # std imports\n import json\n import unittest\n import tempfile\n import os\n from unittest.mock import patch\n \n # local\n import sqlitedict\n@@ -2,8 +2,6 @@\n import json\n import unittest\n import tempfile\n import os\n from unittest.mock import patch\n \n # local\n import sqlitedict\n from sqlitedict import SqliteDict\n \n--- a/piskvorky_sqlitedict/tests/test_autocommit.py\n+++ b/piskvorky_sqlitedict/tests/test_autocommit.py\n@@ -1,6 +1,6 @@\n import os\n import sys\n \n import sqlitedict\n \n-def test():\n+if __name__ == '__main__':\n     \"\"\"Verify autocommit just behaves exactly like autocommit=False\"\"\"\n     path = './tmp.sqlite'\n     sd = sqlitedict.SqliteDict(path, autocommit=True)\n@@ -10,6 +10,8 @@\n     sd['a'] = 1\n     assert(sd['a'] == 1)\n     del sd['a']\n-    assert 'a' not in sd\n+    assert 'a' not in sd\n+\n+    sd.close()\n+    os.remove(path)\n",
    "patch_ground_location": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -568,17 +568,27 @@ def euclidean(a, b):\n def __setitem__(self, key, value):\n     self.conn.execute(\"INSERT OR REPLACE INTO some_table (key, value) VALUES (?, ?)\", (key, value))\n     if self.autocommit:\n         self.commit()\n\n def __delitem__(self, key):\n     self.conn.execute(\"DELETE FROM some_table WHERE key=?\", (key,))\n     if self.autocommit:\n         self.commit()\n",
    "patch_ground_exp": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -191,10 +191,11 @@\n             pickled_value = dumps((value,), PICKLE_PROTOCOL)\n             self.conn.execute('INSERT OR REPLACE INTO kv (key, value) VALUES (?, ?)', (key, pickled_value))\n             if self.autocommit:\n                 self.conn.commit()\n \n     def __delitem__(self, key):\n         with self.conn:\n             cur = self.conn.execute('DELETE FROM kv WHERE key = ?', (key,))\n             if cur.rowcount == 0:\n                 raise KeyError(key)\n             if self.autocommit:\n                 self.conn.commit()\n \n     def __contains__(self, key):\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -35,7 +35,6 @@\n         self.d['abc'] = 'edf'\n         self.assertEqual(self.d['abc'], 'edf')\n         self.assertEqual(len(self.d), 1)\n-        self.d.commit()  # Ensure commit for autocommit=False\n \n     def test_clear_data(self):\n         ''' test_clear_data\n@@ -52,10 +51,15 @@\n         self.d['abc'] = 'rsvp' * 100\n         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n         self.d['abc'] = 100\n         self.assertEqual(self.d['abc'], 100)\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--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -29,6 +29,9 @@\n except FileNotFoundError:\n     long_description = description\n\n+class SetupDevelop(develop):\n+    \"\"\"SetUp Develop class with proper docstring for clarity\"\"\"\n+\n setup(\n     name='sqlitedict',\n     version=__version__,\n\n--- a/piskvorky_sqlitedict/summarize_pr.py\n+++ b/piskvorky_sqlitedict/summarize_pr.py\n@@ -99,7 +99,7 @@\n     except ValueError as e:\n         print(str(e))\n         sys.exit(1)\n \n-    return 0\n+    return\n\n if __name__ == \"__main__\":\n     sys.exit(main())\n",
    "patch_ground_all": "\ndiff --git a/piskvorky_sqlitedict/release/summarize_pr.py b/piskvorky_sqlitedict/release/summarize_pr.py\nindex 1234567..89abcdef 100644\n--- a/piskvorky_sqlitedict/release/summarize_pr.py\n+++ b/piskvorky_sqlitedict/release/summarize_pr.py\n@@ -13,7 +13,6 @@ def summarize_pr(prid):\n             text += f\"- {line.strip()}\\n\"\n \n     print(text)\n-    return text\n \n for prid in sys.argv[1:]:\n     summarize_pr(prid)\ndiff --git a/piskvorky_sqlitedict/setup.py b/piskvorky_sqlitedict/setup.py\nindex 1234567..89abcdef 100644\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -42,7 +42,7 @@ class SetupDevelop(setuptools.command.develop.develop):\n     def run(self):\n         setuptools.command.develop.develop.run(self)\n \n-# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n+SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n \n setup(\n     name='piskvorky_sqlitedict',\ndiff --git a/piskvorky_sqlitedict/sqlitedict.py b/piskvorky_sqlitedict/sqlitedict.py\nindex 1234567..89abcdef 100644\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -311,6 +311,7 @@ class SqliteDict(collections.MutableMapping):\n \n     def __delitem__(self, key):\n         with self.conn:\n             self.conn.execute('DELETE FROM entries WHERE key = ?', (self.encode_key(key),))\n             self.conn.commit()\n+        if self.autocommit:\n+            self.commit()\n \n     def commit(self):\n         with self.conn:\ndiff --git a/piskvorky_sqlitedict/tests/test_temp_db.py b/piskvorky_sqlitedict/tests/test_temp_db.py\nindex 1234567..89abcdef 100644\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -47,9 +47,9 @@ class TempSqliteDictTest(unittest.TestCase):\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'] = 101 # Change from 100 to avoid repetition bugs\n         self.assertEqual(self.d['abc'], 101)\n         self.assertEqual(len(self.d), 1)\n         del self.d['abc']\n@@ -62,7 +62,8 @@ class TempSqliteDictTest(unittest.TestCase):\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()), sorted(['lmno', 'pdq']))\n         self.assertEqual(list(self.d.keys()), sorted(['abc', 'xyz']))\n         self.assertEqual(list(self.d), sorted(['abc', 'xyz']))\n",
    "patch_ground_truth": "--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -311,7 +311,8 @@\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-        self.commit()\n+        if self.autocommit:\n+            self.commit()\n \n     def __delitem__(self, key):\n         if self.flag == 'r':\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -42,7 +42,7 @@\n         setuptools.command.develop.develop.run(self)\n \n \n-# SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n+SetupDevelop.__doc__ = setuptools.command.develop.develop.__doc__\n \n \n setup(\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -46,8 +46,8 @@\n         '''\n         self.d['abc'] = 'rsvp' * 100\n         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n-        self.d['abc'] = 100\n-        self.assertEqual(self.d['abc'], 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@@ -74,7 +74,7 @@\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(self.d), sorted(['q', 'p', 'r', 'v']))\n+        self.assertEqual(sorted(list(self.d)), sorted(['q', 'p', 'r', 'v']))\n \n     def test_handling_errors(self):\n         ''' test_handling_errors\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--- a/piskvorky_sqlitedict/release/summarize_pr.py\n+++ b/piskvorky_sqlitedict/release/summarize_pr.py\n@@ -13,7 +13,6 @@\n         print('  ' + text, file=sys.stderr)\n     else:\n         pyperclip.copy(text)\n-        return text\n \n \n for prid in sys.argv[1:]:\n",
    "message": "======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 615, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 605, 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 473, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nFAIL: test_clear_data (test_named_db.NamedSqliteDictTest)\ntest_clear_data\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 40, in test_clear_data\n    self.assertEqual(len(self.d), 3)\nAssertionError: 4 != 3\n\n======================================================================\nFAIL: 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 25, in test_create_sqlitedict\n    self.assertEqual(dict(self.d), {})\nAssertionError: {'abc': 'edf'} != {}\n- {'abc': 'edf'}\n+ {}\n\n======================================================================\nFAIL: test_handling_errors (test_named_db.NamedSqliteDictTest)\ntest_handling_errors\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 89, in test_handling_errors\n    remove_nonexists(self.d, 'abc')\nAssertionError: KeyError not raised\n\n======================================================================\nFAIL: test_manage_one_record (test_named_db.NamedSqliteDictTest)\ntest_manage_one_record\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 51, in test_manage_one_record\n    self.assertEqual(len(self.d), 1)\nAssertionError: 2 != 1\n\n======================================================================\nFAIL: test_update_records (test_named_db.NamedSqliteDictTest)\ntest_update_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_temp_db.py\", line 73, in test_update_records\n    self.assertEqual(len(self.d), 4)\nAssertionError: 6 != 4\n\n----------------------------------------------------------------------\nRan 88 tests in 1.156s\n\nFAILED (failures=5, errors=1)",
    "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_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.assertIsInstance(self.d, sqlitedict.SqliteDict)\n25         self.assertEqual(dict(self.d), {})\n26         self.assertEqual(list(self.d), [])\n27         self.assertEqual(len(self.d), 0)\n28 \n29     def test_assign_values(self):\n30         ''' test_assign_values\n31         '''\n32         self.d['abc'] = 'edf'\n33         self.assertEqual(self.d['abc'], 'edf')\n34         self.assertEqual(len(self.d), 1)\n35 \n36     def test_clear_data(self):\n37         ''' test_clear_data\n38         '''\n39         self.d.update(a=1, b=2, c=3)\n40         self.assertEqual(len(self.d), 3)\n41         self.d.clear()\n42         self.assertEqual(len(self.d), 0)\n43 \n44     def test_manage_one_record(self):\n45         ''' test_manage_one_record\n46         '''\n47         self.d['abc'] = 'rsvp' * 100\n48         self.assertEqual(self.d['abc'], 'rsvp' * 100)\n49         self.d['abc'] = 100\n50         self.assertEqual(self.d['abc'], 100)\n51         self.assertEqual(len(self.d), 1)\n52         del self.d['abc']\n53         self.assertEqual(len(self.d), 0)\n54         self.assertTrue(not self.d)\n55 \n56     def test_manage_few_records(self):\n57         ''' test_manage_few_records\n58         '''\n59         self.d['abc'] = 'lmno'\n60         self.d['xyz'] = 'pdq'\n61         self.assertEqual(len(self.d), 2)\n62         if major_version == 2:\n63             self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n64         self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n65         self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n66         self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n67         self.assertEqual(list(self.d), ['abc', 'xyz'])\n68 \n69     def test_update_records(self):\n70         ''' test_update_records\n71         '''\n72         self.d.update([('v', 'w')], p='x', q='y', r='z')\n73         self.assertEqual(len(self.d), 4)\n74         # As far as I know dicts does not need to return\n75         # the elements in a specified order (sort() is required )\n76         self.assertEqual(sorted(self.d.items()), sorted([('q', 'y'), ('p', 'x'), ('r', 'z'), ('v', 'w')]))\n77         self.assertEqual(sorted(self.d), sorted(['q', 'p', 'r', 'v']))\n78 \n79     def test_handling_errors(self):\n80         ''' test_handling_errors\n81         '''\n82         def get_value(d, k):\n83             return d[k]\n84 \n85         def remove_nonexists(d, k):\n86             del d[k]\n87 \n88         with self.assertRaises(KeyError):\n89             remove_nonexists(self.d, 'abc')\n90         with self.assertRaises(KeyError):\n91             get_value(self.d, 'abc')"
      },
      {
        "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 im(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
        "content": "1 import os\n2 import sys\n3 \n4 import sqlitedict\n5 \n6 \n7 def test():\n8     \"Verify autocommit just be(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/setup.py",
        "content": "1 #!/usr/bin/env python\n2 # -*- c(...truncated)"
      }
    ],
    "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12"
  },
  "Score": {
    "Difficulty": "Easy",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 4,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 9,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Autocommit Feature Not Working Correctly and Minor Bugs in Test Cases",
      "description": "Users have reported an issue where the `autocommit` feature of the SqliteDict class does not work as expected. Specifically, changes are not being committed as they should be when `autocommit` is enabled. This has potential data integrity issues as users rely on autocommit to ensure their data is saved without manually calling the commit function.\n\nAdditionally, there are minor bugs in the test cases for key value assignments and list sorting when retrieving keys. These issues do not immediately cause failures but may lead to data inconsistency in certain edge cases.\n\nThe issue manifests in the following ways:\n1. Assigning values to the SqliteDict does not trigger an automatic commit when `autocommit` is enabled.\n2. Test cases, particularly `test_manage_one_record` and `test_manage_few_records`, have inconsistencies in value assignments and checks.\n3. The `__delitem__` method should check for `autocommit` to ensure deletions are committed when the feature is enabled.\n4. The setup script does not correctly set the docstring for `SetupDevelop` class, leading to potential confusion and incomplete documentation.\n5. The `summarize_pr.py` script might return an unnecessary value that should be handled or removed to prevent runtime errors or unintended behavior.\n\nSteps to Reproduce:\n1. Initialize a SqliteDict with `autocommit=True`.\n2. Attempt to assign a value and see if it gets automatically committed without manually calling commit.\n3. Run the provided test cases and observe any inconsistencies or failures.\n\nExpected Behavior:\n1. Values should be automatically committed upon assignment when `autocommit` is enabled.\n2. Test cases should correctly handle value assignments and retrievals without inconsistencies.\n3. Deletions should also trigger a commit if `autocommit` is enabled.\n4. The setup script should correctly set the docstring for clarity.\n5. The `summarize_pr.py` script should not return unnecessary values.\n\nImpact:\nThis issue impacts the reliability and data integrity of applications relying on SqliteDict for persistent storage and could lead to undetected bugs in test cases and scripts.",
      "explanation": "### Summary of the Issue:\n\nThe issue addresses several problems in the `SqliteDict` class and its related test cases:\n1. **Autocommit Feature**: The `autocommit` feature of `SqliteDict` does not work as expected. When `autocommit` is enabled, changes are not being automatically committed, potentially leading to data integrity issues.\n2. **Test Case Bugs**: Minor bugs exist in the test cases for key-value assignments and key retrieval, which could lead to data inconsistencies.\n3. **`__delitem__` Method**: This method does not check the `autocommit` flag, causing deletion operations not to be committed automatically.\n4. **Setup Script**: The setup script fails to set the docstring for the `SetupDevelop` class correctly.\n5. **Utility Script `summarize_pr.py`**: The script has a functionality where it returns an unnecessary value, potentially leading to runtime errors.\n\n### Commit Content and Explanation:\n\nThe commit introduces changes to various parts of the codebase to address each of the identified issues:\n\n1. **Autocommit Implementation**: \n   - The commit adds a conditional check to ensure that commits are only triggered if the `autocommit` attribute is set to `True`. This change ensures that whenever a value is assigned or deleted, the changes are committed automatically, preventing potential data integrity issues.\n   \n2. **Test Case Fixes**:\n   - The commit updates the test cases to ensure that key-value assignments and key retrievals handle `SqliteDict` entries correctly. Specifically:\n     - Correcting the assigned values and their assertions to be consistent.\n     - Adjusting the key sorting procedure to convert dictionary keys to a list before sorting, ensuring it works as intended across different Python versions.\n   \n3. **`__delitem__` Method Adjustment**:\n   - The `__delitem__` method was modified to check for `autocommit` and perform a commit after deletion if `autocommit` is enabled. This ensures automatic data persistence upon deletion operations.\n\n4. **Setup Script Docstring**:\n   - The docstring for the `SetupDevelop` class is correctly set, ensuring that the documentation is comprehensive and aids in understanding the setup process.\n   \n5. **Removal of Unnecessary Return Value in Utility Script**:\n   - The commit removes the unnecessary return value from the `summarize_pr.py` script. By doing this, potential runtime errors and unintended behaviors are avoided, enhancing the script's robustness and reliability.\n\n### Solution Explanation:\n\nThe solution provided in the commit effectively addresses the issues:\n\n- **Autocommit Feature**: The conditional commit logic ensures that changes are immediately committed to the database when `autocommit` is enabled. This change significantly enhances data integrity by ensuring that each modification is persisted without needing explicit commits from the user.\n\n- **Test Case Reliability**: By fixing inconsistencies in the test cases, especially around key-value assignments and retrievals, the commit ensures the test suite accurately reflects the intended functionality of `SqliteDict`. This reduces the chance of undetected bugs and guarantees that the class works correctly in various scenarios.\n\n- **Deletion and Autocommit Integration**: The modification to the `__delitem__` method to respect the `autocommit` flag ensures that detrimental changes (like deletions) are also automatically persisted. This is crucial for data consistency and integrity, especially in applications relying heavily on frequent deletions.\n\n- **Improved Documentation in Setup Script**: Setting the correct docstring in the setup script helps developers better understand the setup process, leading to fewer misunderstandings and smoother environment preparation.\n\n- **Utility Script Enhancement**: By removing the unnecessary return value, the commit improves the stability of the `summarize_pr.py` script, preventing errors that may arise from handling unintended return values.\n\nOverall, the commit systematically addresses each aspect of the reported issue with targeted changes, ensuring the `SqliteDict` works as expected and improving the overall robustness and reliability of the codebase."
    }
  }
}