## $Id: //depot/libnet/Net/FTP/A.pm#16 $
## Package to read/write on ASCII data connections
##
package Net::FTP::A;
use strict;
use vars qw(@ISA $buf $VERSION);
use Carp;
require Net::FTP::dataconn;
@ISA = qw(Net::FTP::dataconn);
$VERSION = "1.15";
sub read {
my $data = shift;
local *buf = \$_[0]; shift;
my $size = shift || croak 'read($buf,$size,[$offset])';
my $timeout = @_ ? shift : $data->timeout;
if (length(${*$data}) < $size && !${*$data}{'net_ftp_eof'}) {
my $blksize = ${*$data}{'net_ftp_blksize'};
$blksize = $size if $size > $blksize;
my $l = 0;
my $n;
READ:
{
my $readbuf = defined(${*$data}{'net_ftp_cr'}) ? "\015" : '';
$data->can_read($timeout) or
croak "Timeout";
if ($n = sysread($data, $readbuf, $blksize, length $readbuf)) {
${*$data}{'net_ftp_bytesread'} += $n;
${*$data}{'net_ftp_cr'} = substr($readbuf,-1) eq "\015"
? chop($readbuf)
: undef;
}
else {
return undef
unless defined $n;
${*$data}{'net_ftp_eof'} = 1;
}
$readbuf =~ s/\015\012/\n/sgo;
${*$data} .= $readbuf;
unless (length(${*$data})) {
redo READ
if($n > 0);
$size = length(${*$data})
if($n == 0);
}
}
}
$buf = substr(${*$data},0,$size);
substr(${*$data},0,$size) = '';
length $buf;
}
sub write {
my $data = shift;
local *buf = \$_[0]; shift;
my $size = shift || croak 'write($buf,$size,[$timeout])';
my $timeout = @_ ? shift : $data->timeout;
(my $tmp = substr($buf,0,$size)) =~ s/\n/\015\012/sg;
# If the remote server has closed the connection we will be signal'd
# when we write. This can happen if the disk on the remote server fills up
local $SIG{PIPE} = 'IGNORE' unless $^O eq 'MacOS';
my $len = length($tmp);
my $off = 0;
my $wrote = 0;
my $blksize = ${*$data}{'net_ftp_blksize'};
while($len) {
$data->can_write($timeout) or
croak "Timeout";
$off += $wrote;
$wrote = syswrite($data, substr($tmp,$off), $len > $blksize ? $blksize : $len);
return undef
unless defined($wrote);
$len -= $wrote;
}
$size;
}
1;
|