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
119
120
121
122
123
124
125
126
| | #!perl -w
# Copyright (C) 2020 all contributors <meta@public-inbox.org>
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
# unit tests (no network) for IMAP, see t/imapd.t for end-to-end tests
use strict;
use Test::More;
use PublicInbox::IMAP;
use PublicInbox::IMAPD;
use PublicInbox::TestCommon;
require_mods(qw(DBD::SQLite));
require_git 2.6;
use POSIX qw(strftime);
{
my $parse_date = \&PublicInbox::IMAP::parse_date;
is(strftime('%Y-%m-%d', gmtime($parse_date->('02-Oct-1993'))),
'1993-10-02', 'parse_date works');
is(strftime('%Y-%m-%d', gmtime($parse_date->('2-Oct-1993'))),
'1993-10-02', 'parse_date works w/o leading zero');
is($parse_date->('2-10-1993'), undef, 'bad month');
# from what I can tell, RFC 3501 says nothing about date-month
# case-insensitivity, so be case-sensitive for now
is($parse_date->('02-oct-1993'), undef, 'case-sensitive month');
}
my ($tmpdir, $for_destroy) = tmpdir();
my $cfgfile = "$tmpdir/config";
{
open my $fh, '>', $cfgfile or BAIL_OUT $!;
print $fh <<EOF or BAIL_OUT $!;
[publicinbox "a"]
inboxdir = $tmpdir/a
newsgroup = x.y.z
[publicinbox "b"]
inboxdir = $tmpdir/b
newsgroup = x.z.y
[publicinbox "c"]
inboxdir = $tmpdir/c
newsgroup = IGNORE.THIS
EOF
close $fh or BAIL_OUT $!;
local $ENV{PI_CONFIG} = $cfgfile;
for my $x (qw(a b c)) {
ok(run_script(['-init', '-Lbasic', '-V2', $x, "$tmpdir/$x",
"https://example.com/$x", "$x\@example.com"]),
"init $x");
}
my $imapd = PublicInbox::IMAPD->new;
my @w;
local $SIG{__WARN__} = sub { push @w, @_ };
$imapd->refresh_groups;
my $self = { imapd => $imapd };
is(scalar(@w), 1, 'got a warning for upper-case');
like($w[0], qr/IGNORE\.THIS/, 'warned about upper-case');
my $res = PublicInbox::IMAP::cmd_list($self, 'tag', 'x', '%');
is(scalar($$res =~ tr/\n/\n/), 2, 'only one result');
like($$res, qr/ x\r\ntag OK/, 'saw expected');
$res = PublicInbox::IMAP::cmd_list($self, 'tag', 'x.', '%');
is(scalar($$res =~ tr/\n/\n/), 3, 'only one result');
is(scalar(my @x = ($$res =~ m/ x\.[zy]\r\n/g)), 2, 'match expected');
$res = PublicInbox::IMAP::cmd_list($self, 't', 'x.(?{die "RCE"})', '%');
like($$res, qr/\At OK /, 'refname does not match attempted RCE');
$res = PublicInbox::IMAP::cmd_list($self, 't', '', '(?{die "RCE"})%');
like($$res, qr/\At OK /, 'wildcard does not match attempted RCE');
}
{
my $partial_prepare = \&PublicInbox::IMAP::partial_prepare;
my $x = {};
my $r = $partial_prepare->($x, [], my $p = 'BODY[9]');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[9]<5>');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[9]<5.1>');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[1.1]');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[HEADER.FIELDS (DATE FROM)]');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[HEADER.FIELDS.NOT (TO)]');
ok($r, $p);
$r = $partial_prepare->($x, [], $p = 'BODY[HEDDER.FIELDS.NOT (TO)]');
ok(!$r, "rejected misspelling $p");
$r = $partial_prepare->($x, [], $p = 'BODY[1.1.HEADER.FIELDS (TO)]');
ok($r, $p);
my $partial_body = \&PublicInbox::IMAP::partial_body;
my $partial_hdr_get = \&PublicInbox::IMAP::partial_hdr_get;
my $partial_hdr_not = \&PublicInbox::IMAP::partial_hdr_not;
my $hdrs_regexp = \&PublicInbox::IMAP::hdrs_regexp;
is_deeply($x, {
'BODY[9]' => [ $partial_body, 9, undef, undef, undef ],
'BODY[9]<5>' => [ $partial_body, 9, undef, 5, undef ],
'BODY[9]<5.1>' => [ $partial_body, 9, undef, 5, 1 ],
'BODY[1.1]' => [ $partial_body, '1.1', undef, undef, undef ],
'BODY[HEADER.FIELDS (DATE FROM)]' => [ $partial_hdr_get,
undef, $hdrs_regexp->('DATE FROM'),
undef, undef ],
'BODY[HEADER.FIELDS.NOT (TO)]' => [ $partial_hdr_not,
undef, $hdrs_regexp->('TO'),
undef, undef ],
'BODY[1.1.HEADER.FIELDS (TO)]' => [ $partial_hdr_get,
'1.1', $hdrs_regexp->('TO'),
undef, undef ],
}, 'structure matches expected');
}
{
my $fetch_compile = \&PublicInbox::IMAP::fetch_compile;
my ($cb, $ops, $partial) = $fetch_compile->(['BODY[]']);
is($partial, undef, 'no partial fetch data');
is_deeply($ops,
[ 'BODY[]', \&PublicInbox::IMAP::emit_rfc822 ],
'proper key and op compiled for BODY[]');
($cb, $ops, $partial) = $fetch_compile->(['BODY', 'BODY[]']);
is_deeply($ops, [
'BODY[]', \&PublicInbox::IMAP::emit_rfc822,
undef, \&PublicInbox::IMAP::op_eml_new,
'BODY', \&PublicInbox::IMAP::emit_body,
], 'placed op_eml_new before emit_body');
}
done_testing;
|