Я запускаю почтовый сервер, на котором у меня работают курьерские imap и pop3. Недавно я переместил некоторые локальные сообщения со своей машины (ранее загруженные с помощью pop3) на сервер, чтобы использовать imap. Для этого я использовал Mail.app в Mac OS X 10.5.8.
В результате на почтовом сервере появился ряд файлов, которые выглядят следующим образом:
1324697191.M91227P15574V000000000000CA00I0004B07D_556.hostname,S=5622:2,S
1324697192.M322096P15574V000000000000CA00I0004B07F_557.hostname,S=225691:2,RS
1324697196.M144018P15574V000000000000CA00I0004B081_558.hostname,S=7702:2,RS
1324697197.M715598P15574V000000000000CA00I0004B083_559.hostname,S=15741:2,S
1324697199.M327587P15574V000000000000CA00I0004B085_560.hostname,S=8744:2,RS
Время получения этих сообщений в порядке, указанном выше:
01/15/2010
01/09/2009
07/13/2010
02/21/2010
05/06/2010
Теперь это не проблема с настольными почтовыми клиентами - они просто получают все сообщения, сортируют их, и все в порядке. Несколько поисков в Google по этой теме дали цепочки, в которых люди пытались получить свои сообщения в правильном порядке, и предложенным средством было «отсортировать их на стороне клиента».
К сожалению, в iOS 5.0.1 почтовое приложение предполагает, что порядок сообщений, которые оно получает с сервера imap, уже отсортирован по дате. Из-за того, что сообщения в этом примере имеют имя файла, которое не соответствует их метке времени получения, а вместо этого соответствует метке времени повторной загрузки, это означает, что сообщения отображаются в неправильном порядке на устройстве iOS. Чтобы исправить это, мне нужно загрузить ВСЮ почту, несколько раз нажав кнопку «Загрузить больше сообщений».
Я хотел бы просто иметь возможность перестроить эти файлы сообщений так, чтобы отметка времени в имени файла (которая выглядит как первый набор цифр перед точкой) соответствовала отметке времени получения. Затем они будут отсортированы в правильном порядке, когда устройство iOS пытается их загрузить. Я недостаточно знаю, как курьер организует сообщения - могу ли я просто написать сценарий, который заменяет первый набор цифр в имени файла на временную метку unix времени получения сообщения?
Спасибо!
Спасибо @mailq. Оказывается, что помимо метки времени в начале имени файла сообщения, courier также смотрит на метку времени модификации самого файла. Мне также пришлось установить это на полученную метку времени.
Вот сценарий, который, учитывая кучу сообщений, копирует их в выходной каталог после переписывания имени файла и времени модификации, чтобы они содержали метку времени получения.
#!/usr/bin/env perl
use Email::Simple;
use Date::Parse;
use Getopt::Long;
use File::Path qw(make_path);
use File::Copy qw(copy);
use File::stat;
my $outfolder = "";
$result = GetOptions("output-dir=s" => \$outfolder);
# create directories if needed
if (length($outfolder) > 0) {
make_path($outfolder);
}
foreach my $file (@ARGV) {
my $text = "";
# read file as one string
{
local $/=undef;
open FILE, "$file" or die "Couldn't open file: $!";
binmode FILE;
$text = <FILE>;
close FILE;
}
# use Email::Simple to parse the Received header
my $email = Email::Simple->new($text);
my @received = $email->header("Received");
# Find the latest receive time
my $latestTime = 0;
my $latestTimeStr = "";
foreach my $r (@received) {
if ($r =~ /[^;]*;(.*)$/) {
my $time = str2time($1);
if ($time > $latestTime) {
$latestTime = $time;
$latestTimeStr = $1;
}
}
}
# if this is a sent message, it doesn't have a received header. Use the
# Date header instead.
if ($latestTime == 0) {
my $date = $email->header("Date");
my $time = str2time($date);
if ($time > $latestTime) {
$latestTime = $time;
$latestTimeStr = $date;
}
}
# If we found one, rename or tell about the rename
if ($latestTime != 0) {
if ($file =~ /([0-9]*)(\..*$)/) {
my $newfilename = $latestTime . $2;
if (length($outfolder) == 0) {
print "Would Copy $file ($latestTimeStr) -> \n ";
print "$newfilename\n";
} else {
print "Copied $file ($latestTimeStr) -> \n ";
print "$outfolder/$newfilename\n";
# use the latest received timestamp as the atime and mtime
copy($file, "$outfolder/$newfilename");
utime($latestTime, $latestTime, "$outfolder/$newfilename");
}
}
} else {
print "Couldn't find receive time for " . $file . "\n";
}
}
Используйте такой сценарий:
perl rename.pl cur/*
Когда вы уверены, что он поступит правильно:
perl rename.pl cur/* --output-dir cur_renamed
Тогда вам просто нужно поменять местами cur_renamed
с участием cur
удалите ваш courierimapuiddb
файл и, возможно, перезапустите ваши почтовые клиенты. Для моего устройства iOS мне пришлось удалить учетную запись электронной почты, а затем выполнить повторную синхронизацию с iTunes, чтобы она правильно очистила кеш.