Merge pull request #234 from jaredj/writeable-dir

Add perms test to Qpsmtpd::DB::File::DBM::dir()
This commit is contained in:
Matt Simerson 2015-03-10 09:59:03 -07:00
commit 11a9154552
4 changed files with 121 additions and 22 deletions

View File

@ -180,12 +180,47 @@ sub flush {
} }
sub dir { sub dir {
my ( $self, @arg ) = @_; my ( $self, $dir ) = @_;
return $self->{dir} if $self->{dir} and ! @arg; if ( $dir ) {
for my $d ( $self->candidate_dirs(@arg) ) { $self->validate_dir($dir);
next if ! $self->validate_dir($d); return $self->{dir} = $dir;
return $self->{dir} = $d; # first match wins
} }
return $self->{dir} if $self->{dir};
my @err;
for my $d ( $self->candidate_dirs ) {
# Ignore invalid directories for static default directories
my $is_valid;
eval { $is_valid = $self->validate_dir($d); };
if ($@) {
push @err, $@;
next;
}
else {
$self->{dir} = $d; # first match wins
last;
}
}
if ( !$self->{dir} ) {
my $err = join "\n",
"Unable to find a useable database directory!",
"",
@err;
die $err;
}
if (@err) {
my $err = join "\n",
"Encountered errors while selecting database directory:",
"",
@err,
"Selected database directory: $self->{dir}. Data is now stored in:",
"",
$self->path,
"",
"It is recommended to manually specify a useable database directory",
"and move any important data into this directory.\n";
warn $err;
}
return $self->{dir};
} }
sub candidate_dirs { sub candidate_dirs {
@ -199,8 +234,9 @@ sub candidate_dirs {
sub validate_dir { sub validate_dir {
my ( $self, $d ) = @_; my ( $self, $d ) = @_;
return 0 if ! $d; die "Empty DB directory supplied\n" if ! $d;
return 0 if ! -d $d; die "DB directory '$d' does not exist\n" if ! -d $d;
die "DB directory '$d' is not writeable\n" if ! -w $d;
return 1; return 1;
} }

View File

@ -266,12 +266,17 @@ sub init_dbm {
name => 'greylist', name => 'greylist',
class => 'Qpsmtpd::DB::File::DBM' class => 'Qpsmtpd::DB::File::DBM'
); );
my $cdir = $self->{_args}{db_dir};
$cdir = $1 if $cdir and $cdir =~ m{^([-a-zA-Z0-9./_]+)$};
# greylisting-specific hints for where to store the greylist DB
my $db_dir = $self->db->dir( $cdir, '/var/lib/qpsmtpd/greylisting' );
$self->db->nfs_locking( $self->{_args}{nfslock} ); $self->db->nfs_locking( $self->{_args}{nfslock} );
# Add to the default list of possible DB directories
$self->db->candidate_dirs('/var/lib/qpsmtpd/greylisting');
if ( my $dir_arg = $self->{_args}{db_dir} ) {
# user-supplied db dir
$dir_arg = $1 if $dir_arg =~ m{^([-a-zA-Z0-9./_]+)$};
$self->db->dir($dir_arg);
}
my $db_dir = $self->db->dir;
# Work around old DBM filename # Work around old DBM filename
my $oldname = 'denysoft_greylist'; my $oldname = 'denysoft_greylist';
if ( ! -f "$db_dir/greylist.dbm" && -f "$db_dir/$oldname.dbm" ) { if ( ! -f "$db_dir/greylist.dbm" && -f "$db_dir/$oldname.dbm" ) {

View File

@ -358,9 +358,10 @@ sub test_init_dbm {
my ($self) = @_; my ($self) = @_;
delete $self->{db}; delete $self->{db};
delete $self->{_args}{redis}; delete $self->{_args}{redis};
$self->{_args}{db_dir} = 't/tmp';
$self->init_db; $self->init_db;
is( $self->db->name, 'greylist', 'init_dbm() sets correct db name' ); is( $self->db->name, 'greylist', 'init_dbm() sets correct db name' );
is( $self->db->path, 't/config/greylist.dbm', 'init_dbm() sets correct path' ); is( $self->db->path, 't/tmp/greylist.dbm', 'init_dbm() sets correct path' );
is( ref $self->db, 'Qpsmtpd::DB::File::DBM', 'init_dbm() gives DBM object' ); is( ref $self->db, 'Qpsmtpd::DB::File::DBM', 'init_dbm() gives DBM object' );
} }

View File

@ -119,22 +119,79 @@ sub __candidate_dirs {
} }
sub __validate_dir { sub __validate_dir {
is( $db->validate_dir(), 0, 'validate_dir(): false on no input' ); eval { $db->validate_dir(); };
is( $db->validate_dir(undef), 0, 'validate_dir(): false on undef' ); is( $@, "Empty DB directory supplied\n",
is( $db->validate_dir('invalid'), 0, 'validate_dir(): die on no input' );
'validate_dir(): false for non-existent directory' ); eval { $db->validate_dir(undef); };
is( $@, "Empty DB directory supplied\n",
'validate_dir(): die on undef' );
eval { $db->validate_dir(''); };
is( $@, "Empty DB directory supplied\n",
'validate_dir(): die on empty string' );
eval { $db->validate_dir('invalid'); };
is( $@, "DB directory 'invalid' does not exist\n",
'validate_dir(): die on non-existent directory' );
is( $db->validate_dir('t/tmp'), 1, is( $db->validate_dir('t/tmp'), 1,
'validate_dir(): true for real directory' ); 'validate_dir(): true for real directory' );
mkdir 't/tmp/wtest', 0555;
eval { $db->validate_dir('t/tmp/wtest') };
is( $@, "DB directory 't/tmp/wtest' is not writeable\n",
'validate_dir(): die on non-writeable directory' );
chmod 0777, 't/tmp/wtest';
is( $db->validate_dir('t/tmp/wtest'), 1,
'validate_dir(): true for writeable directory' );
rmdir 't/tmp/wtest';
} }
sub __dir { sub __dir {
my $db2 = Qpsmtpd::DB::File::DBM->new( name => 'dirtest' ); my $db2 = Qpsmtpd::DB::File::DBM->new( name => 'dirtest' );
{
local $SIG{__WARN__} = sub {
warn @_ if $_[0] !~ /selecting database directory/;
};
is( $db2->dir(), 't/config', 'default directory' ); is( $db2->dir(), 't/config', 'default directory' );
is( $db2->dir('_invalid','t/Test'), 't/Test', 'skip invalid candidate dirs' ); delete $db2->{dir};
$db2->candidate_dirs('_invalid','t/Test');
is( $db2->dir, 't/Test', 'skip invalid candidate dirs' );
$db2->{dir} = '_cached'; $db2->{dir} = '_cached';
is( $db2->dir(), '_cached', 'cached directory' ); is( $db2->dir(), '_cached', 'cached directory' );
is( $db2->dir('t/Test'), 't/Test', 'passing candidate dirs resets cache' ); is( $db2->dir('t/Test'), 't/Test', 'passing candidate dirs resets cache' );
is( $db2->dir('_invalid'), 't/config', 'invalid candidate dirs reverts to default' ); delete $db2->{dir};
$db2->candidate_dirs('_invalid');
is( $db2->dir, 't/config', 'invalid candidate dirs reverts to default' );
eval { $db2->dir('_invalid'); };
is( $@, "DB directory '_invalid' does not exist\n", 'die on invalid dir' );
}
{
delete $db2->{dir};
my $warned;
local $SIG{__WARN__} = sub {
warn @_ if $_[0] !~ /selecting database directory/;
$warned .= join '', @_;
};
$db2->candidate_dirs('_invalid2','t/Test');
is( $db2->dir(), 't/Test', 'default directory' );
my $expected_warning =
"Encountered errors while selecting database directory:
DB directory '_invalid2' does not exist
Selected database directory: t/Test. Data is now stored in:
t/Test/dirtest.dbm
It is recommended to manually specify a useable database directory
and move any important data into this directory.\n";
is( $warned, $expected_warning, 'Emit warning on bad directories' );
delete $db2->{dir};
$db2->{candidate_dirs} = ['/___invalid___'];
my $expected_err =
"Unable to find a useable database directory!
DB directory '/___invalid___' does not exist\n";
eval { $db2->dir() };
is( $@, $expected_err, 'Die on no valid directories' );
}
} }
sub __untie_gotcha { sub __untie_gotcha {