Converted issues from mbox to maildir mailboxes
authorDmitriy Morozov <morozov@cs.duke.edu>
Wed, 16 Apr 2008 17:22:20 -0400
changeset 22 88f126b2dd08
parent 21 5b3579dc7abf
child 23 a42b9c661bc1
Converted issues from mbox to maildir mailboxes
.issues/8e4d2c2d12eb0169
.issues/8e4d2c2d12eb0169/new/1208380912.M221138P23014Q1.metatron
.issues/8e4d2c2d12eb0169/new/1208380912.M235104P23014Q2.metatron
.issues/95536ae767c2743a
.issues/95536ae767c2743a/new/1208380912.M239768P23014Q3.metatron
.issues/b7cdd0ec985471b7
.issues/b7cdd0ec985471b7/new/1208380912.M245624P23014Q4.metatron
.issues/b7cdd0ec985471b7/new/1208380912.M248109P23014Q5.metatron
.issues/c568146d3275c22b
.issues/c568146d3275c22b/new/1208380912.M298419P23014Q6.metatron
.issues/c568146d3275c22b/new/1208380912.M301451P23014Q7.metatron
.issues/c568146d3275c22b/new/1208380912.M303416P23014Q8.metatron
.issues/c568146d3275c22b/new/1208380912.M305572P23014Q9.metatron
.issues/c568146d3275c22b/new/1208380912.M308451P23014Q10.metatron
.issues/c568146d3275c22b/new/1208380912.M318159P23014Q11.metatron
.issues/c568146d3275c22b/new/1208380912.M320063P23014Q12.metatron
.issues/c568146d3275c22b/new/1208380912.M323632P23014Q13.metatron
.issues/c568146d3275c22b/new/1208380912.M326045P23014Q14.metatron
.issues/c568146d3275c22b/new/1208380912.M328161P23014Q15.metatron
.issues/c568146d3275c22b/new/1208380912.M330420P23014Q16.metatron
.issues/c568146d3275c22b/new/1208380912.M332474P23014Q17.metatron
.issues/c568146d3275c22b/new/1208380912.M334583P23014Q18.metatron
.issues/ef1b616f31345634
.issues/ef1b616f31345634/new/1208380912.M349704P23014Q19.metatron
--- a/.issues/8e4d2c2d12eb0169	Wed Apr 16 17:21:40 2008 -0400
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,30 +0,0 @@
-From artemis Sat Dec 29 07:57:17 2007
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sat, 29 Dec 2007 02:52:52 -0500
-State: fixed
-Subject: filters
-Message-Id: <8e4d2c2d12eb0169-0-artemis@metatron>
-
-Add ability to define filters in .issues/.filter. ilist should be able
-to restrict listing of issues based on a filter.
-
-.issues/.filter could look as follows:
-[critical-1.0]
-due-in=1.0
-priority=critical
-
-[gui]
-component=gui
-
-Then hg ilist -f critical-1.0 would show only issues with properties
-due-in and priority set to 1.0 and critical in the respective headers.
-
-From MAILER-DAEMON Sun Dec 30 13:37:54 2007
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sun, 30 Dec 2007 08:37:54 -0500
-Subject: properties changes (state)
-Message-Id: <8e4d2c2d12eb0169-1-artemis@metatron>
-In-Reply-To: <8e4d2c2d12eb0169-0-artemis@metatron>
-References: <8e4d2c2d12eb0169-0-artemis@metatron>
-
-state=fixed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/8e4d2c2d12eb0169/new/1208380912.M221138P23014Q1.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,19 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sat, 29 Dec 2007 02:52:52 -0500
+State: fixed
+Subject: filters
+Message-Id: <8e4d2c2d12eb0169-0-artemis@metatron>
+
+Add ability to define filters in .issues/.filter. ilist should be able
+to restrict listing of issues based on a filter.
+
+.issues/.filter could look as follows:
+[critical-1.0]
+due-in=1.0
+priority=critical
+
+[gui]
+component=gui
+
+Then hg ilist -f critical-1.0 would show only issues with properties
+due-in and priority set to 1.0 and critical in the respective headers.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/8e4d2c2d12eb0169/new/1208380912.M235104P23014Q2.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,8 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sun, 30 Dec 2007 08:37:54 -0500
+Subject: properties changes (state)
+Message-Id: <8e4d2c2d12eb0169-1-artemis@metatron>
+In-Reply-To: <8e4d2c2d12eb0169-0-artemis@metatron>
+References: <8e4d2c2d12eb0169-0-artemis@metatron>
+
+state=fixed
--- a/.issues/95536ae767c2743a	Wed Apr 16 17:21:40 2008 -0400
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,10 +0,0 @@
-From artemis Sat Dec 29 07:52:09 2007
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sat, 29 Dec 2007 02:50:26 -0500
-State: new
-Subject: attachments in iadd
-Message-Id: <95536ae767c2743a-0-artemis@metatron>
-
-Add ability to attach files in iadd command. Perhaps use -a flag for
-it (allowing multiple attachments). Store as a multipart message. List
-all parts of messages in ishow.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/95536ae767c2743a/new/1208380912.M239768P23014Q3.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,9 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sat, 29 Dec 2007 02:50:26 -0500
+State: new
+Subject: attachments in iadd
+Message-Id: <95536ae767c2743a-0-artemis@metatron>
+
+Add ability to attach files in iadd command. Perhaps use -a flag for
+it (allowing multiple attachments). Store as a multipart message. List
+all parts of messages in ishow.
--- a/.issues/b7cdd0ec985471b7	Wed Apr 16 17:21:40 2008 -0400
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,21 +0,0 @@
-From artemis Fri Jan  4 18:24:46 2008
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Fri, 04 Jan 2008 13:16:44 -0500
-State: fixed
-Subject: Thread messages by In-Reply-To
-Message-Id: <b7cdd0ec985471b7-0-artemis@metatron>
-
-Artemis is threading messages by References header, and assumes that
-there is only one element there. It should thread messages by
-In-Reply-To header (and set it accordingly), and either ignore
-References, or set it be equal to In-Reply-To.
-
-From artemis Sun Jan  6 06:04:32 2008
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sun, 06 Jan 2008 01:04:32 -0500
-Subject: properties changes (state)
-Message-Id: <b7cdd0ec985471b7-f5b5087bf1c336ab-artemis@metatron>
-References: <b7cdd0ec985471b7-0-artemis@metatron>
-In-Reply-To: <b7cdd0ec985471b7-0-artemis@metatron>
-
-state=fixed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/b7cdd0ec985471b7/new/1208380912.M245624P23014Q4.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,10 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Fri, 04 Jan 2008 13:16:44 -0500
+State: fixed
+Subject: Thread messages by In-Reply-To
+Message-Id: <b7cdd0ec985471b7-0-artemis@metatron>
+
+Artemis is threading messages by References header, and assumes that
+there is only one element there. It should thread messages by
+In-Reply-To header (and set it accordingly), and either ignore
+References, or set it be equal to In-Reply-To.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/b7cdd0ec985471b7/new/1208380912.M248109P23014Q5.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,8 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sun, 06 Jan 2008 01:04:32 -0500
+Subject: properties changes (state)
+Message-Id: <b7cdd0ec985471b7-f5b5087bf1c336ab-artemis@metatron>
+References: <b7cdd0ec985471b7-0-artemis@metatron>
+In-Reply-To: <b7cdd0ec985471b7-0-artemis@metatron>
+
+state=fixed
--- a/.issues/c568146d3275c22b	Wed Apr 16 17:21:40 2008 -0400
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,2948 +0,0 @@
-From mirko@friedenhagen.de  Fri Jan  4 18:31:44 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-0.2 required=5.0 tests=BAYES_40 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m04NVf7J019792
-	for <morozov@cs.duke.edu>; Fri, 4 Jan 2008 18:31:43 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu2) with ESMTP (Nemesis)
-	id 0MKwtQ-1JAw0l1aId-0006EP; Sat, 05 Jan 2008 00:31:35 +0100
-Message-Id: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: hg ishow fails in clone of Artemis-repo
-Date: Sat, 05 Jan 2008 00:31:27 +0100
-State: fixed
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX1/JysvjyYiL3YHLT6W7yfO6FZ1JMB6PepAft/b
- jnXeHpRbt2ZogH7u/u+9jEuj7D5SulS51Scfvx4LgDgb4pctZs
- sbaVxmR5TQar5n7qj9MGA==
-Status: RO
-X-Status: A
-Content-Length: 1180
-Lines: 35
-
-Hello Dmitriy,
-
-first of all let me thank you for this extension, I have the feeling  
-that it might be really useful.
-
-I have one issue with the clone, "hg ilist" will abort, "hg ishow"  
-does not have any problems:
-
-[mirko@borg Artemis]$ hg log -l1
-changeset:   9:5319c712fa34
-tag:         tip
-user:        Dmitriy Morozov <morozov@cs.duke.edu>
-date:        Sun Dec 30 09:23:23 2007 -0500
-summary:     Fixed Message-Ids of our bugs (wrapped them in <...>)
-
-[mirko@borg Artemis]$ hg ilist
-abort: invalid date: 'Sat, 29 Dec 2007 02:50:26 EST'
-
-[mirko@borg Artemis]$ grep -r 'Sat, 29 Dec 2007 02:50:26 EST' .issues/
-.issues/95536ae767c2743a:Date: Sat, 29 Dec 2007 02:50:26 EST
-
-[mirko@borg Artemis]$ hg ishow 95536ae767c2743a
-======================================================================
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sat, 29 Dec 2007 02:50:26 EST
-Subject: attachments in iadd
-State: new
-
-Add ability to attach files in iadd command. Perhaps use -a flag for
-it (allowing multiple attachments). Store as a multipart message. List
-all parts of messages in ishow.
-----------------------------------------------------------------------
-
-Best Regards
-Mirko
-
-From morozov@cs.duke.edu Fri Jan  4 18:34:36 2008
-Date: Fri, 04 Jan 2008 18:34:36 -0500
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-To: Mirko Friedenhagen <mirko@friedenhagen.de>
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Message-ID: <20080104233436.GA1930@cs.duke.edu>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
-Mime-Version: 1.0
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: inline
-In-Reply-To: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
-User-Agent: Mutt/1.4.2.2i
-Status: RO
-Content-Length: 1475
-Lines: 45
-
-Hi, Mirko,
-
-What version of Python, and more importantly what version of Mercurial
-do you have? I have an idea of where the problem is, but I need the
-version numbers first.
-
-Thanks.
-Dmitriy
-
-On Sat, Jan 05, 2008 at 12:31:27AM +0100, Mirko Friedenhagen wrote:
->Hello Dmitriy,
->
->first of all let me thank you for this extension, I have the feeling  
->that it might be really useful.
->
->I have one issue with the clone, "hg ilist" will abort, "hg ishow"  
->does not have any problems:
->
->[mirko@borg Artemis]$ hg log -l1
->changeset:   9:5319c712fa34
->tag:         tip
->user:        Dmitriy Morozov <morozov@cs.duke.edu>
->date:        Sun Dec 30 09:23:23 2007 -0500
->summary:     Fixed Message-Ids of our bugs (wrapped them in <...>)
->
->[mirko@borg Artemis]$ hg ilist
->abort: invalid date: 'Sat, 29 Dec 2007 02:50:26 EST'
->
->[mirko@borg Artemis]$ grep -r 'Sat, 29 Dec 2007 02:50:26 EST' .issues/
->.issues/95536ae767c2743a:Date: Sat, 29 Dec 2007 02:50:26 EST
->
->[mirko@borg Artemis]$ hg ishow 95536ae767c2743a
->======================================================================
->From: Dmitriy Morozov <morozov@cs.duke.edu>
->Date: Sat, 29 Dec 2007 02:50:26 EST
->Subject: attachments in iadd
->State: new
->
->Add ability to attach files in iadd command. Perhaps use -a flag for
->it (allowing multiple attachments). Store as a multipart message. List
->all parts of messages in ishow.
->----------------------------------------------------------------------
->
->Best Regards
->Mirko
-
-From mirko@friedenhagen.de  Fri Jan  4 18:43:30 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-1.4 required=5.0 tests=AWL,BAYES_00 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.187])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m04NhTah021156
-	for <morozov@cs.duke.edu>; Fri, 4 Jan 2008 18:43:29 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu2) with ESMTP (Nemesis)
-	id 0MKwtQ-1JAwCB2mxq-0006FP; Sat, 05 Jan 2008 00:43:23 +0100
-Message-Id: <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080104233436.GA1930@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Sat, 05 Jan 2008 00:43:23 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX1/lGFmR/n0Vn+KixPR/52KDdWD8RANfnQ68xtl
- cTd+HJmbaC+4c1t/CI5zg5HmPAdrPm3n/0v4CF33rUEBuEJFrC
- NXyP7mzJQBuNykwAF7tNw==
-Status: RO
-X-Status: A
-Content-Length: 1216
-Lines: 40
-
-Am 05.01.2008 um 00:34 schrieb Dmitriy Morozov:
-
-> Hi, Mirko,
->
-> What version of Python, and more importantly what version of Mercurial
-> do you have? I have an idea of where the problem is, but I need the
-> version numbers first.
->
-
-Hello Dmitriy,
-
-now, that's a quick answer ;-).
-
-I am following mercurial-crew (the second parent are minor adjustments  
-(git-summary-template, where I include an Link to the hgweb-root)),  
-but no changes to core.
-
-[mirko@borg mercurial-crew]$ hg log -l2
-changeset:   5824:1048a5ac25ef
-branch:      local
-tag:         tip
-parent:      5807:0176986571fe
-parent:      5823:d852151fb8d4
-user:        Mirko Friedenhagen <mirko.friedenhagen@1und1.de>
-date:        Fri Jan 04 23:22:08 2008 +0100
-summary:     Automated merge with http://hg.intevation.org/mercurial/crew
-
-changeset:   5823:d852151fb8d4
-user:        Bryan O'Sullivan <bos@serpentine.com>
-date:        Fri Jan 04 13:56:31 2008 -0800
-summary:     util: drop params added during experimentation
-
-[mirko@borg mercurial-crew]$ python
-Python 2.5.1 (r251:54863, Oct  5 2007, 21:08:09)
-[GCC 4.0.1 (Apple Inc. build 5465)] on darwin
-Type "help", "copyright", "credits" or "license" for more information.
- >>>
-
-Best Regards
-Mirko
-
-From morozov@cs.duke.edu Sat Jan  5 07:47:08 2008
-Date: Sat, 05 Jan 2008 07:47:08 -0500
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-To: Mirko Friedenhagen <mirko@friedenhagen.de>
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Message-ID: <20080105124708.GA7043@cs.duke.edu>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
-Mime-Version: 1.0
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: inline
-In-Reply-To: <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
-User-Agent: Mutt/1.4.2.2i
-Status: RO
-Content-Length: 1504
-Lines: 50
-
-Hi, Mirko,
-
-I'm unable to replicate the problem with mercrial/crew. Can you send
-me util.py (i.e., mercurial/util.py) from the repository that you are
-actually using?
-
-Thanks.
-Dmitriy
-
-On Sat, Jan 05, 2008 at 12:43:23AM +0100, Mirko Friedenhagen wrote:
->Am 05.01.2008 um 00:34 schrieb Dmitriy Morozov:
->
->>Hi, Mirko,
->>
->>What version of Python, and more importantly what version of Mercurial
->>do you have? I have an idea of where the problem is, but I need the
->>version numbers first.
->>
->
->Hello Dmitriy,
->
->now, that's a quick answer ;-).
->
->I am following mercurial-crew (the second parent are minor adjustments  
->(git-summary-template, where I include an Link to the hgweb-root)),  
->but no changes to core.
->
->[mirko@borg mercurial-crew]$ hg log -l2
->changeset:   5824:1048a5ac25ef
->branch:      local
->tag:         tip
->parent:      5807:0176986571fe
->parent:      5823:d852151fb8d4
->user:        Mirko Friedenhagen <mirko.friedenhagen@1und1.de>
->date:        Fri Jan 04 23:22:08 2008 +0100
->summary:     Automated merge with http://hg.intevation.org/mercurial/crew
->
->changeset:   5823:d852151fb8d4
->user:        Bryan O'Sullivan <bos@serpentine.com>
->date:        Fri Jan 04 13:56:31 2008 -0800
->summary:     util: drop params added during experimentation
->
->[mirko@borg mercurial-crew]$ python
->Python 2.5.1 (r251:54863, Oct  5 2007, 21:08:09)
->[GCC 4.0.1 (Apple Inc. build 5465)] on darwin
->Type "help", "copyright", "credits" or "license" for more information.
->>>>
->
->Best Regards
->Mirko
-
-From mirko@friedenhagen.de  Sat Jan  5 16:18:09 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-0.7 required=5.0 tests=AWL,BAYES_50 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.188])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m05LI8r8027688
-	for <morozov@cs.duke.edu>; Sat, 5 Jan 2008 16:18:08 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu7) with ESMTP (Nemesis)
-	id 0ML2xA-1JBGP22KSu-0000u0; Sat, 05 Jan 2008 22:18:02 +0100
-Message-Id: <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080105124708.GA7043@cs.duke.edu>
-Content-Type: multipart/mixed; boundary=Apple-Mail-27-69865486
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Sat, 05 Jan 2008 22:18:01 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX18U2SPmtxdIc/97m419K3u6bNHAGoJX7Zg8bRM
- nhGQpakEwacktoI+iKEBCOeJWa/w25yyJDtlt4rvwcfLwDf3SB
- BTBFcyJEMtRpthNwJH//w==
-Status: RO
-X-Status: A
-Content-Length: 57277
-Lines: 1782
-
-
---Apple-Mail-27-69865486
-Content-Type: text/plain;
-	charset=US-ASCII;
-	format=flowed;
-	delsp=yes
-Content-Transfer-Encoding: 7bit
-
-Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
-
-> Hi, Mirko,
->
-> I'm unable to replicate the problem with mercrial/crew. Can you send
-> me util.py (i.e., mercurial/util.py) from the repository that you are
-> actually using?
-I did not alter anything in it, see the diff between my local branch  
-and default:
-
-[mirko@borg mercurial-crew]$ hg diff -rdefault
-diff -r d852151fb8d4 hgext/convert/cvs.py
---- a/hgext/convert/cvs.py	Fri Jan 04 13:56:31 2008 -0800
-+++ b/hgext/convert/cvs.py	Sat Jan 05 22:16:04 2008 +0100
-@@ -255,6 +255,12 @@ class convert_cvs(converter_source):
-                      return (data, "x" in mode and "x" or "")
-                  elif line.startswith("E "):
-                      self.ui.warn("cvs server: %s\n" % line[2:])
-+                elif line.endswith("- ignored\n"):
-+                    self.ui.warn("cvs server: %s\n" % line)
-+                    self.readp.readline()
-+                    l = self.readp.readline()
-+                    if l != "error\n":
-+                        raise util.Abort("unknown CVS response: %s" %  
-l)
-                  elif line.startswith("Remove"):
-                      l = self.readp.readline()
-                      l = self.readp.readline()
-diff -r d852151fb8d4 templates/gitweb/summary.tmpl
---- a/templates/gitweb/summary.tmpl	Fri Jan 04 13:56:31 2008 -0800
-+++ b/templates/gitweb/summary.tmpl	Sat Jan 05 22:16:05 2008 +0100
-@@ -7,6 +7,7 @@
-  </head>
-  <body>
-
-+<a href="/">^ Repositories List</a>
-  <div class="page_header">
-  <a href="http://www.selenic.com/mercurial/" title="Mercurial"><div  
-style="float:right;">Mercurial</div></a><a  
-href="{url}summary{sessionvars%urlparameter}">#repo|escape#</a> /  
-summary
-
-
-
---Apple-Mail-27-69865486
-Content-Disposition: attachment;
-	filename=util.py
-Content-Type: text/x-python-script;
-	x-unix-mode=0644;
-	name="util.py"
-Content-Transfer-Encoding: 7bit
-
-"""
-util.py - Mercurial utility functions and platform specfic implementations
-
- Copyright 2005 K. Thananchayan <thananck@yahoo.com>
- Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
- Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
-
-This software may be used and distributed according to the terms
-of the GNU General Public License, incorporated herein by reference.
-
-This contains helper routines that are independent of the SCM core and hide
-platform-specific details from the core.
-"""
-
-from i18n import _
-import cStringIO, errno, getpass, popen2, re, shutil, sys, tempfile, strutil
-import os, stat, threading, time, calendar, ConfigParser, locale, glob, osutil
-import re, urlparse
-
-try:
-    set = set
-    frozenset = frozenset
-except NameError:
-    from sets import Set as set, ImmutableSet as frozenset
-
-try:
-    _encoding = os.environ.get("HGENCODING")
-    if sys.platform == 'darwin' and not _encoding:
-        # On darwin, getpreferredencoding ignores the locale environment and
-        # always returns mac-roman. We override this if the environment is
-        # not C (has been customized by the user).
-        locale.setlocale(locale.LC_CTYPE, '')
-        _encoding = locale.getlocale()[1]
-    if not _encoding:
-        _encoding = locale.getpreferredencoding() or 'ascii'
-except locale.Error:
-    _encoding = 'ascii'
-_encodingmode = os.environ.get("HGENCODINGMODE", "strict")
-_fallbackencoding = 'ISO-8859-1'
-
-def tolocal(s):
-    """
-    Convert a string from internal UTF-8 to local encoding
-
-    All internal strings should be UTF-8 but some repos before the
-    implementation of locale support may contain latin1 or possibly
-    other character sets. We attempt to decode everything strictly
-    using UTF-8, then Latin-1, and failing that, we use UTF-8 and
-    replace unknown characters.
-    """
-    for e in ('UTF-8', _fallbackencoding):
-        try:
-            u = s.decode(e) # attempt strict decoding
-            return u.encode(_encoding, "replace")
-        except LookupError, k:
-            raise Abort(_("%s, please check your locale settings") % k)
-        except UnicodeDecodeError:
-            pass
-    u = s.decode("utf-8", "replace") # last ditch
-    return u.encode(_encoding, "replace")
-
-def fromlocal(s):
-    """
-    Convert a string from the local character encoding to UTF-8
-
-    We attempt to decode strings using the encoding mode set by
-    HGENCODINGMODE, which defaults to 'strict'. In this mode, unknown
-    characters will cause an error message. Other modes include
-    'replace', which replaces unknown characters with a special
-    Unicode character, and 'ignore', which drops the character.
-    """
-    try:
-        return s.decode(_encoding, _encodingmode).encode("utf-8")
-    except UnicodeDecodeError, inst:
-        sub = s[max(0, inst.start-10):inst.start+10]
-        raise Abort("decoding near '%s': %s!" % (sub, inst))
-    except LookupError, k:
-        raise Abort(_("%s, please check your locale settings") % k)
-
-def locallen(s):
-    """Find the length in characters of a local string"""
-    return len(s.decode(_encoding, "replace"))
-
-# used by parsedate
-defaultdateformats = (
-    '%Y-%m-%d %H:%M:%S',
-    '%Y-%m-%d %I:%M:%S%p',
-    '%Y-%m-%d %H:%M',
-    '%Y-%m-%d %I:%M%p',
-    '%Y-%m-%d',
-    '%m-%d',
-    '%m/%d',
-    '%m/%d/%y',
-    '%m/%d/%Y',
-    '%a %b %d %H:%M:%S %Y',
-    '%a %b %d %I:%M:%S%p %Y',
-    '%a, %d %b %Y %H:%M:%S',        #  GNU coreutils "/bin/date --rfc-2822"
-    '%b %d %H:%M:%S %Y',
-    '%b %d %I:%M:%S%p %Y',
-    '%b %d %H:%M:%S',
-    '%b %d %I:%M:%S%p',
-    '%b %d %H:%M',
-    '%b %d %I:%M%p',
-    '%b %d %Y',
-    '%b %d',
-    '%H:%M:%S',
-    '%I:%M:%SP',
-    '%H:%M',
-    '%I:%M%p',
-)
-
-extendeddateformats = defaultdateformats + (
-    "%Y",
-    "%Y-%m",
-    "%b",
-    "%b %Y",
-    )
-
-class SignalInterrupt(Exception):
-    """Exception raised on SIGTERM and SIGHUP."""
-
-# differences from SafeConfigParser:
-# - case-sensitive keys
-# - allows values that are not strings (this means that you may not
-#   be able to save the configuration to a file)
-class configparser(ConfigParser.SafeConfigParser):
-    def optionxform(self, optionstr):
-        return optionstr
-
-    def set(self, section, option, value):
-        return ConfigParser.ConfigParser.set(self, section, option, value)
-
-    def _interpolate(self, section, option, rawval, vars):
-        if not isinstance(rawval, basestring):
-            return rawval
-        return ConfigParser.SafeConfigParser._interpolate(self, section,
-                                                          option, rawval, vars)
-
-def cachefunc(func):
-    '''cache the result of function calls'''
-    # XXX doesn't handle keywords args
-    cache = {}
-    if func.func_code.co_argcount == 1:
-        # we gain a small amount of time because
-        # we don't need to pack/unpack the list
-        def f(arg):
-            if arg not in cache:
-                cache[arg] = func(arg)
-            return cache[arg]
-    else:
-        def f(*args):
-            if args not in cache:
-                cache[args] = func(*args)
-            return cache[args]
-
-    return f
-
-def pipefilter(s, cmd):
-    '''filter string S through command CMD, returning its output'''
-    (pin, pout) = os.popen2(cmd, 'b')
-    def writer():
-        try:
-            pin.write(s)
-            pin.close()
-        except IOError, inst:
-            if inst.errno != errno.EPIPE:
-                raise
-
-    # we should use select instead on UNIX, but this will work on most
-    # systems, including Windows
-    w = threading.Thread(target=writer)
-    w.start()
-    f = pout.read()
-    pout.close()
-    w.join()
-    return f
-
-def tempfilter(s, cmd):
-    '''filter string S through a pair of temporary files with CMD.
-    CMD is used as a template to create the real command to be run,
-    with the strings INFILE and OUTFILE replaced by the real names of
-    the temporary files generated.'''
-    inname, outname = None, None
-    try:
-        infd, inname = tempfile.mkstemp(prefix='hg-filter-in-')
-        fp = os.fdopen(infd, 'wb')
-        fp.write(s)
-        fp.close()
-        outfd, outname = tempfile.mkstemp(prefix='hg-filter-out-')
-        os.close(outfd)
-        cmd = cmd.replace('INFILE', inname)
-        cmd = cmd.replace('OUTFILE', outname)
-        code = os.system(cmd)
-        if sys.platform == 'OpenVMS' and code & 1:
-            code = 0
-        if code: raise Abort(_("command '%s' failed: %s") %
-                             (cmd, explain_exit(code)))
-        return open(outname, 'rb').read()
-    finally:
-        try:
-            if inname: os.unlink(inname)
-        except: pass
-        try:
-            if outname: os.unlink(outname)
-        except: pass
-
-filtertable = {
-    'tempfile:': tempfilter,
-    'pipe:': pipefilter,
-    }
-
-def filter(s, cmd):
-    "filter a string through a command that transforms its input to its output"
-    for name, fn in filtertable.iteritems():
-        if cmd.startswith(name):
-            return fn(s, cmd[len(name):].lstrip())
-    return pipefilter(s, cmd)
-
-def binary(s):
-    """return true if a string is binary data using diff's heuristic"""
-    if s and '\0' in s[:4096]:
-        return True
-    return False
-
-def unique(g):
-    """return the uniq elements of iterable g"""
-    return dict.fromkeys(g).keys()
-
-class Abort(Exception):
-    """Raised if a command needs to print an error and exit."""
-
-class UnexpectedOutput(Abort):
-    """Raised to print an error with part of output and exit."""
-
-def always(fn): return True
-def never(fn): return False
-
-def expand_glob(pats):
-    '''On Windows, expand the implicit globs in a list of patterns'''
-    if os.name != 'nt':
-        return list(pats)
-    ret = []
-    for p in pats:
-        kind, name = patkind(p, None)
-        if kind is None:
-            globbed = glob.glob(name)
-            if globbed:
-                ret.extend(globbed)
-                continue
-            # if we couldn't expand the glob, just keep it around
-        ret.append(p)
-    return ret
-
-def patkind(name, dflt_pat='glob'):
-    """Split a string into an optional pattern kind prefix and the
-    actual pattern."""
-    for prefix in 're', 'glob', 'path', 'relglob', 'relpath', 'relre':
-        if name.startswith(prefix + ':'): return name.split(':', 1)
-    return dflt_pat, name
-
-def globre(pat, head='^', tail='$'):
-    "convert a glob pattern into a regexp"
-    i, n = 0, len(pat)
-    res = ''
-    group = False
-    def peek(): return i < n and pat[i]
-    while i < n:
-        c = pat[i]
-        i = i+1
-        if c == '*':
-            if peek() == '*':
-                i += 1
-                res += '.*'
-            else:
-                res += '[^/]*'
-        elif c == '?':
-            res += '.'
-        elif c == '[':
-            j = i
-            if j < n and pat[j] in '!]':
-                j += 1
-            while j < n and pat[j] != ']':
-                j += 1
-            if j >= n:
-                res += '\\['
-            else:
-                stuff = pat[i:j].replace('\\','\\\\')
-                i = j + 1
-                if stuff[0] == '!':
-                    stuff = '^' + stuff[1:]
-                elif stuff[0] == '^':
-                    stuff = '\\' + stuff
-                res = '%s[%s]' % (res, stuff)
-        elif c == '{':
-            group = True
-            res += '(?:'
-        elif c == '}' and group:
-            res += ')'
-            group = False
-        elif c == ',' and group:
-            res += '|'
-        elif c == '\\':
-            p = peek()
-            if p:
-                i += 1
-                res += re.escape(p)
-            else:
-                res += re.escape(c)
-        else:
-            res += re.escape(c)
-    return head + res + tail
-
-_globchars = {'[': 1, '{': 1, '*': 1, '?': 1}
-
-def pathto(root, n1, n2):
-    '''return the relative path from one place to another.
-    root should use os.sep to separate directories
-    n1 should use os.sep to separate directories
-    n2 should use "/" to separate directories
-    returns an os.sep-separated path.
-
-    If n1 is a relative path, it's assumed it's
-    relative to root.
-    n2 should always be relative to root.
-    '''
-    if not n1: return localpath(n2)
-    if os.path.isabs(n1):
-        if os.path.splitdrive(root)[0] != os.path.splitdrive(n1)[0]:
-            return os.path.join(root, localpath(n2))
-        n2 = '/'.join((pconvert(root), n2))
-    a, b = n1.split(os.sep), n2.split('/')
-    a.reverse()
-    b.reverse()
-    while a and b and a[-1] == b[-1]:
-        a.pop()
-        b.pop()
-    b.reverse()
-    return os.sep.join((['..'] * len(a)) + b)
-
-def canonpath(root, cwd, myname):
-    """return the canonical path of myname, given cwd and root"""
-    if root == os.sep:
-        rootsep = os.sep
-    elif root.endswith(os.sep):
-        rootsep = root
-    else:
-        rootsep = root + os.sep
-    name = myname
-    if not os.path.isabs(name):
-        name = os.path.join(root, cwd, name)
-    name = os.path.normpath(name)
-    audit_path = path_auditor(root)
-    if name != rootsep and name.startswith(rootsep):
-        name = name[len(rootsep):]
-        audit_path(name)
-        return pconvert(name)
-    elif name == root:
-        return ''
-    else:
-        # Determine whether `name' is in the hierarchy at or beneath `root',
-        # by iterating name=dirname(name) until that causes no change (can't
-        # check name == '/', because that doesn't work on windows).  For each
-        # `name', compare dev/inode numbers.  If they match, the list `rel'
-        # holds the reversed list of components making up the relative file
-        # name we want.
-        root_st = os.stat(root)
-        rel = []
-        while True:
-            try:
-                name_st = os.stat(name)
-            except OSError:
-                break
-            if samestat(name_st, root_st):
-                if not rel:
-                    # name was actually the same as root (maybe a symlink)
-                    return ''
-                rel.reverse()
-                name = os.path.join(*rel)
-                audit_path(name)
-                return pconvert(name)
-            dirname, basename = os.path.split(name)
-            rel.append(basename)
-            if dirname == name:
-                break
-            name = dirname
-
-        raise Abort('%s not under root' % myname)
-
-def matcher(canonroot, cwd='', names=[], inc=[], exc=[], src=None):
-    return _matcher(canonroot, cwd, names, inc, exc, 'glob', src)
-
-def cmdmatcher(canonroot, cwd='', names=[], inc=[], exc=[], src=None,
-               globbed=False, default=None):
-    default = default or 'relpath'
-    if default == 'relpath' and not globbed:
-        names = expand_glob(names)
-    return _matcher(canonroot, cwd, names, inc, exc, default, src)
-
-def _matcher(canonroot, cwd, names, inc, exc, dflt_pat, src):
-    """build a function to match a set of file patterns
-
-    arguments:
-    canonroot - the canonical root of the tree you're matching against
-    cwd - the current working directory, if relevant
-    names - patterns to find
-    inc - patterns to include
-    exc - patterns to exclude
-    dflt_pat - if a pattern in names has no explicit type, assume this one
-    src - where these patterns came from (e.g. .hgignore)
-
-    a pattern is one of:
-    'glob:<glob>' - a glob relative to cwd
-    're:<regexp>' - a regular expression
-    'path:<path>' - a path relative to canonroot
-    'relglob:<glob>' - an unrooted glob (*.c matches C files in all dirs)
-    'relpath:<path>' - a path relative to cwd
-    'relre:<regexp>' - a regexp that doesn't have to match the start of a name
-    '<something>' - one of the cases above, selected by the dflt_pat argument
-
-    returns:
-    a 3-tuple containing
-    - list of roots (places where one should start a recursive walk of the fs);
-      this often matches the explicit non-pattern names passed in, but also
-      includes the initial part of glob: patterns that has no glob characters
-    - a bool match(filename) function
-    - a bool indicating if any patterns were passed in
-    """
-
-    # a common case: no patterns at all
-    if not names and not inc and not exc:
-        return [], always, False
-
-    def contains_glob(name):
-        for c in name:
-            if c in _globchars: return True
-        return False
-
-    def regex(kind, name, tail):
-        '''convert a pattern into a regular expression'''
-        if not name:
-            return ''
-        if kind == 're':
-            return name
-        elif kind == 'path':
-            return '^' + re.escape(name) + '(?:/|$)'
-        elif kind == 'relglob':
-            return globre(name, '(?:|.*/)', tail)
-        elif kind == 'relpath':
-            return re.escape(name) + '(?:/|$)'
-        elif kind == 'relre':
-            if name.startswith('^'):
-                return name
-            return '.*' + name
-        return globre(name, '', tail)
-
-    def matchfn(pats, tail):
-        """build a matching function from a set of patterns"""
-        if not pats:
-            return
-        try:
-            pat = '(?:%s)' % '|'.join([regex(k, p, tail) for (k, p) in pats])
-            return re.compile(pat).match
-        except OverflowError:
-            # We're using a Python with a tiny regex engine and we
-            # made it explode, so we'll divide the pattern list in two
-            # until it works
-            l = len(pats)
-            if l < 2:
-                raise
-            a, b = matchfn(pats[:l//2], tail), matchfn(pats[l//2:], tail)
-            return lambda s: a(s) or b(s)
-        except re.error:
-            for k, p in pats:
-                try:
-                    re.compile('(?:%s)' % regex(k, p, tail))
-                except re.error:
-                    if src:
-                        raise Abort("%s: invalid pattern (%s): %s" %
-                                    (src, k, p))
-                    else:
-                        raise Abort("invalid pattern (%s): %s" % (k, p))
-            raise Abort("invalid pattern")
-
-    def globprefix(pat):
-        '''return the non-glob prefix of a path, e.g. foo/* -> foo'''
-        root = []
-        for p in pat.split('/'):
-            if contains_glob(p): break
-            root.append(p)
-        return '/'.join(root) or '.'
-
-    def normalizepats(names, default):
-        pats = []
-        roots = []
-        anypats = False
-        for kind, name in [patkind(p, default) for p in names]:
-            if kind in ('glob', 'relpath'):
-                name = canonpath(canonroot, cwd, name)
-            elif kind in ('relglob', 'path'):
-                name = normpath(name)
-
-            pats.append((kind, name))
-
-            if kind in ('glob', 're', 'relglob', 'relre'):
-                anypats = True
-
-            if kind == 'glob':
-                root = globprefix(name)
-                roots.append(root)
-            elif kind in ('relpath', 'path'):
-                roots.append(name or '.')
-            elif kind == 'relglob':
-                roots.append('.')
-        return roots, pats, anypats
-
-    roots, pats, anypats = normalizepats(names, dflt_pat)
-
-    patmatch = matchfn(pats, '$') or always
-    incmatch = always
-    if inc:
-        dummy, inckinds, dummy = normalizepats(inc, 'glob')
-        incmatch = matchfn(inckinds, '(?:/|$)')
-    excmatch = lambda fn: False
-    if exc:
-        dummy, exckinds, dummy = normalizepats(exc, 'glob')
-        excmatch = matchfn(exckinds, '(?:/|$)')
-
-    if not names and inc and not exc:
-        # common case: hgignore patterns
-        match = incmatch
-    else:
-        match = lambda fn: incmatch(fn) and not excmatch(fn) and patmatch(fn)
-
-    return (roots, match, (inc or exc or anypats) and True)
-
-_hgexecutable = None
-
-def hgexecutable():
-    """return location of the 'hg' executable.
-
-    Defaults to $HG or 'hg' in the search path.
-    """
-    if _hgexecutable is None:
-        set_hgexecutable(os.environ.get('HG') or find_exe('hg', 'hg'))
-    return _hgexecutable
-
-def set_hgexecutable(path):
-    """set location of the 'hg' executable"""
-    global _hgexecutable
-    _hgexecutable = path
-
-def system(cmd, environ={}, cwd=None, onerr=None, errprefix=None):
-    '''enhanced shell command execution.
-    run with environment maybe modified, maybe in different dir.
-
-    if command fails and onerr is None, return status.  if ui object,
-    print error message and return status, else raise onerr object as
-    exception.'''
-    def py2shell(val):
-        'convert python object into string that is useful to shell'
-        if val in (None, False):
-            return '0'
-        if val == True:
-            return '1'
-        return str(val)
-    oldenv = {}
-    for k in environ:
-        oldenv[k] = os.environ.get(k)
-    if cwd is not None:
-        oldcwd = os.getcwd()
-    origcmd = cmd
-    if os.name == 'nt':
-        cmd = '"%s"' % cmd
-    try:
-        for k, v in environ.iteritems():
-            os.environ[k] = py2shell(v)
-        os.environ['HG'] = hgexecutable()
-        if cwd is not None and oldcwd != cwd:
-            os.chdir(cwd)
-        rc = os.system(cmd)
-        if sys.platform == 'OpenVMS' and rc & 1:
-            rc = 0
-        if rc and onerr:
-            errmsg = '%s %s' % (os.path.basename(origcmd.split(None, 1)[0]),
-                                explain_exit(rc)[0])
-            if errprefix:
-                errmsg = '%s: %s' % (errprefix, errmsg)
-            try:
-                onerr.warn(errmsg + '\n')
-            except AttributeError:
-                raise onerr(errmsg)
-        return rc
-    finally:
-        for k, v in oldenv.iteritems():
-            if v is None:
-                del os.environ[k]
-            else:
-                os.environ[k] = v
-        if cwd is not None and oldcwd != cwd:
-            os.chdir(oldcwd)
-
-# os.path.lexists is not available on python2.3
-def lexists(filename):
-    "test whether a file with this name exists. does not follow symlinks"
-    try:
-        os.lstat(filename)
-    except:
-        return False
-    return True
-
-def rename(src, dst):
-    """forcibly rename a file"""
-    try:
-        os.rename(src, dst)
-    except OSError, err: # FIXME: check err (EEXIST ?)
-        # on windows, rename to existing file is not allowed, so we
-        # must delete destination first. but if file is open, unlink
-        # schedules it for delete but does not delete it. rename
-        # happens immediately even for open files, so we create
-        # temporary file, delete it, rename destination to that name,
-        # then delete that. then rename is safe to do.
-        fd, temp = tempfile.mkstemp(dir=os.path.dirname(dst) or '.')
-        os.close(fd)
-        os.unlink(temp)
-        os.rename(dst, temp)
-        os.unlink(temp)
-        os.rename(src, dst)
-
-def unlink(f):
-    """unlink and remove the directory if it is empty"""
-    os.unlink(f)
-    # try removing directories that might now be empty
-    try:
-        os.removedirs(os.path.dirname(f))
-    except OSError:
-        pass
-
-def copyfile(src, dest):
-    "copy a file, preserving mode"
-    if os.path.islink(src):
-        try:
-            os.unlink(dest)
-        except:
-            pass
-        os.symlink(os.readlink(src), dest)
-    else:
-        try:
-            shutil.copyfile(src, dest)
-            shutil.copymode(src, dest)
-        except shutil.Error, inst:
-            raise Abort(str(inst))
-
-def copyfiles(src, dst, hardlink=None):
-    """Copy a directory tree using hardlinks if possible"""
-
-    if hardlink is None:
-        hardlink = (os.stat(src).st_dev ==
-                    os.stat(os.path.dirname(dst)).st_dev)
-
-    if os.path.isdir(src):
-        os.mkdir(dst)
-        for name, kind in osutil.listdir(src):
-            srcname = os.path.join(src, name)
-            dstname = os.path.join(dst, name)
-            copyfiles(srcname, dstname, hardlink)
-    else:
-        if hardlink:
-            try:
-                os_link(src, dst)
-            except (IOError, OSError):
-                hardlink = False
-                shutil.copy(src, dst)
-        else:
-            shutil.copy(src, dst)
-
-class path_auditor(object):
-    '''ensure that a filesystem path contains no banned components.
-    the following properties of a path are checked:
-
-    - under top-level .hg
-    - starts at the root of a windows drive
-    - contains ".."
-    - traverses a symlink (e.g. a/symlink_here/b)
-    - inside a nested repository'''
-
-    def __init__(self, root):
-        self.audited = set()
-        self.auditeddir = set()
-        self.root = root
-
-    def __call__(self, path):
-        if path in self.audited:
-            return
-        normpath = os.path.normcase(path)
-        parts = normpath.split(os.sep)
-        if (os.path.splitdrive(path)[0] or parts[0] in ('.hg', '')
-            or os.pardir in parts):
-            raise Abort(_("path contains illegal component: %s") % path)
-        def check(prefix):
-            curpath = os.path.join(self.root, prefix)
-            try:
-                st = os.lstat(curpath)
-            except OSError, err:
-                # EINVAL can be raised as invalid path syntax under win32.
-                # They must be ignored for patterns can be checked too.
-                if err.errno not in (errno.ENOENT, errno.ENOTDIR, errno.EINVAL):
-                    raise
-            else:
-                if stat.S_ISLNK(st.st_mode):
-                    raise Abort(_('path %r traverses symbolic link %r') %
-                                (path, prefix))
-                elif (stat.S_ISDIR(st.st_mode) and
-                      os.path.isdir(os.path.join(curpath, '.hg'))):
-                    raise Abort(_('path %r is inside repo %r') %
-                                (path, prefix))
-
-        prefixes = []
-        for c in strutil.rfindall(normpath, os.sep):
-            prefix = normpath[:c]
-            if prefix in self.auditeddir:
-                break
-            check(prefix)
-            prefixes.append(prefix)
-
-        self.audited.add(path)
-        # only add prefixes to the cache after checking everything: we don't
-        # want to add "foo/bar/baz" before checking if there's a "foo/.hg"
-        self.auditeddir.update(prefixes)
-
-def _makelock_file(info, pathname):
-    ld = os.open(pathname, os.O_CREAT | os.O_WRONLY | os.O_EXCL)
-    os.write(ld, info)
-    os.close(ld)
-
-def _readlock_file(pathname):
-    return posixfile(pathname).read()
-
-def nlinks(pathname):
-    """Return number of hardlinks for the given file."""
-    return os.lstat(pathname).st_nlink
-
-if hasattr(os, 'link'):
-    os_link = os.link
-else:
-    def os_link(src, dst):
-        raise OSError(0, _("Hardlinks not supported"))
-
-def fstat(fp):
-    '''stat file object that may not have fileno method.'''
-    try:
-        return os.fstat(fp.fileno())
-    except AttributeError:
-        return os.stat(fp.name)
-
-posixfile = file
-
-def openhardlinks():
-    '''return true if it is safe to hold open file handles to hardlinks'''
-    return True
-
-getuser_fallback = None
-
-def getuser():
-    '''return name of current user'''
-    try:
-        return getpass.getuser()
-    except ImportError:
-        # import of pwd will fail on windows - try fallback
-        if getuser_fallback:
-            return getuser_fallback()
-    # raised if win32api not available
-    raise Abort(_('user name not available - set USERNAME '
-                  'environment variable'))
-
-def username(uid=None):
-    """Return the name of the user with the given uid.
-
-    If uid is None, return the name of the current user."""
-    try:
-        import pwd
-        if uid is None:
-            uid = os.getuid()
-        try:
-            return pwd.getpwuid(uid)[0]
-        except KeyError:
-            return str(uid)
-    except ImportError:
-        return None
-
-def groupname(gid=None):
-    """Return the name of the group with the given gid.
-
-    If gid is None, return the name of the current group."""
-    try:
-        import grp
-        if gid is None:
-            gid = os.getgid()
-        try:
-            return grp.getgrgid(gid)[0]
-        except KeyError:
-            return str(gid)
-    except ImportError:
-        return None
-
-# File system features
-
-def checkfolding(path):
-    """
-    Check whether the given path is on a case-sensitive filesystem
-
-    Requires a path (like /foo/.hg) ending with a foldable final
-    directory component.
-    """
-    s1 = os.stat(path)
-    d, b = os.path.split(path)
-    p2 = os.path.join(d, b.upper())
-    if path == p2:
-        p2 = os.path.join(d, b.lower())
-    try:
-        s2 = os.stat(p2)
-        if s2 == s1:
-            return False
-        return True
-    except:
-        return True
-
-def checkexec(path):
-    """
-    Check whether the given path is on a filesystem with UNIX-like exec flags
-
-    Requires a directory (like /foo/.hg)
-    """
-
-    # VFAT on some Linux versions can flip mode but it doesn't persist
-    # a FS remount. Frequently we can detect it if files are created
-    # with exec bit on.
-
-    try:
-        EXECFLAGS = stat.S_IXUSR | stat.S_IXGRP | stat.S_IXOTH
-        fh, fn = tempfile.mkstemp("", "", path)
-        try:
-            os.close(fh)
-            m = os.stat(fn).st_mode & 0777
-            new_file_has_exec = m & EXECFLAGS
-            os.chmod(fn, m ^ EXECFLAGS)
-            exec_flags_cannot_flip = ((os.stat(fn).st_mode & 0777) == m)
-        finally:
-            os.unlink(fn)
-    except (IOError, OSError):
-        # we don't care, the user probably won't be able to commit anyway
-        return False
-    return not (new_file_has_exec or exec_flags_cannot_flip)
-
-def execfunc(path, fallback):
-    '''return an is_exec() function with default to fallback'''
-    if checkexec(path):
-        return lambda x: is_exec(os.path.join(path, x))
-    return fallback
-
-def checklink(path):
-    """check whether the given path is on a symlink-capable filesystem"""
-    # mktemp is not racy because symlink creation will fail if the
-    # file already exists
-    name = tempfile.mktemp(dir=path)
-    try:
-        os.symlink(".", name)
-        os.unlink(name)
-        return True
-    except (OSError, AttributeError):
-        return False
-
-def linkfunc(path, fallback):
-    '''return an is_link() function with default to fallback'''
-    if checklink(path):
-        return lambda x: os.path.islink(os.path.join(path, x))
-    return fallback
-
-_umask = os.umask(0)
-os.umask(_umask)
-
-def needbinarypatch():
-    """return True if patches should be applied in binary mode by default."""
-    return os.name == 'nt'
-
-# Platform specific variants
-if os.name == 'nt':
-    import msvcrt
-    nulldev = 'NUL:'
-
-    class winstdout:
-        '''stdout on windows misbehaves if sent through a pipe'''
-
-        def __init__(self, fp):
-            self.fp = fp
-
-        def __getattr__(self, key):
-            return getattr(self.fp, key)
-
-        def close(self):
-            try:
-                self.fp.close()
-            except: pass
-
-        def write(self, s):
-            try:
-                # This is workaround for "Not enough space" error on
-                # writing large size of data to console.
-                limit = 16000
-                l = len(s)
-                start = 0
-                while start < l:
-                    end = start + limit
-                    self.fp.write(s[start:end])
-                    start = end
-            except IOError, inst:
-                if inst.errno != 0: raise
-                self.close()
-                raise IOError(errno.EPIPE, 'Broken pipe')
-
-        def flush(self):
-            try:
-                return self.fp.flush()
-            except IOError, inst:
-                if inst.errno != errno.EINVAL: raise
-                self.close()
-                raise IOError(errno.EPIPE, 'Broken pipe')
-
-    sys.stdout = winstdout(sys.stdout)
-
-    def _is_win_9x():
-        '''return true if run on windows 95, 98 or me.'''
-        try:
-            return sys.getwindowsversion()[3] == 1
-        except AttributeError:
-            return 'command' in os.environ.get('comspec', '')
-
-    def openhardlinks():
-        return not _is_win_9x and "win32api" in locals()
-
-    def system_rcpath():
-        try:
-            return system_rcpath_win32()
-        except:
-            return [r'c:\mercurial\mercurial.ini']
-
-    def user_rcpath():
-        '''return os-specific hgrc search path to the user dir'''
-        try:
-            userrc = user_rcpath_win32()
-        except:
-            userrc = os.path.join(os.path.expanduser('~'), 'mercurial.ini')
-        path = [userrc]
-        userprofile = os.environ.get('USERPROFILE')
-        if userprofile:
-            path.append(os.path.join(userprofile, 'mercurial.ini'))
-        return path
-
-    def parse_patch_output(output_line):
-        """parses the output produced by patch and returns the file name"""
-        pf = output_line[14:]
-        if pf[0] == '`':
-            pf = pf[1:-1] # Remove the quotes
-        return pf
-
-    def sshargs(sshcmd, host, user, port):
-        '''Build argument list for ssh or Plink'''
-        pflag = 'plink' in sshcmd.lower() and '-P' or '-p'
-        args = user and ("%s@%s" % (user, host)) or host
-        return port and ("%s %s %s" % (args, pflag, port)) or args
-
-    def testpid(pid):
-        '''return False if pid dead, True if running or not known'''
-        return True
-
-    def set_flags(f, flags):
-        pass
-
-    def set_binary(fd):
-        msvcrt.setmode(fd.fileno(), os.O_BINARY)
-
-    def pconvert(path):
-        return path.replace("\\", "/")
-
-    def localpath(path):
-        return path.replace('/', '\\')
-
-    def normpath(path):
-        return pconvert(os.path.normpath(path))
-
-    makelock = _makelock_file
-    readlock = _readlock_file
-
-    def samestat(s1, s2):
-        return False
-
-    # A sequence of backslashes is special iff it precedes a double quote:
-    # - if there's an even number of backslashes, the double quote is not
-    #   quoted (i.e. it ends the quoted region)
-    # - if there's an odd number of backslashes, the double quote is quoted
-    # - in both cases, every pair of backslashes is unquoted into a single
-    #   backslash
-    # (See http://msdn2.microsoft.com/en-us/library/a1y7w461.aspx )
-    # So, to quote a string, we must surround it in double quotes, double
-    # the number of backslashes that preceed double quotes and add another
-    # backslash before every double quote (being careful with the double
-    # quote we've appended to the end)
-    _quotere = None
-    def shellquote(s):
-        global _quotere
-        if _quotere is None:
-            _quotere = re.compile(r'(\\*)("|\\$)')
-        return '"%s"' % _quotere.sub(r'\1\1\\\2', s)
-
-    def quotecommand(cmd):
-        """Build a command string suitable for os.popen* calls."""
-        # The extra quotes are needed because popen* runs the command
-        # through the current COMSPEC. cmd.exe suppress enclosing quotes.
-        return '"' + cmd + '"'
-
-    def popen(command):
-        # Work around "popen spawned process may not write to stdout
-        # under windows"
-        # http://bugs.python.org/issue1366
-        command += " 2> %s" % nulldev
-        return os.popen(quotecommand(command))
-
-    def explain_exit(code):
-        return _("exited with status %d") % code, code
-
-    # if you change this stub into a real check, please try to implement the
-    # username and groupname functions above, too.
-    def isowner(fp, st=None):
-        return True
-
-    def find_in_path(name, path, default=None):
-        '''find name in search path. path can be string (will be split
-        with os.pathsep), or iterable thing that returns strings.  if name
-        found, return path to name. else return default. name is looked up
-        using cmd.exe rules, using PATHEXT.'''
-        if isinstance(path, str):
-            path = path.split(os.pathsep)
-
-        pathext = os.environ.get('PATHEXT', '.COM;.EXE;.BAT;.CMD')
-        pathext = pathext.lower().split(os.pathsep)
-        isexec = os.path.splitext(name)[1].lower() in pathext
-
-        for p in path:
-            p_name = os.path.join(p, name)
-
-            if isexec and os.path.exists(p_name):
-                return p_name
-
-            for ext in pathext:
-                p_name_ext = p_name + ext
-                if os.path.exists(p_name_ext):
-                    return p_name_ext
-        return default
-
-    def set_signal_handler():
-        try:
-            set_signal_handler_win32()
-        except NameError:
-            pass
-
-    try:
-        # override functions with win32 versions if possible
-        from util_win32 import *
-        if not _is_win_9x():
-            posixfile = posixfile_nt
-    except ImportError:
-        pass
-
-else:
-    nulldev = '/dev/null'
-
-    def rcfiles(path):
-        rcs = [os.path.join(path, 'hgrc')]
-        rcdir = os.path.join(path, 'hgrc.d')
-        try:
-            rcs.extend([os.path.join(rcdir, f)
-                        for f, kind in osutil.listdir(rcdir)
-                        if f.endswith(".rc")])
-        except OSError:
-            pass
-        return rcs
-
-    def system_rcpath():
-        path = []
-        # old mod_python does not set sys.argv
-        if len(getattr(sys, 'argv', [])) > 0:
-            path.extend(rcfiles(os.path.dirname(sys.argv[0]) +
-                                  '/../etc/mercurial'))
-        path.extend(rcfiles('/etc/mercurial'))
-        return path
-
-    def user_rcpath():
-        return [os.path.expanduser('~/.hgrc')]
-
-    def parse_patch_output(output_line):
-        """parses the output produced by patch and returns the file name"""
-        pf = output_line[14:]
-        if os.sys.platform == 'OpenVMS':
-            if pf[0] == '`':
-                pf = pf[1:-1] # Remove the quotes
-        else:
-           if pf.startswith("'") and pf.endswith("'") and " " in pf:
-                pf = pf[1:-1] # Remove the quotes
-        return pf
-
-    def sshargs(sshcmd, host, user, port):
-        '''Build argument list for ssh'''
-        args = user and ("%s@%s" % (user, host)) or host
-        return port and ("%s -p %s" % (args, port)) or args
-
-    def is_exec(f):
-        """check whether a file is executable"""
-        return (os.lstat(f).st_mode & 0100 != 0)
-
-    def set_flags(f, flags):
-        s = os.lstat(f).st_mode
-        x = "x" in flags
-        l = "l" in flags
-        if l:
-            if not stat.S_ISLNK(s):
-                # switch file to link
-                data = file(f).read()
-                os.unlink(f)
-                os.symlink(data, f)
-            # no chmod needed at this point
-            return
-        if stat.S_ISLNK(s):
-            # switch link to file
-            data = os.readlink(f)
-            os.unlink(f)
-            file(f, "w").write(data)
-            s = 0666 & ~_umask # avoid restatting for chmod
-
-        sx = s & 0100
-        if x and not sx:
-            # Turn on +x for every +r bit when making a file executable
-            # and obey umask.
-            os.chmod(f, s | (s & 0444) >> 2 & ~_umask)
-        elif not x and sx:
-            # Turn off all +x bits
-            os.chmod(f, s & 0666)
-
-    def set_binary(fd):
-        pass
-
-    def pconvert(path):
-        return path
-
-    def localpath(path):
-        return path
-
-    normpath = os.path.normpath
-    samestat = os.path.samestat
-
-    def makelock(info, pathname):
-        try:
-            os.symlink(info, pathname)
-        except OSError, why:
-            if why.errno == errno.EEXIST:
-                raise
-            else:
-                _makelock_file(info, pathname)
-
-    def readlock(pathname):
-        try:
-            return os.readlink(pathname)
-        except OSError, why:
-            if why.errno in (errno.EINVAL, errno.ENOSYS):
-                return _readlock_file(pathname)
-            else:
-                raise
-
-    def shellquote(s):
-        if os.sys.platform == 'OpenVMS':
-            return '"%s"' % s
-        else:
-            return "'%s'" % s.replace("'", "'\\''")
-
-    def quotecommand(cmd):
-        return cmd
-
-    def popen(command):
-        return os.popen(command)
-
-    def testpid(pid):
-        '''return False if pid dead, True if running or not sure'''
-        if os.sys.platform == 'OpenVMS':
-            return True
-        try:
-            os.kill(pid, 0)
-            return True
-        except OSError, inst:
-            return inst.errno != errno.ESRCH
-
-    def explain_exit(code):
-        """return a 2-tuple (desc, code) describing a process's status"""
-        if os.WIFEXITED(code):
-            val = os.WEXITSTATUS(code)
-            return _("exited with status %d") % val, val
-        elif os.WIFSIGNALED(code):
-            val = os.WTERMSIG(code)
-            return _("killed by signal %d") % val, val
-        elif os.WIFSTOPPED(code):
-            val = os.WSTOPSIG(code)
-            return _("stopped by signal %d") % val, val
-        raise ValueError(_("invalid exit code"))
-
-    def isowner(fp, st=None):
-        """Return True if the file object f belongs to the current user.
-
-        The return value of a util.fstat(f) may be passed as the st argument.
-        """
-        if st is None:
-            st = fstat(fp)
-        return st.st_uid == os.getuid()
-
-    def find_in_path(name, path, default=None):
-        '''find name in search path. path can be string (will be split
-        with os.pathsep), or iterable thing that returns strings.  if name
-        found, return path to name. else return default.'''
-        if isinstance(path, str):
-            path = path.split(os.pathsep)
-        for p in path:
-            p_name = os.path.join(p, name)
-            if os.path.exists(p_name):
-                return p_name
-        return default
-
-    def set_signal_handler():
-        pass
-
-def find_exe(name, default=None):
-    '''find path of an executable.
-    if name contains a path component, return it as is.  otherwise,
-    use normal executable search path.'''
-
-    if os.sep in name or sys.platform == 'OpenVMS':
-        # don't check the executable bit.  if the file isn't
-        # executable, whoever tries to actually run it will give a
-        # much more useful error message.
-        return name
-    return find_in_path(name, os.environ.get('PATH', ''), default=default)
-
-def _buildencodefun():
-    e = '_'
-    win_reserved = [ord(x) for x in '\\:*?"<>|']
-    cmap = dict([ (chr(x), chr(x)) for x in xrange(127) ])
-    for x in (range(32) + range(126, 256) + win_reserved):
-        cmap[chr(x)] = "~%02x" % x
-    for x in range(ord("A"), ord("Z")+1) + [ord(e)]:
-        cmap[chr(x)] = e + chr(x).lower()
-    dmap = {}
-    for k, v in cmap.iteritems():
-        dmap[v] = k
-    def decode(s):
-        i = 0
-        while i < len(s):
-            for l in xrange(1, 4):
-                try:
-                    yield dmap[s[i:i+l]]
-                    i += l
-                    break
-                except KeyError:
-                    pass
-            else:
-                raise KeyError
-    return (lambda s: "".join([cmap[c] for c in s]),
-            lambda s: "".join(list(decode(s))))
-
-encodefilename, decodefilename = _buildencodefun()
-
-def encodedopener(openerfn, fn):
-    def o(path, *args, **kw):
-        return openerfn(fn(path), *args, **kw)
-    return o
-
-def mktempcopy(name, emptyok=False):
-    """Create a temporary file with the same contents from name
-
-    The permission bits are copied from the original file.
-
-    If the temporary file is going to be truncated immediately, you
-    can use emptyok=True as an optimization.
-
-    Returns the name of the temporary file.
-    """
-    d, fn = os.path.split(name)
-    fd, temp = tempfile.mkstemp(prefix='.%s-' % fn, dir=d)
-    os.close(fd)
-    # Temporary files are created with mode 0600, which is usually not
-    # what we want.  If the original file already exists, just copy
-    # its mode.  Otherwise, manually obey umask.
-    try:
-        st_mode = os.lstat(name).st_mode & 0777
-    except OSError, inst:
-        if inst.errno != errno.ENOENT:
-            raise
-        st_mode = 0666 & ~_umask
-    os.chmod(temp, st_mode)
-    if emptyok:
-        return temp
-    try:
-        try:
-            ifp = posixfile(name, "rb")
-        except IOError, inst:
-            if inst.errno == errno.ENOENT:
-                return temp
-            if not getattr(inst, 'filename', None):
-                inst.filename = name
-            raise
-        ofp = posixfile(temp, "wb")
-        for chunk in filechunkiter(ifp):
-            ofp.write(chunk)
-        ifp.close()
-        ofp.close()
-    except:
-        try: os.unlink(temp)
-        except: pass
-        raise
-    return temp
-
-class atomictempfile(posixfile):
-    """file-like object that atomically updates a file
-
-    All writes will be redirected to a temporary copy of the original
-    file.  When rename is called, the copy is renamed to the original
-    name, making the changes visible.
-    """
-    def __init__(self, name, mode):
-        self.__name = name
-        self.temp = mktempcopy(name, emptyok=('w' in mode))
-        posixfile.__init__(self, self.temp, mode)
-
-    def rename(self):
-        if not self.closed:
-            posixfile.close(self)
-            rename(self.temp, localpath(self.__name))
-
-    def __del__(self):
-        if not self.closed:
-            try:
-                os.unlink(self.temp)
-            except: pass
-            posixfile.close(self)
-
-class opener(object):
-    """Open files relative to a base directory
-
-    This class is used to hide the details of COW semantics and
-    remote file access from higher level code.
-    """
-    def __init__(self, base, audit=True):
-        self.base = base
-        if audit:
-            self.audit_path = path_auditor(base)
-        else:
-            self.audit_path = always
-
-    def __getattr__(self, name):
-        if name == '_can_symlink':
-            self._can_symlink = checklink(self.base)
-            return self._can_symlink
-        raise AttributeError(name)
-
-    def __call__(self, path, mode="r", text=False, atomictemp=False):
-        self.audit_path(path)
-        f = os.path.join(self.base, path)
-
-        if not text and "b" not in mode:
-            mode += "b" # for that other OS
-
-        if mode[0] != "r":
-            try:
-                nlink = nlinks(f)
-            except OSError:
-                nlink = 0
-                d = os.path.dirname(f)
-                if not os.path.isdir(d):
-                    os.makedirs(d)
-            if atomictemp:
-                return atomictempfile(f, mode)
-            if nlink > 1:
-                rename(mktempcopy(f), f)
-        return posixfile(f, mode)
-
-    def symlink(self, src, dst):
-        self.audit_path(dst)
-        linkname = os.path.join(self.base, dst)
-        try:
-            os.unlink(linkname)
-        except OSError:
-            pass
-
-        dirname = os.path.dirname(linkname)
-        if not os.path.exists(dirname):
-            os.makedirs(dirname)
-
-        if self._can_symlink:
-            try:
-                os.symlink(src, linkname)
-            except OSError, err:
-                raise OSError(err.errno, _('could not symlink to %r: %s') %
-                              (src, err.strerror), linkname)
-        else:
-            f = self(dst, "w")
-            f.write(src)
-            f.close()
-
-class chunkbuffer(object):
-    """Allow arbitrary sized chunks of data to be efficiently read from an
-    iterator over chunks of arbitrary size."""
-
-    def __init__(self, in_iter):
-        """in_iter is the iterator that's iterating over the input chunks.
-        targetsize is how big a buffer to try to maintain."""
-        self.iter = iter(in_iter)
-        self.buf = ''
-        self.targetsize = 2**16
-
-    def read(self, l):
-        """Read L bytes of data from the iterator of chunks of data.
-        Returns less than L bytes if the iterator runs dry."""
-        if l > len(self.buf) and self.iter:
-            # Clamp to a multiple of self.targetsize
-            targetsize = max(l, self.targetsize)
-            collector = cStringIO.StringIO()
-            collector.write(self.buf)
-            collected = len(self.buf)
-            for chunk in self.iter:
-                collector.write(chunk)
-                collected += len(chunk)
-                if collected >= targetsize:
-                    break
-            if collected < targetsize:
-                self.iter = False
-            self.buf = collector.getvalue()
-        if len(self.buf) == l:
-            s, self.buf = str(self.buf), ''
-        else:
-            s, self.buf = self.buf[:l], buffer(self.buf, l)
-        return s
-
-def filechunkiter(f, size=65536, limit=None):
-    """Create a generator that produces the data in the file size
-    (default 65536) bytes at a time, up to optional limit (default is
-    to read all data).  Chunks may be less than size bytes if the
-    chunk is the last chunk in the file, or the file is a socket or
-    some other type of file that sometimes reads less data than is
-    requested."""
-    assert size >= 0
-    assert limit is None or limit >= 0
-    while True:
-        if limit is None: nbytes = size
-        else: nbytes = min(limit, size)
-        s = nbytes and f.read(nbytes)
-        if not s: break
-        if limit: limit -= len(s)
-        yield s
-
-def makedate():
-    lt = time.localtime()
-    if lt[8] == 1 and time.daylight:
-        tz = time.altzone
-    else:
-        tz = time.timezone
-    return time.mktime(lt), tz
-
-def datestr(date=None, format='%a %b %d %H:%M:%S %Y', timezone=True, timezone_format=" %+03d%02d"):
-    """represent a (unixtime, offset) tuple as a localized time.
-    unixtime is seconds since the epoch, and offset is the time zone's
-    number of seconds away from UTC. if timezone is false, do not
-    append time zone to string."""
-    t, tz = date or makedate()
-    s = time.strftime(format, time.gmtime(float(t) - tz))
-    if timezone:
-        s += timezone_format % (-tz / 3600, ((-tz % 3600) / 60))
-    return s
-
-def strdate(string, format, defaults=[]):
-    """parse a localized time string and return a (unixtime, offset) tuple.
-    if the string cannot be parsed, ValueError is raised."""
-    def timezone(string):
-        tz = string.split()[-1]
-        if tz[0] in "+-" and len(tz) == 5 and tz[1:].isdigit():
-            tz = int(tz)
-            offset = - 3600 * (tz / 100) - 60 * (tz % 100)
-            return offset
-        if tz == "GMT" or tz == "UTC":
-            return 0
-        return None
-
-    # NOTE: unixtime = localunixtime + offset
-    offset, date = timezone(string), string
-    if offset != None:
-        date = " ".join(string.split()[:-1])
-
-    # add missing elements from defaults
-    for part in defaults:
-        found = [True for p in part if ("%"+p) in format]
-        if not found:
-            date += "@" + defaults[part]
-            format += "@%" + part[0]
-
-    timetuple = time.strptime(date, format)
-    localunixtime = int(calendar.timegm(timetuple))
-    if offset is None:
-        # local timezone
-        unixtime = int(time.mktime(timetuple))
-        offset = unixtime - localunixtime
-    else:
-        unixtime = localunixtime + offset
-    return unixtime, offset
-
-def parsedate(string, formats=None, defaults=None):
-    """parse a localized time string and return a (unixtime, offset) tuple.
-    The date may be a "unixtime offset" string or in one of the specified
-    formats."""
-    if not string:
-        return 0, 0
-    if not formats:
-        formats = defaultdateformats
-    string = string.strip()
-    try:
-        when, offset = map(int, string.split(' '))
-    except ValueError:
-        # fill out defaults
-        if not defaults:
-            defaults = {}
-        now = makedate()
-        for part in "d mb yY HI M S".split():
-            if part not in defaults:
-                if part[0] in "HMS":
-                    defaults[part] = "00"
-                elif part[0] in "dm":
-                    defaults[part] = "1"
-                else:
-                    defaults[part] = datestr(now, "%" + part[0], False)
-
-        for format in formats:
-            try:
-                when, offset = strdate(string, format, defaults)
-            except ValueError:
-                pass
-            else:
-                break
-        else:
-            raise Abort(_('invalid date: %r ') % string)
-    # validate explicit (probably user-specified) date and
-    # time zone offset. values must fit in signed 32 bits for
-    # current 32-bit linux runtimes. timezones go from UTC-12
-    # to UTC+14
-    if abs(when) > 0x7fffffff:
-        raise Abort(_('date exceeds 32 bits: %d') % when)
-    if offset < -50400 or offset > 43200:
-        raise Abort(_('impossible time zone offset: %d') % offset)
-    return when, offset
-
-def matchdate(date):
-    """Return a function that matches a given date match specifier
-
-    Formats include:
-
-    '{date}' match a given date to the accuracy provided
-
-    '<{date}' on or before a given date
-
-    '>{date}' on or after a given date
-
-    """
-
-    def lower(date):
-        return parsedate(date, extendeddateformats)[0]
-
-    def upper(date):
-        d = dict(mb="12", HI="23", M="59", S="59")
-        for days in "31 30 29".split():
-            try:
-                d["d"] = days
-                return parsedate(date, extendeddateformats, d)[0]
-            except:
-                pass
-        d["d"] = "28"
-        return parsedate(date, extendeddateformats, d)[0]
-
-    if date[0] == "<":
-        when = upper(date[1:])
-        return lambda x: x <= when
-    elif date[0] == ">":
-        when = lower(date[1:])
-        return lambda x: x >= when
-    elif date[0] == "-":
-        try:
-            days = int(date[1:])
-        except ValueError:
-            raise Abort(_("invalid day spec: %s") % date[1:])
-        when = makedate()[0] - days * 3600 * 24
-        return lambda x: x >= when
-    elif " to " in date:
-        a, b = date.split(" to ")
-        start, stop = lower(a), upper(b)
-        return lambda x: x >= start and x <= stop
-    else:
-        start, stop = lower(date), upper(date)
-        return lambda x: x >= start and x <= stop
-
-def shortuser(user):
-    """Return a short representation of a user name or email address."""
-    f = user.find('@')
-    if f >= 0:
-        user = user[:f]
-    f = user.find('<')
-    if f >= 0:
-        user = user[f+1:]
-    f = user.find(' ')
-    if f >= 0:
-        user = user[:f]
-    f = user.find('.')
-    if f >= 0:
-        user = user[:f]
-    return user
-
-def ellipsis(text, maxlength=400):
-    """Trim string to at most maxlength (default: 400) characters."""
-    if len(text) <= maxlength:
-        return text
-    else:
-        return "%s..." % (text[:maxlength-3])
-
-def walkrepos(path):
-    '''yield every hg repository under path, recursively.'''
-    def errhandler(err):
-        if err.filename == path:
-            raise err
-
-    for root, dirs, files in os.walk(path, onerror=errhandler):
-        for d in dirs:
-            if d == '.hg':
-                yield root
-                dirs[:] = []
-                break
-
-_rcpath = None
-
-def os_rcpath():
-    '''return default os-specific hgrc search path'''
-    path = system_rcpath()
-    path.extend(user_rcpath())
-    path = [os.path.normpath(f) for f in path]
-    return path
-
-def rcpath():
-    '''return hgrc search path. if env var HGRCPATH is set, use it.
-    for each item in path, if directory, use files ending in .rc,
-    else use item.
-    make HGRCPATH empty to only look in .hg/hgrc of current repo.
-    if no HGRCPATH, use default os-specific path.'''
-    global _rcpath
-    if _rcpath is None:
-        if 'HGRCPATH' in os.environ:
-            _rcpath = []
-            for p in os.environ['HGRCPATH'].split(os.pathsep):
-                if not p: continue
-                if os.path.isdir(p):
-                    for f, kind in osutil.listdir(p):
-                        if f.endswith('.rc'):
-                            _rcpath.append(os.path.join(p, f))
-                else:
-                    _rcpath.append(p)
-        else:
-            _rcpath = os_rcpath()
-    return _rcpath
-
-def bytecount(nbytes):
-    '''return byte count formatted as readable string, with units'''
-
-    units = (
-        (100, 1<<30, _('%.0f GB')),
-        (10, 1<<30, _('%.1f GB')),
-        (1, 1<<30, _('%.2f GB')),
-        (100, 1<<20, _('%.0f MB')),
-        (10, 1<<20, _('%.1f MB')),
-        (1, 1<<20, _('%.2f MB')),
-        (100, 1<<10, _('%.0f KB')),
-        (10, 1<<10, _('%.1f KB')),
-        (1, 1<<10, _('%.2f KB')),
-        (1, 1, _('%.0f bytes')),
-        )
-
-    for multiplier, divisor, format in units:
-        if nbytes >= divisor * multiplier:
-            return format % (nbytes / float(divisor))
-    return units[-1][2] % nbytes
-
-def drop_scheme(scheme, path):
-    sc = scheme + ':'
-    if path.startswith(sc):
-        path = path[len(sc):]
-        if path.startswith('//'):
-            path = path[2:]
-    return path
-
-def uirepr(s):
-    # Avoid double backslash in Windows path repr()
-    return repr(s).replace('\\\\', '\\')
-
-def hidepassword(url):
-    '''hide user credential in a url string'''
-    scheme, netloc, path, params, query, fragment = urlparse.urlparse(url)
-    netloc = re.sub('([^:]*):([^@]*)@(.*)', r'\1:***@\3', netloc)
-    return urlparse.urlunparse((scheme, netloc, path, params, query, fragment))
-
-def removeauth(url):
-    '''remove all authentication information from a url string'''
-    scheme, netloc, path, params, query, fragment = urlparse.urlparse(url)
-    netloc = netloc[netloc.find('@')+1:]
-    return urlparse.urlunparse((scheme, netloc, path, params, query, fragment))
-
---Apple-Mail-27-69865486--
-
-From morozov@cs.duke.edu Sun Jan  6 00:45:57 2008
-Date: Sun, 06 Jan 2008 00:45:57 -0500
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-To: Mirko Friedenhagen <mirko@friedenhagen.de>
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Message-ID: <20080106054557.GA12219@cs.duke.edu>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
-Mime-Version: 1.0
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: inline
-In-Reply-To: <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
-User-Agent: Mutt/1.4.2.2i
-Status: RO
-X-Status: A
-Content-Length: 675
-Lines: 20
-
-Hi, Mirko,
-
-Very strange. I still don't see what the problem is. Try the attached
-version of artemis.py. It's not really a fix: it's the fix of the
-immediate problem, but if you try to use -d flag of ilist, the problem
-will come back. Meanwhile, I'll think about it some more.
-
-Best,
-Dmitriy
-
-On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
->Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
->
->>Hi, Mirko,
->>
->>I'm unable to replicate the problem with mercrial/crew. Can you send
->>me util.py (i.e., mercurial/util.py) from the repository that you are
->>actually using?
->I did not alter anything in it, see the diff between my local branch  
->and default:
-
-From morozov@cs.duke.edu Sun Jan  6 00:46:30 2008
-Date: Sun, 06 Jan 2008 00:46:30 -0500
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-To: Mirko Friedenhagen <mirko@friedenhagen.de>
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Message-ID: <20080106054630.GB12219@cs.duke.edu>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu>
-Mime-Version: 1.0
-Content-Type: multipart/mixed; boundary="oyUTqETQ0mS9luUI"
-Content-Disposition: inline
-In-Reply-To: <20080106054557.GA12219@cs.duke.edu>
-User-Agent: Mutt/1.4.2.2i
-Status: RO
-Content-Length: 9448
-Lines: 306
-
-
---oyUTqETQ0mS9luUI
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: inline
-
-Forgot the attachment. Dmitriy
-
-On Sun, Jan 06, 2008 at 12:45:57AM -0500, Dmitriy Morozov wrote:
->Hi, Mirko,
->
->Very strange. I still don't see what the problem is. Try the attached
->version of artemis.py. It's not really a fix: it's the fix of the
->immediate problem, but if you try to use -d flag of ilist, the problem
->will come back. Meanwhile, I'll think about it some more.
->
->Best,
->Dmitriy
->
->On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
->>Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
->>
->>>Hi, Mirko,
->>>
->>>I'm unable to replicate the problem with mercrial/crew. Can you send
->>>me util.py (i.e., mercurial/util.py) from the repository that you are
->>>actually using?
->>I did not alter anything in it, see the diff between my local branch  
->>and default:
-
---oyUTqETQ0mS9luUI
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: attachment; filename="artemis.py"
-
-# Author: Dmitriy Morozov <hg@foxcub.org>, 2007
-
-"""A very simple and lightweight issue tracker for Mercurial."""
-
-from mercurial import hg, util
-from mercurial.i18n import _
-import os, time, random, mailbox, glob, socket, ConfigParser
-
-
-state = {'new': 'new', 'fixed': 'fixed'}
-state['default'] = state['new']
-issues_dir = ".issues"
-filter_prefix = ".filter"
-date_format = '%a, %d %b %Y %H:%M:%S %Z'
-
-
-def ilist(ui, repo, **opts):
-	"""List issues associated with the project"""
-
-	# Process options
-	show_all = opts['all']
-	properties = []
-	match_date, date_match = False, lambda x: True
-	if opts['date']: 
-		match_date, date_match = True, util.matchdate(opts['date'])
-
-	# Find issues
-	issues_path = os.path.join(repo.root, issues_dir)
-	if not os.path.exists(issues_path): return
-
-	issues = glob.glob(os.path.join(issues_path, '*'))
-	
-	# Process filter
-	if opts['filter']:
-		filters = glob.glob(os.path.join(issues_path, filter_prefix + '*'))
-		config = ConfigParser.SafeConfigParser()
-		config.read(filters)
-		if not config.has_section(opts['filter']): 
-			ui.warning('No filter %s defined\n', opts['filter'])
-		else:
-			properties += config.items(opts['filter'])
-	
-	_get_properties(opts['property'])
-	
-	for issue in issues:
-		mbox = mailbox.mbox(issue)
-		property_match = True
-		for property,value in properties: 
-			property_match = property_match and (mbox[0][property] == value)
-		if not show_all and (not properties or not property_match) and (properties or mbox[0]['State'].upper() == state['fixed'].upper()): continue 
-
-
-		if match_date and not date_match(util.parsedate(mbox[0]['date'], [date_format])[0]): continue
-		ui.write("%s (%d) [%s]: %s\n" % (issue[len(issues_path)+1:], # +1 for trailing /
-										 len(mbox)-1,				 # number of replies (-1 for self)
-										 mbox[0]['State'],
-										 mbox[0]['Subject']))
-	
-
-def iadd(ui, repo, id = None, comment = 0):
-	"""Adds a new issue, or comment to an existing issue ID or its comment COMMENT"""
-	
-	comment = int(comment)
-
-	# First, make sure issues have a directory
-	issues_path = os.path.join(repo.root, issues_dir)
-	if not os.path.exists(issues_path): os.mkdir(issues_path)
-
-	if id:
-		issue_fn, issue_id = _find_issue(ui, repo, id)
-		if not issue_fn: 
-			ui.warn('No such issue\n')
-			return
-	
-	user = ui.username()
-
-	default_issue_text  = 		"From: %s\nDate: %s\n" % (user, time.strftime(date_format))
-	if not id: 
-		default_issue_text += 	"State: %s\n" % state['default']
-	default_issue_text +=		"Subject: brief description\n\n"
-	default_issue_text += 		"Detailed description."
-
-	issue = ui.edit(default_issue_text, user)
-	if issue.strip() == '':
-		ui.warn('Empty issue, ignoring\n')
-		return
-	if issue.strip() == default_issue_text:
-		ui.warn('Unchanged issue text, ignoring\n')
-		return
-
-	# Create the message
-	msg = mailbox.mboxMessage(issue)
-	msg.set_from('artemis', True)
-	
-	# Pick random filename
-	if not id:
-		issue_fn = issues_path
-		while os.path.exists(issue_fn):
-			issue_id = _random_id() 
-			issue_fn = os.path.join(issues_path, issue_id)
-	# else: issue_fn already set
-
-	# Add message to the mailbox
-	mbox = mailbox.mbox(issue_fn)
-	if id and comment not in mbox: 
-		ui.warn('No such comment number in mailbox, commenting on the issue itself\n')
-	if not id:
-		msg.add_header('Message-Id', "<%s-0-artemis@%s>" % (issue_id, socket.gethostname()))
-	else:
-		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (issue_id, _random_id(), socket.gethostname()))
-		msg.add_header('References', mbox[(comment < len(mbox) and comment) or 0]['Message-Id'])
-	mbox.add(msg)
-	mbox.close()
-
-	# If adding issue, add the new mailbox to the repository
-	if not id: 
-		repo.add([issue_fn[(len(repo.root)+1):]])			# +1 for the trailing /
-		ui.status('Added new issue %s\n' % issue_id)
-
-
-def ishow(ui, repo, id, comment = 0, **opts):
-	"""Shows issue ID, or possibly its comment COMMENT"""
-	
-	comment = int(comment)
-	issue, id = _find_issue(ui, repo, id)
-	if not issue: return
-	mbox = mailbox.mbox(issue)
-
-	if opts['all']:
-		ui.write('='*70 + '\n')
-		for i in xrange(len(mbox)):
-			_write_message(ui, mbox[i], i)
-			ui.write('-'*70 + '\n')
-		return
-
-	_show_mbox(ui, mbox, comment)
-
-
-def iupdate(ui, repo, id, **opts):
-	"""Update properties of issue ID"""
-
-	issue, id = _find_issue(ui, repo, id)
-	if not issue: return
-
-	properties = _get_properties(opts['property'])
-	
-	# Read the issue
-	mbox = mailbox.mbox(issue)
-	msg = mbox[0]
-
-	# Fix the properties
-	properties_text = ''
-	for property, value in properties:
-		msg.replace_header(property, value)
-		properties_text += '%s=%s\n' % (property, value)
-	mbox[0] = msg
-
-	# Write down a comment about updated properties
-	if properties and not opts['no_property_comment']:
-		user = ui.username()
-		properties_text  = 	"From: %s\nDate: %s\nSubject: properties changes (%s)\n\n%s" % \
-							(user, time.strftime(date_format),
-							 _pretty_list(list(set([property for property, value in properties]))), 
-							 properties_text)
-		msg = mailbox.mboxMessage(properties_text)
-		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (id, _random_id(), socket.gethostname()))
-		msg.add_header('References', mbox[0]['Message-Id'])
-		msg.set_from('artemis', True)
-		mbox.add(msg)
-	mbox.flush()
-
-	# Show updated message
-	_show_mbox(ui, mbox, 0)
-
-
-def _find_issue(ui, repo, id):
-	issues_path = os.path.join(repo.root, issues_dir)
-	if not os.path.exists(issues_path): return False
-
-	issues = glob.glob(os.path.join(issues_path, id + '*'))
-
-	if len(issues) == 0:
-		return False, 0
-	elif len(issues) > 1:
-		ui.status("Multiple choices:\n")
-		for i in issues: ui.status('  ', i[len(issues_path)+1:], '\n')
-		return False, 0
-	
-	return issues[0], issues[0][len(issues_path)+1:]
-
-def _get_properties(property_list):
-	return [p.split('=') for p in property_list]
-	
-def _write_message(ui, message, index = 0):
-	if index: ui.write("Comment: %d\n" % index)
-	if ui.verbose:
-		ui.write(message.as_string().strip() + '\n')
-	else:
-		if 'From' in message: ui.write('From: %s\n' % message['From'])
-		if 'Date' in message: ui.write('Date: %s\n' % message['Date'])
-		if 'Subject' in message: ui.write('Subject: %s\n' % message['Subject'])
-		if 'State' in message: ui.write('State: %s\n' % message['State'])
-		ui.write('\n' + message.get_payload().strip() + '\n')
-
-def _show_mbox(ui, mbox, comment):
-	# Output the issue (or comment)
-	if comment >= len(mbox): 
-		comment = 0
-		ui.warn('Comment out of range, showing the issue itself\n')
-	msg = mbox[comment]
-	ui.write('='*70 + '\n')
-	if comment:
-		ui.write('Subject: %s\n' % mbox[0]['Subject'])
-		ui.write('State: %s\n' % mbox[0]['State'])
-		ui.write('-'*70 + '\n')
-	_write_message(ui, msg, comment)
-	ui.write('-'*70 + '\n')
-
-	# Read the mailbox into the messages and children dictionaries
-	messages = {}
-	children = {}
-	for i in xrange(len(mbox)):
-		m = mbox[i]
-		messages[m['Message-Id']] = (i,m)
-		children.setdefault(m['References'], []).append(m['Message-Id'])
-	children[None] = []				# Safeguard against infinte loop on empty Message-Id
-
-	# Iterate over children
-	id = msg['Message-Id'] 
-	id_stack = (id in children and map(lambda x: (x, 1), reversed(children[id]))) or []
-	if not id_stack: return
-	ui.write('Comments:\n')
-	while id_stack:
-		id,offset = id_stack.pop()
-		id_stack += (id in children and map(lambda x: (x, offset+1), reversed(children[id]))) or []
-		index, msg = messages[id]
-		ui.write('  '*offset + ('%d: ' % index) + msg['Subject'] + '\n')
-	ui.write('-'*70 + '\n')
-
-def _pretty_list(lst):
-	s = ''
-	for i in lst:
-		s += i + ', '
-	return s[:-2]
-
-def _random_id():
-	return "%x" % random.randint(2**63, 2**64-1)
-
-
-cmdtable = {
-	'ilist':	(ilist, 
-				 [('a', 'all', False, 
-				   'list all issues (by default only those with state new)'),
-				  ('p', 'property', [], 
-				   'list issues with specific field values (e.g., -p state=fixed)'),
-				  ('d', 'date', '', 'restrict to issues matching the date (e.g., -d ">12/28/2007)"'),
-				  ('f', 'filter', '', 'restrict to pre-defined filter (in %s/%s*)' % (issues_dir, filter_prefix))], 
-				 _('hg ilist [OPTIONS]')),
-	'iadd':   	(iadd,  
-				 [], 
-				 _('hg iadd [ID] [COMMENT]')),
-	'ishow':  	(ishow, 
-				 [('a', 'all', None, 'list all comments')], 
-				 _('hg ishow [OPTIONS] ID [COMMENT]')),
-	'iupdate':	(iupdate,
-				 [('p', 'property', [], 
-				   'update properties (e.g., -p state=fixed)'),
-				  ('n', 'no-property-comment', None, 
-				   'do not add a comment about changed properties')],
-				 _('hg iupdate [OPTIONS] ID'))
-}
-
---oyUTqETQ0mS9luUI--
-
-From morozov@cs.duke.edu Sun Jan  6 01:23:03 2008
-Date: Sun, 06 Jan 2008 01:23:03 -0500
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-To: Mirko Friedenhagen <mirko@friedenhagen.de>
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Message-ID: <20080106062303.GA12860@cs.duke.edu>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu>
-Mime-Version: 1.0
-Content-Type: text/plain; charset=koi8-r
-Content-Disposition: inline
-In-Reply-To: <20080106054630.GB12219@cs.duke.edu>
-User-Agent: Mutt/1.4.2.2i
-Status: RO
-Content-Length: 9705
-Lines: 303
-
-Actually, you can just pull the changes from the Artemis repository on
-the Web. I've put them there as well.
-
-Best,
-Dmitriy
-
-On Sun, Jan 06, 2008 at 12:46:30AM -0500, Dmitriy Morozov wrote:
->Forgot the attachment. Dmitriy
->
->On Sun, Jan 06, 2008 at 12:45:57AM -0500, Dmitriy Morozov wrote:
->>Hi, Mirko,
->>
->>Very strange. I still don't see what the problem is. Try the attached
->>version of artemis.py. It's not really a fix: it's the fix of the
->>immediate problem, but if you try to use -d flag of ilist, the problem
->>will come back. Meanwhile, I'll think about it some more.
->>
->>Best,
->>Dmitriy
->>
->>On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
->>>Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
->>>
->>>>Hi, Mirko,
->>>>
->>>>I'm unable to replicate the problem with mercrial/crew. Can you send
->>>>me util.py (i.e., mercurial/util.py) from the repository that you are
->>>>actually using?
->>>I did not alter anything in it, see the diff between my local branch  
->>>and default:
-
-># Author: Dmitriy Morozov <hg@foxcub.org>, 2007
->
->"""A very simple and lightweight issue tracker for Mercurial."""
->
->from mercurial import hg, util
->from mercurial.i18n import _
->import os, time, random, mailbox, glob, socket, ConfigParser
->
->
->state = {'new': 'new', 'fixed': 'fixed'}
->state['default'] = state['new']
->issues_dir = ".issues"
->filter_prefix = ".filter"
->date_format = '%a, %d %b %Y %H:%M:%S %Z'
->
->
->def ilist(ui, repo, **opts):
->	"""List issues associated with the project"""
->
->	# Process options
->	show_all = opts['all']
->	properties = []
->	match_date, date_match = False, lambda x: True
->	if opts['date']: 
->		match_date, date_match = True, util.matchdate(opts['date'])
->
->	# Find issues
->	issues_path = os.path.join(repo.root, issues_dir)
->	if not os.path.exists(issues_path): return
->
->	issues = glob.glob(os.path.join(issues_path, '*'))
->	
->	# Process filter
->	if opts['filter']:
->		filters = glob.glob(os.path.join(issues_path, filter_prefix + '*'))
->		config = ConfigParser.SafeConfigParser()
->		config.read(filters)
->		if not config.has_section(opts['filter']): 
->			ui.warning('No filter %s defined\n', opts['filter'])
->		else:
->			properties += config.items(opts['filter'])
->	
->	_get_properties(opts['property'])
->	
->	for issue in issues:
->		mbox = mailbox.mbox(issue)
->		property_match = True
->		for property,value in properties: 
->			property_match = property_match and (mbox[0][property] == value)
->		if not show_all and (not properties or not property_match) and (properties or mbox[0]['State'].upper() == state['fixed'].upper()): continue 
->
->
->		if match_date and not date_match(util.parsedate(mbox[0]['date'], [date_format])[0]): continue
->		ui.write("%s (%d) [%s]: %s\n" % (issue[len(issues_path)+1:], # +1 for trailing /
->										 len(mbox)-1,				 # number of replies (-1 for self)
->										 mbox[0]['State'],
->										 mbox[0]['Subject']))
->	
->
->def iadd(ui, repo, id = None, comment = 0):
->	"""Adds a new issue, or comment to an existing issue ID or its comment COMMENT"""
->	
->	comment = int(comment)
->
->	# First, make sure issues have a directory
->	issues_path = os.path.join(repo.root, issues_dir)
->	if not os.path.exists(issues_path): os.mkdir(issues_path)
->
->	if id:
->		issue_fn, issue_id = _find_issue(ui, repo, id)
->		if not issue_fn: 
->			ui.warn('No such issue\n')
->			return
->	
->	user = ui.username()
->
->	default_issue_text  = 		"From: %s\nDate: %s\n" % (user, time.strftime(date_format))
->	if not id: 
->		default_issue_text += 	"State: %s\n" % state['default']
->	default_issue_text +=		"Subject: brief description\n\n"
->	default_issue_text += 		"Detailed description."
->
->	issue = ui.edit(default_issue_text, user)
->	if issue.strip() == '':
->		ui.warn('Empty issue, ignoring\n')
->		return
->	if issue.strip() == default_issue_text:
->		ui.warn('Unchanged issue text, ignoring\n')
->		return
->
->	# Create the message
->	msg = mailbox.mboxMessage(issue)
->	msg.set_from('artemis', True)
->	
->	# Pick random filename
->	if not id:
->		issue_fn = issues_path
->		while os.path.exists(issue_fn):
->			issue_id = _random_id() 
->			issue_fn = os.path.join(issues_path, issue_id)
->	# else: issue_fn already set
->
->	# Add message to the mailbox
->	mbox = mailbox.mbox(issue_fn)
->	if id and comment not in mbox: 
->		ui.warn('No such comment number in mailbox, commenting on the issue itself\n')
->	if not id:
->		msg.add_header('Message-Id', "<%s-0-artemis@%s>" % (issue_id, socket.gethostname()))
->	else:
->		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (issue_id, _random_id(), socket.gethostname()))
->		msg.add_header('References', mbox[(comment < len(mbox) and comment) or 0]['Message-Id'])
->	mbox.add(msg)
->	mbox.close()
->
->	# If adding issue, add the new mailbox to the repository
->	if not id: 
->		repo.add([issue_fn[(len(repo.root)+1):]])			# +1 for the trailing /
->		ui.status('Added new issue %s\n' % issue_id)
->
->
->def ishow(ui, repo, id, comment = 0, **opts):
->	"""Shows issue ID, or possibly its comment COMMENT"""
->	
->	comment = int(comment)
->	issue, id = _find_issue(ui, repo, id)
->	if not issue: return
->	mbox = mailbox.mbox(issue)
->
->	if opts['all']:
->		ui.write('='*70 + '\n')
->		for i in xrange(len(mbox)):
->			_write_message(ui, mbox[i], i)
->			ui.write('-'*70 + '\n')
->		return
->
->	_show_mbox(ui, mbox, comment)
->
->
->def iupdate(ui, repo, id, **opts):
->	"""Update properties of issue ID"""
->
->	issue, id = _find_issue(ui, repo, id)
->	if not issue: return
->
->	properties = _get_properties(opts['property'])
->	
->	# Read the issue
->	mbox = mailbox.mbox(issue)
->	msg = mbox[0]
->
->	# Fix the properties
->	properties_text = ''
->	for property, value in properties:
->		msg.replace_header(property, value)
->		properties_text += '%s=%s\n' % (property, value)
->	mbox[0] = msg
->
->	# Write down a comment about updated properties
->	if properties and not opts['no_property_comment']:
->		user = ui.username()
->		properties_text  = 	"From: %s\nDate: %s\nSubject: properties changes (%s)\n\n%s" % \
->							(user, time.strftime(date_format),
->							 _pretty_list(list(set([property for property, value in properties]))), 
->							 properties_text)
->		msg = mailbox.mboxMessage(properties_text)
->		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (id, _random_id(), socket.gethostname()))
->		msg.add_header('References', mbox[0]['Message-Id'])
->		msg.set_from('artemis', True)
->		mbox.add(msg)
->	mbox.flush()
->
->	# Show updated message
->	_show_mbox(ui, mbox, 0)
->
->
->def _find_issue(ui, repo, id):
->	issues_path = os.path.join(repo.root, issues_dir)
->	if not os.path.exists(issues_path): return False
->
->	issues = glob.glob(os.path.join(issues_path, id + '*'))
->
->	if len(issues) == 0:
->		return False, 0
->	elif len(issues) > 1:
->		ui.status("Multiple choices:\n")
->		for i in issues: ui.status('  ', i[len(issues_path)+1:], '\n')
->		return False, 0
->	
->	return issues[0], issues[0][len(issues_path)+1:]
->
->def _get_properties(property_list):
->	return [p.split('=') for p in property_list]
->	
->def _write_message(ui, message, index = 0):
->	if index: ui.write("Comment: %d\n" % index)
->	if ui.verbose:
->		ui.write(message.as_string().strip() + '\n')
->	else:
->		if 'From' in message: ui.write('From: %s\n' % message['From'])
->		if 'Date' in message: ui.write('Date: %s\n' % message['Date'])
->		if 'Subject' in message: ui.write('Subject: %s\n' % message['Subject'])
->		if 'State' in message: ui.write('State: %s\n' % message['State'])
->		ui.write('\n' + message.get_payload().strip() + '\n')
->
->def _show_mbox(ui, mbox, comment):
->	# Output the issue (or comment)
->	if comment >= len(mbox): 
->		comment = 0
->		ui.warn('Comment out of range, showing the issue itself\n')
->	msg = mbox[comment]
->	ui.write('='*70 + '\n')
->	if comment:
->		ui.write('Subject: %s\n' % mbox[0]['Subject'])
->		ui.write('State: %s\n' % mbox[0]['State'])
->		ui.write('-'*70 + '\n')
->	_write_message(ui, msg, comment)
->	ui.write('-'*70 + '\n')
->
->	# Read the mailbox into the messages and children dictionaries
->	messages = {}
->	children = {}
->	for i in xrange(len(mbox)):
->		m = mbox[i]
->		messages[m['Message-Id']] = (i,m)
->		children.setdefault(m['References'], []).append(m['Message-Id'])
->	children[None] = []				# Safeguard against infinte loop on empty Message-Id
->
->	# Iterate over children
->	id = msg['Message-Id'] 
->	id_stack = (id in children and map(lambda x: (x, 1), reversed(children[id]))) or []
->	if not id_stack: return
->	ui.write('Comments:\n')
->	while id_stack:
->		id,offset = id_stack.pop()
->		id_stack += (id in children and map(lambda x: (x, offset+1), reversed(children[id]))) or []
->		index, msg = messages[id]
->		ui.write('  '*offset + ('%d: ' % index) + msg['Subject'] + '\n')
->	ui.write('-'*70 + '\n')
->
->def _pretty_list(lst):
->	s = ''
->	for i in lst:
->		s += i + ', '
->	return s[:-2]
->
->def _random_id():
->	return "%x" % random.randint(2**63, 2**64-1)
->
->
->cmdtable = {
->	'ilist':	(ilist, 
->				 [('a', 'all', False, 
->				   'list all issues (by default only those with state new)'),
->				  ('p', 'property', [], 
->				   'list issues with specific field values (e.g., -p state=fixed)'),
->				  ('d', 'date', '', 'restrict to issues matching the date (e.g., -d ">12/28/2007)"'),
->				  ('f', 'filter', '', 'restrict to pre-defined filter (in %s/%s*)' % (issues_dir, filter_prefix))], 
->				 _('hg ilist [OPTIONS]')),
->	'iadd':   	(iadd,  
->				 [], 
->				 _('hg iadd [ID] [COMMENT]')),
->	'ishow':  	(ishow, 
->				 [('a', 'all', None, 'list all comments')], 
->				 _('hg ishow [OPTIONS] ID [COMMENT]')),
->	'iupdate':	(iupdate,
->				 [('p', 'property', [], 
->				   'update properties (e.g., -p state=fixed)'),
->				  ('n', 'no-property-comment', None, 
->				   'do not add a comment about changed properties')],
->				 _('hg iupdate [OPTIONS] ID'))
->}
-
-
-From mirko@friedenhagen.de  Mon Jan  7 14:48:39 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-1.8 required=5.0 tests=AWL,BAYES_00 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.174])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07Jmcp6010818
-	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 14:48:38 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu4) with ESMTP (Nemesis)
-	id 0ML21M-1JBxxU3FPf-0006GX; Mon, 07 Jan 2008 20:48:32 +0100
-Message-Id: <BCEC116C-7113-4873-B098-05ACDE943A72@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Mon, 07 Jan 2008 20:48:28 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX180vJsLizprsH5HFEaivi29mMVU4KUVP4sc9S/
- 8gvUFq6zpfnOswNDLfqqQUEOo71JI9E82xPOCeBWYO6HsENm3e
- aYk5cubORJodMGX7f+ung==
-Status: RO
-Content-Length: 417
-Lines: 16
-
-Am 06.01.2008 um 07:23 schrieb Dmitriy Morozov:
-> Actually, you can just pull the changes from the Artemis repository on
-> the Web. I've put them there as well.
->
-
-Thanks, I just pulled to e78a97664dba and ilist is now working without  
-option -d.
-I just checked wether this was a "LANG"-related problem but the error  
-still occurs with LANGs set to:
-de_DE.UTF-8
-C
-en_US
-so it must be something else.
-
-Regards
-Mirko
-
-From mirko@friedenhagen.de  Mon Jan  7 15:51:53 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-2.0 required=5.0 tests=AWL,BAYES_00 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07KpqWA020073
-	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 15:51:52 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu7) with ESMTP (Nemesis)
-	id 0ML2xA-1JBywk2eIR-0000fy; Mon, 07 Jan 2008 21:51:46 +0100
-Message-Id: <1ACD99E9-9100-4C7D-916C-F3516EC9CF66@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Mon, 07 Jan 2008 21:51:45 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX18QJcYZDo6Q37uJfFynUsBGK2GGByd94i/JhM6
- I9MfCwpXS9DUnY0O0ZIhHan+nj+Gx7xd+r8zc31Da+yFftsUtL
- FD+bmLhzp9S5QOyPPuamA==
-Status: RO
-Content-Length: 265
-Lines: 12
-
-Hello Dmitry,
-
-I was digging a bit:
-http://www.faqs.org/rfcs/rfc2822.html
-states a numeric timezone must be used (see "A.6.2. Obsolete dates",  
-section 4.3. and 3.3.).
-
-Though I do not understand, why the other dates are parsed without a  
-problem.
-
-Regards
-Mirko
-
-From mirko@friedenhagen.de  Mon Jan  7 16:27:35 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-2.1 required=5.0 tests=AWL,BAYES_00 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07LRYaF024588
-	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 16:27:34 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu0) with ESMTP (Nemesis)
-	id 0MKwh2-1JBzVI0J5o-00061N; Mon, 07 Jan 2008 22:27:28 +0100
-Message-Id: <13CC4175-53F2-4ED1-94E7-075922A02DF7@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Mon, 07 Jan 2008 22:27:28 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX1+U2upB6j5zBd5UJpAi32pjepkqEt+f2EUVXLk
- 4xDmYE1gswRWEpvxtO62gW/Yu3xHjgzpqm+BfgIVmuRIVXzof7
- YHAIOIdRdjxlxATKGndHQ==
-Status: RO
-Content-Length: 357
-Lines: 12
-
-I propably found the error:
-
-According to /usr/lib/python2.5/_strptime.py, '%Z' is restricted to:
-[mirko@borg mercurial-crew]$ python -c 'import _strptime; print  
-_strptime.TimeRE()["Z"]'
-(?P<Z>cest|utc|cet|gmt)
-
-This is propably only true for my local timezone as is EST for  
-yours :-). Switching to numeric timezones should do the trick.
-
-Regards
-Mirko
-
-From mirko@friedenhagen.de  Mon Jan  7 16:31:41 2008
-Return-Path: <mirko@friedenhagen.de>
-X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
-X-Spam-Level: 
-X-Spam-Status: No, score=-2.2 required=5.0 tests=AWL,BAYES_00 autolearn=ham
-	version=3.2.2
-Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
-	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07LVeM0025030
-	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 16:31:41 -0500 (EST)
-Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
-	by mrelayeu.kundenserver.de (node=mrelayeu3) with ESMTP (Nemesis)
-	id 0MKxQS-1JBzZH0fuC-0005IG; Mon, 07 Jan 2008 22:31:35 +0100
-Message-Id: <2DBEA693-A27E-40AC-8823-B1A34B9A9A15@friedenhagen.de>
-From: Mirko Friedenhagen <mirko@friedenhagen.de>
-To: Dmitriy Morozov <morozov@cs.duke.edu>
-In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
-Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
-Content-Transfer-Encoding: 7bit
-Mime-Version: 1.0 (Apple Message framework v915)
-Subject: Re: hg ishow fails in clone of Artemis-repo
-Date: Mon, 07 Jan 2008 22:31:33 +0100
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
-X-Mailer: Apple Mail (2.915)
-X-Provags-ID: V01U2FsdGVkX19o42Jm2/riqN1HqzwFbS0PWATW9585MzhlPx4
- nqNci/NQIPJL26EJ9q3ok9stQ9Ydm/zHoAGeVyF7JabTf7Ex5I
- kv+DcoQ6Zh8mmw9XDGSOw==
-Status: RO
-Content-Length: 496
-Lines: 13
-
-Last one:
-
-[mirko@borg mercurial-crew]$ grep -A3 Z /usr/lib/python2.5/_strptime.py
-         replacement_pairs.extend([(tz, "%Z") for tz_values in  
-self.timezone
-                                                 for tz in tz_values])
-         for offset,directive in ((0,'%c'), (1,'%x'), (2,'%X')):
-             current_format = date_time[offset]
---
-             'Z': self.__seqToRE((tz for tz_names in  
-self.locale_time.timezone
-                                         for tz in tz_names),
-...
-
-
-From artemis Tue Jan  8 18:28:21 2008
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Tue, 08 Jan 2008 13:28:21 -0500
-Subject: properties changes (state)
-Message-Id: <c568146d3275c22b-8f211e40531795e1-artemis@metatron>
-References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
-In-Reply-To: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
-
-state=fixed
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M298419P23014Q6.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,64 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-0.2 required=5.0 tests=BAYES_40 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m04NVf7J019792
+	for <morozov@cs.duke.edu>; Fri, 4 Jan 2008 18:31:43 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu2) with ESMTP (Nemesis)
+	id 0MKwtQ-1JAw0l1aId-0006EP; Sat, 05 Jan 2008 00:31:35 +0100
+Message-Id: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: hg ishow fails in clone of Artemis-repo
+Date: Sat, 05 Jan 2008 00:31:27 +0100
+State: fixed
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX1/JysvjyYiL3YHLT6W7yfO6FZ1JMB6PepAft/b
+ jnXeHpRbt2ZogH7u/u+9jEuj7D5SulS51Scfvx4LgDgb4pctZs
+ sbaVxmR5TQar5n7qj9MGA==
+Status: RO
+X-Status: A
+Content-Length: 1180
+Lines: 35
+
+Hello Dmitriy,
+
+first of all let me thank you for this extension, I have the feeling  
+that it might be really useful.
+
+I have one issue with the clone, "hg ilist" will abort, "hg ishow"  
+does not have any problems:
+
+[mirko@borg Artemis]$ hg log -l1
+changeset:   9:5319c712fa34
+tag:         tip
+user:        Dmitriy Morozov <morozov@cs.duke.edu>
+date:        Sun Dec 30 09:23:23 2007 -0500
+summary:     Fixed Message-Ids of our bugs (wrapped them in <...>)
+
+[mirko@borg Artemis]$ hg ilist
+abort: invalid date: 'Sat, 29 Dec 2007 02:50:26 EST'
+
+[mirko@borg Artemis]$ grep -r 'Sat, 29 Dec 2007 02:50:26 EST' .issues/
+.issues/95536ae767c2743a:Date: Sat, 29 Dec 2007 02:50:26 EST
+
+[mirko@borg Artemis]$ hg ishow 95536ae767c2743a
+======================================================================
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sat, 29 Dec 2007 02:50:26 EST
+Subject: attachments in iadd
+State: new
+
+Add ability to attach files in iadd command. Perhaps use -a flag for
+it (allowing multiple attachments). Store as a multipart message. List
+all parts of messages in ishow.
+----------------------------------------------------------------------
+
+Best Regards
+Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M301451P23014Q7.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,60 @@
+Date: Fri, 04 Jan 2008 18:34:36 -0500
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+To: Mirko Friedenhagen <mirko@friedenhagen.de>
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Message-ID: <20080104233436.GA1930@cs.duke.edu>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
+Mime-Version: 1.0
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: inline
+In-Reply-To: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
+User-Agent: Mutt/1.4.2.2i
+Status: RO
+Content-Length: 1475
+Lines: 45
+
+Hi, Mirko,
+
+What version of Python, and more importantly what version of Mercurial
+do you have? I have an idea of where the problem is, but I need the
+version numbers first.
+
+Thanks.
+Dmitriy
+
+On Sat, Jan 05, 2008 at 12:31:27AM +0100, Mirko Friedenhagen wrote:
+>Hello Dmitriy,
+>
+>first of all let me thank you for this extension, I have the feeling  
+>that it might be really useful.
+>
+>I have one issue with the clone, "hg ilist" will abort, "hg ishow"  
+>does not have any problems:
+>
+>[mirko@borg Artemis]$ hg log -l1
+>changeset:   9:5319c712fa34
+>tag:         tip
+>user:        Dmitriy Morozov <morozov@cs.duke.edu>
+>date:        Sun Dec 30 09:23:23 2007 -0500
+>summary:     Fixed Message-Ids of our bugs (wrapped them in <...>)
+>
+>[mirko@borg Artemis]$ hg ilist
+>abort: invalid date: 'Sat, 29 Dec 2007 02:50:26 EST'
+>
+>[mirko@borg Artemis]$ grep -r 'Sat, 29 Dec 2007 02:50:26 EST' .issues/
+>.issues/95536ae767c2743a:Date: Sat, 29 Dec 2007 02:50:26 EST
+>
+>[mirko@borg Artemis]$ hg ishow 95536ae767c2743a
+>======================================================================
+>From: Dmitriy Morozov <morozov@cs.duke.edu>
+>Date: Sat, 29 Dec 2007 02:50:26 EST
+>Subject: attachments in iadd
+>State: new
+>
+>Add ability to attach files in iadd command. Perhaps use -a flag for
+>it (allowing multiple attachments). Store as a multipart message. List
+>all parts of messages in ishow.
+>----------------------------------------------------------------------
+>
+>Best Regards
+>Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M303416P23014Q8.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,70 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-1.4 required=5.0 tests=AWL,BAYES_00 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.187])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m04NhTah021156
+	for <morozov@cs.duke.edu>; Fri, 4 Jan 2008 18:43:29 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu2) with ESMTP (Nemesis)
+	id 0MKwtQ-1JAwCB2mxq-0006FP; Sat, 05 Jan 2008 00:43:23 +0100
+Message-Id: <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080104233436.GA1930@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Sat, 05 Jan 2008 00:43:23 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX1/lGFmR/n0Vn+KixPR/52KDdWD8RANfnQ68xtl
+ cTd+HJmbaC+4c1t/CI5zg5HmPAdrPm3n/0v4CF33rUEBuEJFrC
+ NXyP7mzJQBuNykwAF7tNw==
+Status: RO
+X-Status: A
+Content-Length: 1216
+Lines: 40
+
+Am 05.01.2008 um 00:34 schrieb Dmitriy Morozov:
+
+> Hi, Mirko,
+>
+> What version of Python, and more importantly what version of Mercurial
+> do you have? I have an idea of where the problem is, but I need the
+> version numbers first.
+>
+
+Hello Dmitriy,
+
+now, that's a quick answer ;-).
+
+I am following mercurial-crew (the second parent are minor adjustments  
+(git-summary-template, where I include an Link to the hgweb-root)),  
+but no changes to core.
+
+[mirko@borg mercurial-crew]$ hg log -l2
+changeset:   5824:1048a5ac25ef
+branch:      local
+tag:         tip
+parent:      5807:0176986571fe
+parent:      5823:d852151fb8d4
+user:        Mirko Friedenhagen <mirko.friedenhagen@1und1.de>
+date:        Fri Jan 04 23:22:08 2008 +0100
+summary:     Automated merge with http://hg.intevation.org/mercurial/crew
+
+changeset:   5823:d852151fb8d4
+user:        Bryan O'Sullivan <bos@serpentine.com>
+date:        Fri Jan 04 13:56:31 2008 -0800
+summary:     util: drop params added during experimentation
+
+[mirko@borg mercurial-crew]$ python
+Python 2.5.1 (r251:54863, Oct  5 2007, 21:08:09)
+[GCC 4.0.1 (Apple Inc. build 5465)] on darwin
+Type "help", "copyright", "credits" or "license" for more information.
+ >>>
+
+Best Regards
+Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M305572P23014Q9.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,65 @@
+Date: Sat, 05 Jan 2008 07:47:08 -0500
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+To: Mirko Friedenhagen <mirko@friedenhagen.de>
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Message-ID: <20080105124708.GA7043@cs.duke.edu>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
+Mime-Version: 1.0
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: inline
+In-Reply-To: <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de>
+User-Agent: Mutt/1.4.2.2i
+Status: RO
+Content-Length: 1504
+Lines: 50
+
+Hi, Mirko,
+
+I'm unable to replicate the problem with mercrial/crew. Can you send
+me util.py (i.e., mercurial/util.py) from the repository that you are
+actually using?
+
+Thanks.
+Dmitriy
+
+On Sat, Jan 05, 2008 at 12:43:23AM +0100, Mirko Friedenhagen wrote:
+>Am 05.01.2008 um 00:34 schrieb Dmitriy Morozov:
+>
+>>Hi, Mirko,
+>>
+>>What version of Python, and more importantly what version of Mercurial
+>>do you have? I have an idea of where the problem is, but I need the
+>>version numbers first.
+>>
+>
+>Hello Dmitriy,
+>
+>now, that's a quick answer ;-).
+>
+>I am following mercurial-crew (the second parent are minor adjustments  
+>(git-summary-template, where I include an Link to the hgweb-root)),  
+>but no changes to core.
+>
+>[mirko@borg mercurial-crew]$ hg log -l2
+>changeset:   5824:1048a5ac25ef
+>branch:      local
+>tag:         tip
+>parent:      5807:0176986571fe
+>parent:      5823:d852151fb8d4
+>user:        Mirko Friedenhagen <mirko.friedenhagen@1und1.de>
+>date:        Fri Jan 04 23:22:08 2008 +0100
+>summary:     Automated merge with http://hg.intevation.org/mercurial/crew
+>
+>changeset:   5823:d852151fb8d4
+>user:        Bryan O'Sullivan <bos@serpentine.com>
+>date:        Fri Jan 04 13:56:31 2008 -0800
+>summary:     util: drop params added during experimentation
+>
+>[mirko@borg mercurial-crew]$ python
+>Python 2.5.1 (r251:54863, Oct  5 2007, 21:08:09)
+>[GCC 4.0.1 (Apple Inc. build 5465)] on darwin
+>Type "help", "copyright", "credits" or "license" for more information.
+>>>>
+>
+>Best Regards
+>Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M308451P23014Q10.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,1811 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-0.7 required=5.0 tests=AWL,BAYES_50 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.188])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m05LI8r8027688
+	for <morozov@cs.duke.edu>; Sat, 5 Jan 2008 16:18:08 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu7) with ESMTP (Nemesis)
+	id 0ML2xA-1JBGP22KSu-0000u0; Sat, 05 Jan 2008 22:18:02 +0100
+Message-Id: <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080105124708.GA7043@cs.duke.edu>
+Content-Type: multipart/mixed; boundary=Apple-Mail-27-69865486
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Sat, 05 Jan 2008 22:18:01 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX18U2SPmtxdIc/97m419K3u6bNHAGoJX7Zg8bRM
+ nhGQpakEwacktoI+iKEBCOeJWa/w25yyJDtlt4rvwcfLwDf3SB
+ BTBFcyJEMtRpthNwJH//w==
+Status: RO
+X-Status: A
+Content-Length: 57277
+Lines: 1782
+
+
+--Apple-Mail-27-69865486
+Content-Type: text/plain;
+	charset=US-ASCII;
+	format=flowed;
+	delsp=yes
+Content-Transfer-Encoding: 7bit
+
+Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
+
+> Hi, Mirko,
+>
+> I'm unable to replicate the problem with mercrial/crew. Can you send
+> me util.py (i.e., mercurial/util.py) from the repository that you are
+> actually using?
+I did not alter anything in it, see the diff between my local branch  
+and default:
+
+[mirko@borg mercurial-crew]$ hg diff -rdefault
+diff -r d852151fb8d4 hgext/convert/cvs.py
+--- a/hgext/convert/cvs.py	Fri Jan 04 13:56:31 2008 -0800
++++ b/hgext/convert/cvs.py	Sat Jan 05 22:16:04 2008 +0100
+@@ -255,6 +255,12 @@ class convert_cvs(converter_source):
+                      return (data, "x" in mode and "x" or "")
+                  elif line.startswith("E "):
+                      self.ui.warn("cvs server: %s\n" % line[2:])
++                elif line.endswith("- ignored\n"):
++                    self.ui.warn("cvs server: %s\n" % line)
++                    self.readp.readline()
++                    l = self.readp.readline()
++                    if l != "error\n":
++                        raise util.Abort("unknown CVS response: %s" %  
+l)
+                  elif line.startswith("Remove"):
+                      l = self.readp.readline()
+                      l = self.readp.readline()
+diff -r d852151fb8d4 templates/gitweb/summary.tmpl
+--- a/templates/gitweb/summary.tmpl	Fri Jan 04 13:56:31 2008 -0800
++++ b/templates/gitweb/summary.tmpl	Sat Jan 05 22:16:05 2008 +0100
+@@ -7,6 +7,7 @@
+  </head>
+  <body>
+
++<a href="/">^ Repositories List</a>
+  <div class="page_header">
+  <a href="http://www.selenic.com/mercurial/" title="Mercurial"><div  
+style="float:right;">Mercurial</div></a><a  
+href="{url}summary{sessionvars%urlparameter}">#repo|escape#</a> /  
+summary
+
+
+
+--Apple-Mail-27-69865486
+Content-Disposition: attachment;
+	filename=util.py
+Content-Type: text/x-python-script;
+	x-unix-mode=0644;
+	name="util.py"
+Content-Transfer-Encoding: 7bit
+
+"""
+util.py - Mercurial utility functions and platform specfic implementations
+
+ Copyright 2005 K. Thananchayan <thananck@yahoo.com>
+ Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
+ Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com>
+
+This software may be used and distributed according to the terms
+of the GNU General Public License, incorporated herein by reference.
+
+This contains helper routines that are independent of the SCM core and hide
+platform-specific details from the core.
+"""
+
+from i18n import _
+import cStringIO, errno, getpass, popen2, re, shutil, sys, tempfile, strutil
+import os, stat, threading, time, calendar, ConfigParser, locale, glob, osutil
+import re, urlparse
+
+try:
+    set = set
+    frozenset = frozenset
+except NameError:
+    from sets import Set as set, ImmutableSet as frozenset
+
+try:
+    _encoding = os.environ.get("HGENCODING")
+    if sys.platform == 'darwin' and not _encoding:
+        # On darwin, getpreferredencoding ignores the locale environment and
+        # always returns mac-roman. We override this if the environment is
+        # not C (has been customized by the user).
+        locale.setlocale(locale.LC_CTYPE, '')
+        _encoding = locale.getlocale()[1]
+    if not _encoding:
+        _encoding = locale.getpreferredencoding() or 'ascii'
+except locale.Error:
+    _encoding = 'ascii'
+_encodingmode = os.environ.get("HGENCODINGMODE", "strict")
+_fallbackencoding = 'ISO-8859-1'
+
+def tolocal(s):
+    """
+    Convert a string from internal UTF-8 to local encoding
+
+    All internal strings should be UTF-8 but some repos before the
+    implementation of locale support may contain latin1 or possibly
+    other character sets. We attempt to decode everything strictly
+    using UTF-8, then Latin-1, and failing that, we use UTF-8 and
+    replace unknown characters.
+    """
+    for e in ('UTF-8', _fallbackencoding):
+        try:
+            u = s.decode(e) # attempt strict decoding
+            return u.encode(_encoding, "replace")
+        except LookupError, k:
+            raise Abort(_("%s, please check your locale settings") % k)
+        except UnicodeDecodeError:
+            pass
+    u = s.decode("utf-8", "replace") # last ditch
+    return u.encode(_encoding, "replace")
+
+def fromlocal(s):
+    """
+    Convert a string from the local character encoding to UTF-8
+
+    We attempt to decode strings using the encoding mode set by
+    HGENCODINGMODE, which defaults to 'strict'. In this mode, unknown
+    characters will cause an error message. Other modes include
+    'replace', which replaces unknown characters with a special
+    Unicode character, and 'ignore', which drops the character.
+    """
+    try:
+        return s.decode(_encoding, _encodingmode).encode("utf-8")
+    except UnicodeDecodeError, inst:
+        sub = s[max(0, inst.start-10):inst.start+10]
+        raise Abort("decoding near '%s': %s!" % (sub, inst))
+    except LookupError, k:
+        raise Abort(_("%s, please check your locale settings") % k)
+
+def locallen(s):
+    """Find the length in characters of a local string"""
+    return len(s.decode(_encoding, "replace"))
+
+# used by parsedate
+defaultdateformats = (
+    '%Y-%m-%d %H:%M:%S',
+    '%Y-%m-%d %I:%M:%S%p',
+    '%Y-%m-%d %H:%M',
+    '%Y-%m-%d %I:%M%p',
+    '%Y-%m-%d',
+    '%m-%d',
+    '%m/%d',
+    '%m/%d/%y',
+    '%m/%d/%Y',
+    '%a %b %d %H:%M:%S %Y',
+    '%a %b %d %I:%M:%S%p %Y',
+    '%a, %d %b %Y %H:%M:%S',        #  GNU coreutils "/bin/date --rfc-2822"
+    '%b %d %H:%M:%S %Y',
+    '%b %d %I:%M:%S%p %Y',
+    '%b %d %H:%M:%S',
+    '%b %d %I:%M:%S%p',
+    '%b %d %H:%M',
+    '%b %d %I:%M%p',
+    '%b %d %Y',
+    '%b %d',
+    '%H:%M:%S',
+    '%I:%M:%SP',
+    '%H:%M',
+    '%I:%M%p',
+)
+
+extendeddateformats = defaultdateformats + (
+    "%Y",
+    "%Y-%m",
+    "%b",
+    "%b %Y",
+    )
+
+class SignalInterrupt(Exception):
+    """Exception raised on SIGTERM and SIGHUP."""
+
+# differences from SafeConfigParser:
+# - case-sensitive keys
+# - allows values that are not strings (this means that you may not
+#   be able to save the configuration to a file)
+class configparser(ConfigParser.SafeConfigParser):
+    def optionxform(self, optionstr):
+        return optionstr
+
+    def set(self, section, option, value):
+        return ConfigParser.ConfigParser.set(self, section, option, value)
+
+    def _interpolate(self, section, option, rawval, vars):
+        if not isinstance(rawval, basestring):
+            return rawval
+        return ConfigParser.SafeConfigParser._interpolate(self, section,
+                                                          option, rawval, vars)
+
+def cachefunc(func):
+    '''cache the result of function calls'''
+    # XXX doesn't handle keywords args
+    cache = {}
+    if func.func_code.co_argcount == 1:
+        # we gain a small amount of time because
+        # we don't need to pack/unpack the list
+        def f(arg):
+            if arg not in cache:
+                cache[arg] = func(arg)
+            return cache[arg]
+    else:
+        def f(*args):
+            if args not in cache:
+                cache[args] = func(*args)
+            return cache[args]
+
+    return f
+
+def pipefilter(s, cmd):
+    '''filter string S through command CMD, returning its output'''
+    (pin, pout) = os.popen2(cmd, 'b')
+    def writer():
+        try:
+            pin.write(s)
+            pin.close()
+        except IOError, inst:
+            if inst.errno != errno.EPIPE:
+                raise
+
+    # we should use select instead on UNIX, but this will work on most
+    # systems, including Windows
+    w = threading.Thread(target=writer)
+    w.start()
+    f = pout.read()
+    pout.close()
+    w.join()
+    return f
+
+def tempfilter(s, cmd):
+    '''filter string S through a pair of temporary files with CMD.
+    CMD is used as a template to create the real command to be run,
+    with the strings INFILE and OUTFILE replaced by the real names of
+    the temporary files generated.'''
+    inname, outname = None, None
+    try:
+        infd, inname = tempfile.mkstemp(prefix='hg-filter-in-')
+        fp = os.fdopen(infd, 'wb')
+        fp.write(s)
+        fp.close()
+        outfd, outname = tempfile.mkstemp(prefix='hg-filter-out-')
+        os.close(outfd)
+        cmd = cmd.replace('INFILE', inname)
+        cmd = cmd.replace('OUTFILE', outname)
+        code = os.system(cmd)
+        if sys.platform == 'OpenVMS' and code & 1:
+            code = 0
+        if code: raise Abort(_("command '%s' failed: %s") %
+                             (cmd, explain_exit(code)))
+        return open(outname, 'rb').read()
+    finally:
+        try:
+            if inname: os.unlink(inname)
+        except: pass
+        try:
+            if outname: os.unlink(outname)
+        except: pass
+
+filtertable = {
+    'tempfile:': tempfilter,
+    'pipe:': pipefilter,
+    }
+
+def filter(s, cmd):
+    "filter a string through a command that transforms its input to its output"
+    for name, fn in filtertable.iteritems():
+        if cmd.startswith(name):
+            return fn(s, cmd[len(name):].lstrip())
+    return pipefilter(s, cmd)
+
+def binary(s):
+    """return true if a string is binary data using diff's heuristic"""
+    if s and '\0' in s[:4096]:
+        return True
+    return False
+
+def unique(g):
+    """return the uniq elements of iterable g"""
+    return dict.fromkeys(g).keys()
+
+class Abort(Exception):
+    """Raised if a command needs to print an error and exit."""
+
+class UnexpectedOutput(Abort):
+    """Raised to print an error with part of output and exit."""
+
+def always(fn): return True
+def never(fn): return False
+
+def expand_glob(pats):
+    '''On Windows, expand the implicit globs in a list of patterns'''
+    if os.name != 'nt':
+        return list(pats)
+    ret = []
+    for p in pats:
+        kind, name = patkind(p, None)
+        if kind is None:
+            globbed = glob.glob(name)
+            if globbed:
+                ret.extend(globbed)
+                continue
+            # if we couldn't expand the glob, just keep it around
+        ret.append(p)
+    return ret
+
+def patkind(name, dflt_pat='glob'):
+    """Split a string into an optional pattern kind prefix and the
+    actual pattern."""
+    for prefix in 're', 'glob', 'path', 'relglob', 'relpath', 'relre':
+        if name.startswith(prefix + ':'): return name.split(':', 1)
+    return dflt_pat, name
+
+def globre(pat, head='^', tail='$'):
+    "convert a glob pattern into a regexp"
+    i, n = 0, len(pat)
+    res = ''
+    group = False
+    def peek(): return i < n and pat[i]
+    while i < n:
+        c = pat[i]
+        i = i+1
+        if c == '*':
+            if peek() == '*':
+                i += 1
+                res += '.*'
+            else:
+                res += '[^/]*'
+        elif c == '?':
+            res += '.'
+        elif c == '[':
+            j = i
+            if j < n and pat[j] in '!]':
+                j += 1
+            while j < n and pat[j] != ']':
+                j += 1
+            if j >= n:
+                res += '\\['
+            else:
+                stuff = pat[i:j].replace('\\','\\\\')
+                i = j + 1
+                if stuff[0] == '!':
+                    stuff = '^' + stuff[1:]
+                elif stuff[0] == '^':
+                    stuff = '\\' + stuff
+                res = '%s[%s]' % (res, stuff)
+        elif c == '{':
+            group = True
+            res += '(?:'
+        elif c == '}' and group:
+            res += ')'
+            group = False
+        elif c == ',' and group:
+            res += '|'
+        elif c == '\\':
+            p = peek()
+            if p:
+                i += 1
+                res += re.escape(p)
+            else:
+                res += re.escape(c)
+        else:
+            res += re.escape(c)
+    return head + res + tail
+
+_globchars = {'[': 1, '{': 1, '*': 1, '?': 1}
+
+def pathto(root, n1, n2):
+    '''return the relative path from one place to another.
+    root should use os.sep to separate directories
+    n1 should use os.sep to separate directories
+    n2 should use "/" to separate directories
+    returns an os.sep-separated path.
+
+    If n1 is a relative path, it's assumed it's
+    relative to root.
+    n2 should always be relative to root.
+    '''
+    if not n1: return localpath(n2)
+    if os.path.isabs(n1):
+        if os.path.splitdrive(root)[0] != os.path.splitdrive(n1)[0]:
+            return os.path.join(root, localpath(n2))
+        n2 = '/'.join((pconvert(root), n2))
+    a, b = n1.split(os.sep), n2.split('/')
+    a.reverse()
+    b.reverse()
+    while a and b and a[-1] == b[-1]:
+        a.pop()
+        b.pop()
+    b.reverse()
+    return os.sep.join((['..'] * len(a)) + b)
+
+def canonpath(root, cwd, myname):
+    """return the canonical path of myname, given cwd and root"""
+    if root == os.sep:
+        rootsep = os.sep
+    elif root.endswith(os.sep):
+        rootsep = root
+    else:
+        rootsep = root + os.sep
+    name = myname
+    if not os.path.isabs(name):
+        name = os.path.join(root, cwd, name)
+    name = os.path.normpath(name)
+    audit_path = path_auditor(root)
+    if name != rootsep and name.startswith(rootsep):
+        name = name[len(rootsep):]
+        audit_path(name)
+        return pconvert(name)
+    elif name == root:
+        return ''
+    else:
+        # Determine whether `name' is in the hierarchy at or beneath `root',
+        # by iterating name=dirname(name) until that causes no change (can't
+        # check name == '/', because that doesn't work on windows).  For each
+        # `name', compare dev/inode numbers.  If they match, the list `rel'
+        # holds the reversed list of components making up the relative file
+        # name we want.
+        root_st = os.stat(root)
+        rel = []
+        while True:
+            try:
+                name_st = os.stat(name)
+            except OSError:
+                break
+            if samestat(name_st, root_st):
+                if not rel:
+                    # name was actually the same as root (maybe a symlink)
+                    return ''
+                rel.reverse()
+                name = os.path.join(*rel)
+                audit_path(name)
+                return pconvert(name)
+            dirname, basename = os.path.split(name)
+            rel.append(basename)
+            if dirname == name:
+                break
+            name = dirname
+
+        raise Abort('%s not under root' % myname)
+
+def matcher(canonroot, cwd='', names=[], inc=[], exc=[], src=None):
+    return _matcher(canonroot, cwd, names, inc, exc, 'glob', src)
+
+def cmdmatcher(canonroot, cwd='', names=[], inc=[], exc=[], src=None,
+               globbed=False, default=None):
+    default = default or 'relpath'
+    if default == 'relpath' and not globbed:
+        names = expand_glob(names)
+    return _matcher(canonroot, cwd, names, inc, exc, default, src)
+
+def _matcher(canonroot, cwd, names, inc, exc, dflt_pat, src):
+    """build a function to match a set of file patterns
+
+    arguments:
+    canonroot - the canonical root of the tree you're matching against
+    cwd - the current working directory, if relevant
+    names - patterns to find
+    inc - patterns to include
+    exc - patterns to exclude
+    dflt_pat - if a pattern in names has no explicit type, assume this one
+    src - where these patterns came from (e.g. .hgignore)
+
+    a pattern is one of:
+    'glob:<glob>' - a glob relative to cwd
+    're:<regexp>' - a regular expression
+    'path:<path>' - a path relative to canonroot
+    'relglob:<glob>' - an unrooted glob (*.c matches C files in all dirs)
+    'relpath:<path>' - a path relative to cwd
+    'relre:<regexp>' - a regexp that doesn't have to match the start of a name
+    '<something>' - one of the cases above, selected by the dflt_pat argument
+
+    returns:
+    a 3-tuple containing
+    - list of roots (places where one should start a recursive walk of the fs);
+      this often matches the explicit non-pattern names passed in, but also
+      includes the initial part of glob: patterns that has no glob characters
+    - a bool match(filename) function
+    - a bool indicating if any patterns were passed in
+    """
+
+    # a common case: no patterns at all
+    if not names and not inc and not exc:
+        return [], always, False
+
+    def contains_glob(name):
+        for c in name:
+            if c in _globchars: return True
+        return False
+
+    def regex(kind, name, tail):
+        '''convert a pattern into a regular expression'''
+        if not name:
+            return ''
+        if kind == 're':
+            return name
+        elif kind == 'path':
+            return '^' + re.escape(name) + '(?:/|$)'
+        elif kind == 'relglob':
+            return globre(name, '(?:|.*/)', tail)
+        elif kind == 'relpath':
+            return re.escape(name) + '(?:/|$)'
+        elif kind == 'relre':
+            if name.startswith('^'):
+                return name
+            return '.*' + name
+        return globre(name, '', tail)
+
+    def matchfn(pats, tail):
+        """build a matching function from a set of patterns"""
+        if not pats:
+            return
+        try:
+            pat = '(?:%s)' % '|'.join([regex(k, p, tail) for (k, p) in pats])
+            return re.compile(pat).match
+        except OverflowError:
+            # We're using a Python with a tiny regex engine and we
+            # made it explode, so we'll divide the pattern list in two
+            # until it works
+            l = len(pats)
+            if l < 2:
+                raise
+            a, b = matchfn(pats[:l//2], tail), matchfn(pats[l//2:], tail)
+            return lambda s: a(s) or b(s)
+        except re.error:
+            for k, p in pats:
+                try:
+                    re.compile('(?:%s)' % regex(k, p, tail))
+                except re.error:
+                    if src:
+                        raise Abort("%s: invalid pattern (%s): %s" %
+                                    (src, k, p))
+                    else:
+                        raise Abort("invalid pattern (%s): %s" % (k, p))
+            raise Abort("invalid pattern")
+
+    def globprefix(pat):
+        '''return the non-glob prefix of a path, e.g. foo/* -> foo'''
+        root = []
+        for p in pat.split('/'):
+            if contains_glob(p): break
+            root.append(p)
+        return '/'.join(root) or '.'
+
+    def normalizepats(names, default):
+        pats = []
+        roots = []
+        anypats = False
+        for kind, name in [patkind(p, default) for p in names]:
+            if kind in ('glob', 'relpath'):
+                name = canonpath(canonroot, cwd, name)
+            elif kind in ('relglob', 'path'):
+                name = normpath(name)
+
+            pats.append((kind, name))
+
+            if kind in ('glob', 're', 'relglob', 'relre'):
+                anypats = True
+
+            if kind == 'glob':
+                root = globprefix(name)
+                roots.append(root)
+            elif kind in ('relpath', 'path'):
+                roots.append(name or '.')
+            elif kind == 'relglob':
+                roots.append('.')
+        return roots, pats, anypats
+
+    roots, pats, anypats = normalizepats(names, dflt_pat)
+
+    patmatch = matchfn(pats, '$') or always
+    incmatch = always
+    if inc:
+        dummy, inckinds, dummy = normalizepats(inc, 'glob')
+        incmatch = matchfn(inckinds, '(?:/|$)')
+    excmatch = lambda fn: False
+    if exc:
+        dummy, exckinds, dummy = normalizepats(exc, 'glob')
+        excmatch = matchfn(exckinds, '(?:/|$)')
+
+    if not names and inc and not exc:
+        # common case: hgignore patterns
+        match = incmatch
+    else:
+        match = lambda fn: incmatch(fn) and not excmatch(fn) and patmatch(fn)
+
+    return (roots, match, (inc or exc or anypats) and True)
+
+_hgexecutable = None
+
+def hgexecutable():
+    """return location of the 'hg' executable.
+
+    Defaults to $HG or 'hg' in the search path.
+    """
+    if _hgexecutable is None:
+        set_hgexecutable(os.environ.get('HG') or find_exe('hg', 'hg'))
+    return _hgexecutable
+
+def set_hgexecutable(path):
+    """set location of the 'hg' executable"""
+    global _hgexecutable
+    _hgexecutable = path
+
+def system(cmd, environ={}, cwd=None, onerr=None, errprefix=None):
+    '''enhanced shell command execution.
+    run with environment maybe modified, maybe in different dir.
+
+    if command fails and onerr is None, return status.  if ui object,
+    print error message and return status, else raise onerr object as
+    exception.'''
+    def py2shell(val):
+        'convert python object into string that is useful to shell'
+        if val in (None, False):
+            return '0'
+        if val == True:
+            return '1'
+        return str(val)
+    oldenv = {}
+    for k in environ:
+        oldenv[k] = os.environ.get(k)
+    if cwd is not None:
+        oldcwd = os.getcwd()
+    origcmd = cmd
+    if os.name == 'nt':
+        cmd = '"%s"' % cmd
+    try:
+        for k, v in environ.iteritems():
+            os.environ[k] = py2shell(v)
+        os.environ['HG'] = hgexecutable()
+        if cwd is not None and oldcwd != cwd:
+            os.chdir(cwd)
+        rc = os.system(cmd)
+        if sys.platform == 'OpenVMS' and rc & 1:
+            rc = 0
+        if rc and onerr:
+            errmsg = '%s %s' % (os.path.basename(origcmd.split(None, 1)[0]),
+                                explain_exit(rc)[0])
+            if errprefix:
+                errmsg = '%s: %s' % (errprefix, errmsg)
+            try:
+                onerr.warn(errmsg + '\n')
+            except AttributeError:
+                raise onerr(errmsg)
+        return rc
+    finally:
+        for k, v in oldenv.iteritems():
+            if v is None:
+                del os.environ[k]
+            else:
+                os.environ[k] = v
+        if cwd is not None and oldcwd != cwd:
+            os.chdir(oldcwd)
+
+# os.path.lexists is not available on python2.3
+def lexists(filename):
+    "test whether a file with this name exists. does not follow symlinks"
+    try:
+        os.lstat(filename)
+    except:
+        return False
+    return True
+
+def rename(src, dst):
+    """forcibly rename a file"""
+    try:
+        os.rename(src, dst)
+    except OSError, err: # FIXME: check err (EEXIST ?)
+        # on windows, rename to existing file is not allowed, so we
+        # must delete destination first. but if file is open, unlink
+        # schedules it for delete but does not delete it. rename
+        # happens immediately even for open files, so we create
+        # temporary file, delete it, rename destination to that name,
+        # then delete that. then rename is safe to do.
+        fd, temp = tempfile.mkstemp(dir=os.path.dirname(dst) or '.')
+        os.close(fd)
+        os.unlink(temp)
+        os.rename(dst, temp)
+        os.unlink(temp)
+        os.rename(src, dst)
+
+def unlink(f):
+    """unlink and remove the directory if it is empty"""
+    os.unlink(f)
+    # try removing directories that might now be empty
+    try:
+        os.removedirs(os.path.dirname(f))
+    except OSError:
+        pass
+
+def copyfile(src, dest):
+    "copy a file, preserving mode"
+    if os.path.islink(src):
+        try:
+            os.unlink(dest)
+        except:
+            pass
+        os.symlink(os.readlink(src), dest)
+    else:
+        try:
+            shutil.copyfile(src, dest)
+            shutil.copymode(src, dest)
+        except shutil.Error, inst:
+            raise Abort(str(inst))
+
+def copyfiles(src, dst, hardlink=None):
+    """Copy a directory tree using hardlinks if possible"""
+
+    if hardlink is None:
+        hardlink = (os.stat(src).st_dev ==
+                    os.stat(os.path.dirname(dst)).st_dev)
+
+    if os.path.isdir(src):
+        os.mkdir(dst)
+        for name, kind in osutil.listdir(src):
+            srcname = os.path.join(src, name)
+            dstname = os.path.join(dst, name)
+            copyfiles(srcname, dstname, hardlink)
+    else:
+        if hardlink:
+            try:
+                os_link(src, dst)
+            except (IOError, OSError):
+                hardlink = False
+                shutil.copy(src, dst)
+        else:
+            shutil.copy(src, dst)
+
+class path_auditor(object):
+    '''ensure that a filesystem path contains no banned components.
+    the following properties of a path are checked:
+
+    - under top-level .hg
+    - starts at the root of a windows drive
+    - contains ".."
+    - traverses a symlink (e.g. a/symlink_here/b)
+    - inside a nested repository'''
+
+    def __init__(self, root):
+        self.audited = set()
+        self.auditeddir = set()
+        self.root = root
+
+    def __call__(self, path):
+        if path in self.audited:
+            return
+        normpath = os.path.normcase(path)
+        parts = normpath.split(os.sep)
+        if (os.path.splitdrive(path)[0] or parts[0] in ('.hg', '')
+            or os.pardir in parts):
+            raise Abort(_("path contains illegal component: %s") % path)
+        def check(prefix):
+            curpath = os.path.join(self.root, prefix)
+            try:
+                st = os.lstat(curpath)
+            except OSError, err:
+                # EINVAL can be raised as invalid path syntax under win32.
+                # They must be ignored for patterns can be checked too.
+                if err.errno not in (errno.ENOENT, errno.ENOTDIR, errno.EINVAL):
+                    raise
+            else:
+                if stat.S_ISLNK(st.st_mode):
+                    raise Abort(_('path %r traverses symbolic link %r') %
+                                (path, prefix))
+                elif (stat.S_ISDIR(st.st_mode) and
+                      os.path.isdir(os.path.join(curpath, '.hg'))):
+                    raise Abort(_('path %r is inside repo %r') %
+                                (path, prefix))
+
+        prefixes = []
+        for c in strutil.rfindall(normpath, os.sep):
+            prefix = normpath[:c]
+            if prefix in self.auditeddir:
+                break
+            check(prefix)
+            prefixes.append(prefix)
+
+        self.audited.add(path)
+        # only add prefixes to the cache after checking everything: we don't
+        # want to add "foo/bar/baz" before checking if there's a "foo/.hg"
+        self.auditeddir.update(prefixes)
+
+def _makelock_file(info, pathname):
+    ld = os.open(pathname, os.O_CREAT | os.O_WRONLY | os.O_EXCL)
+    os.write(ld, info)
+    os.close(ld)
+
+def _readlock_file(pathname):
+    return posixfile(pathname).read()
+
+def nlinks(pathname):
+    """Return number of hardlinks for the given file."""
+    return os.lstat(pathname).st_nlink
+
+if hasattr(os, 'link'):
+    os_link = os.link
+else:
+    def os_link(src, dst):
+        raise OSError(0, _("Hardlinks not supported"))
+
+def fstat(fp):
+    '''stat file object that may not have fileno method.'''
+    try:
+        return os.fstat(fp.fileno())
+    except AttributeError:
+        return os.stat(fp.name)
+
+posixfile = file
+
+def openhardlinks():
+    '''return true if it is safe to hold open file handles to hardlinks'''
+    return True
+
+getuser_fallback = None
+
+def getuser():
+    '''return name of current user'''
+    try:
+        return getpass.getuser()
+    except ImportError:
+        # import of pwd will fail on windows - try fallback
+        if getuser_fallback:
+            return getuser_fallback()
+    # raised if win32api not available
+    raise Abort(_('user name not available - set USERNAME '
+                  'environment variable'))
+
+def username(uid=None):
+    """Return the name of the user with the given uid.
+
+    If uid is None, return the name of the current user."""
+    try:
+        import pwd
+        if uid is None:
+            uid = os.getuid()
+        try:
+            return pwd.getpwuid(uid)[0]
+        except KeyError:
+            return str(uid)
+    except ImportError:
+        return None
+
+def groupname(gid=None):
+    """Return the name of the group with the given gid.
+
+    If gid is None, return the name of the current group."""
+    try:
+        import grp
+        if gid is None:
+            gid = os.getgid()
+        try:
+            return grp.getgrgid(gid)[0]
+        except KeyError:
+            return str(gid)
+    except ImportError:
+        return None
+
+# File system features
+
+def checkfolding(path):
+    """
+    Check whether the given path is on a case-sensitive filesystem
+
+    Requires a path (like /foo/.hg) ending with a foldable final
+    directory component.
+    """
+    s1 = os.stat(path)
+    d, b = os.path.split(path)
+    p2 = os.path.join(d, b.upper())
+    if path == p2:
+        p2 = os.path.join(d, b.lower())
+    try:
+        s2 = os.stat(p2)
+        if s2 == s1:
+            return False
+        return True
+    except:
+        return True
+
+def checkexec(path):
+    """
+    Check whether the given path is on a filesystem with UNIX-like exec flags
+
+    Requires a directory (like /foo/.hg)
+    """
+
+    # VFAT on some Linux versions can flip mode but it doesn't persist
+    # a FS remount. Frequently we can detect it if files are created
+    # with exec bit on.
+
+    try:
+        EXECFLAGS = stat.S_IXUSR | stat.S_IXGRP | stat.S_IXOTH
+        fh, fn = tempfile.mkstemp("", "", path)
+        try:
+            os.close(fh)
+            m = os.stat(fn).st_mode & 0777
+            new_file_has_exec = m & EXECFLAGS
+            os.chmod(fn, m ^ EXECFLAGS)
+            exec_flags_cannot_flip = ((os.stat(fn).st_mode & 0777) == m)
+        finally:
+            os.unlink(fn)
+    except (IOError, OSError):
+        # we don't care, the user probably won't be able to commit anyway
+        return False
+    return not (new_file_has_exec or exec_flags_cannot_flip)
+
+def execfunc(path, fallback):
+    '''return an is_exec() function with default to fallback'''
+    if checkexec(path):
+        return lambda x: is_exec(os.path.join(path, x))
+    return fallback
+
+def checklink(path):
+    """check whether the given path is on a symlink-capable filesystem"""
+    # mktemp is not racy because symlink creation will fail if the
+    # file already exists
+    name = tempfile.mktemp(dir=path)
+    try:
+        os.symlink(".", name)
+        os.unlink(name)
+        return True
+    except (OSError, AttributeError):
+        return False
+
+def linkfunc(path, fallback):
+    '''return an is_link() function with default to fallback'''
+    if checklink(path):
+        return lambda x: os.path.islink(os.path.join(path, x))
+    return fallback
+
+_umask = os.umask(0)
+os.umask(_umask)
+
+def needbinarypatch():
+    """return True if patches should be applied in binary mode by default."""
+    return os.name == 'nt'
+
+# Platform specific variants
+if os.name == 'nt':
+    import msvcrt
+    nulldev = 'NUL:'
+
+    class winstdout:
+        '''stdout on windows misbehaves if sent through a pipe'''
+
+        def __init__(self, fp):
+            self.fp = fp
+
+        def __getattr__(self, key):
+            return getattr(self.fp, key)
+
+        def close(self):
+            try:
+                self.fp.close()
+            except: pass
+
+        def write(self, s):
+            try:
+                # This is workaround for "Not enough space" error on
+                # writing large size of data to console.
+                limit = 16000
+                l = len(s)
+                start = 0
+                while start < l:
+                    end = start + limit
+                    self.fp.write(s[start:end])
+                    start = end
+            except IOError, inst:
+                if inst.errno != 0: raise
+                self.close()
+                raise IOError(errno.EPIPE, 'Broken pipe')
+
+        def flush(self):
+            try:
+                return self.fp.flush()
+            except IOError, inst:
+                if inst.errno != errno.EINVAL: raise
+                self.close()
+                raise IOError(errno.EPIPE, 'Broken pipe')
+
+    sys.stdout = winstdout(sys.stdout)
+
+    def _is_win_9x():
+        '''return true if run on windows 95, 98 or me.'''
+        try:
+            return sys.getwindowsversion()[3] == 1
+        except AttributeError:
+            return 'command' in os.environ.get('comspec', '')
+
+    def openhardlinks():
+        return not _is_win_9x and "win32api" in locals()
+
+    def system_rcpath():
+        try:
+            return system_rcpath_win32()
+        except:
+            return [r'c:\mercurial\mercurial.ini']
+
+    def user_rcpath():
+        '''return os-specific hgrc search path to the user dir'''
+        try:
+            userrc = user_rcpath_win32()
+        except:
+            userrc = os.path.join(os.path.expanduser('~'), 'mercurial.ini')
+        path = [userrc]
+        userprofile = os.environ.get('USERPROFILE')
+        if userprofile:
+            path.append(os.path.join(userprofile, 'mercurial.ini'))
+        return path
+
+    def parse_patch_output(output_line):
+        """parses the output produced by patch and returns the file name"""
+        pf = output_line[14:]
+        if pf[0] == '`':
+            pf = pf[1:-1] # Remove the quotes
+        return pf
+
+    def sshargs(sshcmd, host, user, port):
+        '''Build argument list for ssh or Plink'''
+        pflag = 'plink' in sshcmd.lower() and '-P' or '-p'
+        args = user and ("%s@%s" % (user, host)) or host
+        return port and ("%s %s %s" % (args, pflag, port)) or args
+
+    def testpid(pid):
+        '''return False if pid dead, True if running or not known'''
+        return True
+
+    def set_flags(f, flags):
+        pass
+
+    def set_binary(fd):
+        msvcrt.setmode(fd.fileno(), os.O_BINARY)
+
+    def pconvert(path):
+        return path.replace("\\", "/")
+
+    def localpath(path):
+        return path.replace('/', '\\')
+
+    def normpath(path):
+        return pconvert(os.path.normpath(path))
+
+    makelock = _makelock_file
+    readlock = _readlock_file
+
+    def samestat(s1, s2):
+        return False
+
+    # A sequence of backslashes is special iff it precedes a double quote:
+    # - if there's an even number of backslashes, the double quote is not
+    #   quoted (i.e. it ends the quoted region)
+    # - if there's an odd number of backslashes, the double quote is quoted
+    # - in both cases, every pair of backslashes is unquoted into a single
+    #   backslash
+    # (See http://msdn2.microsoft.com/en-us/library/a1y7w461.aspx )
+    # So, to quote a string, we must surround it in double quotes, double
+    # the number of backslashes that preceed double quotes and add another
+    # backslash before every double quote (being careful with the double
+    # quote we've appended to the end)
+    _quotere = None
+    def shellquote(s):
+        global _quotere
+        if _quotere is None:
+            _quotere = re.compile(r'(\\*)("|\\$)')
+        return '"%s"' % _quotere.sub(r'\1\1\\\2', s)
+
+    def quotecommand(cmd):
+        """Build a command string suitable for os.popen* calls."""
+        # The extra quotes are needed because popen* runs the command
+        # through the current COMSPEC. cmd.exe suppress enclosing quotes.
+        return '"' + cmd + '"'
+
+    def popen(command):
+        # Work around "popen spawned process may not write to stdout
+        # under windows"
+        # http://bugs.python.org/issue1366
+        command += " 2> %s" % nulldev
+        return os.popen(quotecommand(command))
+
+    def explain_exit(code):
+        return _("exited with status %d") % code, code
+
+    # if you change this stub into a real check, please try to implement the
+    # username and groupname functions above, too.
+    def isowner(fp, st=None):
+        return True
+
+    def find_in_path(name, path, default=None):
+        '''find name in search path. path can be string (will be split
+        with os.pathsep), or iterable thing that returns strings.  if name
+        found, return path to name. else return default. name is looked up
+        using cmd.exe rules, using PATHEXT.'''
+        if isinstance(path, str):
+            path = path.split(os.pathsep)
+
+        pathext = os.environ.get('PATHEXT', '.COM;.EXE;.BAT;.CMD')
+        pathext = pathext.lower().split(os.pathsep)
+        isexec = os.path.splitext(name)[1].lower() in pathext
+
+        for p in path:
+            p_name = os.path.join(p, name)
+
+            if isexec and os.path.exists(p_name):
+                return p_name
+
+            for ext in pathext:
+                p_name_ext = p_name + ext
+                if os.path.exists(p_name_ext):
+                    return p_name_ext
+        return default
+
+    def set_signal_handler():
+        try:
+            set_signal_handler_win32()
+        except NameError:
+            pass
+
+    try:
+        # override functions with win32 versions if possible
+        from util_win32 import *
+        if not _is_win_9x():
+            posixfile = posixfile_nt
+    except ImportError:
+        pass
+
+else:
+    nulldev = '/dev/null'
+
+    def rcfiles(path):
+        rcs = [os.path.join(path, 'hgrc')]
+        rcdir = os.path.join(path, 'hgrc.d')
+        try:
+            rcs.extend([os.path.join(rcdir, f)
+                        for f, kind in osutil.listdir(rcdir)
+                        if f.endswith(".rc")])
+        except OSError:
+            pass
+        return rcs
+
+    def system_rcpath():
+        path = []
+        # old mod_python does not set sys.argv
+        if len(getattr(sys, 'argv', [])) > 0:
+            path.extend(rcfiles(os.path.dirname(sys.argv[0]) +
+                                  '/../etc/mercurial'))
+        path.extend(rcfiles('/etc/mercurial'))
+        return path
+
+    def user_rcpath():
+        return [os.path.expanduser('~/.hgrc')]
+
+    def parse_patch_output(output_line):
+        """parses the output produced by patch and returns the file name"""
+        pf = output_line[14:]
+        if os.sys.platform == 'OpenVMS':
+            if pf[0] == '`':
+                pf = pf[1:-1] # Remove the quotes
+        else:
+           if pf.startswith("'") and pf.endswith("'") and " " in pf:
+                pf = pf[1:-1] # Remove the quotes
+        return pf
+
+    def sshargs(sshcmd, host, user, port):
+        '''Build argument list for ssh'''
+        args = user and ("%s@%s" % (user, host)) or host
+        return port and ("%s -p %s" % (args, port)) or args
+
+    def is_exec(f):
+        """check whether a file is executable"""
+        return (os.lstat(f).st_mode & 0100 != 0)
+
+    def set_flags(f, flags):
+        s = os.lstat(f).st_mode
+        x = "x" in flags
+        l = "l" in flags
+        if l:
+            if not stat.S_ISLNK(s):
+                # switch file to link
+                data = file(f).read()
+                os.unlink(f)
+                os.symlink(data, f)
+            # no chmod needed at this point
+            return
+        if stat.S_ISLNK(s):
+            # switch link to file
+            data = os.readlink(f)
+            os.unlink(f)
+            file(f, "w").write(data)
+            s = 0666 & ~_umask # avoid restatting for chmod
+
+        sx = s & 0100
+        if x and not sx:
+            # Turn on +x for every +r bit when making a file executable
+            # and obey umask.
+            os.chmod(f, s | (s & 0444) >> 2 & ~_umask)
+        elif not x and sx:
+            # Turn off all +x bits
+            os.chmod(f, s & 0666)
+
+    def set_binary(fd):
+        pass
+
+    def pconvert(path):
+        return path
+
+    def localpath(path):
+        return path
+
+    normpath = os.path.normpath
+    samestat = os.path.samestat
+
+    def makelock(info, pathname):
+        try:
+            os.symlink(info, pathname)
+        except OSError, why:
+            if why.errno == errno.EEXIST:
+                raise
+            else:
+                _makelock_file(info, pathname)
+
+    def readlock(pathname):
+        try:
+            return os.readlink(pathname)
+        except OSError, why:
+            if why.errno in (errno.EINVAL, errno.ENOSYS):
+                return _readlock_file(pathname)
+            else:
+                raise
+
+    def shellquote(s):
+        if os.sys.platform == 'OpenVMS':
+            return '"%s"' % s
+        else:
+            return "'%s'" % s.replace("'", "'\\''")
+
+    def quotecommand(cmd):
+        return cmd
+
+    def popen(command):
+        return os.popen(command)
+
+    def testpid(pid):
+        '''return False if pid dead, True if running or not sure'''
+        if os.sys.platform == 'OpenVMS':
+            return True
+        try:
+            os.kill(pid, 0)
+            return True
+        except OSError, inst:
+            return inst.errno != errno.ESRCH
+
+    def explain_exit(code):
+        """return a 2-tuple (desc, code) describing a process's status"""
+        if os.WIFEXITED(code):
+            val = os.WEXITSTATUS(code)
+            return _("exited with status %d") % val, val
+        elif os.WIFSIGNALED(code):
+            val = os.WTERMSIG(code)
+            return _("killed by signal %d") % val, val
+        elif os.WIFSTOPPED(code):
+            val = os.WSTOPSIG(code)
+            return _("stopped by signal %d") % val, val
+        raise ValueError(_("invalid exit code"))
+
+    def isowner(fp, st=None):
+        """Return True if the file object f belongs to the current user.
+
+        The return value of a util.fstat(f) may be passed as the st argument.
+        """
+        if st is None:
+            st = fstat(fp)
+        return st.st_uid == os.getuid()
+
+    def find_in_path(name, path, default=None):
+        '''find name in search path. path can be string (will be split
+        with os.pathsep), or iterable thing that returns strings.  if name
+        found, return path to name. else return default.'''
+        if isinstance(path, str):
+            path = path.split(os.pathsep)
+        for p in path:
+            p_name = os.path.join(p, name)
+            if os.path.exists(p_name):
+                return p_name
+        return default
+
+    def set_signal_handler():
+        pass
+
+def find_exe(name, default=None):
+    '''find path of an executable.
+    if name contains a path component, return it as is.  otherwise,
+    use normal executable search path.'''
+
+    if os.sep in name or sys.platform == 'OpenVMS':
+        # don't check the executable bit.  if the file isn't
+        # executable, whoever tries to actually run it will give a
+        # much more useful error message.
+        return name
+    return find_in_path(name, os.environ.get('PATH', ''), default=default)
+
+def _buildencodefun():
+    e = '_'
+    win_reserved = [ord(x) for x in '\\:*?"<>|']
+    cmap = dict([ (chr(x), chr(x)) for x in xrange(127) ])
+    for x in (range(32) + range(126, 256) + win_reserved):
+        cmap[chr(x)] = "~%02x" % x
+    for x in range(ord("A"), ord("Z")+1) + [ord(e)]:
+        cmap[chr(x)] = e + chr(x).lower()
+    dmap = {}
+    for k, v in cmap.iteritems():
+        dmap[v] = k
+    def decode(s):
+        i = 0
+        while i < len(s):
+            for l in xrange(1, 4):
+                try:
+                    yield dmap[s[i:i+l]]
+                    i += l
+                    break
+                except KeyError:
+                    pass
+            else:
+                raise KeyError
+    return (lambda s: "".join([cmap[c] for c in s]),
+            lambda s: "".join(list(decode(s))))
+
+encodefilename, decodefilename = _buildencodefun()
+
+def encodedopener(openerfn, fn):
+    def o(path, *args, **kw):
+        return openerfn(fn(path), *args, **kw)
+    return o
+
+def mktempcopy(name, emptyok=False):
+    """Create a temporary file with the same contents from name
+
+    The permission bits are copied from the original file.
+
+    If the temporary file is going to be truncated immediately, you
+    can use emptyok=True as an optimization.
+
+    Returns the name of the temporary file.
+    """
+    d, fn = os.path.split(name)
+    fd, temp = tempfile.mkstemp(prefix='.%s-' % fn, dir=d)
+    os.close(fd)
+    # Temporary files are created with mode 0600, which is usually not
+    # what we want.  If the original file already exists, just copy
+    # its mode.  Otherwise, manually obey umask.
+    try:
+        st_mode = os.lstat(name).st_mode & 0777
+    except OSError, inst:
+        if inst.errno != errno.ENOENT:
+            raise
+        st_mode = 0666 & ~_umask
+    os.chmod(temp, st_mode)
+    if emptyok:
+        return temp
+    try:
+        try:
+            ifp = posixfile(name, "rb")
+        except IOError, inst:
+            if inst.errno == errno.ENOENT:
+                return temp
+            if not getattr(inst, 'filename', None):
+                inst.filename = name
+            raise
+        ofp = posixfile(temp, "wb")
+        for chunk in filechunkiter(ifp):
+            ofp.write(chunk)
+        ifp.close()
+        ofp.close()
+    except:
+        try: os.unlink(temp)
+        except: pass
+        raise
+    return temp
+
+class atomictempfile(posixfile):
+    """file-like object that atomically updates a file
+
+    All writes will be redirected to a temporary copy of the original
+    file.  When rename is called, the copy is renamed to the original
+    name, making the changes visible.
+    """
+    def __init__(self, name, mode):
+        self.__name = name
+        self.temp = mktempcopy(name, emptyok=('w' in mode))
+        posixfile.__init__(self, self.temp, mode)
+
+    def rename(self):
+        if not self.closed:
+            posixfile.close(self)
+            rename(self.temp, localpath(self.__name))
+
+    def __del__(self):
+        if not self.closed:
+            try:
+                os.unlink(self.temp)
+            except: pass
+            posixfile.close(self)
+
+class opener(object):
+    """Open files relative to a base directory
+
+    This class is used to hide the details of COW semantics and
+    remote file access from higher level code.
+    """
+    def __init__(self, base, audit=True):
+        self.base = base
+        if audit:
+            self.audit_path = path_auditor(base)
+        else:
+            self.audit_path = always
+
+    def __getattr__(self, name):
+        if name == '_can_symlink':
+            self._can_symlink = checklink(self.base)
+            return self._can_symlink
+        raise AttributeError(name)
+
+    def __call__(self, path, mode="r", text=False, atomictemp=False):
+        self.audit_path(path)
+        f = os.path.join(self.base, path)
+
+        if not text and "b" not in mode:
+            mode += "b" # for that other OS
+
+        if mode[0] != "r":
+            try:
+                nlink = nlinks(f)
+            except OSError:
+                nlink = 0
+                d = os.path.dirname(f)
+                if not os.path.isdir(d):
+                    os.makedirs(d)
+            if atomictemp:
+                return atomictempfile(f, mode)
+            if nlink > 1:
+                rename(mktempcopy(f), f)
+        return posixfile(f, mode)
+
+    def symlink(self, src, dst):
+        self.audit_path(dst)
+        linkname = os.path.join(self.base, dst)
+        try:
+            os.unlink(linkname)
+        except OSError:
+            pass
+
+        dirname = os.path.dirname(linkname)
+        if not os.path.exists(dirname):
+            os.makedirs(dirname)
+
+        if self._can_symlink:
+            try:
+                os.symlink(src, linkname)
+            except OSError, err:
+                raise OSError(err.errno, _('could not symlink to %r: %s') %
+                              (src, err.strerror), linkname)
+        else:
+            f = self(dst, "w")
+            f.write(src)
+            f.close()
+
+class chunkbuffer(object):
+    """Allow arbitrary sized chunks of data to be efficiently read from an
+    iterator over chunks of arbitrary size."""
+
+    def __init__(self, in_iter):
+        """in_iter is the iterator that's iterating over the input chunks.
+        targetsize is how big a buffer to try to maintain."""
+        self.iter = iter(in_iter)
+        self.buf = ''
+        self.targetsize = 2**16
+
+    def read(self, l):
+        """Read L bytes of data from the iterator of chunks of data.
+        Returns less than L bytes if the iterator runs dry."""
+        if l > len(self.buf) and self.iter:
+            # Clamp to a multiple of self.targetsize
+            targetsize = max(l, self.targetsize)
+            collector = cStringIO.StringIO()
+            collector.write(self.buf)
+            collected = len(self.buf)
+            for chunk in self.iter:
+                collector.write(chunk)
+                collected += len(chunk)
+                if collected >= targetsize:
+                    break
+            if collected < targetsize:
+                self.iter = False
+            self.buf = collector.getvalue()
+        if len(self.buf) == l:
+            s, self.buf = str(self.buf), ''
+        else:
+            s, self.buf = self.buf[:l], buffer(self.buf, l)
+        return s
+
+def filechunkiter(f, size=65536, limit=None):
+    """Create a generator that produces the data in the file size
+    (default 65536) bytes at a time, up to optional limit (default is
+    to read all data).  Chunks may be less than size bytes if the
+    chunk is the last chunk in the file, or the file is a socket or
+    some other type of file that sometimes reads less data than is
+    requested."""
+    assert size >= 0
+    assert limit is None or limit >= 0
+    while True:
+        if limit is None: nbytes = size
+        else: nbytes = min(limit, size)
+        s = nbytes and f.read(nbytes)
+        if not s: break
+        if limit: limit -= len(s)
+        yield s
+
+def makedate():
+    lt = time.localtime()
+    if lt[8] == 1 and time.daylight:
+        tz = time.altzone
+    else:
+        tz = time.timezone
+    return time.mktime(lt), tz
+
+def datestr(date=None, format='%a %b %d %H:%M:%S %Y', timezone=True, timezone_format=" %+03d%02d"):
+    """represent a (unixtime, offset) tuple as a localized time.
+    unixtime is seconds since the epoch, and offset is the time zone's
+    number of seconds away from UTC. if timezone is false, do not
+    append time zone to string."""
+    t, tz = date or makedate()
+    s = time.strftime(format, time.gmtime(float(t) - tz))
+    if timezone:
+        s += timezone_format % (-tz / 3600, ((-tz % 3600) / 60))
+    return s
+
+def strdate(string, format, defaults=[]):
+    """parse a localized time string and return a (unixtime, offset) tuple.
+    if the string cannot be parsed, ValueError is raised."""
+    def timezone(string):
+        tz = string.split()[-1]
+        if tz[0] in "+-" and len(tz) == 5 and tz[1:].isdigit():
+            tz = int(tz)
+            offset = - 3600 * (tz / 100) - 60 * (tz % 100)
+            return offset
+        if tz == "GMT" or tz == "UTC":
+            return 0
+        return None
+
+    # NOTE: unixtime = localunixtime + offset
+    offset, date = timezone(string), string
+    if offset != None:
+        date = " ".join(string.split()[:-1])
+
+    # add missing elements from defaults
+    for part in defaults:
+        found = [True for p in part if ("%"+p) in format]
+        if not found:
+            date += "@" + defaults[part]
+            format += "@%" + part[0]
+
+    timetuple = time.strptime(date, format)
+    localunixtime = int(calendar.timegm(timetuple))
+    if offset is None:
+        # local timezone
+        unixtime = int(time.mktime(timetuple))
+        offset = unixtime - localunixtime
+    else:
+        unixtime = localunixtime + offset
+    return unixtime, offset
+
+def parsedate(string, formats=None, defaults=None):
+    """parse a localized time string and return a (unixtime, offset) tuple.
+    The date may be a "unixtime offset" string or in one of the specified
+    formats."""
+    if not string:
+        return 0, 0
+    if not formats:
+        formats = defaultdateformats
+    string = string.strip()
+    try:
+        when, offset = map(int, string.split(' '))
+    except ValueError:
+        # fill out defaults
+        if not defaults:
+            defaults = {}
+        now = makedate()
+        for part in "d mb yY HI M S".split():
+            if part not in defaults:
+                if part[0] in "HMS":
+                    defaults[part] = "00"
+                elif part[0] in "dm":
+                    defaults[part] = "1"
+                else:
+                    defaults[part] = datestr(now, "%" + part[0], False)
+
+        for format in formats:
+            try:
+                when, offset = strdate(string, format, defaults)
+            except ValueError:
+                pass
+            else:
+                break
+        else:
+            raise Abort(_('invalid date: %r ') % string)
+    # validate explicit (probably user-specified) date and
+    # time zone offset. values must fit in signed 32 bits for
+    # current 32-bit linux runtimes. timezones go from UTC-12
+    # to UTC+14
+    if abs(when) > 0x7fffffff:
+        raise Abort(_('date exceeds 32 bits: %d') % when)
+    if offset < -50400 or offset > 43200:
+        raise Abort(_('impossible time zone offset: %d') % offset)
+    return when, offset
+
+def matchdate(date):
+    """Return a function that matches a given date match specifier
+
+    Formats include:
+
+    '{date}' match a given date to the accuracy provided
+
+    '<{date}' on or before a given date
+
+    '>{date}' on or after a given date
+
+    """
+
+    def lower(date):
+        return parsedate(date, extendeddateformats)[0]
+
+    def upper(date):
+        d = dict(mb="12", HI="23", M="59", S="59")
+        for days in "31 30 29".split():
+            try:
+                d["d"] = days
+                return parsedate(date, extendeddateformats, d)[0]
+            except:
+                pass
+        d["d"] = "28"
+        return parsedate(date, extendeddateformats, d)[0]
+
+    if date[0] == "<":
+        when = upper(date[1:])
+        return lambda x: x <= when
+    elif date[0] == ">":
+        when = lower(date[1:])
+        return lambda x: x >= when
+    elif date[0] == "-":
+        try:
+            days = int(date[1:])
+        except ValueError:
+            raise Abort(_("invalid day spec: %s") % date[1:])
+        when = makedate()[0] - days * 3600 * 24
+        return lambda x: x >= when
+    elif " to " in date:
+        a, b = date.split(" to ")
+        start, stop = lower(a), upper(b)
+        return lambda x: x >= start and x <= stop
+    else:
+        start, stop = lower(date), upper(date)
+        return lambda x: x >= start and x <= stop
+
+def shortuser(user):
+    """Return a short representation of a user name or email address."""
+    f = user.find('@')
+    if f >= 0:
+        user = user[:f]
+    f = user.find('<')
+    if f >= 0:
+        user = user[f+1:]
+    f = user.find(' ')
+    if f >= 0:
+        user = user[:f]
+    f = user.find('.')
+    if f >= 0:
+        user = user[:f]
+    return user
+
+def ellipsis(text, maxlength=400):
+    """Trim string to at most maxlength (default: 400) characters."""
+    if len(text) <= maxlength:
+        return text
+    else:
+        return "%s..." % (text[:maxlength-3])
+
+def walkrepos(path):
+    '''yield every hg repository under path, recursively.'''
+    def errhandler(err):
+        if err.filename == path:
+            raise err
+
+    for root, dirs, files in os.walk(path, onerror=errhandler):
+        for d in dirs:
+            if d == '.hg':
+                yield root
+                dirs[:] = []
+                break
+
+_rcpath = None
+
+def os_rcpath():
+    '''return default os-specific hgrc search path'''
+    path = system_rcpath()
+    path.extend(user_rcpath())
+    path = [os.path.normpath(f) for f in path]
+    return path
+
+def rcpath():
+    '''return hgrc search path. if env var HGRCPATH is set, use it.
+    for each item in path, if directory, use files ending in .rc,
+    else use item.
+    make HGRCPATH empty to only look in .hg/hgrc of current repo.
+    if no HGRCPATH, use default os-specific path.'''
+    global _rcpath
+    if _rcpath is None:
+        if 'HGRCPATH' in os.environ:
+            _rcpath = []
+            for p in os.environ['HGRCPATH'].split(os.pathsep):
+                if not p: continue
+                if os.path.isdir(p):
+                    for f, kind in osutil.listdir(p):
+                        if f.endswith('.rc'):
+                            _rcpath.append(os.path.join(p, f))
+                else:
+                    _rcpath.append(p)
+        else:
+            _rcpath = os_rcpath()
+    return _rcpath
+
+def bytecount(nbytes):
+    '''return byte count formatted as readable string, with units'''
+
+    units = (
+        (100, 1<<30, _('%.0f GB')),
+        (10, 1<<30, _('%.1f GB')),
+        (1, 1<<30, _('%.2f GB')),
+        (100, 1<<20, _('%.0f MB')),
+        (10, 1<<20, _('%.1f MB')),
+        (1, 1<<20, _('%.2f MB')),
+        (100, 1<<10, _('%.0f KB')),
+        (10, 1<<10, _('%.1f KB')),
+        (1, 1<<10, _('%.2f KB')),
+        (1, 1, _('%.0f bytes')),
+        )
+
+    for multiplier, divisor, format in units:
+        if nbytes >= divisor * multiplier:
+            return format % (nbytes / float(divisor))
+    return units[-1][2] % nbytes
+
+def drop_scheme(scheme, path):
+    sc = scheme + ':'
+    if path.startswith(sc):
+        path = path[len(sc):]
+        if path.startswith('//'):
+            path = path[2:]
+    return path
+
+def uirepr(s):
+    # Avoid double backslash in Windows path repr()
+    return repr(s).replace('\\\\', '\\')
+
+def hidepassword(url):
+    '''hide user credential in a url string'''
+    scheme, netloc, path, params, query, fragment = urlparse.urlparse(url)
+    netloc = re.sub('([^:]*):([^@]*)@(.*)', r'\1:***@\3', netloc)
+    return urlparse.urlunparse((scheme, netloc, path, params, query, fragment))
+
+def removeauth(url):
+    '''remove all authentication information from a url string'''
+    scheme, netloc, path, params, query, fragment = urlparse.urlparse(url)
+    netloc = netloc[netloc.find('@')+1:]
+    return urlparse.urlunparse((scheme, netloc, path, params, query, fragment))
+
+--Apple-Mail-27-69865486--
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M318159P23014Q11.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,36 @@
+Date: Sun, 06 Jan 2008 00:45:57 -0500
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+To: Mirko Friedenhagen <mirko@friedenhagen.de>
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Message-ID: <20080106054557.GA12219@cs.duke.edu>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
+Mime-Version: 1.0
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: inline
+In-Reply-To: <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de>
+User-Agent: Mutt/1.4.2.2i
+Status: RO
+X-Status: A
+Content-Length: 675
+Lines: 20
+
+Hi, Mirko,
+
+Very strange. I still don't see what the problem is. Try the attached
+version of artemis.py. It's not really a fix: it's the fix of the
+immediate problem, but if you try to use -d flag of ilist, the problem
+will come back. Meanwhile, I'll think about it some more.
+
+Best,
+Dmitriy
+
+On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
+>Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
+>
+>>Hi, Mirko,
+>>
+>>I'm unable to replicate the problem with mercrial/crew. Can you send
+>>me util.py (i.e., mercurial/util.py) from the repository that you are
+>>actually using?
+>I did not alter anything in it, see the diff between my local branch  
+>and default:
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M320063P23014Q12.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,321 @@
+Date: Sun, 06 Jan 2008 00:46:30 -0500
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+To: Mirko Friedenhagen <mirko@friedenhagen.de>
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Message-ID: <20080106054630.GB12219@cs.duke.edu>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu>
+Mime-Version: 1.0
+Content-Type: multipart/mixed; boundary="oyUTqETQ0mS9luUI"
+Content-Disposition: inline
+In-Reply-To: <20080106054557.GA12219@cs.duke.edu>
+User-Agent: Mutt/1.4.2.2i
+Status: RO
+Content-Length: 9448
+Lines: 306
+
+
+--oyUTqETQ0mS9luUI
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: inline
+
+Forgot the attachment. Dmitriy
+
+On Sun, Jan 06, 2008 at 12:45:57AM -0500, Dmitriy Morozov wrote:
+>Hi, Mirko,
+>
+>Very strange. I still don't see what the problem is. Try the attached
+>version of artemis.py. It's not really a fix: it's the fix of the
+>immediate problem, but if you try to use -d flag of ilist, the problem
+>will come back. Meanwhile, I'll think about it some more.
+>
+>Best,
+>Dmitriy
+>
+>On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
+>>Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
+>>
+>>>Hi, Mirko,
+>>>
+>>>I'm unable to replicate the problem with mercrial/crew. Can you send
+>>>me util.py (i.e., mercurial/util.py) from the repository that you are
+>>>actually using?
+>>I did not alter anything in it, see the diff between my local branch  
+>>and default:
+
+--oyUTqETQ0mS9luUI
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: attachment; filename="artemis.py"
+
+# Author: Dmitriy Morozov <hg@foxcub.org>, 2007
+
+"""A very simple and lightweight issue tracker for Mercurial."""
+
+from mercurial import hg, util
+from mercurial.i18n import _
+import os, time, random, mailbox, glob, socket, ConfigParser
+
+
+state = {'new': 'new', 'fixed': 'fixed'}
+state['default'] = state['new']
+issues_dir = ".issues"
+filter_prefix = ".filter"
+date_format = '%a, %d %b %Y %H:%M:%S %Z'
+
+
+def ilist(ui, repo, **opts):
+	"""List issues associated with the project"""
+
+	# Process options
+	show_all = opts['all']
+	properties = []
+	match_date, date_match = False, lambda x: True
+	if opts['date']: 
+		match_date, date_match = True, util.matchdate(opts['date'])
+
+	# Find issues
+	issues_path = os.path.join(repo.root, issues_dir)
+	if not os.path.exists(issues_path): return
+
+	issues = glob.glob(os.path.join(issues_path, '*'))
+	
+	# Process filter
+	if opts['filter']:
+		filters = glob.glob(os.path.join(issues_path, filter_prefix + '*'))
+		config = ConfigParser.SafeConfigParser()
+		config.read(filters)
+		if not config.has_section(opts['filter']): 
+			ui.warning('No filter %s defined\n', opts['filter'])
+		else:
+			properties += config.items(opts['filter'])
+	
+	_get_properties(opts['property'])
+	
+	for issue in issues:
+		mbox = mailbox.mbox(issue)
+		property_match = True
+		for property,value in properties: 
+			property_match = property_match and (mbox[0][property] == value)
+		if not show_all and (not properties or not property_match) and (properties or mbox[0]['State'].upper() == state['fixed'].upper()): continue 
+
+
+		if match_date and not date_match(util.parsedate(mbox[0]['date'], [date_format])[0]): continue
+		ui.write("%s (%d) [%s]: %s\n" % (issue[len(issues_path)+1:], # +1 for trailing /
+										 len(mbox)-1,				 # number of replies (-1 for self)
+										 mbox[0]['State'],
+										 mbox[0]['Subject']))
+	
+
+def iadd(ui, repo, id = None, comment = 0):
+	"""Adds a new issue, or comment to an existing issue ID or its comment COMMENT"""
+	
+	comment = int(comment)
+
+	# First, make sure issues have a directory
+	issues_path = os.path.join(repo.root, issues_dir)
+	if not os.path.exists(issues_path): os.mkdir(issues_path)
+
+	if id:
+		issue_fn, issue_id = _find_issue(ui, repo, id)
+		if not issue_fn: 
+			ui.warn('No such issue\n')
+			return
+	
+	user = ui.username()
+
+	default_issue_text  = 		"From: %s\nDate: %s\n" % (user, time.strftime(date_format))
+	if not id: 
+		default_issue_text += 	"State: %s\n" % state['default']
+	default_issue_text +=		"Subject: brief description\n\n"
+	default_issue_text += 		"Detailed description."
+
+	issue = ui.edit(default_issue_text, user)
+	if issue.strip() == '':
+		ui.warn('Empty issue, ignoring\n')
+		return
+	if issue.strip() == default_issue_text:
+		ui.warn('Unchanged issue text, ignoring\n')
+		return
+
+	# Create the message
+	msg = mailbox.mboxMessage(issue)
+	msg.set_from('artemis', True)
+	
+	# Pick random filename
+	if not id:
+		issue_fn = issues_path
+		while os.path.exists(issue_fn):
+			issue_id = _random_id() 
+			issue_fn = os.path.join(issues_path, issue_id)
+	# else: issue_fn already set
+
+	# Add message to the mailbox
+	mbox = mailbox.mbox(issue_fn)
+	if id and comment not in mbox: 
+		ui.warn('No such comment number in mailbox, commenting on the issue itself\n')
+	if not id:
+		msg.add_header('Message-Id', "<%s-0-artemis@%s>" % (issue_id, socket.gethostname()))
+	else:
+		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (issue_id, _random_id(), socket.gethostname()))
+		msg.add_header('References', mbox[(comment < len(mbox) and comment) or 0]['Message-Id'])
+	mbox.add(msg)
+	mbox.close()
+
+	# If adding issue, add the new mailbox to the repository
+	if not id: 
+		repo.add([issue_fn[(len(repo.root)+1):]])			# +1 for the trailing /
+		ui.status('Added new issue %s\n' % issue_id)
+
+
+def ishow(ui, repo, id, comment = 0, **opts):
+	"""Shows issue ID, or possibly its comment COMMENT"""
+	
+	comment = int(comment)
+	issue, id = _find_issue(ui, repo, id)
+	if not issue: return
+	mbox = mailbox.mbox(issue)
+
+	if opts['all']:
+		ui.write('='*70 + '\n')
+		for i in xrange(len(mbox)):
+			_write_message(ui, mbox[i], i)
+			ui.write('-'*70 + '\n')
+		return
+
+	_show_mbox(ui, mbox, comment)
+
+
+def iupdate(ui, repo, id, **opts):
+	"""Update properties of issue ID"""
+
+	issue, id = _find_issue(ui, repo, id)
+	if not issue: return
+
+	properties = _get_properties(opts['property'])
+	
+	# Read the issue
+	mbox = mailbox.mbox(issue)
+	msg = mbox[0]
+
+	# Fix the properties
+	properties_text = ''
+	for property, value in properties:
+		msg.replace_header(property, value)
+		properties_text += '%s=%s\n' % (property, value)
+	mbox[0] = msg
+
+	# Write down a comment about updated properties
+	if properties and not opts['no_property_comment']:
+		user = ui.username()
+		properties_text  = 	"From: %s\nDate: %s\nSubject: properties changes (%s)\n\n%s" % \
+							(user, time.strftime(date_format),
+							 _pretty_list(list(set([property for property, value in properties]))), 
+							 properties_text)
+		msg = mailbox.mboxMessage(properties_text)
+		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (id, _random_id(), socket.gethostname()))
+		msg.add_header('References', mbox[0]['Message-Id'])
+		msg.set_from('artemis', True)
+		mbox.add(msg)
+	mbox.flush()
+
+	# Show updated message
+	_show_mbox(ui, mbox, 0)
+
+
+def _find_issue(ui, repo, id):
+	issues_path = os.path.join(repo.root, issues_dir)
+	if not os.path.exists(issues_path): return False
+
+	issues = glob.glob(os.path.join(issues_path, id + '*'))
+
+	if len(issues) == 0:
+		return False, 0
+	elif len(issues) > 1:
+		ui.status("Multiple choices:\n")
+		for i in issues: ui.status('  ', i[len(issues_path)+1:], '\n')
+		return False, 0
+	
+	return issues[0], issues[0][len(issues_path)+1:]
+
+def _get_properties(property_list):
+	return [p.split('=') for p in property_list]
+	
+def _write_message(ui, message, index = 0):
+	if index: ui.write("Comment: %d\n" % index)
+	if ui.verbose:
+		ui.write(message.as_string().strip() + '\n')
+	else:
+		if 'From' in message: ui.write('From: %s\n' % message['From'])
+		if 'Date' in message: ui.write('Date: %s\n' % message['Date'])
+		if 'Subject' in message: ui.write('Subject: %s\n' % message['Subject'])
+		if 'State' in message: ui.write('State: %s\n' % message['State'])
+		ui.write('\n' + message.get_payload().strip() + '\n')
+
+def _show_mbox(ui, mbox, comment):
+	# Output the issue (or comment)
+	if comment >= len(mbox): 
+		comment = 0
+		ui.warn('Comment out of range, showing the issue itself\n')
+	msg = mbox[comment]
+	ui.write('='*70 + '\n')
+	if comment:
+		ui.write('Subject: %s\n' % mbox[0]['Subject'])
+		ui.write('State: %s\n' % mbox[0]['State'])
+		ui.write('-'*70 + '\n')
+	_write_message(ui, msg, comment)
+	ui.write('-'*70 + '\n')
+
+	# Read the mailbox into the messages and children dictionaries
+	messages = {}
+	children = {}
+	for i in xrange(len(mbox)):
+		m = mbox[i]
+		messages[m['Message-Id']] = (i,m)
+		children.setdefault(m['References'], []).append(m['Message-Id'])
+	children[None] = []				# Safeguard against infinte loop on empty Message-Id
+
+	# Iterate over children
+	id = msg['Message-Id'] 
+	id_stack = (id in children and map(lambda x: (x, 1), reversed(children[id]))) or []
+	if not id_stack: return
+	ui.write('Comments:\n')
+	while id_stack:
+		id,offset = id_stack.pop()
+		id_stack += (id in children and map(lambda x: (x, offset+1), reversed(children[id]))) or []
+		index, msg = messages[id]
+		ui.write('  '*offset + ('%d: ' % index) + msg['Subject'] + '\n')
+	ui.write('-'*70 + '\n')
+
+def _pretty_list(lst):
+	s = ''
+	for i in lst:
+		s += i + ', '
+	return s[:-2]
+
+def _random_id():
+	return "%x" % random.randint(2**63, 2**64-1)
+
+
+cmdtable = {
+	'ilist':	(ilist, 
+				 [('a', 'all', False, 
+				   'list all issues (by default only those with state new)'),
+				  ('p', 'property', [], 
+				   'list issues with specific field values (e.g., -p state=fixed)'),
+				  ('d', 'date', '', 'restrict to issues matching the date (e.g., -d ">12/28/2007)"'),
+				  ('f', 'filter', '', 'restrict to pre-defined filter (in %s/%s*)' % (issues_dir, filter_prefix))], 
+				 _('hg ilist [OPTIONS]')),
+	'iadd':   	(iadd,  
+				 [], 
+				 _('hg iadd [ID] [COMMENT]')),
+	'ishow':  	(ishow, 
+				 [('a', 'all', None, 'list all comments')], 
+				 _('hg ishow [OPTIONS] ID [COMMENT]')),
+	'iupdate':	(iupdate,
+				 [('p', 'property', [], 
+				   'update properties (e.g., -p state=fixed)'),
+				  ('n', 'no-property-comment', None, 
+				   'do not add a comment about changed properties')],
+				 _('hg iupdate [OPTIONS] ID'))
+}
+
+--oyUTqETQ0mS9luUI--
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M323632P23014Q13.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,318 @@
+Date: Sun, 06 Jan 2008 01:23:03 -0500
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+To: Mirko Friedenhagen <mirko@friedenhagen.de>
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Message-ID: <20080106062303.GA12860@cs.duke.edu>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu>
+Mime-Version: 1.0
+Content-Type: text/plain; charset=koi8-r
+Content-Disposition: inline
+In-Reply-To: <20080106054630.GB12219@cs.duke.edu>
+User-Agent: Mutt/1.4.2.2i
+Status: RO
+Content-Length: 9705
+Lines: 303
+
+Actually, you can just pull the changes from the Artemis repository on
+the Web. I've put them there as well.
+
+Best,
+Dmitriy
+
+On Sun, Jan 06, 2008 at 12:46:30AM -0500, Dmitriy Morozov wrote:
+>Forgot the attachment. Dmitriy
+>
+>On Sun, Jan 06, 2008 at 12:45:57AM -0500, Dmitriy Morozov wrote:
+>>Hi, Mirko,
+>>
+>>Very strange. I still don't see what the problem is. Try the attached
+>>version of artemis.py. It's not really a fix: it's the fix of the
+>>immediate problem, but if you try to use -d flag of ilist, the problem
+>>will come back. Meanwhile, I'll think about it some more.
+>>
+>>Best,
+>>Dmitriy
+>>
+>>On Sat, Jan 05, 2008 at 10:18:01PM +0100, Mirko Friedenhagen wrote:
+>>>Am 05.01.2008 um 13:47 schrieb Dmitriy Morozov:
+>>>
+>>>>Hi, Mirko,
+>>>>
+>>>>I'm unable to replicate the problem with mercrial/crew. Can you send
+>>>>me util.py (i.e., mercurial/util.py) from the repository that you are
+>>>>actually using?
+>>>I did not alter anything in it, see the diff between my local branch  
+>>>and default:
+
+># Author: Dmitriy Morozov <hg@foxcub.org>, 2007
+>
+>"""A very simple and lightweight issue tracker for Mercurial."""
+>
+>from mercurial import hg, util
+>from mercurial.i18n import _
+>import os, time, random, mailbox, glob, socket, ConfigParser
+>
+>
+>state = {'new': 'new', 'fixed': 'fixed'}
+>state['default'] = state['new']
+>issues_dir = ".issues"
+>filter_prefix = ".filter"
+>date_format = '%a, %d %b %Y %H:%M:%S %Z'
+>
+>
+>def ilist(ui, repo, **opts):
+>	"""List issues associated with the project"""
+>
+>	# Process options
+>	show_all = opts['all']
+>	properties = []
+>	match_date, date_match = False, lambda x: True
+>	if opts['date']: 
+>		match_date, date_match = True, util.matchdate(opts['date'])
+>
+>	# Find issues
+>	issues_path = os.path.join(repo.root, issues_dir)
+>	if not os.path.exists(issues_path): return
+>
+>	issues = glob.glob(os.path.join(issues_path, '*'))
+>	
+>	# Process filter
+>	if opts['filter']:
+>		filters = glob.glob(os.path.join(issues_path, filter_prefix + '*'))
+>		config = ConfigParser.SafeConfigParser()
+>		config.read(filters)
+>		if not config.has_section(opts['filter']): 
+>			ui.warning('No filter %s defined\n', opts['filter'])
+>		else:
+>			properties += config.items(opts['filter'])
+>	
+>	_get_properties(opts['property'])
+>	
+>	for issue in issues:
+>		mbox = mailbox.mbox(issue)
+>		property_match = True
+>		for property,value in properties: 
+>			property_match = property_match and (mbox[0][property] == value)
+>		if not show_all and (not properties or not property_match) and (properties or mbox[0]['State'].upper() == state['fixed'].upper()): continue 
+>
+>
+>		if match_date and not date_match(util.parsedate(mbox[0]['date'], [date_format])[0]): continue
+>		ui.write("%s (%d) [%s]: %s\n" % (issue[len(issues_path)+1:], # +1 for trailing /
+>										 len(mbox)-1,				 # number of replies (-1 for self)
+>										 mbox[0]['State'],
+>										 mbox[0]['Subject']))
+>	
+>
+>def iadd(ui, repo, id = None, comment = 0):
+>	"""Adds a new issue, or comment to an existing issue ID or its comment COMMENT"""
+>	
+>	comment = int(comment)
+>
+>	# First, make sure issues have a directory
+>	issues_path = os.path.join(repo.root, issues_dir)
+>	if not os.path.exists(issues_path): os.mkdir(issues_path)
+>
+>	if id:
+>		issue_fn, issue_id = _find_issue(ui, repo, id)
+>		if not issue_fn: 
+>			ui.warn('No such issue\n')
+>			return
+>	
+>	user = ui.username()
+>
+>	default_issue_text  = 		"From: %s\nDate: %s\n" % (user, time.strftime(date_format))
+>	if not id: 
+>		default_issue_text += 	"State: %s\n" % state['default']
+>	default_issue_text +=		"Subject: brief description\n\n"
+>	default_issue_text += 		"Detailed description."
+>
+>	issue = ui.edit(default_issue_text, user)
+>	if issue.strip() == '':
+>		ui.warn('Empty issue, ignoring\n')
+>		return
+>	if issue.strip() == default_issue_text:
+>		ui.warn('Unchanged issue text, ignoring\n')
+>		return
+>
+>	# Create the message
+>	msg = mailbox.mboxMessage(issue)
+>	msg.set_from('artemis', True)
+>	
+>	# Pick random filename
+>	if not id:
+>		issue_fn = issues_path
+>		while os.path.exists(issue_fn):
+>			issue_id = _random_id() 
+>			issue_fn = os.path.join(issues_path, issue_id)
+>	# else: issue_fn already set
+>
+>	# Add message to the mailbox
+>	mbox = mailbox.mbox(issue_fn)
+>	if id and comment not in mbox: 
+>		ui.warn('No such comment number in mailbox, commenting on the issue itself\n')
+>	if not id:
+>		msg.add_header('Message-Id', "<%s-0-artemis@%s>" % (issue_id, socket.gethostname()))
+>	else:
+>		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (issue_id, _random_id(), socket.gethostname()))
+>		msg.add_header('References', mbox[(comment < len(mbox) and comment) or 0]['Message-Id'])
+>	mbox.add(msg)
+>	mbox.close()
+>
+>	# If adding issue, add the new mailbox to the repository
+>	if not id: 
+>		repo.add([issue_fn[(len(repo.root)+1):]])			# +1 for the trailing /
+>		ui.status('Added new issue %s\n' % issue_id)
+>
+>
+>def ishow(ui, repo, id, comment = 0, **opts):
+>	"""Shows issue ID, or possibly its comment COMMENT"""
+>	
+>	comment = int(comment)
+>	issue, id = _find_issue(ui, repo, id)
+>	if not issue: return
+>	mbox = mailbox.mbox(issue)
+>
+>	if opts['all']:
+>		ui.write('='*70 + '\n')
+>		for i in xrange(len(mbox)):
+>			_write_message(ui, mbox[i], i)
+>			ui.write('-'*70 + '\n')
+>		return
+>
+>	_show_mbox(ui, mbox, comment)
+>
+>
+>def iupdate(ui, repo, id, **opts):
+>	"""Update properties of issue ID"""
+>
+>	issue, id = _find_issue(ui, repo, id)
+>	if not issue: return
+>
+>	properties = _get_properties(opts['property'])
+>	
+>	# Read the issue
+>	mbox = mailbox.mbox(issue)
+>	msg = mbox[0]
+>
+>	# Fix the properties
+>	properties_text = ''
+>	for property, value in properties:
+>		msg.replace_header(property, value)
+>		properties_text += '%s=%s\n' % (property, value)
+>	mbox[0] = msg
+>
+>	# Write down a comment about updated properties
+>	if properties and not opts['no_property_comment']:
+>		user = ui.username()
+>		properties_text  = 	"From: %s\nDate: %s\nSubject: properties changes (%s)\n\n%s" % \
+>							(user, time.strftime(date_format),
+>							 _pretty_list(list(set([property for property, value in properties]))), 
+>							 properties_text)
+>		msg = mailbox.mboxMessage(properties_text)
+>		msg.add_header('Message-Id', "<%s-%s-artemis@%s>" % (id, _random_id(), socket.gethostname()))
+>		msg.add_header('References', mbox[0]['Message-Id'])
+>		msg.set_from('artemis', True)
+>		mbox.add(msg)
+>	mbox.flush()
+>
+>	# Show updated message
+>	_show_mbox(ui, mbox, 0)
+>
+>
+>def _find_issue(ui, repo, id):
+>	issues_path = os.path.join(repo.root, issues_dir)
+>	if not os.path.exists(issues_path): return False
+>
+>	issues = glob.glob(os.path.join(issues_path, id + '*'))
+>
+>	if len(issues) == 0:
+>		return False, 0
+>	elif len(issues) > 1:
+>		ui.status("Multiple choices:\n")
+>		for i in issues: ui.status('  ', i[len(issues_path)+1:], '\n')
+>		return False, 0
+>	
+>	return issues[0], issues[0][len(issues_path)+1:]
+>
+>def _get_properties(property_list):
+>	return [p.split('=') for p in property_list]
+>	
+>def _write_message(ui, message, index = 0):
+>	if index: ui.write("Comment: %d\n" % index)
+>	if ui.verbose:
+>		ui.write(message.as_string().strip() + '\n')
+>	else:
+>		if 'From' in message: ui.write('From: %s\n' % message['From'])
+>		if 'Date' in message: ui.write('Date: %s\n' % message['Date'])
+>		if 'Subject' in message: ui.write('Subject: %s\n' % message['Subject'])
+>		if 'State' in message: ui.write('State: %s\n' % message['State'])
+>		ui.write('\n' + message.get_payload().strip() + '\n')
+>
+>def _show_mbox(ui, mbox, comment):
+>	# Output the issue (or comment)
+>	if comment >= len(mbox): 
+>		comment = 0
+>		ui.warn('Comment out of range, showing the issue itself\n')
+>	msg = mbox[comment]
+>	ui.write('='*70 + '\n')
+>	if comment:
+>		ui.write('Subject: %s\n' % mbox[0]['Subject'])
+>		ui.write('State: %s\n' % mbox[0]['State'])
+>		ui.write('-'*70 + '\n')
+>	_write_message(ui, msg, comment)
+>	ui.write('-'*70 + '\n')
+>
+>	# Read the mailbox into the messages and children dictionaries
+>	messages = {}
+>	children = {}
+>	for i in xrange(len(mbox)):
+>		m = mbox[i]
+>		messages[m['Message-Id']] = (i,m)
+>		children.setdefault(m['References'], []).append(m['Message-Id'])
+>	children[None] = []				# Safeguard against infinte loop on empty Message-Id
+>
+>	# Iterate over children
+>	id = msg['Message-Id'] 
+>	id_stack = (id in children and map(lambda x: (x, 1), reversed(children[id]))) or []
+>	if not id_stack: return
+>	ui.write('Comments:\n')
+>	while id_stack:
+>		id,offset = id_stack.pop()
+>		id_stack += (id in children and map(lambda x: (x, offset+1), reversed(children[id]))) or []
+>		index, msg = messages[id]
+>		ui.write('  '*offset + ('%d: ' % index) + msg['Subject'] + '\n')
+>	ui.write('-'*70 + '\n')
+>
+>def _pretty_list(lst):
+>	s = ''
+>	for i in lst:
+>		s += i + ', '
+>	return s[:-2]
+>
+>def _random_id():
+>	return "%x" % random.randint(2**63, 2**64-1)
+>
+>
+>cmdtable = {
+>	'ilist':	(ilist, 
+>				 [('a', 'all', False, 
+>				   'list all issues (by default only those with state new)'),
+>				  ('p', 'property', [], 
+>				   'list issues with specific field values (e.g., -p state=fixed)'),
+>				  ('d', 'date', '', 'restrict to issues matching the date (e.g., -d ">12/28/2007)"'),
+>				  ('f', 'filter', '', 'restrict to pre-defined filter (in %s/%s*)' % (issues_dir, filter_prefix))], 
+>				 _('hg ilist [OPTIONS]')),
+>	'iadd':   	(iadd,  
+>				 [], 
+>				 _('hg iadd [ID] [COMMENT]')),
+>	'ishow':  	(ishow, 
+>				 [('a', 'all', None, 'list all comments')], 
+>				 _('hg ishow [OPTIONS] ID [COMMENT]')),
+>	'iupdate':	(iupdate,
+>				 [('p', 'property', [], 
+>				   'update properties (e.g., -p state=fixed)'),
+>				  ('n', 'no-property-comment', None, 
+>				   'do not add a comment about changed properties')],
+>				 _('hg iupdate [OPTIONS] ID'))
+>}
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M326045P23014Q14.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,45 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-1.8 required=5.0 tests=AWL,BAYES_00 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.174])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07Jmcp6010818
+	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 14:48:38 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu4) with ESMTP (Nemesis)
+	id 0ML21M-1JBxxU3FPf-0006GX; Mon, 07 Jan 2008 20:48:32 +0100
+Message-Id: <BCEC116C-7113-4873-B098-05ACDE943A72@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Mon, 07 Jan 2008 20:48:28 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX180vJsLizprsH5HFEaivi29mMVU4KUVP4sc9S/
+ 8gvUFq6zpfnOswNDLfqqQUEOo71JI9E82xPOCeBWYO6HsENm3e
+ aYk5cubORJodMGX7f+ung==
+Status: RO
+Content-Length: 417
+Lines: 16
+
+Am 06.01.2008 um 07:23 schrieb Dmitriy Morozov:
+> Actually, you can just pull the changes from the Artemis repository on
+> the Web. I've put them there as well.
+>
+
+Thanks, I just pulled to e78a97664dba and ilist is now working without  
+option -d.
+I just checked wether this was a "LANG"-related problem but the error  
+still occurs with LANGs set to:
+de_DE.UTF-8
+C
+en_US
+so it must be something else.
+
+Regards
+Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M328161P23014Q15.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,41 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-2.0 required=5.0 tests=AWL,BAYES_00 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07KpqWA020073
+	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 15:51:52 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu7) with ESMTP (Nemesis)
+	id 0ML2xA-1JBywk2eIR-0000fy; Mon, 07 Jan 2008 21:51:46 +0100
+Message-Id: <1ACD99E9-9100-4C7D-916C-F3516EC9CF66@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Mon, 07 Jan 2008 21:51:45 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX18QJcYZDo6Q37uJfFynUsBGK2GGByd94i/JhM6
+ I9MfCwpXS9DUnY0O0ZIhHan+nj+Gx7xd+r8zc31Da+yFftsUtL
+ FD+bmLhzp9S5QOyPPuamA==
+Status: RO
+Content-Length: 265
+Lines: 12
+
+Hello Dmitry,
+
+I was digging a bit:
+http://www.faqs.org/rfcs/rfc2822.html
+states a numeric timezone must be used (see "A.6.2. Obsolete dates",  
+section 4.3. and 3.3.).
+
+Though I do not understand, why the other dates are parsed without a  
+problem.
+
+Regards
+Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M330420P23014Q16.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,41 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-2.1 required=5.0 tests=AWL,BAYES_00 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07LRYaF024588
+	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 16:27:34 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu0) with ESMTP (Nemesis)
+	id 0MKwh2-1JBzVI0J5o-00061N; Mon, 07 Jan 2008 22:27:28 +0100
+Message-Id: <13CC4175-53F2-4ED1-94E7-075922A02DF7@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Mon, 07 Jan 2008 22:27:28 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX1+U2upB6j5zBd5UJpAi32pjepkqEt+f2EUVXLk
+ 4xDmYE1gswRWEpvxtO62gW/Yu3xHjgzpqm+BfgIVmuRIVXzof7
+ YHAIOIdRdjxlxATKGndHQ==
+Status: RO
+Content-Length: 357
+Lines: 12
+
+I propably found the error:
+
+According to /usr/lib/python2.5/_strptime.py, '%Z' is restricted to:
+[mirko@borg mercurial-crew]$ python -c 'import _strptime; print  
+_strptime.TimeRE()["Z"]'
+(?P<Z>cest|utc|cet|gmt)
+
+This is propably only true for my local timezone as is EST for  
+yours :-). Switching to numeric timezones should do the trick.
+
+Regards
+Mirko
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M332474P23014Q17.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,43 @@
+Return-Path: <mirko@friedenhagen.de>
+X-Spam-Checker-Version: SpamAssassin 3.2.2 (2007-07-23) on one.cs.duke.edu
+X-Spam-Level: 
+X-Spam-Status: No, score=-2.2 required=5.0 tests=AWL,BAYES_00 autolearn=ham
+	version=3.2.2
+Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.177])
+	by one.cs.duke.edu (8.14.0/8.14.0) with ESMTP id m07LVeM0025030
+	for <morozov@cs.duke.edu>; Mon, 7 Jan 2008 16:31:41 -0500 (EST)
+Received: from borg.local (HSI-KBW-085-216-123-176.hsi.kabelbw.de [85.216.123.176])
+	by mrelayeu.kundenserver.de (node=mrelayeu3) with ESMTP (Nemesis)
+	id 0MKxQS-1JBzZH0fuC-0005IG; Mon, 07 Jan 2008 22:31:35 +0100
+Message-Id: <2DBEA693-A27E-40AC-8823-B1A34B9A9A15@friedenhagen.de>
+From: Mirko Friedenhagen <mirko@friedenhagen.de>
+To: Dmitriy Morozov <morozov@cs.duke.edu>
+In-Reply-To: <20080106062303.GA12860@cs.duke.edu>
+Content-Type: text/plain; charset=US-ASCII; format=flowed; delsp=yes
+Content-Transfer-Encoding: 7bit
+Mime-Version: 1.0 (Apple Message framework v915)
+Subject: Re: hg ishow fails in clone of Artemis-repo
+Date: Mon, 07 Jan 2008 22:31:33 +0100
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de> <20080104233436.GA1930@cs.duke.edu> <312C0172-3EF2-488A-A993-CF56C183F9A0@friedenhagen.de> <20080105124708.GA7043@cs.duke.edu> <8B223BED-8A46-4129-A4F3-76B50A890874@friedenhagen.de> <20080106054557.GA12219@cs.duke.edu> <20080106054630.GB12219@cs.duke.edu> <20080106062303.GA12860@cs.duke.edu>
+X-Mailer: Apple Mail (2.915)
+X-Provags-ID: V01U2FsdGVkX19o42Jm2/riqN1HqzwFbS0PWATW9585MzhlPx4
+ nqNci/NQIPJL26EJ9q3ok9stQ9Ydm/zHoAGeVyF7JabTf7Ex5I
+ kv+DcoQ6Zh8mmw9XDGSOw==
+Status: RO
+Content-Length: 496
+Lines: 13
+
+Last one:
+
+[mirko@borg mercurial-crew]$ grep -A3 Z /usr/lib/python2.5/_strptime.py
+         replacement_pairs.extend([(tz, "%Z") for tz_values in  
+self.timezone
+                                                 for tz in tz_values])
+         for offset,directive in ((0,'%c'), (1,'%x'), (2,'%X')):
+             current_format = date_time[offset]
+--
+             'Z': self.__seqToRE((tz for tz_names in  
+self.locale_time.timezone
+                                         for tz in tz_names),
+...
+
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/c568146d3275c22b/new/1208380912.M334583P23014Q18.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,8 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Tue, 08 Jan 2008 13:28:21 -0500
+Subject: properties changes (state)
+Message-Id: <c568146d3275c22b-8f211e40531795e1-artemis@metatron>
+References: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
+In-Reply-To: <D08973FE-D4D3-471D-A0E3-4CE2C4C4677F@friedenhagen.de>
+
+state=fixed
--- a/.issues/ef1b616f31345634	Wed Apr 16 17:21:40 2008 -0400
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,15 +0,0 @@
-From artemis Sat Dec 29 21:16:55 2007
-From: Dmitriy Morozov <morozov@cs.duke.edu>
-Date: Sat, 29 Dec 2007 16:12:43 -0500
-State: new
-Subject: intergration with commit
-Message-Id: <ef1b616f31345634-0-artemis@metatron>
-
-When an issue id is mentioned in a commit message, the text of the
-message should be added as a comment to the issue (prior to the
-commit).
-
-Similarly the subjects of the issues (with their ids) should be added
-to the initial commit message given by mercurial in the editor. They
-should appear in the ignored lines of the message (lines starting with
-HG, so that the user is reminded what issue each id stands for.
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/.issues/ef1b616f31345634/new/1208380912.M349704P23014Q19.metatron	Wed Apr 16 17:22:20 2008 -0400
@@ -0,0 +1,14 @@
+From: Dmitriy Morozov <morozov@cs.duke.edu>
+Date: Sat, 29 Dec 2007 16:12:43 -0500
+State: new
+Subject: intergration with commit
+Message-Id: <ef1b616f31345634-0-artemis@metatron>
+
+When an issue id is mentioned in a commit message, the text of the
+message should be added as a comment to the issue (prior to the
+commit).
+
+Similarly the subjects of the issues (with their ids) should be added
+to the initial commit message given by mercurial in the editor. They
+should appear in the ignored lines of the message (lines starting with
+HG, so that the user is reminded what issue each id stands for.