Ancient

 view release on metacpan or  search on metacpan

t/8006-file-platform.t  view on Meta::CPAN

#!/usr/bin/env perl
use strict;
use warnings;
use Test::More;
use File::Temp qw(tempdir);
use Config;

use_ok('file');

my $tmpdir = tempdir(CLEANUP => 1);
my $is_windows = $^O eq 'MSWin32';
my $is_unix = !$is_windows;

diag("Running on: $^O");
diag("Perl version: $]");

# ============================================
# Platform detection
# ============================================

subtest 'platform identification' => sub {
    ok(defined $^O, 'OS is defined');
    diag("Operating system: $^O");
};

# ============================================
# Path separator handling
# ============================================

subtest 'path separators' => sub {
    # Forward slashes work everywhere
    my $path1 = file::join('a', 'b', 'c');
    ok($path1 =~ m{a.b.c}, 'join produces path-like string');

    # Test dirname/basename with forward slashes
    is(file::basename('/path/to/file.txt'), 'file.txt', 'basename with forward slashes');
    is(file::dirname('/path/to/file.txt'), '/path/to', 'dirname with forward slashes');

    SKIP: {
        skip "Windows path tests", 2 unless $is_windows;

        # Windows should also handle backslashes
        is(file::basename('C:\\path\\to\\file.txt'), 'file.txt', 'basename with backslashes');
        is(file::dirname('C:\\path\\to\\file.txt'), 'C:\\path\\to', 'dirname with backslashes');
    }
};

# ============================================
# Symlink tests (Unix only)
# ============================================

SKIP: {
    skip "Symlink tests require Unix", 6 unless $is_unix;

    subtest 'symlinks' => sub {
        my $target = "$tmpdir/symlink_target.txt";
        my $link = "$tmpdir/symlink_link.txt";

        file::spew($target, "target content");

        SKIP: {
            skip "symlink not available", 5 unless eval { symlink($target, $link) };

            ok(file::exists($link), 'symlink exists');
            ok(file::is_link($link), 'is_link returns true for symlink');
            ok(!file::is_link($target), 'is_link returns false for regular file');

            # Reading through symlink should work
            is(file::slurp($link), "target content", 'can read through symlink');

            # File tests on symlink
            ok(file::is_file($link), 'symlink to file is_file');
        }
    };
}

# ============================================
# Permission tests (Unix only)
# ============================================

SKIP: {
    skip "Permission tests require Unix", 1 unless $is_unix;

    subtest 'unix permissions' => sub {
        my $file = "$tmpdir/perm_test.txt";
        file::spew($file, "permission test");

        # Test chmod
        ok(file::chmod($file, 0644), 'chmod 0644');
        my $mode = file::mode($file);
        is($mode & 0777, 0644, 'mode is 0644');

        ok(file::chmod($file, 0755), 'chmod 0755');
        $mode = file::mode($file);
        is($mode & 0777, 0755, 'mode is 0755');

        # Test is_executable
        ok(file::is_executable($file), 'is_executable after chmod 0755');

        file::chmod($file, 0644);
        ok(!file::is_executable($file), 'not executable after chmod 0644');
    };
}

# ============================================
# Case sensitivity
# ============================================

subtest 'case sensitivity' => sub {
    my $lower = "$tmpdir/casefile.txt";
    my $upper = "$tmpdir/CaseFile.txt";

    file::spew($lower, "lowercase");

    SKIP: {
        # On case-insensitive systems (macOS default, Windows), these are same file
        skip "Case-insensitive filesystem", 2 if file::exists($upper) && !$is_unix;

        # On case-sensitive systems, these are different files
        if (!file::exists($upper)) {
            file::spew($upper, "UPPERCASE");
            ok(file::exists($lower), 'lowercase file exists');
            ok(file::exists($upper), 'uppercase file exists');
            isnt(file::slurp($lower), file::slurp($upper), 'different content');
        }
    }

    pass('case sensitivity test completed');
};

# ============================================
# Line ending handling
# ============================================

