Test-MockFile
view release on metacpan or search on metacpan
use strict;
use warnings;
use Test2::Bundle::Extended;
use Test2::Tools::Explain;
use Test2::Plugin::NoWarnings;
use Test2::Tools::Exception qw< lives dies >;
use Test::MockFile ();
use Errno qw/ENOENT EPERM/;
my $euid = $>;
my $egid = int $);
my $filename = __FILE__;
my $file = Test::MockFile->file( $filename, 'whatevs' );
my $is_root = $> == 0 || $) =~ /( ^ | \s ) 0 ( \s | $)/xms;
my $top_gid;
my $next_gid;
if ( !$is_root ) {
my @groups;
( $top_gid, @groups ) = split /\s+/xms, $);
# root can have $) set to "0 0"
($next_gid) = grep $_ != $top_gid, @groups;
}
# Three scenarios:
# 1. If you're root, switch to +9999
# 2. If you're not root, do you have another group to use?
# 3. If you're not root and have no other group, switch to -1
subtest(
'Default ownership' => sub {
my $dir_foo = Test::MockFile->dir('/foo');
my $file_bar = Test::MockFile->file( '/foo/bar', 'content' );
ok( -d '/foo', 'Directory /foo exists' );
ok( -f '/foo/bar', 'File /foo/bar exists' );
foreach my $path (qw< /foo /foo/bar >) {
is(
( stat $path )[4],
$euid,
"$path set UID correctly to $euid",
);
is(
( stat $path )[5],
$egid,
"$path set GID correctly to $egid",
);
}
}
);
subtest(
'Change ownership of file to someone else' => sub {
note("\$>: $>, \$): $)");
my $chown_cb = sub {
my ( $args, $message ) = @_;
$! = 0;
if ($is_root) {
ok( chown( @{$args} ), $message );
is( $! + 0, 0, 'chown succeeded' );
is( "$!", '', 'No failure' );
}
else {
ok( !chown( @{$args} ), $message );
is( $! + 0, 1, "chown failed (EPERM): \$>:$>, \$):$)" );
}
};
$chown_cb->(
[ $euid + 9999, $egid + 9999, $filename ],
'chown file to some high, probably unavailable, UID/GID',
);
$chown_cb->(
[ $euid, $egid + 9999, $filename ],
'chown file to some high, probably unavailable, GID',
);
$chown_cb->(
[ $euid + 9999, $egid, $filename ],
'chown file to some high, probably unavailable, UID',
);
$chown_cb->(
[ 0, 0, $filename ],
'chown file to root',
);
$chown_cb->(
[ $euid, 0, $filename ],
'chown file to root GID',
);
$chown_cb->(
[ 0, $egid, $filename ],
'chown file to root UID',
);
}
);
subtest(
'chown with bareword (nonexistent file)' => sub {
no strict;
my $bareword_file = Test::MockFile->file('RANDOM_FILE_THAT_WILL_NOT_EXIST');
is( $! + 0, 0, '$! starts clean' );
ok(
!chown( $euid, $egid, RANDOM_FILE_THAT_WILL_NOT_EXIST ),
'Using bareword treats it as string',
);
is( $! + 0, 2, 'Correct ENOENT error' );
}
);
subtest(
'chown only user, only group, both' => sub {
is( $! + 0, 0, '$! starts clean' );
ok(
chown( $euid, -1, $filename ),
'chown\'ing file to only UID',
);
is( $! + 0, 0, '$! still clean' );
ok(
chown( -1, $egid, $filename ),
'chown\'ing file to only GID',
);
is( $! + 0, 0, '$! still clean' );
ok(
chown( $euid, $egid, $filename ),
'chown\'ing file to both UID and GID',
);
is( $! + 0, 0, '$! still clean' );
}
);
subtest(
'chown to different group of same user' => sub {
# See if this user has another group available
# (we might be on a user that has only one group)
$next_gid
or skip_all('This user only has one group');
is( $top_gid, $egid, 'Skipping the first GID' );
isnt( $next_gid, $egid, 'Testing a different GID' );
is( $! + 0, 0, '$! starts clean' );
ok(
chown( -1, $next_gid, $filename ),
'chown\'ing file to a different GID',
);
is( $! + 0, 0, '$! stays clean' );
}
);
subtest(
'chown on typeglob / filehandle' => sub {
my $filename = '/tmp/not-a-file';
my $file = Test::MockFile->file($filename);
open my $fh, '>', $filename
or die;
print {$fh} "whatevs\n"
or die;
my ( $exp_euid, $exp_egid ) = $is_root ? ( $euid + 9999, $egid + 9999 ) : ( $euid, $egid );
if ($is_root) {
is( $! + 0, 0, '$! starts clean' );
is( chown( $exp_euid, $exp_egid, $fh ), 1, 'root chown on a file handle works' );
is( $! + 0, 0, '$! stays clean' );
}
else {
is( $! + 0, 0, '$! starts clean' );
is( chown( $exp_euid, $exp_egid, $fh ), 1, 'Non-root chown on a file handle works' );
is( $! + 0, 0, '$! stays clean' );
}
close $fh
or die;
my (
$dev, $ino, $mode, $nlink, $uid, $gid, $rdev, $size,
$atime, $mtime, $ctime, $blksize, $blocks
) = stat($filename);
is( $uid, $exp_euid, "Owner of the file is now there" );
is( $gid, $exp_egid, "Group of the file is now there" );
}
);
subtest(
'chown does not reset $!' => sub {
my $file = Test::MockFile->file( '/foo' => 'bar' );
$! = 3;
is( $! + 0, 3, '$! is set to 3 for our test' );
ok( chown( -1, -1, '/foo' ), 'Successfully run chown' );
is( $! + 0, 3, '$! is still 3 (not reset by chown)' );
}
);
subtest(
'chown -1 preserves per-file ownership, not process identity' => sub {
# Create a file with non-default ownership
my $custom_uid = 12345;
my $custom_gid = 67890;
my $file = Test::MockFile->file(
'/chown_test_preserve' => 'data',
{ uid => $custom_uid, gid => $custom_gid },
);
# Use root mock user so permission checks don't interfere with
# the -1 preservation semantics being tested here.
Test::MockFile->set_user( 0, 0 );
# chown(-1, -1) should keep the custom values, not replace with $> / $)
ok( chown( -1, -1, '/chown_test_preserve' ), 'chown(-1, -1) succeeds' );
my @st = stat('/chown_test_preserve');
is( $st[4], $custom_uid, 'uid preserved (not replaced with process uid)' );
is( $st[5], $custom_gid, 'gid preserved (not replaced with process gid)' );
# chown($new_uid, -1) should change uid but preserve gid
ok( chown( 99, -1, '/chown_test_preserve' ), 'chown(99, -1) succeeds' );
@st = stat('/chown_test_preserve');
is( $st[4], 99, 'uid changed to 99' );
is( $st[5], $custom_gid, 'gid still preserved after uid-only change' );
# chown(-1, $new_gid) should preserve uid but change gid
ok( chown( -1, 42, '/chown_test_preserve' ), 'chown(-1, 42) succeeds' );
@st = stat('/chown_test_preserve');
is( $st[4], 99, 'uid still preserved after gid-only change' );
is( $st[5], 42, 'gid changed to 42' );
Test::MockFile->clear_user;
}
);
subtest(
'chown uid-only and gid-only permission checks' => sub {
my $file = Test::MockFile->file(
'/chown_perm_test' => 'data',
{ uid => 1000, gid => 1000 },
);
# Non-root user cannot change uid to a different user (uid-only)
Test::MockFile->set_user( 1000, 1000 );
$! = 0;
is( chown( 2000, -1, '/chown_perm_test' ), 0, 'non-root cannot chown uid-only to different user' );
is( $! + 0, EPERM, 'errno is EPERM for uid-only chown' );
# Non-root user cannot change gid to a group they are not in (gid-only)
$! = 0;
is( chown( -1, 9999, '/chown_perm_test' ), 0, 'non-root cannot chown gid-only to foreign group' );
is( $! + 0, EPERM, 'errno is EPERM for gid-only chown' );
# Non-root user CAN change gid to a group they belong to
Test::MockFile->set_user( 1000, 1000, 2000 );
$! = 0;
is( chown( -1, 2000, '/chown_perm_test' ), 1, 'non-root can chown gid to own group' );
is( $! + 0, 0, 'no error for allowed gid change' );
# Non-root user CAN chown uid to self (no-op)
$! = 0;
is( chown( 1000, -1, '/chown_perm_test' ), 1, 'non-root can chown uid to self' );
is( $! + 0, 0, 'no error for uid self-chown' );
Test::MockFile->clear_user;
}
);
subtest(
'chown with broken symlink in multi-file list does not confess' => sub {
my $link = Test::MockFile->symlink( '/nonexistent_target', '/chown_broken_link' );
my $file = Test::MockFile->file( '/chown_real_file', 'content' );
# chown on a mix of regular file + broken symlink should NOT die.
# The broken symlink should silently fail with ENOENT, and the
# regular file should succeed.
my ( $result, $errno );
ok(
lives { $result = chown( $>, int($)), '/chown_broken_link', '/chown_real_file' ); $errno = $! + 0 },
'chown with broken symlink + regular file does not confess',
);
is( $result, 1, 'chown returns 1 (one file changed)' );
is( $errno, ENOENT, 'errno set to ENOENT for the broken symlink' );
}
);
subtest(
'chown with only broken symlink' => sub {
my $link = Test::MockFile->symlink( '/nowhere', '/chown_only_broken' );
my ( $result, $errno );
ok(
lives { $result = chown( $>, int($)), '/chown_only_broken' ); $errno = $! + 0 },
'chown with only a broken symlink does not confess',
);
is( $result, 0, 'chown returns 0 (no files changed)' );
is( $errno, ENOENT, 'errno set to ENOENT' );
}
);
done_testing();
exit;
( run in 0.868 second using v1.01-cache-2.11-cpan-5511b514fd6 )