{
  "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    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\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_overwrite_using_flag_n (test_core.NamedSqliteDictCreateOrReuseTest)\nRe-opening of a database with flag='c' destroys it all.\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 131, in test_overwrite_using_flag_n\n    next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 616, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 606, in check_raise_error\n    reraise(e_type, e_value, e_tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 46, in reraise\n    raise value.with_traceback(tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 474, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nERROR: test_assign_values (test_named_db.CreateNewSqliteDictTest)\ntest_assign_values\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_clear_data (test_named_db.CreateNewSqliteDictTest)\ntest_clear_data\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.CreateNewSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_handling_errors (test_named_db.CreateNewSqliteDictTest)\ntest_handling_errors\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_manage_few_records (test_named_db.CreateNewSqliteDictTest)\ntest_manage_few_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_manage_one_record (test_named_db.CreateNewSqliteDictTest)\ntest_manage_one_record\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_update_records (test_named_db.CreateNewSqliteDictTest)\ntest_update_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n----------------------------------------------------------------------\nRan 88 tests in 1.128s\n\nFAILED (errors=9)",
  "Patch": "--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -143,7 +143,7 @@\n \n \n class SqliteDict(DictClass):\n-    VALID_FLAGS = ['c', 'r', 'w']\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--- 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": {
        "143": "\n",
        "144": "\n",
        "145": "class SqliteDict(DictClass):\n",
        "146": "    VALID_FLAGS = ['c', 'r', 'w']\n"
      },
      "content_change": {
        "146": "    VALID_FLAGS = ['c', 'r', 'w']\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": "Enhance SqliteDict Functionality, Documentation, and Tests",
    "description": "Several issues have been identified that need addressing to improve the functionality, documentation, and testing of the SqliteDict project:\n\n1. **Addition of New Operation Mode**: The current implementation of `SqliteDict` lacks an operation mode for creating new databases. This is necessary to prevent unintentional overwriting of existing databases, which can lead to data loss.\n\n2. **Incomplete Documentation for Development Setup**: The documentation string for setup in the `SetupDevelop` class is commented out, which hinders the understanding of its behavior. Proper documentation is essential for comprehending its purpose and usage.\n\n3. **Update Test Consistency**: Inconsistent test cases for temporary databases can lead to discrepancies and false positives/negatives in test failures. Specifically, the testing of record management should be standardized to expect string values.\n\n4. **Improvement in Key Sorting During Tests**: The sorting of dictionary keys in tests doesn't consider type conversion, which raises compatibility issues across different Python versions.\n\n5. **Redundant Assertions in Flag Handling**: The assertion for checking an unrecognized flag's presence in the dictionary keys without value verification is redundant. This should be reviewed to ensure the integrity and completeness of test cases.\n\n6. **Clipboard Copy Function in PR Summarizer**: The `summarize_pr.py` script includes a conditional clipboard copy operation that doesn't return any text, causing it to fail in some environments. Consider streamlining or removing the feature to reduce complexity and improve reliability.\n\nAddressing these issues will enhance the robustness, maintainability, and usability of the `SqliteDict` library.",
    "explanation": "### Summary of the Issue\n\nThe GitHub issue focuses on enhancing the `SqliteDict` library by addressing several identified problems related to its functionality, documentation, and testing framework. The main issues are:\n\n1. Adding a new operation mode to the `SqliteDict` to create new databases without overwriting existing ones.\n2. Completing the documentation for the `SetupDevelop` class.\n3. Standardizing test cases to improve consistency, specifically around the temporary databases and handling of string values.\n4. Improving key sorting in tests to ensure compatibility across different Python versions.\n5. Removing redundant assertions in the test cases for handling unrecognized flags.\n6. Simplifying or eliminating the clipboard copy feature in the `summarize_pr.py` script to improve reliability.\n\n### Content of the Commit\n\nThe commit addresses each of the issues mentioned in the summary. Here's a detailed explanation:\n\n1. **Addition of New Operation Mode**: \n   - **Issue**: The existing `SqliteDict` lacked an operation mode for creating new databases (`flag='n'`), which is necessary to prevent unintentional overwriting of existing databases.\n   - **Commit Solution**: The commit introduces a new flag `n` to the valid flags list. This flag will allow users to explicitly indicate that a new database should be created, erasing any existing databases. \n\n2. **Incomplete Documentation for Development Setup**:\n   - **Issue**: The documentation for the `SetupDevelop` class was commented out, making it difficult for developers to understand its purpose.\n   - **Commit Solution**: The commit includes the `SetupDevelop` class’s documentation string to explain its usage, ensuring developers can properly read and understand its functionality.\n\n3. **Update Test Consistency**:\n   - **Issue**: Inconsistent test cases around handling of temporary databases and expected values led to discrepancies.\n   - **Commit Solution**: The test cases have been modified to consistently expect string values, aligning them across different scenarios to avoid false positives or negatives.\n\n4. **Improvement in Key Sorting During Tests**:\n   - **Issue**: Dictionary key sorting in tests did not consider type conversion, causing issues across different Python versions.\n   - **Commit Solution**: The commit adjusts the tests to use appropriate sorting mechanisms that ensure cross-version compatibility by explicitly converting keys to a list before sorting them.\n\n5. **Redundant Assertions in Flag Handling**:\n   - **Issue**: Redundant assertions in the test cases were checking for the presence of unrecognized flags in the dictionary without value verification.\n   - **Commit Solution**: The commit removes these redundant assertions to streamline the test cases, enhancing their clarity and relevance.\n\n6. **Clipboard Copy Function in PR Summarizer**:\n   - **Issue**: The clipboard copy function in `summarize_pr.py` didn’t return any text, leading to failures in some environments.\n   - **Commit Solution**: The commit streamlines this function by simply copying the text without attempting to return it, reducing complexity and improving script reliability.\n\n### Detailed Explanation of the Solution\n\n#### Addition of New Operation Mode\nBy incorporating a new flag `n` into the valid flags list, developers can now create new databases with confidence. This mode explicitly ensures that any existing database with the same name will be erased and rebuilt from scratch. This prevents accidental data loss from overwriting. This update improves data safety and aligns with best practices in database management.\n\n#### Documentation Improvement for Development Setup\nThe `SetupDevelop` class is crucial for setting up the development environment, especially for ensuring necessary dependencies like 'nose' and 'coverage.py' are in place for testing. Adding the documentation back provides clear guidance to developers, reducing setup time and errors, and ensuring a smoother development workflow.\n\n#### Test Consistency and Key Sorting\nEnsuring consistency in tests, particularly around handling string values for temporary databases, fraud attempts are minimized. The improved key sorting method explicitly converts keys to lists before sorting them. This guarantees the sorting logic remains consistent and feasible across different Python versions, enhancing test reliability and coverage.\n\n#### Redundant Assertions Removal\nEliminating redundant assertions in the flag handling tests reduces noise in the test output. This cleanup enhances the integrity of the tests, ensuring they are meaningful and focused on validating the core functionality without unnecessary checks.\n\n#### Clipboard Copy Function Simplification\nThe adjustment in the `summarize_pr.py` script simplifies the clipboard copying mechanism by removing the return text step. This reduction in complexity enhances the script's reliability and ensures it operates correctly across diverse environments, offering a straightforward copy-and-paste functionality to users.\n\n### Conclusion\n\nThe addressed issues improve the robustness, maintainability, and usability of the `SqliteDict` library. Through detailed commits that systematically tackle each problem—from adding new operational modes and improving documentation to standardizing tests and simplifying utility scripts—the library is now better equipped for both developers and users. These changes collectively enhance the development experience, reduce potential for errors, and ensure the library remains compatible and efficient across different scenarios and Python versions."
  },
  "Explain": "### Summary of the Issue\n\nThe GitHub issue focuses on enhancing the `SqliteDict` library by addressing several identified problems related to its functionality, documentation, and testing framework. The main issues are:\n\n1. Adding a new operation mode to the `SqliteDict` to create new databases without overwriting existing ones.\n2. Completing the documentation for the `SetupDevelop` class.\n3. Standardizing test cases to improve consistency, specifically around the temporary databases and handling of string values.\n4. Improving key sorting in tests to ensure compatibility across different Python versions.\n5. Removing redundant assertions in the test cases for handling unrecognized flags.\n6. Simplifying or eliminating the clipboard copy feature in the `summarize_pr.py` script to improve reliability.\n\n### Content of the Commit\n\nThe commit addresses each of the issues mentioned in the summary. Here's a detailed explanation:\n\n1. **Addition of New Operation Mode**: \n   - **Issue**: The existing `SqliteDict` lacked an operation mode for creating new databases (`flag='n'`), which is necessary to prevent unintentional overwriting of existing databases.\n   - **Commit Solution**: The commit introduces a new flag `n` to the valid flags list. This flag will allow users to explicitly indicate that a new database should be created, erasing any existing databases. \n\n2. **Incomplete Documentation for Development Setup**:\n   - **Issue**: The documentation for the `SetupDevelop` class was commented out, making it difficult for developers to understand its purpose.\n   - **Commit Solution**: The commit includes the `SetupDevelop` class’s documentation string to explain its usage, ensuring developers can properly read and understand its functionality.\n\n3. **Update Test Consistency**:\n   - **Issue**: Inconsistent test cases around handling of temporary databases and expected values led to discrepancies.\n   - **Commit Solution**: The test cases have been modified to consistently expect string values, aligning them across different scenarios to avoid false positives or negatives.\n\n4. **Improvement in Key Sorting During Tests**:\n   - **Issue**: Dictionary key sorting in tests did not consider type conversion, causing issues across different Python versions.\n   - **Commit Solution**: The commit adjusts the tests to use appropriate sorting mechanisms that ensure cross-version compatibility by explicitly converting keys to a list before sorting them.\n\n5. **Redundant Assertions in Flag Handling**:\n   - **Issue**: Redundant assertions in the test cases were checking for the presence of unrecognized flags in the dictionary without value verification.\n   - **Commit Solution**: The commit removes these redundant assertions to streamline the test cases, enhancing their clarity and relevance.\n\n6. **Clipboard Copy Function in PR Summarizer**:\n   - **Issue**: The clipboard copy function in `summarize_pr.py` didn’t return any text, leading to failures in some environments.\n   - **Commit Solution**: The commit streamlines this function by simply copying the text without attempting to return it, reducing complexity and improving script reliability.\n\n### Detailed Explanation of the Solution\n\n#### Addition of New Operation Mode\nBy incorporating a new flag `n` into the valid flags list, developers can now create new databases with confidence. This mode explicitly ensures that any existing database with the same name will be erased and rebuilt from scratch. This prevents accidental data loss from overwriting. This update improves data safety and aligns with best practices in database management.\n\n#### Documentation Improvement for Development Setup\nThe `SetupDevelop` class is crucial for setting up the development environment, especially for ensuring necessary dependencies like 'nose' and 'coverage.py' are in place for testing. Adding the documentation back provides clear guidance to developers, reducing setup time and errors, and ensuring a smoother development workflow.\n\n#### Test Consistency and Key Sorting\nEnsuring consistency in tests, particularly around handling string values for temporary databases, fraud attempts are minimized. The improved key sorting method explicitly converts keys to lists before sorting them. This guarantees the sorting logic remains consistent and feasible across different Python versions, enhancing test reliability and coverage.\n\n#### Redundant Assertions Removal\nEliminating redundant assertions in the flag handling tests reduces noise in the test output. This cleanup enhances the integrity of the tests, ensuring they are meaningful and focused on validating the core functionality without unnecessary checks.\n\n#### Clipboard Copy Function Simplification\nThe adjustment in the `summarize_pr.py` script simplifies the clipboard copying mechanism by removing the return text step. This reduction in complexity enhances the script's reliability and ensures it operates correctly across diverse environments, offering a straightforward copy-and-paste functionality to users.\n\n### Conclusion\n\nThe addressed issues improve the robustness, maintainability, and usability of the `SqliteDict` library. Through detailed commits that systematically tackle each problem—from adding new operational modes and improving documentation to standardizing tests and simplifying utility scripts—the library is now better equipped for both developers and users. These changes collectively enhance the development experience, reduce potential for errors, and ensure the library remains compatible and efficient across different scenarios and Python versions.",
  "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/setup.py",
      "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the(...truncated)"
    },
    {
      "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
      "content": "1 import os\n2 import sys\n3 \n4 import sqlit(...truncated)"
    }
  ],
  "TokenAll": 13037,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Easy",
    "issue_origin": {
      "title": "Potential compatibility issue with Python 3",
      "description": "In the function `test_manage_few_records` in `piskvorky_sqlitedict/tests/test_temp_db.py`, the method `iteritems()` is used, which is not available in Python 3. Use `items()` instead to ensure compatibility with both Python 2 and Python 3.",
      "explanation": "### Issue Summary\n\nThe issue at hand involves a potential compatibility problem with Python 3 in the `test_manage_few_records` function located in `piskvorky_sqlitedict/tests/test_temp_db.py`. Specifically, the method `iteritems()` is used to iterate over dictionary items, but this method is not available in Python 3. The function should use `items()` instead to ensure compatibility with both Python 2 and Python 3.\n\n### Detailed Analysis of the Code and Issue\n\nFrom the provided code base, here's a focus on the relevant parts:\n\n1. **File: `piskvorky_sqlitedict/tests/test_temp_db.py`:**\n\n    - Line 62-67:\n    ```python\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    ```\n\n    The `iteritems()` method was used inside a Python 2 check, with `items()` called unconditionally. Let's understand the changes needed:\n\n2. **Iterating over Dictionary items:**\n    - In Python 2, `iteritems()` returns an iterator over the dictionary’s (key, value) pairs.\n    - In Python 3, `iteritems()` was removed, and only `items()` returns a view of the dictionary’s items.\n\n### Proposed Solution\n\nGiven that the code already distinguishes between Python 2 and Python 3 (`major_version` check), the solution can be simplified by updating the conditional logic to handle this scenario cleanly. \n\n#### Commit Content\n\n```python\n62  if major_version == 2:\n63      self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n64  else:\n65      self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n66  self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n67  self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\n68  self.assertEqual(list(self.d), ['abc', 'xyz'])\n```\n\nThe proposed solution is to add an `else` branch that calls `self.d.items()` when the `major_version` is not 2 (i.e., Python 3). This ensures that the `items()` method is only called in Python 3 environments, allowing the test to pass in both Python 2 and Python 3.\n\n### Explanation of How the Commit Solves the Issue\n\n- **Backward Compatibility**: By keeping the `iteritems()` call inside a Python 2 conditional (`major_version == 2`), compatibility with Python 2 is maintained.\n  \n- **Forward Compatibility**: The `else` branch ensures that in Python 3, the `items()` method is used, which is the correct method for that version. This ensures that developers working in a Python 3 environment will not encounter errors related to non-existent methods.\n\n- **Testing**: This solution enables the same logical test (`self.assertEqual(...)`) to be performed regardless of the Python version, ensuring the behavior of `sqlitedict` is consistently verified across both Python 2 and Python 3 environments.\n\nBy implementing this change, the code base is effectively made compatible with both versions of Python, thereby addressing the issue comprehensively and ensuring broader compatibility of the test suite."
    },
    "issue_message": {
      "title": "Unrecognized flag in SqliteDict initialization",
      "description": "The code raises a RuntimeError for the unrecognized flag 'n' in the SqliteDict class's __init__ method.\nPlease update the SqliteDict class to handle the 'n' flag, which should correspond to creating a new database file.\nError occurs on line 195 in sqlitedict.py.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `SqliteDict` class in the `sqlitedict.py` file, which is throwing a `RuntimeError` due to an unrecognized flag `'n'` during initialization. The flag `'n'` is supposed to indicate that a new SQLite database file should be created, however, the current implementation does not recognize this flag and thus raises an error.\n\n### Cause of the Issue\n\nThe cause of the issue is that the `SqliteDict` class does not currently handle the flag `'n'`. The initialization method (`__init__`) of the `SqliteDict` checks for valid flags, but `'n'` is missing from this check. When the flag `'n'` is provided, it results in the `RuntimeError` being raised.\n\n### Detailed Commit Content\n\nLet's assume the following commit addresses the issue (hypothetical commit content for illustration):\n\n```python\n--- a/sqlitedict.py\n+++ b/sqlitedict.py\n@@ -192,13 +192,21 @@ class SqliteDict(collections.MutableMapping):\n             raise ValueError('dbname cannot be None if db is an existing database')\n \n         # initialize instance state\n+        if flag == 'n':\n+            # Create new database file\n+            if os.path.exists(filename):\n+                os.remove(filename)\n+        elif flag == 'c':\n+            # Open existing database or create one if it doesn't exist\n+            pass\n+        elif flag == 'w':\n+            # Open existing database only\n             pass\n-        elif flag == 'w':\n-            # Fail if database file does not exist\n+        else:\n+            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n+        \n+        # Connect to the sqlite database\n         self.conn = sqlite3.connect(self.filename, check_same_thread=False)\n \n-        if flag not in ('c', 'w'):\n-            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n-\n         self.conn.execute(MAKE_TABLE)\n         self.conn.commit()\n         self.autocommit = autocommit\n```\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Flag Handling**:\n    - The commit adds an explicit check for the flag `'n'`:\n        ```python\n        if flag == 'n':\n            if os.path.exists(filename):\n                os.remove(filename)\n        ```\n    - This code snippet checks if the flag is `'n'`. If true, it ensures any pre-existing file with the same name is deleted using `os.remove(filename)`, effectively creating a new database file when the `SqliteDict` instance is initialized.\n\n2. **Reordering Flag Checks**:\n    - The flag checking has been reordered to first check for `'n'`, then handle `'c'` and `'w'`. This makes sure the appropriate action is taken based on the flag without overlapping or missing any required actions.\n\n3. **Connection to SQLite Database**:\n    - After handling the flags, the code establishes a connection to the SQLite database using:\n        ```python\n        self.conn = sqlite3.connect(self.filename, check_same_thread=False)\n        ```\n\n4. **Creating the Table**:\n    - Post connection, it ensures the table is created by executing the `MAKE_TABLE` statement:\n        ```python\n        self.conn.execute(MAKE_TABLE)\n        ```\n\nBy incorporating these changes, the code properly handles the `'n'` flag, thereby resolving the `RuntimeError` issue. It also ensures backward compatibility with the previously supported flags `'c'` and `'w'`.\n\n### Conclusion\n\nThe provided commit appropriately addresses the issue of the unrecognized `'n'` flag by adding necessary handling for the flag within the `__init__` method of the `SqliteDict` class. This allows creating new database files when the flag `'n'` is used, thus resolving the `RuntimeError` and aligning with the intended functionality described in the issue report."
    },
    "issue_ground": {
      "title": "Enhance SqliteDict Functionality, Documentation, and Tests",
      "description": "Several issues have been identified that need addressing to improve the functionality, documentation, and testing of the SqliteDict project:\n\n1. **Addition of New Operation Mode**: The current implementation of `SqliteDict` lacks an operation mode for creating new databases. This is necessary to prevent unintentional overwriting of existing databases, which can lead to data loss.\n\n2. **Incomplete Documentation for Development Setup**: The documentation string for setup in the `SetupDevelop` class is commented out, which hinders the understanding of its behavior. Proper documentation is essential for comprehending its purpose and usage.\n\n3. **Update Test Consistency**: Inconsistent test cases for temporary databases can lead to discrepancies and false positives/negatives in test failures. Specifically, the testing of record management should be standardized to expect string values.\n\n4. **Improvement in Key Sorting During Tests**: The sorting of dictionary keys in tests doesn't consider type conversion, which raises compatibility issues across different Python versions.\n\n5. **Redundant Assertions in Flag Handling**: The assertion for checking an unrecognized flag's presence in the dictionary keys without value verification is redundant. This should be reviewed to ensure the integrity and completeness of test cases.\n\n6. **Clipboard Copy Function in PR Summarizer**: The `summarize_pr.py` script includes a conditional clipboard copy operation that doesn't return any text, causing it to fail in some environments. Consider streamlining or removing the feature to reduce complexity and improve reliability.\n\nAddressing these issues will enhance the robustness, maintainability, and usability of the `SqliteDict` library.",
      "explanation": "### Summary of the Issue\n\nThe `SqliteDict` project has several identified issues that impact its functionality, documentation, and testing. These include:\n\n1. **New Operation Mode**: There is a need for an operation mode that allows the creation of new databases without overwriting existing ones.\n2. **Incomplete Documentation**: The lack of comprehensive documentation for development setup hinders the understanding of the `SetupDevelop` class.\n3. **Test Consistency**: Inconsistent test cases for temporary databases can lead to discrepancies in test results.\n4. **Key Sorting in Tests**: The current key sorting doesn't consider type conversion, causing compatibility issues.\n5. **Redundant Assertions**: Some assertions are redundant and may need review.\n6. **Clipboard Copy Function**: The clipboard copy functionality in the `summarize_pr.py` script causes failures in some environments.\n\n### Detailed Content of the Commit\n\n#### The Major Additions and Changes\n\n1. **New Operation Mode**: Implementation of a new flag `n` to create a new database without overwriting an existing one.\n   \n   In `sqlitedict.py`:\n   ```python\n   # Adding support for the new flag 'n'\n   VALID_FLAGS = {'r', 'w', 'c', 'n'}\n   \n   flag = kwargs.pop('flag', 'c')\n   if flag not in VALID_FLAGS:\n       raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n   \n   if flag == 'n' and os.path.exists(self.filename):\n       raise RuntimeError(\"Database already exists: %s\" % self.filename)\n   ```\n\n2. **Improved Documentation**: Restoring the documentation for the `SetupDevelop` class by uncommenting docstrings.\n\n   In `setup.py`:\n   ```python\n   # Uncommenting the docstring\n   \"\"\"\n   Usage:\n   python setup.py develop\n   \"\"\"\n   ```\n\n3. **Test Consistency**: Standardizing test cases to use string values for record management.\n\n   In `test_temp_db.py`:\n   ```python\n   def test_manage_one_record(self):\n       self.d['abc'] = 'rsvp' * 100  # Ensuring uniform string type\n       self.assertEqual(self.d['abc'], 'rsvp' * 100)\n   ```\n\n4. **Key Sorting During Tests**: Ensuring keys are sorted as strings for compatibility.\n\n   In `test_temp_db.py`:\n   ```python\n   self.assertEqual(sorted(self.d.keys(), key=str), ['abc', 'xyz'])\n   ```\n\n5. **Redundant Assertions**: Removing redundant assertions regarding unrecognized flags and refining the checks.\n\n   In `sqlitedict.py`:\n   ```python\n   # Removing redundant assertions\n   if flag == 'n' and os.path.exists(self.filename):\n       raise RuntimeError(\"Database already exists: %s\" % self.filename)\n   ```\n\n6. **Clipboard Copy Function**: Streamlining the clipboard copy functionality for reliability.\n\n   In `summarize_pr.py`:\n   ```python\n   # Simplifying the clipboard copy functionality\n   try:\n       clipboard.copy(summary)\n   except Exception as e:\n       print(\"Clipboard function failed: \", e)\n   ```\n\n### Explanation of How the Commit Solves the Issue\n\n1. **New Operation Mode**:\n   - **Problem**: Existing functionality could unintentionally overwrite existing databases.\n   - **Solution**: The `n` flag ensures a database creation mode that raises an error if the database already exists, thereby preventing accidental data loss.\n\n2. **Improved Documentation**:\n   - **Problem**: Commented-out docstrings hinder understanding of how to set up the development environment.\n   - **Solution**: Un-commenting the docstrings provides clear guidance on usage, enhancing developer experience and reducing confusion.\n\n3. **Test Consistency**:\n   - **Problem**: Inconsistencies in test cases can lead to false positives/negatives.\n   - **Solution**: Standardizing data types (strings) in tests ensures consistent and accurate test results.\n\n4. **Key Sorting During Tests**:\n   - **Problem**: Incompatibility issues arising from different Python versions due to key sorting.\n   - **Solution**: Sorting keys as strings ensures uniform behavior across Python versions, improving cross-version compatibility.\n\n5. **Redundant Assertions**:\n   - **Problem**: Redundant checks for unrecognized flags without value verification add unnecessary complexity.\n   - **Solution**: Removing these assertions simplifies the code and focuses on more meaningful checks that maintain integrity.\n\n6. **Clipboard Copy Function**:\n   - **Problem**: Conditional clipboard copy operations can fail in some environments.\n   - **Solution**: A simplified and more robust clipboard copy implementation reduces complexity and increases reliability.\n\n### Conclusion\n\nThe commit addresses multiple issues by enhancing functionality, improving documentation, and refining tests. Each change is aimed at making the `SqliteDict` project more robust, maintainable, and user-friendly."
    },
    "issue_ground_truth": {
      "title": "Enhance SqliteDict Functionality, Documentation, and Tests",
      "description": "Several issues have been identified that need addressing to improve the functionality, documentation, and testing of the SqliteDict project:\n\n1. **Addition of New Operation Mode**: The current implementation of `SqliteDict` lacks an operation mode for creating new databases. This is necessary to prevent unintentional overwriting of existing databases, which can lead to data loss.\n\n2. **Incomplete Documentation for Development Setup**: The documentation string for setup in the `SetupDevelop` class is commented out, which hinders the understanding of its behavior. Proper documentation is essential for comprehending its purpose and usage.\n\n3. **Update Test Consistency**: Inconsistent test cases for temporary databases can lead to discrepancies and false positives/negatives in test failures. Specifically, the testing of record management should be standardized to expect string values.\n\n4. **Improvement in Key Sorting During Tests**: The sorting of dictionary keys in tests doesn't consider type conversion, which raises compatibility issues across different Python versions.\n\n5. **Redundant Assertions in Flag Handling**: The assertion for checking an unrecognized flag's presence in the dictionary keys without value verification is redundant. This should be reviewed to ensure the integrity and completeness of test cases.\n\n6. **Clipboard Copy Function in PR Summarizer**: The `summarize_pr.py` script includes a conditional clipboard copy operation that doesn't return any text, causing it to fail in some environments. Consider streamlining or removing the feature to reduce complexity and improve reliability.\n\nAddressing these issues will enhance the robustness, maintainability, and usability of the `SqliteDict` library.",
      "explanation": "### Summary of the Issue\n\nThe GitHub issue focuses on enhancing the `SqliteDict` library by addressing several identified problems related to its functionality, documentation, and testing framework. The main issues are:\n\n1. Adding a new operation mode to the `SqliteDict` to create new databases without overwriting existing ones.\n2. Completing the documentation for the `SetupDevelop` class.\n3. Standardizing test cases to improve consistency, specifically around the temporary databases and handling of string values.\n4. Improving key sorting in tests to ensure compatibility across different Python versions.\n5. Removing redundant assertions in the test cases for handling unrecognized flags.\n6. Simplifying or eliminating the clipboard copy feature in the `summarize_pr.py` script to improve reliability.\n\n### Content of the Commit\n\nThe commit addresses each of the issues mentioned in the summary. Here's a detailed explanation:\n\n1. **Addition of New Operation Mode**: \n   - **Issue**: The existing `SqliteDict` lacked an operation mode for creating new databases (`flag='n'`), which is necessary to prevent unintentional overwriting of existing databases.\n   - **Commit Solution**: The commit introduces a new flag `n` to the valid flags list. This flag will allow users to explicitly indicate that a new database should be created, erasing any existing databases. \n\n2. **Incomplete Documentation for Development Setup**:\n   - **Issue**: The documentation for the `SetupDevelop` class was commented out, making it difficult for developers to understand its purpose.\n   - **Commit Solution**: The commit includes the `SetupDevelop` class’s documentation string to explain its usage, ensuring developers can properly read and understand its functionality.\n\n3. **Update Test Consistency**:\n   - **Issue**: Inconsistent test cases around handling of temporary databases and expected values led to discrepancies.\n   - **Commit Solution**: The test cases have been modified to consistently expect string values, aligning them across different scenarios to avoid false positives or negatives.\n\n4. **Improvement in Key Sorting During Tests**:\n   - **Issue**: Dictionary key sorting in tests did not consider type conversion, causing issues across different Python versions.\n   - **Commit Solution**: The commit adjusts the tests to use appropriate sorting mechanisms that ensure cross-version compatibility by explicitly converting keys to a list before sorting them.\n\n5. **Redundant Assertions in Flag Handling**:\n   - **Issue**: Redundant assertions in the test cases were checking for the presence of unrecognized flags in the dictionary without value verification.\n   - **Commit Solution**: The commit removes these redundant assertions to streamline the test cases, enhancing their clarity and relevance.\n\n6. **Clipboard Copy Function in PR Summarizer**:\n   - **Issue**: The clipboard copy function in `summarize_pr.py` didn’t return any text, leading to failures in some environments.\n   - **Commit Solution**: The commit streamlines this function by simply copying the text without attempting to return it, reducing complexity and improving script reliability.\n\n### Detailed Explanation of the Solution\n\n#### Addition of New Operation Mode\nBy incorporating a new flag `n` into the valid flags list, developers can now create new databases with confidence. This mode explicitly ensures that any existing database with the same name will be erased and rebuilt from scratch. This prevents accidental data loss from overwriting. This update improves data safety and aligns with best practices in database management.\n\n#### Documentation Improvement for Development Setup\nThe `SetupDevelop` class is crucial for setting up the development environment, especially for ensuring necessary dependencies like 'nose' and 'coverage.py' are in place for testing. Adding the documentation back provides clear guidance to developers, reducing setup time and errors, and ensuring a smoother development workflow.\n\n#### Test Consistency and Key Sorting\nEnsuring consistency in tests, particularly around handling string values for temporary databases, fraud attempts are minimized. The improved key sorting method explicitly converts keys to lists before sorting them. This guarantees the sorting logic remains consistent and feasible across different Python versions, enhancing test reliability and coverage.\n\n#### Redundant Assertions Removal\nEliminating redundant assertions in the flag handling tests reduces noise in the test output. This cleanup enhances the integrity of the tests, ensuring they are meaningful and focused on validating the core functionality without unnecessary checks.\n\n#### Clipboard Copy Function Simplification\nThe adjustment in the `summarize_pr.py` script simplifies the clipboard copying mechanism by removing the return text step. This reduction in complexity enhances the script's reliability and ensures it operates correctly across diverse environments, offering a straightforward copy-and-paste functionality to users.\n\n### Conclusion\n\nThe addressed issues improve the robustness, maintainability, and usability of the `SqliteDict` library. Through detailed commits that systematically tackle each problem—from adding new operational modes and improving documentation to standardizing tests and simplifying utility scripts—the library is now better equipped for both developers and users. These changes collectively enhance the development experience, reduce potential for errors, and ensure the library remains compatible and efficient across different scenarios and Python versions."
    },
    "location_origin": [
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "56": "test_manage_few_records"
        },
        "content_all": {
          "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": "        else:\n",
          "65": "            self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n",
          "66": "        self.assertEqual(list(self.d.values()), ['lmno', 'pdq'])\n"
        },
        "content_change": {
          "63": "            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n",
          "64": "        else:\n",
          "65": "            self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "172": "__init__"
        },
        "content_all": {
          "171": "        self.inner_tablename = tablename",
          "172": "        if tablename is None:",
          "173": "            self.tablename = os.path.basename(filename) if filename else 'unnamed_table_' + str(random.random())[2:]",
          "174": "        else:",
          "175": "            self.tablename = tablename"
        },
        "content_change": {
          "172": "        if flag == 'n':"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "195": "__init__"
        },
        "content_all": {
          "192": "        # initialize instance state",
          "193": "        if flag == 'n':",
          "194": "            # Create new database file",
          "195": "            if os.path.exists(filename):",
          "196": "                os.remove(filename)",
          "197": "        elif flag == 'c':",
          "198": "            # Open existing database or create one if it doesn't exist"
        },
        "content_change": {
          "195": "            if os.path.exists(filename):"
        }
      },
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "229": "__init__"
        },
        "content_all": {
          "228": "        self.conn = sqlite3.connect(self.filename, check_same_thread=False)",
          "229": "        if flag not in ('c', 'w', 'n'):",
          "230": "            raise RuntimeError(\"Unrecognized flag: %s\" % flag)",
          "231": "        self.conn.execute(MAKE_TABLE)",
          "232": "        self.conn.commit()"
        },
        "content_change": {
          "229": "        if flag not in ('c', 'w', 'n'):"
        }
      }
    ],
    "location_ground": [
      {
        "location": "```json\n[\n    {\n        \"file\": \"piskvorky_sqlitedict/sqlitedict.py\",\n        \"function\": {\n            \"50\": \"main function handling SqliteDict initialization and interactions\"\n        },\n        \"content_all\": {\n            \"48\": \"\\n\",\n            \"49\": \"\\n\",\n            \"50\": \"try:\",\n            \"51\": \"    from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\",\n            \"52\": \"except ImportError:\",\n            \"53\": \"    from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\",\n            \"54\": \"\\n\",\n            \"255\": \"        VALID_FLAGS = {'r', 'w', 'c', 'n'}\\n\",\n            \"256\": \"\\n\",\n            \"257\": \"        flag = kwargs.pop('flag', 'c')\\n\",\n            \"258\": \"        if flag not in VALID_FLAGS:\\n\",\n            \"259\": \"            raise RuntimeError(\\\"Unrecognized flag: %s\\\" % flag)\\n\",\n            \"260\": \"\\n\",\n            \"261\": \"        if flag == 'n' and os.path.exists(self.filename):\\n\",\n            \"262\": \"            raise RuntimeError(\\\"Database already exists: %s\\\" % self.filename)\\n\",\n            \"263\": \"\\n\"\n        },\n        \"content_change\": {\n            \"255\": \"        VALID_FLAGS = {'r', 'w', 'c', 'n'}\\n\",\n            \"257\": \"        flag = kwargs.pop('flag', 'c')\\n\",\n            \"258\": \"        if flag not in VALID_FLAGS:\\n\",\n            \"259\": \"            raise RuntimeError(\\\"Unrecognized flag: %s\\\" % flag)\\n\",\n            \"261\": \"        if flag == 'n' and os.path.exists(self.filename):\\n\",\n            \"262\": \"            raise RuntimeError(\\\"Database already exists: %s\\\" % self.filename)\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/setup.py\",\n        \"function\": {\n            \"0\": \"SetupDevelop\"\n        },\n        \"content_all\": {\n            \"0\": \"\\n\",\n            \"1\": \"#!/usr/bin/env python\\n\",\n            \"2\": \"# -*- coding: utf-8 -*-\\n\",\n            \"3\": \"#\\n\",\n            \"4\": \"# This code is distributed under the terms and conditions\\n\",\n            \"5\": \"# from the Apache License, Version 2.0\\n\",\n            \"8\": \"class SetupDevelop:\\n\",\n            \"9\": \"'\\n\",\n            \"10\": \"SetupDevelop class\\n\",\n            \"11\": \"Usage:\\n\",\n            \"12\": \"    python setup.py develop\\n\",\n            \"13\": \"\\n\",\n            \"14\": \"    '\\n\"\n        },\n        \"content_change\": {\n            \"8\": \"class SetupDevelop:\\n\",\n            \"9\": \"\\\"\\\"\\\"\\n\",\n            \"10\": \"SetupDevelop class\\n\",\n            \"11\": \"Usage:\\n\",\n            \"12\": \"    python setup.py develop\\n\",\n            \"13\": \"\\\"\\\"\\\"\\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            \"44\": \"    def test_manage_one_record(self):\\n\",\n            \"45\": \"        ''' test_manage_one_record\\n\",\n            \"46\": \"        '''\\n\",\n            \"47\": \"        self.d['abc'] = 'rsvp' * 100\\n\",\n            \"48\": \"        self.assertEqual(self.d['abc'], 'rsvp' * 100)\\n\",\n            \"49\": \"        self.d['abc'] = 100\\n\",\n            \"50\": \"        self.assertEqual(self.d['abc'], 100)\\n\",\n            \"51\": \"        self.assertEqual(len(self.d), 1)\\n\",\n            \"52\": \"        del self.d['abc']\\n\",\n            \"53\": \"        self.assertEqual(len(self.d), 0)\\n\",\n            \"54\": \"        self.assertTrue(not self.d)\\n\"\n        },\n        \"content_change\": {\n            \"47\": \"        self.d['abc'] = 'rsvp' * 100\\n\",\n            \"48\": \"        self.assertEqual(self.d['abc'], 'rsvp' * 100)\\n\"\n        }\n    },\n    {\n        \"file\": \"piskvorky_sqlitedict/tests/test_temp_db.py\",\n        \"function\": {\n            \"66\": \"test_manage_few_records\"\n        },\n        \"content_all\": {\n            \"66\": \"        self.assertEqual(list(self.d.keys()), ['abc', 'xyz'])\\n\",\n"
      }
    ],
    "location_ground_exp": [
      {
        "file": "piskvorky_sqlitedict/sqlitedict.py",
        "function": {
          "42": "reraise"
        },
        "content_all": {
          "39": "__version__ = '2.1.0'",
          "40": "",
          "41": "",
          "42": "def reraise(tp, value, tb=None):",
          "43": "    if value is None:",
          "44": "        value = tp()",
          "45": "    if value.__traceback__ is not tb:",
          "46": "        raise value.with_traceback(tb)",
          "47": "    raise value"
        },
        "content_change": {
          "43": "    if value is None:",
          "44": "        value = tp()",
          "45": "    if value.__traceback__ is not tb:",
          "46": "        raise value.with_traceback(tb)",
          "47": "    raise value"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_temp_db.py",
        "function": {
          "13": "TempSqliteDictTest"
        },
        "content_all": {
          "56": "    def test_manage_few_records(self):",
          "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')])"
        },
        "content_change": {
          "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')])"
        }
      },
      {
        "file": "piskvorky_sqlitedict/tests/test_core.py",
        "function": {
          "9": "import sqlitedict"
        },
        "content_all": {
          "8": "# local",
          "9": "import sqlitedict",
          "10": "from sqlitedict import SqliteDict",
          "11": "",
          "12": ""
        },
        "content_change": {
          "9": "import sqlitedict",
          "10": "from sqlitedict import SqliteDict"
        }
      }
    ],
    "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": {
          "143": "\n",
          "144": "\n",
          "145": "class SqliteDict(DictClass):\n",
          "146": "    VALID_FLAGS = ['c', 'r', 'w']\n"
        },
        "content_change": {
          "146": "    VALID_FLAGS = ['c', 'r', 'w']\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@@ -61,9 +61,11 @@\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+        if major_version == 2:  # Python 2\n+            self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n+        else:  # Python 3\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",
    "patch_im": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -189,12 +189,21 @@ class SqliteDict(collections.MutableMapping):\n         if flag == 'r':\n             raise ValueError('dbname cannot be None if db is an existing database')\n \n-        # initialize instance state\n+        # Handle new flag 'n': create a new database file\n+        if flag == 'n':\n+            if os.path.exists(filename):\n+                os.remove(filename)\n+        elif flag == 'c':\n+            # Open existing database or create one if it doesn't exist\n+            pass\n+        elif flag == 'w':\n+            # Open existing database only\n+            pass\n         else:\n             raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n \n+        # Connect to the SQLite database\n         self.conn = sqlite3.connect(self.filename, check_same_thread=False)\n \n-        if flag not in ('c', 'w'):\n-            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n-\n         self.conn.execute(MAKE_TABLE)\n         self.conn.commit()\n         self.autocommit = autocommit\n",
    "patch_il": "\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -62,7 +62,9 @@\n         if major_version == 2:\n             self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n         else:\n-            self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n+            self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n+\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",
    "patch_iml": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -192,14 +192,22 @@ class SqliteDict(collections.MutableMapping):\n         # initialize instance state\n         if flag == 'n':\n             # Create new database file\n-            if os.path.exists(filename):\n-                os.remove(filename)\n+            if os.path.exists(self.filename):\n+                os.remove(self.filename)\n+        elif flag == 'c':\n+            # Open existing database or create one if it doesn't exist\n+            pass\n+        elif flag == 'w':\n+            # Open existing database only\n             pass\n-        elif flag == 'c':\n-            # Open existing database or create one if it doesn't exist\n-            pass\n-        elif flag == 'w':\n-            # Open existing database only\n+        else:\n+            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n+\n+        # Connect to the sqlite database\n+        self.conn = sqlite3.connect(self.filename, check_same_thread=False)\n+\n+        if flag not in ('c', 'w', 'n'):\n+            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n+\n         self.conn.execute(MAKE_TABLE)\n         self.conn.commit()\n         self.autocommit = autocommit\n",
    "patch_ground": "```diff\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -1,27 +1,35 @@\n #!/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 \"\"\"\n A lightweight wrapper around Python's sqlite3 database, with a dict-like interface\n and 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 \n Pickle is used internally to serialize the values. Keys are strings.\n \n If you don't use autocommit (default is no autocommit for performance), then\n don't forget to call `mydict.commit()` when done with a transaction.\n \n \"\"\"\n \n import sqlite3\n import os\n import sys\n import tempfile\n import threading\n import logging\n import traceback\n from base64 import b64decode, b64encode\n import weakref\n \n __version__ = '2.1.0'\n \n@@ -50,10 +50,18 @@ def reraise(tp, value, tb=None):\n     from cPickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n except ImportError:\n     from pickle import dumps, loads, HIGHEST_PROTOCOL as PICKLE_PROTOCOL\n \n # some Python 3 vs 2 imports\n try:\n     from StringIO import StringIO\n except ImportError:\n     from io import StringIO\n \n VALID_FLAGS = {'r', 'w', 'c', 'n'}\n\n class SqliteDict(object):\n     ...\n     def __init__(self, filename='', table='unnamed', autocommit=False, journal_mode='DELETE', **kwargs):\n         ...\n         self.filename = filename\n         self.journal_mode = journal_mode\n         \n         flag = kwargs.pop('flag', 'c')\n         if flag not in VALID_FLAGS:\n             raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n         if flag == 'n' and os.path.exists(self.filename):\n             raise RuntimeError(\"Database already exists: %s\" % self.filename)\n             \n ...\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -1,4 +1,6 @@\n #!/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 \"\"\"\n Usage:\n python setup.py develop\n \"\"\"\n \n ...\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -44,9 +44,10 @@ class TempSqliteDictTest(unittest.TestCase):\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     def test_manage_few_records(self):\n         ''' test_manage_few_records\n         '''\n         self.d['abc'] = 'lmno'\n         self.d['xyz'] = 'pdq'\n@@ -66,7 +67,11 @@ class TempSqliteDictTest(unittest.TestCase):\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+        self.assertEqual(sorted(self.d.items(), key=lambda x: str(x[0])), sorted",
    "patch_ground_location": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -252,6 +252,13 @@\n             raise\n \n         # handle flags\n+        VALID_FLAGS = {'r', 'w', 'c', 'n'}\n+\n+        flag = kwargs.pop('flag', 'c')\n+        if flag not in VALID_FLAGS:\n+            raise RuntimeError(\"Unrecognized flag: %s\" % flag)\n+\n+        if flag == 'n' and os.path.exists(self.filename):\n+            raise RuntimeError(\"Database already exists: %s\" % self.filename)\n \n         self.conn = self.text_factory = self.row_factory = self.commit_on_exit = self.autocommit = self.flag = self.timeout = self.settings = None\n         self.commitlock = CommitLock()\n\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -6,10 +6,9 @@\n try:\n     from setuptools import setup\n except ImportError:\n     from distutils.core import setup\n \n \n class SetupDevelop:\n-    '''\n+    \"\"\"\n     SetupDevelop class\n     Usage:\n         python setup.py develop\n-    '''\n+    \"\"\"\n \n--- piskvorky_sqlitedict/tests/test_temp_db.py\n+++ piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -47,8 +47,7 @@ def 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'], 'rsvp' * 100)  # Removed integer assignment\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@@ -63,7 +62,7 @@ def test_manage_few_records(self):\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(sorted(self.d.keys()), ['abc', 'xyz'])\n         self.assertEqual(list(self.d), ['abc', 'xyz'])\n\n--- a/piskvorky_sqlitedict/summarize_pr.py\n+++ b/piskvorky_sqlitedict/summarize_pr.py\n@@ -XXX,XXX +XXX,XXX @@  # Modify as per the exact location\n+++            # Streamlining clipboard copy functionality\n+            try:\n+                clipboard.copy(summary)\n+            except Exception as e:\n+                print(\"Clipboard function failed: \", e)\n",
    "patch_ground_exp": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -55,6 +55,7 @@ except ImportError:\n \n \n VALID_FLAGS = set('rwc')  # r/w/c/o\n+VALID_FLAGS.add('n')  # n for creating new database\n \n\n class SqliteDict(object):\n     \"\"\"SQLite-backed dictionary for persistent, concurrent mappings.\n\n@@ -152,6 +153,9 @@ class SqliteDict(object):\n             raise ValueError(\"unrecognized flag: %s\" % flag)\n         if flag == 'n':\n             flag = 'w'\n+            if os.path.exists(filename):\n+                os.remove(filename)\n+\n\n\n class SetupDevelop(object):\n     \"\"\"\n@@ -187,4 +191,4 @@ class SetupDevelop(object):\n \n         pass\n \n     \"\"\"\n     # End of commented-out docstring\n\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -60,7 +60,7 @@ class TempSqliteDictTest(unittest.TestCase):\n             self.assertEqual(list(self.d.iteritems()), [('abc', 'lmno'), ('xyz', 'pdq')])\n         else:\n             self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n-        self.assertEqual(list(self.d.items()), [('abc', 'lmno'), ('xyz', 'pdq')])\n+        self.assertEqual(list(sorted(self.d.items())), sorted([('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\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -10,7 +10,9 @@ from unittest.mock import patch\n \n # local\n import sqlitedict\n+import tempfile\n from sqlitedict import SqliteDict\n+import os\n \n class CoreTest(unittest.TestCase):\n \n@@ -22,7 +24,7 @@ class CoreTest(unittest.TestCase):\n \n     def test_create_new_database_flag(self):\n         \"\"\"Test for creating a new database without overwriting the existing one.\"\"\"\n-        with tempfile.NamedTemporaryFile(delete=False) as tmpfile:\n+        tmpfile = tempfile.NamedTemporaryFile(delete=False)\n             tmpfile.close()\n             db = SqliteDict(tmpfile.name, autocommit=True)\n             db['key'] = 'value'\n@@ -30,4 +32,7 @@ class CoreTest(unittest.TestCase):\n         with SqliteDict(tmpfile.name, flag='n', autocommit=True) as db_new:\n             self.assertEqual(len(db_new), 0)\n             self.assertEqual(dict(db_new), {})\n+        os.remove(tmpfile.name)\n \n if __name__ == \"__main__\":\n     unittest.main()\n\n--- a/piskvorky_sqlitedict/tests/test_autocommit.py\n+++ b/piskvorky_sqlitedict/tests/test_autocommit.py\n@@ -6,2 +6,2 @@\n # local\n import sqlitedict\n from sqlitedict import SqliteDict\n import tempfile\n import os\n@@ -8,3 +8,5 @@\n class AutocommitTest(unittest.TestCase):\n \n     def test_autocommit(self):\n         \"\"\"Test autocommit functionality\"\"\"\n         with tempfile.NamedTemporaryFile(delete=False) as tmpfile:\n             tmpfile.close()\n             db = SqliteDict(tmpfile.name, autocommit=True)\n             db['key'] = 'value'\n             self.assertEqual(db['key'], 'value')\n+        os.remove(tmpfile.name)\n\n--- a/piskvorky_sqlitedict/summarize_pr.py\n+++ b/piskvorky_sqlitedict/summarize_pr.py\n@@ -54,7 +54,7 @@ if __name__ == \"__main__\":\n         print(summary)\n         try:\n             if summary:\n-                pyperclip.copy(summary)\n+                import pyperclip; pyperclip.copy(summary)\n         except Exception as e:\n             print(\"Failed to copy text to clipboard: %s\" % e)\n",
    "patch_ground_all": "\n--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -1,7 +1,7 @@\n #!/usr/bin/env python\n # -*- coding: utf-8 -*-\n #\n-# This code is distributed under the terms and conditions\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@@ -143,7 +143,7 @@ class SqliteDict(DictClass):\n         self.conn.text_factory = str\n         self.execute('BEGIN', force=True)\n \n-    VALID_FLAGS = ['c', 'r', 'w']\n+    VALID_FLAGS = ['c', 'r', 'w', 'n']\n \n     def execute(self, command, stands_alone=True, force=False):\n         with self._thread_lock:\n--- a/piskvorky_sqlitedict/setup.py\n+++ b/piskvorky_sqlitedict/setup.py\n@@ -43,7 +43,7 @@ class SetupDevelop(setuptools.command.develop.develop):\n         self.run_check_dependencies()\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\",\n--- a/piskvorky_sqlitedict/tests/test_temp_db.py\n+++ b/piskvorky_sqlitedict/tests/test_temp_db.py\n@@ -62,7 +62,7 @@ class TempSqliteDictTest(unittest.TestCase):\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.keys()), ['abc', 'xyz'])\n         self.assertEqual(list(self.d), ['abc', 'xyz'])\n \n     def test_update_records(self):\n@@ -76,7 +76,7 @@ class TempSqliteDictTest(unittest.TestCase):\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+\n     def test_manage_one_record(self):\n         ''' test_manage_one_record\n         '''\n--- a/piskvorky_sqlitedict/tests/test_core.py\n+++ b/piskvorky_sqlitedict/tests/test_core.py\n@@ -133,7 +133,7 @@ class TestCore(unittest.TestCase):\n         next_db.commit()\n \n         self.assertNotIn('key', next_db)\n-        self.assertNotIn('value', next_db.values())\n+        self.assertNotIn('value', next_db.values())\n \n     def test_capacity(self):\n         '''Ensure that the database can handle capacity limits.'''\n--- a/piskvorky_sqlitedict/release/summarize_pr.py\n+++ b/piskvorky_sqlitedict/release/summarize_pr.py\n@@ -15,7 +15,6 @@ def summarize(prid):\n         print(\"Summary:\")\n         print(text)\n \n-        return text\n         try:\n             import pyperclip\n             pyperclip.copy(text)\n",
    "patch_ground_truth": "--- a/piskvorky_sqlitedict/sqlitedict.py\n+++ b/piskvorky_sqlitedict/sqlitedict.py\n@@ -143,7 +143,7 @@\n \n \n class SqliteDict(DictClass):\n-    VALID_FLAGS = ['c', 'r', 'w']\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--- 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_overwrite_using_flag_n (test_core.NamedSqliteDictCreateOrReuseTest)\nRe-opening of a database with flag='c' destroys it all.\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 131, in test_overwrite_using_flag_n\n    next_db = SqliteDict(filename=fname, tablename='sometable', flag='n')\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_terminate_cannot_delete (test_core.SqliteDictTerminateFailTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_core.py\", line 254, in setUp\n    self.db = SqliteDict(filename=self.fname)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 229, in __init__\n    self.conn.execute(MAKE_TABLE)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 616, in execute\n    self.check_raise_error()\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 606, in check_raise_error\n    reraise(e_type, e_value, e_tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 46, in reraise\n    raise value.with_traceback(tb)\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 474, in _connect\n    conn = sqlite3.connect(self.filename, check_same_thread=False)\nsqlite3.OperationalError: unable to open database file\n\n======================================================================\nERROR: test_assign_values (test_named_db.CreateNewSqliteDictTest)\ntest_assign_values\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_clear_data (test_named_db.CreateNewSqliteDictTest)\ntest_clear_data\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_create_sqlitedict (test_named_db.CreateNewSqliteDictTest)\ntest_create_sqlitedict\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_handling_errors (test_named_db.CreateNewSqliteDictTest)\ntest_handling_errors\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_manage_few_records (test_named_db.CreateNewSqliteDictTest)\ntest_manage_few_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_manage_one_record (test_named_db.CreateNewSqliteDictTest)\ntest_manage_one_record\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n======================================================================\nERROR: test_update_records (test_named_db.CreateNewSqliteDictTest)\ntest_update_records\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/tests/test_named_db.py\", line 26, in setUp\n    self.d = sqlitedict.SqliteDict(filename=db, flag=\"n\")\n  File \"/home/user/Documents/repoben/buggycode/piskvorky_sqlitedict/sqlitedict.py\", line 195, in __init__\n    raise RuntimeError(\"Unrecognized flag: %s\" % flag)\nRuntimeError: Unrecognized flag: n\n\n----------------------------------------------------------------------\nRan 88 tests in 1.128s\n\nFAILED (errors=9)",
    "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/setup.py",
        "content": "1 #!/usr/bin/env python\n2 # -*- coding: utf-8 -*-\n3 #\n4 # This code is distributed under the(...truncated)"
      },
      {
        "path": "piskvorky_sqlitedict/tests/test_autocommit.py",
        "content": "1 import os\n2 import sys\n3 \n4 import sqlit(...truncated)"
      }
    ],
    "CommitSHA": "96e81621fd6ab094efdd86e70fd57efe9d40ca12"
  },
  "Score": {
    "Difficulty": "Easy",
    "issue_origin": {
      "Title": 6,
      "Description": 7,
      "Reproducibility": 5,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6.5,
      "Description": 5.5,
      "Reproducibility": 4,
      "Relevance": 8,
      "Explanation": 7.5,
      "Overall": 6.5
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Enhance SqliteDict Functionality, Documentation, and Tests",
      "description": "Several issues have been identified that need addressing to improve the functionality, documentation, and testing of the SqliteDict project:\n\n1. **Addition of New Operation Mode**: The current implementation of `SqliteDict` lacks an operation mode for creating new databases. This is necessary to prevent unintentional overwriting of existing databases, which can lead to data loss.\n\n2. **Incomplete Documentation for Development Setup**: The documentation string for setup in the `SetupDevelop` class is commented out, which hinders the understanding of its behavior. Proper documentation is essential for comprehending its purpose and usage.\n\n3. **Update Test Consistency**: Inconsistent test cases for temporary databases can lead to discrepancies and false positives/negatives in test failures. Specifically, the testing of record management should be standardized to expect string values.\n\n4. **Improvement in Key Sorting During Tests**: The sorting of dictionary keys in tests doesn't consider type conversion, which raises compatibility issues across different Python versions.\n\n5. **Redundant Assertions in Flag Handling**: The assertion for checking an unrecognized flag's presence in the dictionary keys without value verification is redundant. This should be reviewed to ensure the integrity and completeness of test cases.\n\n6. **Clipboard Copy Function in PR Summarizer**: The `summarize_pr.py` script includes a conditional clipboard copy operation that doesn't return any text, causing it to fail in some environments. Consider streamlining or removing the feature to reduce complexity and improve reliability.\n\nAddressing these issues will enhance the robustness, maintainability, and usability of the `SqliteDict` library.",
      "explanation": "### Summary of the Issue\n\nThe GitHub issue focuses on enhancing the `SqliteDict` library by addressing several identified problems related to its functionality, documentation, and testing framework. The main issues are:\n\n1. Adding a new operation mode to the `SqliteDict` to create new databases without overwriting existing ones.\n2. Completing the documentation for the `SetupDevelop` class.\n3. Standardizing test cases to improve consistency, specifically around the temporary databases and handling of string values.\n4. Improving key sorting in tests to ensure compatibility across different Python versions.\n5. Removing redundant assertions in the test cases for handling unrecognized flags.\n6. Simplifying or eliminating the clipboard copy feature in the `summarize_pr.py` script to improve reliability.\n\n### Content of the Commit\n\nThe commit addresses each of the issues mentioned in the summary. Here's a detailed explanation:\n\n1. **Addition of New Operation Mode**: \n   - **Issue**: The existing `SqliteDict` lacked an operation mode for creating new databases (`flag='n'`), which is necessary to prevent unintentional overwriting of existing databases.\n   - **Commit Solution**: The commit introduces a new flag `n` to the valid flags list. This flag will allow users to explicitly indicate that a new database should be created, erasing any existing databases. \n\n2. **Incomplete Documentation for Development Setup**:\n   - **Issue**: The documentation for the `SetupDevelop` class was commented out, making it difficult for developers to understand its purpose.\n   - **Commit Solution**: The commit includes the `SetupDevelop` class’s documentation string to explain its usage, ensuring developers can properly read and understand its functionality.\n\n3. **Update Test Consistency**:\n   - **Issue**: Inconsistent test cases around handling of temporary databases and expected values led to discrepancies.\n   - **Commit Solution**: The test cases have been modified to consistently expect string values, aligning them across different scenarios to avoid false positives or negatives.\n\n4. **Improvement in Key Sorting During Tests**:\n   - **Issue**: Dictionary key sorting in tests did not consider type conversion, causing issues across different Python versions.\n   - **Commit Solution**: The commit adjusts the tests to use appropriate sorting mechanisms that ensure cross-version compatibility by explicitly converting keys to a list before sorting them.\n\n5. **Redundant Assertions in Flag Handling**:\n   - **Issue**: Redundant assertions in the test cases were checking for the presence of unrecognized flags in the dictionary without value verification.\n   - **Commit Solution**: The commit removes these redundant assertions to streamline the test cases, enhancing their clarity and relevance.\n\n6. **Clipboard Copy Function in PR Summarizer**:\n   - **Issue**: The clipboard copy function in `summarize_pr.py` didn’t return any text, leading to failures in some environments.\n   - **Commit Solution**: The commit streamlines this function by simply copying the text without attempting to return it, reducing complexity and improving script reliability.\n\n### Detailed Explanation of the Solution\n\n#### Addition of New Operation Mode\nBy incorporating a new flag `n` into the valid flags list, developers can now create new databases with confidence. This mode explicitly ensures that any existing database with the same name will be erased and rebuilt from scratch. This prevents accidental data loss from overwriting. This update improves data safety and aligns with best practices in database management.\n\n#### Documentation Improvement for Development Setup\nThe `SetupDevelop` class is crucial for setting up the development environment, especially for ensuring necessary dependencies like 'nose' and 'coverage.py' are in place for testing. Adding the documentation back provides clear guidance to developers, reducing setup time and errors, and ensuring a smoother development workflow.\n\n#### Test Consistency and Key Sorting\nEnsuring consistency in tests, particularly around handling string values for temporary databases, fraud attempts are minimized. The improved key sorting method explicitly converts keys to lists before sorting them. This guarantees the sorting logic remains consistent and feasible across different Python versions, enhancing test reliability and coverage.\n\n#### Redundant Assertions Removal\nEliminating redundant assertions in the flag handling tests reduces noise in the test output. This cleanup enhances the integrity of the tests, ensuring they are meaningful and focused on validating the core functionality without unnecessary checks.\n\n#### Clipboard Copy Function Simplification\nThe adjustment in the `summarize_pr.py` script simplifies the clipboard copying mechanism by removing the return text step. This reduction in complexity enhances the script's reliability and ensures it operates correctly across diverse environments, offering a straightforward copy-and-paste functionality to users.\n\n### Conclusion\n\nThe addressed issues improve the robustness, maintainability, and usability of the `SqliteDict` library. Through detailed commits that systematically tackle each problem—from adding new operational modes and improving documentation to standardizing tests and simplifying utility scripts—the library is now better equipped for both developers and users. These changes collectively enhance the development experience, reduce potential for errors, and ensure the library remains compatible and efficient across different scenarios and Python versions."
    }
  }
}