HEX
Server: Apache
System: Linux server-674799.igrow.ws 5.14.0-611.30.1.el9_7.x86_64 #1 SMP PREEMPT_DYNAMIC Wed Feb 11 06:42:00 EST 2026 x86_64
User: elrashedytravel (1025)
PHP: 8.1.34
Disabled: exec,passthru,shell_exec,system
Upload Files
File: /home/.cpan/build/Path-Tiny-0.146-0/t/basic.t
use 5.008001;
use strict;
use warnings;
use Test::More 0.96;
use File::Spec;
use File::Glob;
use Path::Tiny;
use Cwd;

my $IS_WIN32 = $^O eq 'MSWin32';
my $IS_CYGWIN = $^O eq 'cygwin';

use lib 't/lib';
use TestUtils qw/exception/;

my $file1 = path('foo.txt');
isa_ok( $file1, "Path::Tiny" );
ok $file1->isa('Path::Tiny');
is $file1, 'foo.txt';
ok $file1->is_relative;
is $file1->dirname,  '.';
is $file1->basename, 'foo.txt';

my $file2 = path( 'dir', 'bar.txt' );
is $file2, 'dir/bar.txt';
ok !$file2->is_absolute;
is $file2->dirname,  'dir/';
is $file2->basename, 'bar.txt';

my $dir = path('tmp');
is $dir, 'tmp';
ok !$dir->is_absolute;
is $dir->basename, 'tmp';

my $dir2 = path('/tmp');
is $dir2, '/tmp';
ok $dir2->is_absolute;

my $cat = path( $dir, 'foo' );
is $cat, 'tmp/foo';
$cat = $dir->child('foo');
is $cat, 'tmp/foo';
is $cat->dirname,  'tmp/';
is $cat->basename, 'foo';

$cat = path( $dir2, 'foo' );
is $cat, '/tmp/foo';
$cat = $dir2->child('foo');
is $cat,     '/tmp/foo';
isa_ok $cat, 'Path::Tiny';
is $cat->dirname, '/tmp/';

$cat = $dir2->child('foo');
is $cat,     '/tmp/foo';
isa_ok $cat, 'Path::Tiny';
is $cat->basename, 'foo';

my $sib = $cat->sibling('bar');
is $sib,     '/tmp/bar';
isa_ok $sib, 'Path::Tiny';

my $file = path('/foo//baz/./foo');
is $file, '/foo/baz/foo';
is $file->dirname, '/foo/baz/';
is $file->parent,  '/foo/baz';

{
    my $file = path("foo/bar/baz");
    is( $file->canonpath, File::Spec->canonpath("$file"), "canonpath" );
}

{
    my $dir = path('/foo/bar/baz');
    is $dir->parent, '/foo/bar';
    is $dir->parent->parent, '/foo';
    is $dir->parent->parent->parent, '/';
    is $dir->parent->parent->parent->parent, '/';

    $dir = path('foo/bar/baz');
    is $dir->parent, 'foo/bar';
    is $dir->parent->parent, 'foo';
    is $dir->parent->parent->parent, '.';
    is $dir->parent->parent->parent->parent, '..';
    is $dir->parent->parent->parent->parent->parent, '../..';
}

{
    my $dir = path("foo/");
    is $dir, 'foo';
    is $dir->parent, '.';
}

{
    # Special cases
    for my $bad ( [''], [undef], [], [ '', 'var', 'tmp' ], [ 'foo', '', 'bar' ] ) {
        like( exception { path(@$bad) }, qr/positive-length/, "exception" );
    }
    is( Path::Tiny->cwd,     path( Cwd::getcwd() ) );
    is( path('.')->absolute, path( Cwd::getcwd() ) );
}

{
    my $file = path('/tmp/foo/bar.txt');
    is $file->relative('/tmp'),      'foo/bar.txt';
    is $file->relative('/tmp/foo'),  'bar.txt';
    is $file->relative('/tmp/'),     'foo/bar.txt';
    is $file->relative('/tmp/foo/'), 'bar.txt';

    $file = path('one/two/three');
    is $file->relative('one'), 'two/three';

    $file = path('/one[0/two');
    is $file->relative( '/one[0' ), 'two', 'path with regex special char';
}

{
    my $file = Path::Tiny->new( File::Spec->rootdir );
    my $root = Path::Tiny->rootdir;
    is( $file,               $root,  "rootdir is like path('/')" );
    is( $file->child("lib"), "/lib", "child of rootdir is correct" );
}

# constructor
{
    is( path(qw/foo bar baz/), Path::Tiny->new(qw/foo bar baz/), "path() vs new" );
    is( path(qw/foo bar baz/), path("foo/bar/baz"), "path(a,b,c) vs path('a/b/c')" );
}

