summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBrian McCauley <nobull@mail.com>2001-06-15 08:51:26 +0100
committerJarkko Hietaniemi <jhi@iki.fi>2001-06-15 14:08:19 +0000
commit9f826d6a0fa46286bcc4b159b950e3f1073f9af4 (patch)
tree8e2e1bd51be112a505c91d41dab8b47f4b7339f6
parentb7bcf49446150838af3c7cd1dec335ba45a2fc6b (diff)
downloadperl-9f826d6a0fa46286bcc4b159b950e3f1073f9af4.tar.gz
Re: [ID 20010608.010] File::Find re-entrancy
Message-Id: <200106150923.f5F9NpG02725@wcl-l.bham.ac.uk> p4raw-id: //depot/perl@10607
-rw-r--r--lib/File/Find.pm22
-rwxr-xr-xt/lib/filefind.t30
2 files changed, 43 insertions, 9 deletions
diff --git a/lib/File/Find.pm b/lib/File/Find.pm
index 9ae39ace5d..274c7b8d17 100644
--- a/lib/File/Find.pm
+++ b/lib/File/Find.pm
@@ -281,8 +281,11 @@ my $Is_MacOS;
require File::Basename;
require File::Spec;
-my %SLnkSeen;
-my ($wanted_callback, $avoid_nlink, $bydepth, $no_chdir, $follow,
+# Should ideally be my() not our() but local() currently
+# refuses to operate on lexicals
+
+our %SLnkSeen;
+our ($wanted_callback, $avoid_nlink, $bydepth, $no_chdir, $follow,
$follow_skip, $full_check, $untaint, $untaint_skip, $untaint_pat,
$pre_process, $post_process);
@@ -447,6 +450,15 @@ sub _find_opt {
my $wanted = shift;
die "invalid top directory" unless defined $_[0];
+ # This function must local()ize everything because callbacks may
+ # call find() or finddepth()
+
+ local %SLnkSeen;
+ local ($wanted_callback, $avoid_nlink, $bydepth, $no_chdir, $follow,
+ $follow_skip, $full_check, $untaint, $untaint_skip, $untaint_pat,
+ $pre_process, $post_process);
+ local($dir, $name, $fullname, $prune);
+
my $cwd = $wanted->{bydepth} ? Cwd::fastcwd() : Cwd::cwd();
my $cwd_untainted = $cwd;
my $check_t_cwd = 1;
@@ -463,7 +475,7 @@ sub _find_opt {
$untaint_skip = $wanted->{untaint_skip};
# for compatability reasons (find.pl, find2perl)
- our ($topdir, $topdev, $topino, $topmode, $topnlink);
+ local our ($topdir, $topdev, $topino, $topmode, $topnlink);
# a symbolic link to a directory doesn't increase the link count
$avoid_nlink = $follow || $File::Find::dont_use_nlink;
@@ -1028,17 +1040,13 @@ sub wrap_wanted {
sub find {
my $wanted = shift;
- %SLnkSeen= (); # clear hash first
_find_opt(wrap_wanted($wanted), @_);
- %SLnkSeen= (); # free memory
}
sub finddepth {
my $wanted = wrap_wanted(shift);
- %SLnkSeen= (); # clear hash first
$wanted->{bydepth} = 1;
_find_opt($wanted, @_);
- %SLnkSeen= (); # free memory
}
# default
diff --git a/t/lib/filefind.t b/t/lib/filefind.t
index d07948b0ea..3e73ffcadc 100755
--- a/t/lib/filefind.t
+++ b/t/lib/filefind.t
@@ -9,10 +9,10 @@ my $cwd_untainted;
BEGIN {
chdir 't' if -d 't';
- @INC = '../lib';
+ unshift @INC => '../lib';
for (keys %ENV) { # untaint ENV
- ($ENV{$_}) = keys %{{ map {$_ => 1} $ENV{$_} }};
+ ($ENV{$_}) = $ENV{$_} =~ /(.*)/;
}
$SIG{'__WARN__'} = sub { $warn_msg = $_[0]; warn "# Warn: $_[0]"; }
@@ -215,6 +215,19 @@ if ($^O eq 'MacOS') {
File::Find::find( {wanted => \&wanted, untaint => 1},':fa' );
Check( scalar(keys %Expect) == 0 );
+ print "# check re-entancy\n";
+ %Expect = (':' => 1, 'fsl' => 1, 'fa_ord' => 1, 'fab' => 1, 'fab_ord' => 1,
+ 'faba' => 1, 'faa' => 1, 'faa_ord' => 1);
+ delete $Expect{'fsl'} unless $symlink_exists;
+ %Expect_Dir = (':' => 1, 'fa' => 1, 'faa' => 1, 'fab' => 1, 'faba' => 1,
+ 'fb' => 1, 'fba' => 1);
+ delete @Expect_Dir{'fb','fba'} unless $symlink_exists;
+ File::Find::find( {wanted => sub {
+ wanted();
+ File::Find::find( {wanted => sub {} , untaint => 1 },':' );
+ }, untaint => 1 }, ':fa' );
+ Check( scalar(keys %Expect) == 0 );
+
%Expect=(':fa' => 1, ':fa:fsl' => 1, ':fa:fa_ord' => 1, ':fa:fab' => 1,
':fa:fab:fab_ord' => 1, ':fa:fab:faba' => 1,
':fa:fab:faba:faba_ord' => 1, ':fa:faa' => 1, ':fa:faa:faa_ord' => 1);
@@ -465,6 +478,19 @@ if ($^O eq 'MacOS') {
File::Find::find( {wanted => \&wanted, untaint => 1, untaint_pattern => qr|^(.+)$|},'fa' );
Check( scalar(keys %Expect) == 0 );
+ print "# check re-entancy\n";
+ %Expect = ('.' => 1, 'fsl' => 1, 'fa_ord' => 1, 'fab' => 1, 'fab_ord' => 1,
+ 'faba' => 1, 'faa' => 1, 'faa_ord' => 1);
+ delete $Expect{'fsl'} unless $symlink_exists;
+ %Expect_Dir = ('fa' => 1, 'faa' => 1, 'fab' => 1, 'faba' => 1,
+ 'fb' => 1, 'fba' => 1);
+ delete @Expect_Dir{'fb','fba'} unless $symlink_exists;
+ File::Find::find( {wanted => sub {
+ wanted();
+ File::Find::find( {wanted => sub {} , untaint => 1, untaint_pattern => qr|^(.+)$|},'.' );
+ }, untaint => 1, untaint_pattern => qr|^(.+)$|},'fa' );
+ Check( scalar(keys %Expect) == 0 );
+
%Expect=('fa' => 1, 'fa/fsl' => 1, 'fa/fa_ord' => 1, 'fa/fab' => 1,
'fa/fab/fab_ord' => 1, 'fa/fab/faba' => 1,
'fa/fab/faba/faba_ord' => 1, 'fa/faa' => 1, 'fa/faa/faa_ord' => 1);