* [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe
@ 2023-11-11 9:04 Eric Wong
2023-11-11 9:04 ` [PATCH 1/4] learn: fix redundant ham import on dual matches Eric Wong
` (3 more replies)
0 siblings, 4 replies; 5+ messages in thread
From: Eric Wong @ 2023-11-11 9:04 UTC (permalink / raw)
To: meta
Patch 2 is the important one in this series to drop unique
tokens from public archives.
I noticed our unsubscribe.milter example was adopted on one of
the lists I'm subscribed to, and didn't want my independent
public archive of that list to get sabotaged by somebody else
unsubscribing me via HTTPS.
Oddly, the rest of the unsubscription process via
unsubscribe.psgi (HTTPS) or mailto: didn't seem configured by
that list I'm subscribed to. IOW, they're only running
unsubscribe.milter so far and generating List-Unsubscribe
headers which don't work...
1, 3 and 4 fix some small things I noticed while working on 2.
Eric Wong (4):
learn: fix redundant ham import on dual matches
mda|learn|watch: support dropUniqueUnsubscribe config
mda: fix and test some usage problems
doc: update README.unsubscribe
Documentation/public-inbox-config.pod | 17 ++++++++
Documentation/public-inbox-learn.pod | 19 +++++++++
Documentation/public-inbox-mda.pod | 18 +++++++-
Documentation/public-inbox-watch.pod | 6 ++-
examples/README.unsubscribe | 9 ++--
lib/PublicInbox/Import.pm | 27 ++++++++++++
lib/PublicInbox/LeiToMail.pm | 6 +++
lib/PublicInbox/Watch.pm | 1 +
script/public-inbox-learn | 7 +++-
script/public-inbox-mda | 11 ++++-
script/public-inbox-watch | 2 +
t/lei-import.t | 48 +++++++++++++++++++++-
t/mda.t | 59 +++++++++++++++++++++++++--
t/watch_maildir.t | 30 ++++++++++++--
14 files changed, 242 insertions(+), 18 deletions(-)
^ permalink raw reply [flat|nested] 5+ messages in thread
* [PATCH 1/4] learn: fix redundant ham import on dual matches
2023-11-11 9:04 [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe Eric Wong
@ 2023-11-11 9:04 ` Eric Wong
2023-11-11 9:04 ` [PATCH 2/4] mda|learn|watch: support dropUniqueUnsubscribe config Eric Wong
` (2 subsequent siblings)
3 siblings, 0 replies; 5+ messages in thread
From: Eric Wong @ 2023-11-11 9:04 UTC (permalink / raw)
To: meta
When learning and injecting new messages ham, we want to avoid
wasting cycles importing the same message into an inbox twice
(once for the To/Cc match and once for the List-Id match). Our
existing %seen hash turned out to be ineffective since
PublicInbox::Inbox refs get re-blessed to PublicInbox::InboxWritable.
So we stop letting class name influence the hash key for tracking by
using the reference address instead. We can get the reference address
by performing an arithmetic operation (+ 0) instead of having to
pay the cost of importing Scalar::Util::refaddr.
---
script/public-inbox-learn | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/script/public-inbox-learn b/script/public-inbox-learn
index 54d31cb6..8069d919 100755
--- a/script/public-inbox-learn
+++ b/script/public-inbox-learn
@@ -110,12 +110,12 @@ if ($train eq 'spam' || ($train eq 'rm' && $opt{all})) {
my %seen;
while (my ($addr, $ibx) = each %dests) {
next unless ref($ibx); # $ibx may be 0
- next if $seen{"$ibx"}++;
+ next if $seen{0 + $ibx}++;
remove_or_add($ibx, $train, $mime, $addr);
}
my $dests = PublicInbox::MDA->inboxes_for_list_id($pi_cfg, $mime);
for my $ibx (@$dests) {
- next if $seen{"$ibx"}++;
+ next if $seen{0 + $ibx}++;
remove_or_add($ibx, $train, $mime, $ibx->{-primary_address});
}
}
^ permalink raw reply related [flat|nested] 5+ messages in thread
* [PATCH 2/4] mda|learn|watch: support dropUniqueUnsubscribe config
2023-11-11 9:04 [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe Eric Wong
2023-11-11 9:04 ` [PATCH 1/4] learn: fix redundant ham import on dual matches Eric Wong
@ 2023-11-11 9:04 ` Eric Wong
2023-11-11 9:04 ` [PATCH 3/4] mda: fix and test some usage problems Eric Wong
2023-11-11 9:04 ` [PATCH 4/4] doc: update README.unsubscribe Eric Wong
3 siblings, 0 replies; 5+ messages in thread
From: Eric Wong @ 2023-11-11 9:04 UTC (permalink / raw)
To: meta
List-Unsubscribe headers with unique identifiers (such as those
generated by our examples/unsubscribe.milter) should not
end up in public archives. Add a new config knob to strip
List-Unsubscribe headers if they have the
`List-Unsubscribe-Post: List-Unsubscribe=One-Click'
header.
Unfortunately, this breaks DKIM signatures if the signature
covers either of these List-Unsubscribe* headers. However,
breaking DKIM is the lesser evil compared to any archive reader
being able to stop archival by an independent archivist.
As much as I would like this to be the default, it probably
affects few users at the moment since very few mailing lists
use unique identifiers in List-Unsubscribe (but that number
has grown, recently).
---
Documentation/public-inbox-config.pod | 17 ++++++++++
Documentation/public-inbox-learn.pod | 19 +++++++++++
Documentation/public-inbox-mda.pod | 18 +++++++++-
Documentation/public-inbox-watch.pod | 6 +++-
lib/PublicInbox/Import.pm | 27 +++++++++++++++
lib/PublicInbox/LeiToMail.pm | 6 ++++
lib/PublicInbox/Watch.pm | 1 +
script/public-inbox-learn | 3 ++
script/public-inbox-mda | 4 +++
script/public-inbox-watch | 2 ++
t/lei-import.t | 48 ++++++++++++++++++++++++++-
t/mda.t | 41 ++++++++++++++++++++---
t/watch_maildir.t | 30 +++++++++++++++--
13 files changed, 212 insertions(+), 10 deletions(-)
diff --git a/Documentation/public-inbox-config.pod b/Documentation/public-inbox-config.pod
index 871ac6c5..1ef7f46f 100644
--- a/Documentation/public-inbox-config.pod
+++ b/Documentation/public-inbox-config.pod
@@ -196,6 +196,23 @@ and the path may be "/dev/null" or any empty file.
Multiple files may be specified and will be included in the
order specified.
+=item publicinboxImport.dropUniqueUnsubscribe
+
+Drop C<List-Unsubscribe> headers if the message also includes
+the C<List-Unsubscribe-Post: List-Unsubscribe=One-Click> header
+to signal MUAs to support an instantaneous unsubscribe. This
+is strongly recommended for users creating their own public
+archives of mailing lists they subscribe to, otherwise any
+archive reader can unsubscribe the archivist.
+
+This may break DKIM signatures if the C<List-Unsubscribe*>
+headers are signed, but breaking DKIM signatures is the
+lesser evil compared to allowing any reader to unsubscribe
+the archivist.
+
+This affects L<public-inbox-mda(1)>, L<public-inbox-watch(1)>,
+and L<public-inbox-learn(1)>
+
=item publicinboxmda.spamcheck
This may be set to C<none> to disable the use of SpamAssassin
diff --git a/Documentation/public-inbox-learn.pod b/Documentation/public-inbox-learn.pod
index f776df6b..b08e4bc8 100644
--- a/Documentation/public-inbox-learn.pod
+++ b/Documentation/public-inbox-learn.pod
@@ -73,6 +73,25 @@ Default: ~/.public-inbox/config
=back
+=head1 CONFIGURATION
+
+These configuration knobs should be used in the
+L<public-inbox-config(5)> file.
+
+=over 8
+
+=item publicinboxImport.dropUniqueUnsubscribe
+
+=item publicinbox.<name>.address
+
+=item publicinbox.<name>.listid
+
+=item publicinboxmda.spamcheck
+
+See L<public-inbox-config(5)> for descriptions of these options
+
+=back
+
=head1 CONTACT
Feedback welcome via plain-text mail to L<mailto:meta@public-inbox.org>
diff --git a/Documentation/public-inbox-mda.pod b/Documentation/public-inbox-mda.pod
index 93cb0e9c..edc90287 100644
--- a/Documentation/public-inbox-mda.pod
+++ b/Documentation/public-inbox-mda.pod
@@ -68,6 +68,22 @@ Default: ~/.public-inbox/emergency/
=back
+=head1 CONFIGURATION
+
+Various configuration knobs should be used in the
+L<public-inbox-config(5)> file.
+
+=over 8
+
+=item publicinboxImport.dropUniqueUnsubscribe
+
+=item publicinbox.<name>.address
+
+=item publicinbox.<name>.listid
+
+See L<public-inbox-config(5)> for descriptions of these options
+
+=back
=head1 CONTACT
@@ -78,7 +94,7 @@ L<http://4uok3hntl7oi7b4uf4rtfwefqeexfzil2w6kgk2jn5z2f764irre7byd.onion/meta/>
=head1 COPYRIGHT
-Copyright 2013-2021 all contributors L<mailto:meta@public-inbox.org>
+Copyright all contributors L<mailto:meta@public-inbox.org>
License: AGPL-3.0+ L<https://www.gnu.org/licenses/agpl-3.0.txt>
diff --git a/Documentation/public-inbox-watch.pod b/Documentation/public-inbox-watch.pod
index febda0b1..7c21f7ce 100644
--- a/Documentation/public-inbox-watch.pod
+++ b/Documentation/public-inbox-watch.pod
@@ -66,6 +66,10 @@ L<public-inbox-config(5)> file.
=over 8
+=item publicinboxImport.dropUniqueUnsubscribe
+
+See L<public-inbox-config(5)/publicinboxImport.dropUniqueUnsubscribe>
+
=item publicinbox.<name>.watch
A location to watch. public-inbox 1.5.0 and earlier only supported
@@ -201,7 +205,7 @@ L<http://4uok3hntl7oi7b4uf4rtfwefqeexfzil2w6kgk2jn5z2f764irre7byd.onion/meta/>
=head1 COPYRIGHT
-Copyright 2016-2021 all contributors L<mailto:meta@public-inbox.org>
+Copyright all contributors L<mailto:meta@public-inbox.org>
License: AGPL-3.0+ L<https://www.gnu.org/licenses/agpl-3.0.txt>
diff --git a/lib/PublicInbox/Import.pm b/lib/PublicInbox/Import.pm
index 2d60db55..e4f8615e 100644
--- a/lib/PublicInbox/Import.pm
+++ b/lib/PublicInbox/Import.pm
@@ -321,11 +321,38 @@ sub extract_cmt_info ($;$) {
# kill potentially confusing/misleading headers
our @UNWANTED_HEADERS = (qw(Bytes Lines Content-Length),
qw(Status X-Status));
+our $DROP_UNIQUE_UNSUB;
sub drop_unwanted_headers ($) {
my ($eml) = @_;
for (@UNWANTED_HEADERS, @PublicInbox::MDA::BAD_HEADERS) {
$eml->header_set($_);
}
+
+ # We don't want public-inbox readers to be able to unsubcribe the
+ # address which does archiving. WARNING: this breaks DKIM if the
+ # mailing list sender follows RFC 8058, section 4; but breaking DKIM
+ # (or have senders ignore RFC 8058 sec. 4) is preferable to having
+ # saboteurs unsubscribing independent archivists:
+ if ($DROP_UNIQUE_UNSUB && grep(/\AList-Unsubscribe=One-Click\z/,
+ $eml->header_raw('List-Unsubscribe-Post'))) {
+ for (qw(List-Unsubscribe-Post List-Unsubscribe)) {
+ $eml->header_set($_)
+ }
+ }
+}
+
+sub load_config ($;$) {
+ my ($cfg, $do_exit) = @_;
+ my $v = $cfg->{lc 'publicinboxImport.dropUniqueUnsubscribe'};
+ if (defined $v) {
+ $DROP_UNIQUE_UNSUB = $cfg->git_bool($v) // do {
+ warn <<EOM;
+E: publicinboxImport.dropUniqueUnsubscribe=$v in $cfg->{-f} is not boolean
+EOM
+ $do_exit //= \&CORE::exit;
+ $do_exit->(78); # EX_CONFIG
+ };
+ }
}
# used by V2Writable, too
diff --git a/lib/PublicInbox/LeiToMail.pm b/lib/PublicInbox/LeiToMail.pm
index b73af68a..0d2f586a 100644
--- a/lib/PublicInbox/LeiToMail.pm
+++ b/lib/PublicInbox/LeiToMail.pm
@@ -10,6 +10,7 @@ use PublicInbox::Eml;
use PublicInbox::IO;
use PublicInbox::Git;
use PublicInbox::Spawn qw(spawn);
+use PublicInbox::Import;
use IO::Handle; # ->autoflush
use Fcntl qw(SEEK_SET SEEK_END O_CREAT O_EXCL O_WRONLY);
use PublicInbox::Syscall qw(rename_noreplace);
@@ -672,6 +673,11 @@ sub _pre_augment_v2 {
});
}
PublicInbox::InboxWritable->new($ibx, @creat);
+ local $PublicInbox::Import::DROP_UNIQUE_UNSUB; # only for workers
+ PublicInbox::Import::load_config(PublicInbox::Config->new, sub {
+ $lei->x_it(shift);
+ die "E: can't write v2 inbox with broken config\n";
+ });
$ibx->init_inbox if @creat;
my $v2w = $ibx->importer;
$v2w->wq_workers_start("lei/v2w $dir", 1, $lei->oldset, {lei => $lei},
diff --git a/lib/PublicInbox/Watch.pm b/lib/PublicInbox/Watch.pm
index 1cdf12a5..5253ec94 100644
--- a/lib/PublicInbox/Watch.pm
+++ b/lib/PublicInbox/Watch.pm
@@ -45,6 +45,7 @@ sub new {
my (%mdmap);
my (%imap, %nntp); # url => [inbox objects] or 'watchspam'
my (@imap, @nntp);
+ PublicInbox::Import::load_config($cfg);
# "publicinboxwatch" is the documented namespace
# "publicinboxlearn" is legacy but may be supported
diff --git a/script/public-inbox-learn b/script/public-inbox-learn
index 8069d919..6a1bc890 100755
--- a/script/public-inbox-learn
+++ b/script/public-inbox-learn
@@ -28,6 +28,7 @@ use PublicInbox::Spamcheck::Spamc;
use Getopt::Long qw(:config gnu_getopt no_ignore_case auto_abbrev);
my %opt = (all => 0);
GetOptions(\%opt, qw(all help|h)) or die $help;
+use PublicInbox::Import;
my $train = shift or die $help;
if ($train !~ /\A(?:ham|spam|rm)\z/) {
@@ -37,6 +38,8 @@ die "--all only works with `rm'\n" if $opt{all} && $train ne 'rm';
my $spamc = PublicInbox::Spamcheck::Spamc->new;
my $pi_cfg = PublicInbox::Config->new;
+local $PublicInbox::Import::DROP_UNIQUE_UNSUB;
+PublicInbox::Import::load_config($pi_cfg);
my $err;
my $mime = PublicInbox::Eml->new(do{
defined(my $data = do { local $/; <STDIN> }) or die "read STDIN: $!\n";
diff --git a/script/public-inbox-mda b/script/public-inbox-mda
index cac819ac..04fd8aad 100755
--- a/script/public-inbox-mda
+++ b/script/public-inbox-mda
@@ -16,6 +16,8 @@ use strict;
use Getopt::Long qw(:config gnu_getopt no_ignore_case auto_abbrev);
my ($ems, $emm, $show_help);
my $precheck = 1;
+use PublicInbox::Import;
+local $PublicInbox::Import::DROP_UNIQUE_UNSUB; # does this need a CLI switch?
GetOptions('precheck!' => \$precheck, 'help|h' => \$show_help) or
do { print STDERR $help; exit 1 };
@@ -47,6 +49,8 @@ my $key = 'publicinboxmda.spamcheck';
my $default = 'PublicInbox::Spamcheck::Spamc';
my $spamc = PublicInbox::Spamcheck::get($cfg, $key, $default);
my $dests = [];
+PublicInbox::Import::load_config($cfg, $do_exit);
+
my $recipient = $ENV{ORIGINAL_RECIPIENT};
if (defined $recipient) {
my $ibx = $cfg->lookup($recipient); # first check
diff --git a/script/public-inbox-watch b/script/public-inbox-watch
index d9215de9..9bcd42ed 100755
--- a/script/public-inbox-watch
+++ b/script/public-inbox-watch
@@ -11,6 +11,8 @@ use strict;
use Getopt::Long qw(:config gnu_getopt no_ignore_case auto_abbrev);
use IO::Handle; # ->autoflush
use PublicInbox::Watch;
+use PublicInbox::Import;
+local $PublicInbox::Import::DROP_UNIQUE_UNSUB;
use PublicInbox::Config;
use PublicInbox::DS;
my $do_scan = 1;
diff --git a/t/lei-import.t b/t/lei-import.t
index 1edd607d..bd562617 100644
--- a/t/lei-import.t
+++ b/t/lei-import.t
@@ -3,7 +3,8 @@
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
use v5.12; use PublicInbox::TestCommon;
use PublicInbox::DS qw(now);
-use autodie qw(open close);
+use PublicInbox::IO qw(write_file);
+use autodie qw(open close truncate);
test_lei(sub {
ok(!lei(qw(import -F bogus), 't/plack-qp.eml'), 'fails with bogus format');
like($lei_err, qr/\bis `eml', not --in-format/, 'gave error message');
@@ -180,6 +181,51 @@ SKIP: {
'EIO noted in stderr');
}
+{
+ local $ENV{PI_CONFIG} = "$ENV{HOME}/pi_config";
+ write_file '>', $ENV{PI_CONFIG}, <<EOM;
+[publicinboxImport]
+ dropUniqueUnsubscribe
+EOM
+ my $in = <<EOM;
+List-Unsubscribe: <https://example.com/some-UUID-here/test>
+List-Unsubscribe-Post: List-Unsubscribe=One-Click
+Message-ID: <unsubscribe-1\@example>
+Subject: unsubscribe-1 example
+From: u\@example.com
+To: 2\@example.com
+Date: Fri, 02 Oct 1993 00:00:00 +0000
+
+EOM
+ lei_ok [qw(import -F eml +L:unsub)], undef, { %$lei_opt, 0 => \$in },
+ 'import succeeds w/ List-Unsubscribe';
+ lei_ok qw(q L:unsub -f mboxrd);
+ like $lei_out, qr/some-UUID-here/,
+ 'Unsubscribe header preserved despite PI_CONFIG dropping';
+ lei_ok qw(q L:unsub -o), "v2:$ENV{HOME}/v2-1";
+ lei_ok qw(q s:unsubscribe -f mboxrd --only), "$ENV{HOME}/v2-1";
+ unlike $lei_out, qr/some-UUID-here/,
+ 'Unsubscribe header dropped w/ dropUniqueUnsubscribe';
+ like $lei_out, qr/Message-ID: <unsubscribe-1\@example>/,
+ 'wrote expected message to v2 output';
+
+ # the default for compatibility:
+ truncate $ENV{PI_CONFIG}, 0;
+ lei_ok qw(q L:unsub -o), "v2:$ENV{HOME}/v2-2";
+ lei_ok qw(q s:unsubscribe -f mboxrd --only), "$ENV{HOME}/v2-2";
+ like $lei_out, qr/some-UUID-here/,
+ 'Unsubscribe header preserved by default :<';
+
+ # ensure we can fail
+ write_file '>', $ENV{PI_CONFIG}, <<EOM;
+[publicinboxImport]
+ dropUniqueUnsubscribe = bogus
+EOM
+ ok(!lei(qw(q L:unsub -o), "v2:$ENV{HOME}/v2-3"), 'bad config fails');
+ like $lei_err, qr/is not boolean/, 'non-booleaness noted in stderr';
+ ok !-d "$ENV{HOME}/v2-3", 'v2 directory not created';
+}
+
# see t/lei_to_mail.t for "import -F mbox*"
});
done_testing;
diff --git a/t/mda.t b/t/mda.t
index 83b0b33a..5144f3ca 100644
--- a/t/mda.t
+++ b/t/mda.t
@@ -8,6 +8,7 @@ use PublicInbox::Git;
use PublicInbox::InboxWritable;
use PublicInbox::TestCommon;
use PublicInbox::Import;
+use PublicInbox::IO qw(write_file);
use File::Path qw(remove_tree);
my ($tmpdir, $for_destroy) = tmpdir();
my $home = "$tmpdir/pi-home";
@@ -49,13 +50,11 @@ my $fail_bad_header = sub ($$$) {
is(1, mkdir($pi_home, 0755), "setup ~/.public-inbox");
PublicInbox::Import::init_bare($maindir);
- open my $fh, '>>', $pi_config or die;
- print $fh <<EOF or die;
+ write_file '>>', $pi_config, <<EOF;
[publicinbox "test"]
address = $addr
inboxdir = $maindir
EOF
- close $fh or die;
}
local $ENV{GIT_COMMITTER_NAME} = eval {
@@ -306,10 +305,44 @@ EOF
# ensure -learn rm works after inbox address is updated
($out, $err) = ('', '');
xsys(qw(git config --file), $pi_config, "$cfgpfx.address",
- 'updated-address@example.com');
+ $addr = 'updated-address@example.com');
ok(run_script(['-learn', 'rm'], undef, $rdr), 'rm-ed via -learn');
$cur = $git->qx(qw(diff HEAD~1..HEAD));
like($cur, qr/^-Message-ID: <2lids\@example>/sm, 'changed in git');
+
+ # ensure we can strip List-Unsubscribe
+ $in = <<EOF;
+To: You <you\@example.com>
+List-Id: <$list_id>
+Message-ID: <unsubscribe-1\@example>
+Subject: unsubscribe-1
+From: user <user\@example.com>
+To: $addr
+Date: Fri, 02 Oct 1993 00:00:00 +0000
+List-Unsubscribe: <https://example.com/some-UUID-here/listname>
+List-Unsubscribe-Post: List-Unsubscribe=One-Click
+
+List-Unsubscribe should be stripped
+EOF
+ write_file '>>', $pi_config, <<EOM;
+[publicinboxImport]
+ dropUniqueUnsubscribe
+EOM
+ $out = $err = '';
+ ok(run_script([qw(-mda)], undef, $rdr), 'mda w/ dropUniqueUnsubscribe');
+ $cur = join('', grep(/^\+/, $git->qx(qw(diff HEAD~1..HEAD))));
+ like $cur, qr/Message-ID: <unsubscribe-1/, 'imported new message';
+ unlike $cur, qr/some-UUID-here/, 'List-Unsubscribe gone';
+ unlike $cur, qr/List-Unsubscribe-Post/i, 'List-Unsubscribe-Post gone';
+
+ $in =~ s/unsubscribe-1/unsubscribe-2/g or xbail 'BUG: s// fail';
+ ok(run_script([qw(-learn ham)], undef, $rdr),
+ 'learn ham w/ dropUniqueUnsubscribe');
+ $cur = join('', grep(/^\+/, $git->qx(qw(diff HEAD~1..HEAD))));
+ like $cur, qr/Message-ID: <unsubscribe-2/, 'learn ham';
+ unlike $cur, qr/some-UUID-here/, 'List-Unsubscribe gone on learn ham';
+ unlike $cur, qr/List-Unsubscribe-Post/i,
+ 'List-Unsubscribe-Post gone on learn ham';
}
SKIP: {
diff --git a/t/watch_maildir.t b/t/watch_maildir.t
index 29e9bdc5..69a5e1f3 100644
--- a/t/watch_maildir.t
+++ b/t/watch_maildir.t
@@ -6,6 +6,7 @@ use PublicInbox::Eml;
use Cwd;
use PublicInbox::TestCommon;
use PublicInbox::Import;
+use PublicInbox::IO qw(write_file);
my ($tmpdir, $for_destroy) = tmpdir();
my $git_dir = "$tmpdir/test.git";
my $maildir = "$tmpdir/md";
@@ -143,6 +144,10 @@ More majordomo info at http://vger.kernel.org/majordomo-info.html\n);
my $env = { PI_CONFIG => $cfg_path };
$git->cleanup;
+ write_file '>>', $cfg_path, <<EOM;
+[publicinboxImport]
+ dropUniqueUnsubscribe
+EOM
# n.b. --no-scan is only intended for testing atm
my $wm = start_script([qw(-watch --no-scan)], $env);
no_pollerfd($wm->{pid});
@@ -194,13 +199,32 @@ More majordomo info at http://vger.kernel.org/majordomo-info.html\n);
$em->commit; # wake -watch up
diag 'waiting for -watch to import new message';
PublicInbox::DS::event_loop();
+
+ my $head = $git->qx(qw(cat-file commit HEAD));
+ my $subj = $eml->header('Subject');
+ like($head, qr/^\Q$subj\E/sm, 'new commit made');
+
+ # try dropUniqueUnsubscribe
+ $delivered = 0;
+ $eml->header_set('Message-ID', '<unsubscribe@example>');
+ $eml->header_set('List-Unsubscribe',
+ '<https://example.com/some-UUID-here/test');
+ $eml->header_set('List-Unsubscribe-Post', 'List-Unsubscribe=One-Click');
+ $em = PublicInbox::Emergency->new($maildir);
+ $em->prepare(\($eml->as_string));
+ $em->commit; # wake -watch up
+ diag 'waiting for -watch to import dropUniqueUnsubscribe message';
+ PublicInbox::DS::event_loop();
+ my $cur = $git->qx(qw(diff HEAD~1..HEAD));
+ like $cur, qr/Message-ID: <unsubscribe\@example>/,
+ 'unsubscribe@example imported';
+ unlike $cur, qr/List-Unsubscribe\b/,
+ 'List-Unsubscribe-* headers gone w/ dropUniqueUnsubscribe';
+
$wm->kill;
$wm->join;
$ii->close;
PublicInbox::DS->Reset;
- my $head = $git->qx(qw(cat-file commit HEAD));
- my $subj = $eml->header('Subject');
- like($head, qr/^\Q$subj\E/sm, 'new commit made');
}
sub is_maildir {
^ permalink raw reply related [flat|nested] 5+ messages in thread
* [PATCH 3/4] mda: fix and test some usage problems
2023-11-11 9:04 [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe Eric Wong
2023-11-11 9:04 ` [PATCH 1/4] learn: fix redundant ham import on dual matches Eric Wong
2023-11-11 9:04 ` [PATCH 2/4] mda|learn|watch: support dropUniqueUnsubscribe config Eric Wong
@ 2023-11-11 9:04 ` Eric Wong
2023-11-11 9:04 ` [PATCH 4/4] doc: update README.unsubscribe Eric Wong
3 siblings, 0 replies; 5+ messages in thread
From: Eric Wong @ 2023-11-11 9:04 UTC (permalink / raw)
To: meta
-mda now honors `--help' properly and invocations missing
ORIGINAL_RECIPIENT now fail with EX_NOUSER.
Helped-by: Leah Neukirchen <leah@vuxu.org>
Link: https://public-inbox.org/meta/87msvlguqu.fsf@vuxu.org/
---
script/public-inbox-mda | 7 ++++++-
t/mda.t | 18 ++++++++++++++++++
2 files changed, 24 insertions(+), 1 deletion(-)
diff --git a/script/public-inbox-mda b/script/public-inbox-mda
index 04fd8aad..b2e0908d 100755
--- a/script/public-inbox-mda
+++ b/script/public-inbox-mda
@@ -20,6 +20,10 @@ use PublicInbox::Import;
local $PublicInbox::Import::DROP_UNIQUE_UNSUB; # does this need a CLI switch?
GetOptions('precheck!' => \$precheck, 'help|h' => \$show_help) or
do { print STDERR $help; exit 1 };
+if ($show_help) {
+ print $help;
+ exit;
+}
my $do_exit = sub {
my ($code) = shift;
@@ -59,7 +63,8 @@ if (defined $recipient) {
if (!scalar(@$dests)) {
$dests = PublicInbox::MDA->inboxes_for_list_id($cfg, $eml);
if (!scalar(@$dests) && !defined($recipient)) {
- die "ORIGINAL_RECIPIENT not defined in ENV\n";
+ warn "ORIGINAL_RECIPIENT not defined in ENV\n";
+ $do_exit->(67); # EX_NOUSER
}
scalar(@$dests) or $do_exit->(67); # EX_NOUSER 5.1.1 user unknown
}
diff --git a/t/mda.t b/t/mda.t
index 5144f3ca..1d9e237b 100644
--- a/t/mda.t
+++ b/t/mda.t
@@ -82,6 +82,13 @@ die $@ if $@;
local $ENV{PI_EMERGENCY} = $faildir;
local $ENV{HOME} = $home;
local $ENV{ORIGINAL_RECIPIENT} = $addr;
+ ok(run_script([qw(-mda --help)], undef,
+ { 1 => \my $out, 2 => \my $err }), '-mda --help');
+ like $out, qr/usage:/, 'usage shown w/ --help';
+ ok(!run_script([qw(-mda --bogus)], undef,
+ { 1 => \$out, 2 => \$err }), '-mda --bogus fails');
+ like $err, qr/usage:/, 'usage shown on bogus switch';
+
my $in = <<EOF;
From: Me <me\@example.com>
To: You <you\@example.com>
@@ -91,6 +98,17 @@ Subject: hihi
Date: Thu, 01 Jan 1970 00:00:00 +0000
EOF
+ {
+ local $ENV{PATH} = $main_path;
+ ok(!run_script(['-mda'], { ORIGINAL_RECIPIENT => undef },
+ { 0 => \$in, 2 => \$err }),
+ 'missing ORIGINAL_RECIPIENT fails');
+ is($? >> 8, 67, 'got EX_NOUSER');
+ like $err, qr/\bORIGINAL_RECIPIENT\b/,
+ 'ORIGINAL_RECIPIENT noted in stderr';
+ is unlink(glob("$faildir/*/*")), 1, 'unlinked failed message';
+ }
+
# ensure successful message delivery
{
local $ENV{PATH} = $main_path;
^ permalink raw reply related [flat|nested] 5+ messages in thread
* [PATCH 4/4] doc: update README.unsubscribe
2023-11-11 9:04 [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe Eric Wong
` (2 preceding siblings ...)
2023-11-11 9:04 ` [PATCH 3/4] mda: fix and test some usage problems Eric Wong
@ 2023-11-11 9:04 ` Eric Wong
3 siblings, 0 replies; 5+ messages in thread
From: Eric Wong @ 2023-11-11 9:04 UTC (permalink / raw)
To: meta
The whitelist was only used in the early days of its development
and hasn't existed for a while. I've largely forgotten this
thing exists since it's been working well...
---
examples/README.unsubscribe | 9 ++++-----
1 file changed, 4 insertions(+), 5 deletions(-)
diff --git a/examples/README.unsubscribe b/examples/README.unsubscribe
index 3e80e838..3b407960 100644
--- a/examples/README.unsubscribe
+++ b/examples/README.unsubscribe
@@ -3,10 +3,9 @@ Unsubscribe endpoints for mlmmj users (and possibly Mailman, too)
* examples/unsubscribe.milter filters outgoing messages
and appends an HTTPS URL to the List-Unsubscribe header.
This List-Unsubscribe header should point to the PSGI
- described below.
- Currently, this is only active for a whitelist of test
- addresses in /etc/unsubscribe-milter.whitelist
- with one email address per line.
+ described below. You may edit the archive_addr sub
+ to disable List-Unsubscribe headers for well-known archiver
+ addresses to prevent saboteurs from stopping archival.
* examples/unsubscribe.psgi is a PSGI which needs to run
as the mlmmj user with permission to run mlmmj-unsub.
@@ -36,5 +35,5 @@ in /etc/postfix/main.cf:
# This is not needed for mlmmj since mlmmj uses SMTP:
# non_smtpd_milters = local:/var/spool/postfix/unsubscribe/unsubscribe.sock
-Copyright (C) 2016-2021 all contributors <meta@public-inbox.org>
+Copyright (C) all contributors <meta@public-inbox.org>
License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
^ permalink raw reply related [flat|nested] 5+ messages in thread
end of thread, other threads:[~2023-11-11 9:05 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2023-11-11 9:04 [PATCH 0/4] support publicinboxImport.dropUniqueUnsubscribe Eric Wong
2023-11-11 9:04 ` [PATCH 1/4] learn: fix redundant ham import on dual matches Eric Wong
2023-11-11 9:04 ` [PATCH 2/4] mda|learn|watch: support dropUniqueUnsubscribe config Eric Wong
2023-11-11 9:04 ` [PATCH 3/4] mda: fix and test some usage problems Eric Wong
2023-11-11 9:04 ` [PATCH 4/4] doc: update README.unsubscribe Eric Wong
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).