@INC = qw(t . lib ../lib);
}
+use Config;
require "test.pl";
-plan(tests => 25);
+plan(tests => 41);
-my $IsVMS = $^O eq 'VMS';
+my $IsVMS = $^O eq 'VMS';
+my $IsMacOS = $^O eq 'MacOS';
# Might be a little early in the testing process to start using these,
# but I can't think of a way to write this test without them.
-use File::Spec::Functions qw(:DEFAULT splitdir rel2abs);
+use File::Spec::Functions qw(:DEFAULT splitdir rel2abs splitpath);
# Can't use Cwd::abs_path() because it has different ideas about
-# path seperators than File::Spec.
+# path separators than File::Spec.
sub abs_path {
- $IsVMS ? uc(rel2abs(curdir)) : rel2abs(curdir);
+ my $d = rel2abs(curdir);
+
+ $d = uc($d) if $IsVMS;
+ $d = lc($d) if $^O =~ /^uwin/;
+ $d;
}
my $Cwd = abs_path;
# Let's get to a known position
SKIP: {
- skip("Already in t/", 2) if (splitdir(abs_path))[-1] eq ($IsVMS ? 'T' : 't');
+ my ($vol,$dir) = splitpath(abs_path,1);
+ my $test_dir = $IsVMS ? 'T' : 't';
+ skip("Already in t/", 2) if (splitdir($dir))[-1] eq $test_dir;
- ok( chdir('t'), 'chdir("t")');
- is( abs_path, catdir($Cwd, 't'), ' abs_path() agrees' );
+ ok( chdir($test_dir), 'chdir($test_dir)');
+ is( abs_path, catdir($Cwd, $test_dir), ' abs_path() agrees' );
}
$Cwd = abs_path;
+SKIP: {
+ skip("no fchdir", 9) unless ($Config{d_fchdir} || "") eq "define";
+ ok(opendir(my $dh, "."), "opendir .");
+ ok(open(my $fh, "<", "op"), "open op");
+ ok(chdir($fh), "fchdir op");
+ ok(-f "chdir.t", "verify that we are in op");
+ if (($Config{d_dirfd} || "") eq "define") {
+ ok(chdir($dh), "fchdir back");
+ }
+ else {
+ eval { chdir($dh); };
+ like($@, qr/^The dirfd function is unimplemented at/, "dirfd is unimplemented");
+ chdir "..";
+ }
+
+ # same with bareword file handles
+ no warnings 'once';
+ *DH = $dh;
+ *FH = $fh;
+ ok(chdir FH, "fchdir op bareword");
+ ok(-f "chdir.t", "verify that we are in op");
+ if (($Config{d_dirfd} || "") eq "define") {
+ ok(chdir DH, "fchdir back bareword");
+ }
+ else {
+ eval { chdir(DH); };
+ like($@, qr/^The dirfd function is unimplemented at/, "dirfd is unimplemented");
+ chdir "..";
+ }
+ ok(-d "op", "verify that we are back");
+}
+
+SKIP: {
+ skip("has fchdir", 1) if ($Config{d_fchdir} || "") eq "define";
+ opendir(my $dh, "op");
+ eval { chdir($dh); };
+ like($@, qr/^The fchdir function is unimplemented at/, "fchdir is unimplemented");
+}
+
# The environment variables chdir() pays attention to.
my @magic_envs = qw(HOME LOGDIR SYS$LOGIN);
my($key) = @_;
# Make sure $ENV{'SYS$LOGIN'} is only honored on VMS.
- if( $key eq 'SYS$LOGIN' && !$IsVMS ) {
+ if( $key eq 'SYS$LOGIN' && !$IsVMS && !$IsMacOS ) {
ok( !chdir(), "chdir() on $^O ignores only \$ENV{$key} set" );
is( abs_path, $Cwd, ' abs_path() did not change' );
- pass( " no need to chdir back on $^O" );
+ pass( " no need to test SYS\$LOGIN on $^O" ) for 1..7;
}
else {
ok( chdir(), "chdir() w/ only \$ENV{$key} set" );
# Check the deprecated chdir(undef) feature.
-#line 60
+#line 64
ok( chdir(undef), "chdir(undef) w/ only \$ENV{$key} set" );
is( abs_path, $ENV{$key}, ' abs_path() agrees' );
is( $warning, <<WARNING, ' got uninit & deprecation warning' );
-Use of uninitialized value in chdir at $0 line 60.
-Use of chdir('') or chdir(undef) as chdir() is deprecated at $0 line 60.
+Use of uninitialized value in chdir at $0 line 64.
+Use of chdir('') or chdir(undef) as chdir() is deprecated at $0 line 64.
WARNING
chdir($Cwd);
# Ditto chdir('').
$warning = '';
-#line 72
+#line 76
ok( chdir(''), "chdir('') w/ only \$ENV{$key} set" );
is( abs_path, $ENV{$key}, ' abs_path() agrees' );
is( $warning, <<WARNING, ' got deprecation warning' );
-Use of chdir('') or chdir(undef) as chdir() is deprecated at $0 line 72.
+Use of chdir('') or chdir(undef) as chdir() is deprecated at $0 line 76.
WARNING
chdir($Cwd);
}
}
+my %Saved_Env = ();
+sub clean_env {
+ foreach my $env (@magic_envs) {
+ $Saved_Env{$env} = $ENV{$env};
+
+ # Can't actually delete SYS$ stuff on VMS.
+ next if $IsVMS && $env eq 'SYS$LOGIN';
+ next if $IsVMS && $env eq 'HOME' && !$Config{'d_setenv'};
+
+ unless ($IsMacOS) { # ENV on MacOS is "special" :-)
+ # On VMS, %ENV is many layered.
+ delete $ENV{$env} while exists $ENV{$env};
+ }
+ }
+
+ # The following means we won't really be testing for non-existence,
+ # but in Perl we can only delete from the process table, not the job
+ # table.
+ $ENV{'SYS$LOGIN'} = '' if $IsVMS;
+}
+
+END {
+ no warnings 'uninitialized';
+
+ # Restore the environment for VMS (and doesn't hurt for anyone else)
+ @ENV{@magic_envs} = @Saved_Env{@magic_envs};
+
+ # On VMS this must be deleted or process table is wrong on exit
+ # when this script is run interactively.
+ delete $ENV{'SYS$LOGIN'} if $IsVMS;
+}
+
+
foreach my $key (@magic_envs) {
# We're going to be using undefs a lot here.
no warnings 'uninitialized';
- local %ENV = () if !$IsVMS;
+ clean_env;
$ENV{$key} = catdir $Cwd, ($IsVMS ? 'OP' : 'op');
-
+
check_env($key);
}
{
- local %ENV = () if !$IsVMS;
-
- ok( !chdir(), 'chdir() w/o any ENV set' );
+ clean_env;
+ if (($IsVMS || $IsMacOS) && !$Config{'d_setenv'}) {
+ pass("Can't reset HOME, so chdir() test meaningless");
+ } else {
+ ok( !chdir(), 'chdir() w/o any ENV set' );
+ }
is( abs_path, $Cwd, ' abs_path() agrees' );
}