# tilde processing
{
    # Construct expected paths manually with glob, but normalize with Path::Tiny
    # to work around windows slashes and drive case issues.  Extract the interior
    # paths with ->[0] rather than relying on stringification, which will escape
    # leading tildes.

    my $homedir = path(glob('~'))->[0];
    my $username = path($homedir)->basename;
    my $root_homedir = path(glob('~root'))->[0];
    my $missing_homedir = path(glob('~idontthinkso'))->[0];

    # remove one trailing slash from a path string, if present
    # so the result of concatenating a path that starts with a slash will be correct
    sub S ($) { ( my $p = $_[0] ) =~ s!/\z!!; $p }

    my @tests = (
      # [arg for path(), expected string (undef if eq arg for path()), test string]
        ['~',                        $homedir,                   'Test my homedir' ],
        ['~/',                       $homedir,                   'Test my homedir with trailing "/"' ],
        ['~/foo/bar',              S($homedir).'/foo/bar',       'Test my homedir with longer path' ],
        ['~/foo/bar/',             S($homedir).'/foo/bar',       'Test my homedir, longer path and trailing "/"' ],
        ['~root',                    $root_homedir,              'Test root homedir' ],
        ['~root/',                   $root_homedir,              'Test root homedir with trailing /' ],
        ['~root/foo/bar',          S($root_homedir).'/foo/bar',  'Test root homedir with longer path' ],
        ['~root/foo/bar/',         S($root_homedir).'/foo/bar',  'Test root homedir, longer path and trailing "/"'],
        ['~idontthinkso',            undef,                      'Test homedir of nonexistant user' ],
        ['~idontthinkso',            $missing_homedir,           'Test homedir of nonexistant user (via glob)' ],
        ['~blah blah',               undef,                      'Test space' ],
        ['~this is fun',             undef,                      'Test multiple spaces' ],
        ['~yikes \' apostrophe!',    undef,                      'Test spaces and embedded apostrophe' ],
        ['~hum " quote',             undef,                      'Test spaces and embedded quote' ],
        ['~hello ~there',            undef,                      'Test space-separated tildes' ],
        ["~fun\ttimes",              undef,                      'Test tab' ],
        ["~new\nline",               undef,                      'Test newline' ],
        ['~'.$username.' file',      undef,                      'Test \'~$username file\'' ],
        ['./~',                      '~',                        'Test literal tilde under current directory' ],
        ['~idontthinkso[123]',       undef,                      'Test File::Glob metacharacter ['],
        ['~idontthinkso*',           undef,                      'Test File::Glob metacharacter *'],
        ['~idontthinkso?',           undef,                      'Test File::Glob metacharacter ?'],
        ['~idontthinkso{a}',         undef,                      'Test File::Glob metacharacter {'],
    );

    if (! $IS_WIN32 && ! $IS_CYGWIN ) {
        push @tests, ['~idontthinkso\\x',      undef,                    'Test File::Glob metacharacter \\'];
    }

    for my $test (@tests) {
        my $path = path($test->[0]);
        my $internal_path = $path->[0]; # Avoid stringification adding a "./" prefix
        my $expected = defined $test->[1] ? $test->[1] : $test->[0];
        is($internal_path, $expected, $test->[2]);
        is($path, $expected =~ /^~/ ? "./$expected" : $expected, '... and its stringification');
    }

    is(path('.')->child('~')->[0], '~', 'Test indirect form of literal tilde under current directory');
    is(path('.')->child('~'), './~', '... and its stringification');

    $file = path('/tmp/foo/~root');
    is $file->relative('/tmp/foo')->[0], '~root', 'relative path begins with tilde';
    is $file->relative('/tmp/foo'), "./~root", '... and its stringification is escaped';

    # successful tilde expansion of account names with glob metacharacters is
    # actually untested so far because it would require such accounts to exist
    # so instead we wrap File::Glob::bsd_glob to mock up certain responses:
    my %mock = (
        '~i[dont]{think}so' => '/home/i[dont]{think}so',
        '~idont{think}so'   => '/home/idont{think}so',
        '~i{dont,think}so'  => '/home/i{dont,think}so',
    );
    if ( ! $IS_WIN32 && ! $IS_CYGWIN ) {
        $mock{'~i?dont*think*so?'} = '/home/i?dont*think*so?';
    }
    my $orig_bsd_glob = \&File::Glob::bsd_glob;
    my $do_brace_expansion_only = do { package File::Glob; GLOB_NOCHECK() | GLOB_BRACE() | GLOB_QUOTE() };
    sub mock_bsd_glob {
        my $dequoted = $orig_bsd_glob->( $_[0], $do_brace_expansion_only );
        $mock{ $dequoted } || goto &$orig_bsd_glob;
    }
    no warnings 'redefine'; local *File::Glob::bsd_glob = \&mock_bsd_glob;
    is(File::Glob::bsd_glob('{root}'), 'root', 'double-check of mock_bsd_glob dequoting');
    is(File::Glob::bsd_glob('~root'), $root_homedir, 'double-check of mock_bsd_glob fallback');
    for my $test (sort keys %mock) {
        is(path($test), $mock{ $test }, "tilde expansion with glob metacharacters in account name: $test");
    }
}

# freeze/thaw
{
    my @cases = qw(
        /foo/bar/baz"
        ./~root
    );

    for my $c ( @cases ) {
        my $path = path($c);
        is( Path::Tiny->THAW( "fake", $path->FREEZE("fake") ),
            $path, "FREEZE-THAW roundtrip: $c" );
    }
}

# assertions
{
    my $err = exception {
        path("aljfakdlfadks")->assert( sub { $_->exists } )
    };
    like( $err, qr/failed assertion/, "assert exists" );
    my $path;
    $err = exception {
        $path = path("t")->assert( sub { -d && -r _ } )
    };
    is( $err, '', "no exception if assertion succeeds" );
    isa_ok( $path, "Path::Tiny", "assertion return value" );

    $err = exception {
        path(".")->visit(
            sub { $_[1]->{$_} = { path => $_ } },
            { recurse => 1 },
        );
    };
    is $err, "", 'no exception';
}

done_testing();