Initial release. 0.1
authorPhilip Mak <pmak@aaanime.net>
Tue, 25 Dec 2001 01:16:00 +0000 (02:16 +0100)
committerEmmanuel Lacour <elacour@home-dn.net>
Tue, 25 Dec 2001 01:16:00 +0000 (02:16 +0100)
CHANGES [new file with mode: 0644]
README [new file with mode: 0644]
perfect_maildir.pl [new file with mode: 0755]

diff --git a/CHANGES b/CHANGES
new file mode 100644 (file)
index 0000000..32cf35f
--- /dev/null
+++ b/CHANGES
@@ -0,0 +1,5 @@
+v0.1 Tue, 25 Dec 2001 02:16:15 -0800
+
+Philip Mak <pmak@aaanime.net>
+
+- Initial release
diff --git a/README b/README
new file mode 100644 (file)
index 0000000..abce7a0
--- /dev/null
+++ b/README
@@ -0,0 +1,22 @@
+Usage: perfect_maildir ~/Maildir < mbox
+
+I wrote this script after being unsatisfied with existing mbox to
+maildir converters. By making it "Simple", code complexity is kept
+low thus making it easy to program and debug. At the same time,
+since it only converts one mbox at a time, it is perfect for use in
+a shell ``for'' loop (for example).
+
+As for being "Perfect", to the best of my knowledge this script does
+the conversion correctly in all cases; it will translate "Status"
+and "X-Status" fields into maildir info, and it correctly detects
+where messages begin and end. (This is only version 0.1 so I may
+have messed something up though. Please send me feedback!)
+
+NOTE: The MUA ``mutt'' has a bug/feature where in the message index,
+it claims that all maildir messages have 0 lines unless they have a
+"Lines:" header set. perfect_maildir does not attempt to add the
+"Lines:" header; you may want to reconfigure ``mutt' to display byte
+size instead of lines instead by adding the following line to your
+~/.muttrc file:
+#
+set index_format="%4C %Z %{%b %d} %-15.15L (%4c) %s"
diff --git a/perfect_maildir.pl b/perfect_maildir.pl
new file mode 100755 (executable)
index 0000000..5f5d13e
--- /dev/null
@@ -0,0 +1,106 @@
+#!/usr/bin/perl
+
+# "Simple but Perfect" mbox to Maildir converter v0.1
+# by Philip Mak <pmak@aaanime.net>
+
+# Usage: perfect_maildir ~/Maildir < mbox
+
+# Simple  - only converts one mbox (can use script in one-liners)
+# Perfect - message Flags/X-Flags are converted; "^>From ." line is unescaped
+
+# I wrote this script after being unsatisfied with existing mbox to
+# maildir converters. By making it "Simple", code complexity is kept
+# low thus making it easy to program and debug. At the same time,
+# since it only converts one mbox at a time, it is perfect for use in
+# a shell `for'' loop (for example).
+
+# As for being "Perfect", to the best of my knowledge this script does
+# the conversion correctly in all cases; it will translate "Status"
+# and "X-Status" fields into maildir info, and it correctly detects
+# where messages begin and end. (This is only version 0.1 so I may
+# have messed something up though. Please send me feedback!)
+
+# NOTE: The MUA `mutt'' has a bug/feature where in the message index,
+# it claims that all maildir messages have 0 lines unless they have a
+# "Lines:" header set. perfect_maildir does not attempt to add the
+# "Lines:" header; you may want to reconfigure `mutt' to display byte
+# size instead of lines instead by adding the following line to your
+# ~/.muttrc file:
+#
+# set index_format="%4C %Z %{%b %d} %-15.15L (%4c) %s"
+
+# check for valid arguments
+my ($maildir) = @ARGV;
+if (!$maildir) {
+  print STDERR "Usage: perfect_maildir ~/Maildir < mbox\n";
+  exit 1;
+}
+
+# check for writable maildir
+unless (-w "$maildir/cur") {
+  print STDERR "Cannot write to $maildir/cur\n";
+  exit 1;
+}
+unless (-w "$maildir/new") {
+  print STDERR "Cannot write to $maildir/new\n";
+  exit 1;
+}
+
+my $num = 0;
+my $time = time;
+
+repeat:
+
+# read header
+my $headers = '';
+my $flags = '';
+my $subject = '';
+while (my $line = <STDIN>) {
+  # detect end of headers
+  last if $line eq "\n";
+
+  # strip "From" line from header
+  $headers .= $line unless $line =~ /^From ./;
+
+  # detect flags
+  $flags .= $1 if $line =~ /^Status: ([A-Z]+)/;
+  $flags .= $1 if $line =~ /^X-Status: ([A-Z]+)/;
+  $subject = $1 if $line =~ /^Subject: (.*)$/;
+}
+$num++;
+
+# open output file
+my $file;
+if ($flags =~ /O/) {
+  $file = "$maildir/cur/$time.$num.$ENV{HOSTNAME}";
+  my $extra = '';
+  $extra .= 'F' if $flags =~ /F/; # flagged
+  $extra .= 'R' if $flags =~ /A/; # replied
+  $extra .= 'S' if $flags =~ /R/; # seen
+  $extra .= 'T' if $flags =~ /D/; # trashed
+  $file .= ":2,$extra" if $extra;
+} else {
+  $file = "$maildir/new/$time.$num.$ENV{HOSTNAME}";
+}
+
+# filter out the "DON'T DELETE THIS MESSAGE -- FOLDER INTERNAL DATA" message
+$file = '/dev/null' if ($num == 1 and $subject eq "DON'T DELETE THIS MESSAGE -- FOLDER
+INTERNAL DATA");
+
+open(FILE, ">$file");
+print FILE "$headers\n";
+while (my $line = <STDIN>) {
+  # detect end of message
+  last if $line =~ /^From ./;
+
+  # unescape "From"
+  $line =~ s/^>From (.)/From $1/;
+
+  print FILE $line;
+}
+close(FILE);
+
+goto repeat unless eof(STDIN);
+
+my $elapsed = time - $time;
+print "Inserted $num messages into maildir $maildir in $elapsed seconds\n";