#!./perl BEGIN { chdir 't' if -d 't'; @INC = '../lib'; require './test.pl'; skip_all("VMS too picky about line endings for record-oriented pipes") if $^O eq 'VMS'; } use strict; my $Perl = which_perl(); my $data = <<'EOD'; x yy z EOD (my $data2 = $data) =~ s/\n/\n\n/g; my $t1 = { data => $data, write_c => [1,2,length $data], read_c => [1,2,3,length $data]}; my $t2 = { data => $data2, write_c => [1,2,length $data2], read_c => [1,2,3,length $data2]}; $_->{write_c} = [1..length($_->{data})], $_->{read_c} = [1..length($_->{data})+1, 0xe000] # Need <0xffff for REx for (); # $t1, $t2; my $c; # len write tests, for each: one _all test, and 3 each len+2 $c += @{$_->{write_c}} * (1 + 3*@{$_->{read_c}}) for $t1, $t2; $c *= 3*2*2; # $how_w, file/pipe, 2 reports $c += 6; # Tests with sleep()... print "1..$c\n"; my $set_out = ''; $set_out = "binmode STDOUT, ':crlf'" if defined $main::use_crlf && $main::use_crlf == 1; sub testread ($$$$$$$) { my ($fh, $str, $read_c, $how_r, $write_c, $how_w, $why) = @_; my $buf = ''; if ($how_r eq 'readline_all') { $buf .= $_ while <$fh>; } elsif ($how_r eq 'readline') { $/ = \$read_c; $buf .= $_ while <$fh>; } elsif ($how_r eq 'read') { my($in, $c); $buf .= $in while $c = read($fh, $in, $read_c); } elsif ($how_r eq 'sysread') { my($in, $c); $buf .= $in while $c = sysread($fh, $in, $read_c); } else { die "Unrecognized read: '$how_r'"; } close $fh or die "close: $!"; # The only contamination allowed is with sysread/prints $buf =~ s/\r\n/\n/g if $how_r eq 'sysread' and $how_w =~ /print/; is(length $buf, length $str, "length with wrc=$write_c, rdc=$read_c, $how_w, $how_r, $why"); is($buf, $str, "content with wrc=$write_c, rdc=$read_c, $how_w, $how_r, $why"); } sub testpipe ($$$$$$) { my ($str, $write_c, $read_c, $how_w, $how_r, $why) = @_; (my $quoted = $str) =~ s/\n/\\n/g;; my $fh; if ($how_w eq 'print') { # AUTOFLUSH??? # Should be shell-neutral: open $fh, '-|', qq[$Perl -we "$set_out;print for grep length, split /(.{1,$write_c})/s, qq($quoted)"] or die "open: $!"; } elsif ($how_w eq 'print/flush') { # shell-neutral and miniperl-enabled autoflush? qq(\x24\x7c) eq '$|' if ($::IS_ASCII) { open $fh, '-|', qq[$Perl -we "$set_out;eval qq(\\x24\\x7c = 1) or die;print for grep length, split /(.{1,$write_c})/s, qq($quoted)"] or die "open: $!"; } else { open $fh, '-|', qq[$Perl -we "$set_out;eval qq(\\x5b\\x4f = 1) or die;print for grep length, split /(.{1,$write_c})/s, qq($quoted)"] or die "open: $!"; } } elsif ($how_w eq 'syswrite') { ### How to protect \$_ if ($::IS_ASCII) { open $fh, '-|', qq[$Perl -we "$set_out;eval qq(sub w {syswrite STDOUT, \\x24_} 1) or die; w() for grep length, split /(.{1,$write_c})/s, qq($quoted)"] or die "open: $!"; } else { open $fh, '-|', qq[$Perl -we "$set_out;eval qq(sub w {syswrite STDOUT, \\x5B_} 1) or die; w() for grep length, split /(.{1,$write_c})/s, qq($quoted)"] or die "open: $!"; } } else { die "Unrecognized write: '$how_w'"; } binmode $fh, ':crlf' if defined $main::use_crlf && $main::use_crlf == 1; testread($fh, $str, $read_c, $how_r, $write_c, $how_w, "pipe$why"); } sub testfile ($$$$$$) { my ($str, $write_c, $read_c, $how_w, $how_r, $why) = @_; my @data = grep length, split /(.{1,$write_c})/s, $str; my $filename = tempfile(); open my $fh, '>', $filename or die "open: > $filename: $!"; select $fh; binmode $fh, ':crlf' if defined $main::use_crlf && $main::use_crlf == 1; if ($how_w eq 'print') { # AUTOFLUSH??? $| = 0; print $fh $_ for @data; } elsif ($how_w eq 'print/flush') { $| = 1; print $fh $_ for @data; } elsif ($how_w eq 'syswrite') { syswrite $fh, $_ for @data; } else { die "Unrecognized write: '$how_w'"; } close $fh or die "close: $!"; open $fh, '<', $filename or die "open: < $filename: $!"; binmode $fh, ':crlf' if defined $main::use_crlf && $main::use_crlf == 1; testread($fh, $str, $read_c, $how_r, $write_c, $how_w, "file$why"); } # shell-neutral and miniperl-enabled autoflush? qq(\x24\x7c) eq '$|' my $fh; if ($::IS_ASCII) { open $fh, '-|', qq[$Perl -we "eval qq(\\x24\\x7c = 1) or die; binmode STDOUT; sleep 1, print for split //, qq(a\nb\n\nc\n\n\n)"] or die "open: $!"; } else { open $fh, '-|', qq[$Perl -we "eval qq(\\x5B\\x4f = 1) or die; binmode STDOUT; sleep 1, print for split //, qq(a\nb\n\nc\n\n\n)"] or die "open: $!"; } ok(1, 'open pipe'); binmode $fh, q(:crlf); ok(1, 'binmode'); $c = undef; my @c; push @c, ord $c while $c = getc $fh; ok(1, 'got chars'); is(scalar @c, 9, 'got 9 chars'); is("@c", join(" ", utf8::unicode_to_native(97), utf8::unicode_to_native(10), utf8::unicode_to_native(98), utf8::unicode_to_native(10), utf8::unicode_to_native(10), utf8::unicode_to_native(99), utf8::unicode_to_native(10), utf8::unicode_to_native(10), utf8::unicode_to_native(10)), 'got expected chars'); ok(close($fh), 'close'); for my $s (1..2) { my $t = ($t1, $t2)[$s-1]; my $str = $t->{data}; my $r = $t->{read_c}; my $w = $t->{write_c}; for my $read_c (@$r) { for my $write_c (@$w) { for my $how_r (qw(readline_all readline read sysread)) { next if $how_r eq 'readline_all' and $read_c != 1; for my $how_w (qw(print print/flush syswrite)) { testfile($str, $write_c, $read_c, $how_w, $how_r, $s); testpipe($str, $write_c, $read_c, $how_w, $how_r, $s); } } } } } 1;