NAME
Mailmunge::Action::Stream - stream mail by domain or recipient
ABSTRACT
This class implements methods that let you "stream" mail. This lets you apply different filtering rules per-recipient or per-domain.
SYNOPSIS
package MyFilter;
use base qw(Mailmunge::Filter);
use Mailmunge::Action::Stream;
sub filter_message {
my ($self, $ctx) = @_;
my $action = Mailmunge::Action::Stream->new($self);
if ($action->stream_by_domain($ctx)) {
$self->action_discard($ctx);
return;
}
# Now all recipients are guaranteed to be in the same domain
}
my $filter = MyFilter->new();
$filter->run();
1;
METHODS
Mailmunge::Action::Stream->new($filter)
Constructor. Typically used within a filter file as follows:
my $action = Mailmunge::Action::Stream->new($self);
stream_by_domain($ctx)
If all recipients are in the same domain, returns 0. Otherwise, remails copies of the message with each copy going to a group of recipients in the same domain, and returns 1.
You typically want to discard the original message and skip the rest of your filter processing if stream_by_domain
returns 1; the usual idiom is:
if ($action->stream_by_domain($ctx)) {
$self->action_discard($ctx);
return;
}
The remailed messages will be filtered in a subsequent set of milter calls.
Returns undef if something went wrong.
stream_by_recipient($ctx)
If there is only one recipient, returns 0. Otherwise, remails copies of the message with each copy going to a single recipient.
As with stream_by_domain
, you typically want to discard the original message and skip the rest of your filter processing if stream_by_recipient
returns 1.
Returns undef if something went wrong.
stream_recipients($ctx, $groups)
This is the most general streaming function. $groups
is an arrayref; each element must be an arrayref of recipients. Each recipient in $ctx->recipients
must appear in exactly one member of $groups
and no additional recipients may appear.
If $groups
has only one element, this function does nothing and returns 0. Otherwise, it emails one copy of the message to each group of recipients in $groups
and returns 1.
Returns undef if something went wrong.
STREAMING MECHANISM
When Mailmunge streams a message, it remails copies of it so the copies can be re-scanned.
If your MTA is Sendmail, the copies are put in the Sendmail submission queue and they appear in a subsequent set of filter callbacks in an SMTP session originating from the localhost.
If your MTA is Postfix, the copies are re-injected, but are not sent via SMTP. Instead, Postfix simulates an SMTP session. If you want to stream messages and are using Postfix, you must set the non_smtpd_milters
Postfix configuration variable to match smtpd_milters
. Otherwise, streamed messages will not be filtered!
When Mailmunge::Action::Stream
remails a message, it adds a header of the form:
X-Mailmunge-Remailed: original_ip original_qid
where original_ip
is the value of $ctx->hostip
and original_qid
is $ctx->qid
when the streaming function is invoked.
Mailmunge::Filter
looks for this header. If the connecting IP is the local host, the filter trusts the header and sets $ctx->hostip
and $ctx->hostname
based on original_ip
from the header. It also logs a message:
Resent from queue-ID: original_qid; original IP: original_ip
so the streamed messages can be correlated with the original message.
If the connecting host is not the localhost, then Mailmunge::Filter
ignores the X-Mailmunge-Remailed header. Regardless of where a message origintates, Mailmunge::Filter
deletes all X-Mailmunge-Remailed headers so they don't show up downstream.
AUTHOR
Dianne Skoll <dianne@skollsoft.com>
LICENSE
This code is licensed under the terms of the GNU General Public License, version 2.
Copyright © 2025 Skoll Software Consulting