use
5.20.0;
our
$VERSION
=
'3.20241024'
;
sub
_build_config_smtp {
my
(
$self
,
$handler_config
) =
@_
;
my
$config
= {
'_is_test'
=> 1,
'debug'
=> 1,
'dryrun'
=> 0,
'logtoerr'
=> 1,
'error_log'
=>
'tmp/smtp.err'
,
'connection'
=>
'unix:tmp/authentication_milter_test.sock'
,
'umask'
=>
'0000'
,
'connect_timeout'
=> 55,
'command_timeout'
=> 55,
'content_timeout'
=> 595,
'tempfail_on_error'
=> 1,
'tempfail_on_error_authenticated'
=> 1,
'tempfail_on_error_local'
=> 1,
'tempfail_on_error_trusted'
=> 1,
'#metric_connection'
=>
'unix:tmp/authentication_milter_test_metrics.sock'
,
'#metric_umask'
=>
'0000'
,
'protocol'
=>
'smtp'
,
'smtp'
=> {
'sock_type'
=>
'unix'
,
'sock_path'
=>
'tmp/authentication_milter_smtp_out.sock'
,
'pipeline_limit'
=>
'4'
,
},
'handlers'
=>
$handler_config
,
};
return
$config
;
}
sub
_build_config_milter {
my
(
$self
,
$handler_config
) =
@_
;
my
$config
= {
'_is_test'
=> 1,
'debug'
=> 1,
'dryrun'
=> 0,
'logtoerr'
=> 1,
'error_log'
=>
'tmp/milter.err'
,
'connection'
=>
'unix:tmp/authentication_milter_test.sock'
,
'umask'
=>
'0000'
,
'connect_timeout'
=> 55,
'command_timeout'
=> 55,
'content_timeout'
=> 595,
'tempfail_on_error'
=> 1,
'tempfail_on_error_authenticated'
=> 1,
'tempfail_on_error_local'
=> 1,
'tempfail_on_error_trusted'
=> 1,
'#metric_connection'
=>
'unix:tmp/authentication_milter_test_metrics.sock'
,
'#metric_umask'
=>
'0000'
,
'protocol'
=>
'milter'
,
'handlers'
=>
$handler_config
,
};
return
$config
;
}
sub
new {
my
(
$class
,
$args
) =
@_
;
my
$self
= {};
bless
$self
,
$class
;
$self
->{
'snapshots'
} = {};
foreach
my
$arg
(
qw{ prefix zonefile zonedata }
) {
$self
->{
$arg
} =
$args
->{
$arg
}
if
exists
$args
->{
$arg
};
}
croak
'prefix must be supplies'
if
!
exists
$self
->{
'prefix'
};
croak
'zonefile or zonedata cannot both be supplied'
if
(
exists
$self
->{
'zonefile'
} ) && (
exists
$self
->{
'zonedata'
});
$self
->{
'zonedata'
} =
q{}
if
( !
exists
$self
->{
'zonefile'
} ) && ( !
exists
$self
->{
'zonedata'
});
my
$protocol
=
$args
->{
'protocol'
} //
'smtp'
;
if
(
exists
(
$args
->{
'handler_config'
} ) ) {
if
(
$protocol
eq
'smtp'
) {
set_config(
$self
->_build_config_smtp(
$args
->{
'handler_config'
} ) );
}
else
{
set_config(
$self
->_build_config_milter(
$args
->{
'handler_config'
} ) );
}
}
$Mail::Milter::Authentication::Config::PREFIX
=
$self
->{
'prefix'
};
Mail::Milter::Authentication::Config::setup_config;
my
$config
= get_config();
my
$Resolver
= Net::DNS::Resolver::Mock->new();
$Resolver
->zonefile_read(
$self
->{
'zonefile'
} )
if
exists
$self
->{
'zonefile'
};
$Resolver
->zonefile_parse(
$self
->{
'zonedata'
} )
if
exists
$self
->{
'zonedata'
};
$Mail::Milter::Authentication::Handler::TestResolver
=
$Resolver
;
my
$authmilter
= Mail::Milter::Authentication->new();
$authmilter
->{
'metric'
} = Mail::Milter::Authentication::Metric->new(
$authmilter
);
$authmilter
->{
'config'
} =
$config
;
push
@Mail::Milter::Authentication::ISA
,
'Mail::Milter::Authentication::Protocol::SMTP'
;
$authmilter
->{
'server'
}->{
'ppid'
} =
$PID
;
foreach
my
$name
( @{
$config
->{
'load_handlers'
}} ) {
$authmilter
->load_handler(
$name
);
my
$package
=
"Mail::Milter::Authentication::Handler::$name"
;
my
$object
=
$package
->new(
$authmilter
);
if
(
$object
->can(
'pre_loop_setup'
) ) {
$object
->pre_loop_setup();
}
if
(
$object
->can(
'register_metrics'
) ) {
$authmilter
->{
'metric'
}->register_metrics(
$object
->register_metrics() );
}
}
my
$callbacks_list
= {};
my
$callbacks
= {};
my
$handler
= {};
my
$object
= {};
my
$object_maker
= {};
my
$count
= 0;
$authmilter
->{
'callbacks_list'
} =
$callbacks_list
;
$authmilter
->{
'callbacks'
} =
$callbacks
;
$authmilter
->{
'count'
} =
$count
;
$authmilter
->{
'handler'
} =
$handler
;
$authmilter
->{
'object'
} =
$object
;
$authmilter
->{
'object_maker'
} =
$object_maker
;
$authmilter
->setup_handlers();
$self
->{
'authmilter'
} =
$authmilter
;
$self
->handler()->top_setup_callback();
$self
->snapshot(
'_new'
);
return
$self
;
}
sub
snapshot {
my
(
$self
,
$name
) =
@_
;
my
$snapshot
= clone(
$self
->{
'authmilter'
} );
$self
->{
'snapshots'
}->{
$name
} =
$snapshot
;
}
sub
switch {
my
(
$self
,
$name
) =
@_
;
croak
'unknown snapshot'
if
!
exists
(
$self
->{
'snapshots'
}->{
$name
} );
my
$snapshot
= clone(
$self
->{
'snapshots'
}->{
$name
} );
$self
->{
'authmilter'
} =
$snapshot
;
}
sub
handler {
my
(
$self
) =
@_
;
return
$self
->{
'authmilter'
}->{
'handler'
}->{
'_Handler'
};
}
sub
connect
{
my
(
$self
,
$name
,
$ip
) =
@_
;
my
$authmilter
=
$self
->{
'authmilter'
};
my
$ip_obj
=
eval
{ Net::IP->new(
$ip
) } //
undef
;
$self
->handler()->remap_connect_callback(
$name
,
$ip_obj
);
return
$self
->handler()->top_connect_callback(
$name
,
$self
->handler()->{
'ip_object'
} );
}
sub
helo {
my
(
$self
,
$helo
) =
@_
;
$self
->handler()->remap_helo_callback(
$helo
);
return
$self
->handler()->top_helo_callback(
$self
->handler()->{
'helo_name'
} );
}
sub
mailfrom {
my
(
$self
,
$from
) =
@_
;
return
$self
->handler()->top_envfrom_callback(
$from
);
}
sub
rcptto {
my
(
$self
,
$to
) =
@_
;
return
$self
->handler()->top_envrcpt_callback(
$to
);
}
sub
header {
my
(
$self
,
$key
,
$value
,
$original
) =
@_
;
$original
=
"$key: $value"
if
!
defined
$original
;
return
$self
->handler()->top_header_callback(
$key
,
$value
,
$original
);
}
sub
end_of_headers {
my
(
$self
) =
@_
;
return
$self
->handler()->top_eoh_callback();
}
sub
body {
my
(
$self
,
$body
) =
@_
;
return
$self
->handler()->top_body_callback(
$body
);
}
sub
end_of_message {
my
(
$self
) =
@_
;
return
$self
->handler()->top_eom_callback();
}
sub
close
{
my
(
$self
) =
@_
;
return
$self
->handler()->top_close_callback();
}
sub
abort {
my
(
$self
) =
@_
;
return
$self
->handler()->top_abort_callback();
}
sub
addheader {
my
(
$self
) =
@_
;
return
$self
->handler()->top_addheader_callback();
}
sub
run {
my
(
$self
,
$args
) =
@_
;
$self
->switch(
'_new'
);
my
$returncode
;
$returncode
=
$self
->
connect
(
$args
->{
'connect_name'
},
$args
->{
'connect_ip'
} );
die
'connect'
if
(
$returncode
!= SMFIS_CONTINUE );
$returncode
=
$self
->helo(
$args
->{
'helo'
} );
die
'helo'
if
(
$returncode
!= SMFIS_CONTINUE );
$returncode
=
$self
->mailfrom(
$args
->{
'mailfrom'
} );
die
'mailfrom'
if
(
$returncode
!= SMFIS_CONTINUE );
foreach
my
$rcptto
( @{
$args
->{
'rcptto'
} } ) {
$returncode
=
$self
->rcptto(
$rcptto
);
die
'rcptto '
.
$rcptto
if
(
$returncode
!= SMFIS_CONTINUE );
}
my
$body
=
$args
->{
'body'
};
$body
=~ s/\r?\n/\n/g;
my
@lines
=
split
( /\n/,
$body
);
my
$buffer
=
q{}
;
while
(
my
$line
=
shift
@lines
) {
chomp
$line
;
last
if
$line
eq
q{}
;
if
(
$line
=~ /^\s/ ) {
$buffer
.=
"\n"
.
$line
;
}
else
{
if
(
$buffer
) {
my
(
$key
,
$value
) =
split
(
':'
,
$buffer
, 2 );
$key
=~ s/\s+$//;
$value
=~ s/^\s+//;
$returncode
=
$self
->header(
$key
,
$value
);
die
"header $key: $value"
if
(
$returncode
!= SMFIS_CONTINUE );
}
$buffer
=
$line
;
}
}
if
(
$buffer
) {
my
(
$key
,
$value
) =
split
(
':'
,
$buffer
, 2 );
$key
=~ s/\s+$//;
$value
=~ s/^\s+//;
$returncode
=
$self
->header(
$key
,
$value
);
die
"header $key: $value"
if
(
$returncode
!= SMFIS_CONTINUE );
}
$returncode
=
$self
->end_of_headers();
die
'eoh'
if
(
$returncode
!= SMFIS_CONTINUE );
$returncode
=
$self
->body(
join
(
"\n"
,
@lines
) .
"\n"
);
die
'body'
if
(
$returncode
!= SMFIS_CONTINUE );
$returncode
=
$self
->end_of_message();
die
'body'
if
(
$returncode
!= SMFIS_CONTINUE );
$self
->addheader();
}
sub
get_return {
my
(
$self
) =
@_
;
return
$self
->handler()->get_return();
}
sub
get_reject_mail {
my
(
$self
) =
@_
;
return
$self
->handler()->get_reject_mail();
}
sub
servername {
my
(
$self
) =
@_
;
return
'handlertester.test.authmilter.org'
;
}
sub
get_authresults_header {
my
(
$self
) =
@_
;
my
$c_auth_headers
=
eval
{ clone(
$self
->handler()->{
'c_auth_headers'
}->{
'Authentication-Results'
} ) } // [];
my
$auth_headers
=
eval
{ clone(
$self
->handler()->{
'auth_headers'
}->{
'Authentication-Results'
} ) } // [];
my
@added_ar_headers
= ( @{
$c_auth_headers
}, @{
$auth_headers
} );
my
$header
= Mail::AuthenticationResults::Header->new()->set_value( Mail::AuthenticationResults::Header::AuthServID->new()->set_value(
$self
->servername() ) );
foreach
my
$ar_header
(
@added_ar_headers
) {
eval
{
$ar_header
->orphan(); };
$header
->add_child(
$ar_header
);
}
return
$header
;
}
1;