public-inbox.git  about / heads / tags
an "archives first" approach to mailing lists
blob 3b6cb29926eee322355365631ff636c3d261eb3b 4557 bytes (raw)
$ git show HEAD:t/lei-import-imap.t	# shows this blob on the CLI

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
 
#!perl -w
# Copyright (C) 2021 all contributors <meta@public-inbox.org>
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
use strict; use v5.10.1; use PublicInbox::TestCommon;
require_mods(qw(lei -imapd Mail::IMAPClient));
my ($ro_home, $cfg_path) = setup_public_inboxes;
my ($tmpdir, $for_destroy) = tmpdir;
my $sock = tcp_server;
my $cmd = [ '-imapd', '-W0', "--stdout=$tmpdir/1", "--stderr=$tmpdir/2" ];
my $env = { PI_CONFIG => $cfg_path };
my $td = start_script($cmd, $env, { 3 => $sock }) or BAIL_OUT("-imapd: $?");
my $host_port = tcp_host_port($sock);
undef $sock;
test_lei({ tmpdir => $tmpdir }, sub {
	my $url = "imap://$host_port/t.v2.0";
	my $url_orig = $url;
	lei_ok(qw(ls-mail-source), "imap://$host_port/");
	like($lei_out, qr/^t\.v2\.0$/ms, 'shows mailbox');
	lei_ok(qw(ls-mail-source), $url);
	is($lei_out, "t.v2.0\n", 'shows only mailbox with filter');
	lei_ok(qw(ls-mail-source -l), "imap://$host_port/");
	is(ref(json_utf8->decode($lei_out)), 'ARRAY', 'ls-mail-source JSON');

	lei_ok(qw(q z:1..));
	my $out = json_utf8->decode($lei_out);
	is_deeply($out, [ undef ], 'nothing imported, yet');

	lei_ok('inspect', $url);
	is_deeply(json_utf8->decode($lei_out), {}, 'no inspect stats, yet');

	lei_ok('import', $url);
	lei_ok('inspect', $url);
	my $res = json_utf8->decode($lei_out);
	is(scalar keys %$res, 1, 'got one key in inspect URL');
	my $re = qr!\Aimap://;AUTH=ANONYMOUS\@\Q$host_port\E
			/t\.v2\.0;UIDVALIDITY=\d+!x;
	like((keys %$res)[0], qr/$re\z/, 'got expanded key');

	lei_ok 'ls-mail-sync';
	like($lei_out, qr!$re\n\z!, 'ls-mail-sync');
	chomp(my $u = $lei_out);
	lei_ok('import', $u, \'UIDVALIDITY match in URL');
	$url = $u;
	$u =~ s/;UIDVALIDITY=(\d+)\s*/;UIDVALIDITY=9$1/s;
	ok(!lei('import', $u), 'UIDVALIDITY mismatch in URL rejected');
	like($lei_err, qr/UIDVALIDITY mismatch/, 'mismatch noted');

	lei_ok('inspect', $url);
	my $inspect = json_utf8->decode($lei_out);
	my @k = keys %$inspect;
	is(scalar(@k), 1, 'one URL resolved');
	is($k[0], $url, 'inspect URL matches');
	my $stats = $inspect->{$k[0]};
	is_deeply([ sort keys %$stats ],
		[ qw(uid.count uid.max uid.min) ], 'keys match');
	ok($stats->{'uid.min'} < $stats->{'uid.max'}, 'min < max');
	ok($stats->{'uid.count'} > 0, 'count > 0');

	lei_ok('lcat', $url);
	is(scalar(grep(/^# blob:/, split(/\n/ms, $lei_out))),
		$stats->{'uid.count'}, 'lcat on URL dumps folder');
	lei_ok qw(lcat -f json), $url;
	$out = json_utf8->decode($lei_out);
	is(scalar(@$out) - 1, $stats->{'uid.count'}, 'lcat JSON dumps folder');

	lei_ok(qw(q z:1..));
	$out = json_utf8->decode($lei_out);
	ok(scalar(@$out) > 1, 'got imported messages');
	is(pop @$out, undef, 'trailing JSON null element was null');
	my %r;
	for (@$out) { $r{ref($_)}++ }
	is_deeply(\%r, { 'HASH' => scalar(@$out) }, 'all hashes');
	lei_ok([qw(tag +kw:seen), $url], undef, undef);

	my $f = "$ENV{HOME}/.local/share/lei/store/mail_sync.sqlite3";
	ok(-s $f, 'mail_sync tracked for redundant imports');
	lei_ok('inspect', "blob:$out->[5]->{blob}");
	my $x = json_utf8->decode($lei_out);
	is(ref($x->{'lei/store'}), 'ARRAY', 'lei/store in inspect');
	is(ref($x->{'mail-sync'}), 'HASH', 'sync in inspect');
	is(ref($x->{'mail-sync'}->{$k[0]}), 'ARRAY', 'UID arrays in inspect')
		or diag explain($x);

	my $psgi_attach = 'cfa3622cbeffc9bd6b0fc66c4d60d420ba74f60d';
	lei_ok('blob', $psgi_attach);
	like($lei_out, qr!^Content-Type: multipart/mixed;!sm, 'got full blob');
	lei_ok('blob', "$psgi_attach:2");
	is($lei_out, "b64\xde\xad\xbe\xef\n", 'got attachment');

	lei_ok 'forget-mail-sync', $url;
	lei_ok 'ls-mail-sync';
	unlike($lei_out, qr!\Q$host_port\E!, 'sync info gone after forget');
	my $uid_url = "$url/;UID=".$stats->{'uid.max'};
	lei_ok 'import', $uid_url;
	lei_ok 'ls-mail-sync';
	is($lei_out, "$url\n", 'ls-mail-sync added URL w/o UID');
	lei_ok 'inspect', $uid_url;
	$lei_out =~ /([a-f0-9]{40,})/ or
		xbail 'inspect missed blob with UID URL';
	my $blob = $1;
	lei_ok 'lcat', $uid_url;
	like $lei_out, qr/^Subject: /sm,
		'lcat shows mail text with UID URL';
	like $lei_out, qr/\bblob:$blob\b/, 'lcat showed blob';
	my $orig = $lei_out;
	lei_ok 'lcat', "blob:$blob";
	is($lei_out, $orig, 'lcat understands blob:...');
	lei_ok qw(lcat -f json), $uid_url;
	$out = json_utf8->decode($lei_out);
	is(scalar(@$out), 2, 'got JSON') or diag explain($out);
	lei_ok qw(lcat), $url_orig;
	is($lei_out, $orig, 'lcat w/o UID works');

	ok(!lei(qw(export-kw), $url_orig), 'export-kw fails on read-only IMAP');
	like($lei_err, qr/does not support/, 'error noted in failure');
});

done_testing;

git clone https://public-inbox.org/public-inbox.git
git clone http://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/public-inbox.git