[PATCH] python: fix get_filenames() and make it actually usable
authorJan Malakhovski <oxij@oxij.org>
Fri, 31 Jul 2015 01:36:26 +0000 (01:36 +0000)
committerW. Trevor King <wking@tremily.us>
Sat, 20 Aug 2016 21:49:13 +0000 (14:49 -0700)
6c/4e4dd8a4b6f06d0bb9044e96e2910e4407ee75 [new file with mode: 0644]

diff --git a/6c/4e4dd8a4b6f06d0bb9044e96e2910e4407ee75 b/6c/4e4dd8a4b6f06d0bb9044e96e2910e4407ee75
new file mode 100644 (file)
index 0000000..b25370a
--- /dev/null
@@ -0,0 +1,166 @@
+Return-Path: <oxij@oxij.org>\r
+X-Original-To: notmuch@notmuchmail.org\r
+Delivered-To: notmuch@notmuchmail.org\r
+Received: from localhost (localhost [127.0.0.1])\r
+ by arlo.cworth.org (Postfix) with ESMTP id CFDBF6DE0B44\r
+ for <notmuch@notmuchmail.org>; Thu, 30 Jul 2015 18:44:05 -0700 (PDT)\r
+X-Virus-Scanned: Debian amavisd-new at cworth.org\r
+X-Spam-Flag: NO\r
+X-Spam-Score: -0.001\r
+X-Spam-Level: \r
+X-Spam-Status: No, score=-0.001 tagged_above=-999 required=5\r
+ tests=[SPF_PASS=-0.001] autolearn=disabled\r
+Received: from arlo.cworth.org ([127.0.0.1])\r
+ by localhost (arlo.cworth.org [127.0.0.1]) (amavisd-new, port 10024)\r
+ with ESMTP id LrlbbbVVE8Wv for <notmuch@notmuchmail.org>;\r
+ Thu, 30 Jul 2015 18:44:03 -0700 (PDT)\r
+X-Greylist: delayed 437 seconds by postgrey-1.35 at arlo;\r
+ Thu, 30 Jul 2015 18:44:02 PDT\r
+Received: from tricoro.koumakan.jp (tricoro.koumakan.jp [195.154.188.176])\r
+ by arlo.cworth.org (Postfix) with ESMTPS id BD9DD6DE0B20\r
+ for <notmuch@notmuchmail.org>; Thu, 30 Jul 2015 18:44:02 -0700 (PDT)\r
+Received: from localhost (ppp91-122-53-24.pppoe.avangarddsl.ru [91.122.53.24])\r
+ (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128\r
+ bits)) (No client certificate requested)\r
+ by tricoro.koumakan.jp (Postfix) with ESMTPSA id 9F1F6103D;\r
+ Fri, 31 Jul 2015 10:36:41 +0900 (JST)\r
+From: Jan Malakhovski <oxij@oxij.org>\r
+To: notmuch@notmuchmail.org\r
+Subject: [PATCH] python: fix get_filenames() and make it actually usable\r
+Date: Fri, 31 Jul 2015 01:36:26 +0000\r
+Message-Id: <1438306586-7597-1-git-send-email-oxij@oxij.org>\r
+X-Mailer: git-send-email 2.4.1\r
+X-BeenThere: notmuch@notmuchmail.org\r
+X-Mailman-Version: 2.1.18\r
+Precedence: list\r
+List-Id: "Use and development of the notmuch mail system."\r
+ <notmuch.notmuchmail.org>\r
+List-Unsubscribe: <http://notmuchmail.org/mailman/options/notmuch>,\r
+ <mailto:notmuch-request@notmuchmail.org?subject=unsubscribe>\r
+List-Archive: <http://notmuchmail.org/pipermail/notmuch/>\r
+List-Post: <mailto:notmuch@notmuchmail.org>\r
+List-Help: <mailto:notmuch-request@notmuchmail.org?subject=help>\r
+List-Subscribe: <http://notmuchmail.org/mailman/listinfo/notmuch>,\r
+ <mailto:notmuch-request@notmuchmail.org?subject=subscribe>\r
+X-List-Received-Date: Fri, 31 Jul 2015 01:44:05 -0000\r
+\r
+The problem with the previous implementation is that different\r
+versions of python exhaust __iter__() differently and the\r
+implementation that can be exhausted is not only absolutely unusable\r
+in the user code, but it also can not be consistently used with both\r
+python 2.* and 3.*.\r
+\r
+This doesn't change the interface.\r
+---\r
+ bindings/python/notmuch/filenames.py | 84 +++++++++++-------------------------\r
+ 1 file changed, 26 insertions(+), 58 deletions(-)\r
+\r
+diff --git a/bindings/python/notmuch/filenames.py b/bindings/python/notmuch/filenames.py\r
+index 229f414..e2b8886 100644\r
+--- a/bindings/python/notmuch/filenames.py\r
++++ b/bindings/python/notmuch/filenames.py\r
+@@ -65,6 +65,18 @@ class Filenames(Python3StringMixIn):\r
+     _get.argtypes = [NotmuchFilenamesP]\r
+     _get.restype = c_char_p\r
\r
++    _valid = nmlib.notmuch_filenames_valid\r
++    _valid.argtypes = [NotmuchFilenamesP]\r
++    _valid.restype = bool\r
++\r
++    _move_to_next = nmlib.notmuch_filenames_move_to_next\r
++    _move_to_next.argtypes = [NotmuchFilenamesP]\r
++    _move_to_next.restype = None\r
++\r
++    _destroy = nmlib.notmuch_filenames_destroy\r
++    _destroy.argtypes = [NotmuchFilenamesP]\r
++    _destroy.restype = None\r
++\r
+     def __init__(self, files_p, parent):\r
+         """\r
+         :param files_p: A pointer to an underlying *notmuch_tags_t*\r
+@@ -83,68 +95,24 @@ class Filenames(Python3StringMixIn):\r
+         if not files_p:\r
+             raise NullPointerError()\r
\r
+-        self._files_p = files_p\r
++        self._list = []\r
++        while self._valid(files_p):\r
++            file_ = self._get(files_p)\r
++            self._move_to_next(files_p)\r
++            self._list.append(file_.decode('utf-8', 'ignore'))\r
++        self._destroy(files_p)\r
++\r
+         #save reference to parent object so we keep it alive\r
+         self._parent = parent\r
\r
+     def __iter__(self):\r
+-        """ Make Filenames an iterator """\r
+-        return self\r
+-\r
+-    _valid = nmlib.notmuch_filenames_valid\r
+-    _valid.argtypes = [NotmuchFilenamesP]\r
+-    _valid.restype = bool\r
+-\r
+-    _move_to_next = nmlib.notmuch_filenames_move_to_next\r
+-    _move_to_next.argtypes = [NotmuchFilenamesP]\r
+-    _move_to_next.restype = None\r
+-\r
+-    def __next__(self):\r
+-        if not self._files_p:\r
+-            raise NotInitializedError()\r
+-\r
+-        if not self._valid(self._files_p):\r
+-            self._files_p = None\r
+-            raise StopIteration\r
+-\r
+-        file_ = Filenames._get(self._files_p)\r
+-        self._move_to_next(self._files_p)\r
+-        return file_.decode('utf-8', 'ignore')\r
+-    next = __next__ # python2.x iterator protocol compatibility\r
+-\r
+-    def __unicode__(self):\r
+-        """Represent Filenames() as newline-separated list of full paths\r
+-\r
+-        .. note::\r
+-\r
+-            This method exhausts the iterator object, so you will not be able to\r
+-            iterate over them again.\r
+-        """\r
+-        return "\n".join(self)\r
+-\r
+-    _destroy = nmlib.notmuch_filenames_destroy\r
+-    _destroy.argtypes = [NotmuchMessageP]\r
+-    _destroy.restype = None\r
+-\r
+-    def __del__(self):\r
+-        """Close and free the notmuch filenames"""\r
+-        if self._files_p:\r
+-            self._destroy(self._files_p)\r
++        """Make Filenames an iterator"""\r
++        return self._list.__iter__()\r
\r
+     def __len__(self):\r
+-        """len(:class:`Filenames`) returns the number of contained files\r
++        """len(:class:`Filenames`) returns the number of contained files"""\r
++        return self._list.__len__()\r
\r
+-        .. note::\r
+-\r
+-            This method exhausts the iterator object, so you will not be able to\r
+-            iterate over them again.\r
+-        """\r
+-        if not self._files_p:\r
+-            raise NotInitializedError()\r
+-\r
+-        i = 0\r
+-        while self._valid(self._files_p):\r
+-            self._move_to_next(self._files_p)\r
+-            i += 1\r
+-        self._files_p = None\r
+-        return i\r
++    def __unicode__(self):\r
++        """Represent Filenames() as newline-separated list of full paths"""\r
++        return "\n".join(self)\r
+-- \r
+2.4.1\r
+\r