subtest 'line endings' => sub {
    my $unix_file = "$tmpdir/unix_lines.txt";
    my $win_file = "$tmpdir/win_lines.txt";
    my $mac_file = "$tmpdir/mac_lines.txt";

    # Unix: \n
    file::spew($unix_file, "line1\nline2\nline3");
    my $unix_lines = file::lines($unix_file);
    is(scalar(@$unix_lines), 3, 'unix line count');
    is($unix_lines->[0], 'line1', 'unix line 1');

    # Windows: \r\n (note: file module splits on \n only)
    file::spew($win_file, "line1\r\nline2\r\nline3");
    my $win_lines = file::lines($win_file);
    is(scalar(@$win_lines), 3, 'windows line count');
    # Lines will have \r at end
    like($win_lines->[0], qr/^line1/, 'windows line 1 starts correctly');

    # Old Mac: \r only
    file::spew($mac_file, "line1\rline2\rline3");
    my $mac_lines = file::lines($mac_file);
    is(scalar(@$mac_lines), 1, 'mac lines (no \\n) is one line');
};

# ============================================
# Binary mode consistency
# ============================================

subtest 'binary mode' => sub {
    my $file = "$tmpdir/binary_mode.dat";

    # Write binary data
    my $binary = join('', map { chr($_) } 0..255);
    file::spew($file, $binary);

    # Read with slurp
    my $read1 = file::slurp($file);
    is(length($read1), 256, 'slurp binary length');

    # Read with slurp_raw
    my $read2 = file::slurp_raw($file);
    is(length($read2), 256, 'slurp_raw binary length');

    # Both should be identical
    is($read1, $read2, 'slurp and slurp_raw identical for binary');
    is($read1, $binary, 'binary content preserved');
};

# ============================================
# Temporary directory behavior
# ============================================

subtest 'temp directory' => sub {
    ok(file::is_dir($tmpdir), 'temp directory exists');
    ok(file::is_writable($tmpdir), 'temp directory is writable');

    my $nested = "$tmpdir/nested/deep/dir";
    # file::mkdir doesn't do recursive mkdir
    ok(!file::mkdir($nested), 'mkdir fails for nested (no recursive)');

    # Create step by step
    ok(file::mkdir("$tmpdir/nested"), 'mkdir nested');
    ok(file::mkdir("$tmpdir/nested/deep"), 'mkdir nested/deep');
    ok(file::mkdir("$tmpdir/nested/deep/dir"), 'mkdir nested/deep/dir');
    ok(file::is_dir($nested), 'nested directory created');
};

# ============================================
# File size limits
# ============================================

subtest 'file sizes' => sub {
    # Empty file
    my $empty = "$tmpdir/size_empty.txt";
    file::spew($empty, "");
    is(file::size($empty), 0, 'empty file size');

    # Small file
    my $small = "$tmpdir/size_small.txt";
    file::spew($small, "hello");
    is(file::size($small), 5, 'small file size');

    # Medium file
    my $medium = "$tmpdir/size_medium.txt";
    file::spew($medium, "x" x 10000);
    is(file::size($medium), 10000, 'medium file size');

    SKIP: {
        skip "Large file test slow", 1 unless $ENV{TEST_LARGE_FILES};

        # Large file (10MB)
        my $large = "$tmpdir/size_large.txt";
        file::spew($large, "x" x (10 * 1024 * 1024));
        is(file::size($large), 10 * 1024 * 1024, 'large file size');
    }
};

# ============================================
# Special files (Unix only)
# ============================================

SKIP: {
    skip "Special file tests require Unix", 1 unless $is_unix;

    subtest 'special files' => sub {
        # /dev/null
        SKIP: {
            skip "/dev/null not available", 3 unless -e '/dev/null';

            ok(file::exists('/dev/null'), '/dev/null exists');
            # Can write to /dev/null
            ok(file::spew('/dev/null', "test"), 'can write to /dev/null');
            # Size of /dev/null is 0
            is(file::size('/dev/null'), 0, '/dev/null size is 0');
        }

        # /dev/zero - reading would be problematic, just check exists
        SKIP: {
            skip "/dev/zero not available", 1 unless -e '/dev/zero';

            ok(file::exists('/dev/zero'), '/dev/zero exists');
        }
    };
}

# ============================================
# Atomic operations across platforms
# ============================================

subtest 'atomic operations' => sub {
    my $file = "$tmpdir/atomic_platform.txt";

    # Multiple atomic writes
    for my $i (1..10) {
        ok(file::atomic_spew($file, "version $i"), "atomic write $i");
    }

    is(file::slurp($file), "version 10", 'final atomic content');

    # Verify no temp files left behind
    my $entries = file::readdir($tmpdir);
    my @temps = grep { /\.tmp\.\d+/ } @$entries;
    is(scalar(@temps), 0, 'no temp files left behind');
};

done_testing();



( run in 1.293 second using v1.01-cache-2.11-cpan-524268b4103 )