summaryrefslogtreecommitdiff
path: root/lib/Net/FTP.pm
diff options
context:
space:
mode:
authorPerl 5 Porters <perl5-porters@africa.nicoh.com>1996-12-20 11:14:00 +1200
committerChip Salzenberg <chip@atlantic.net>1996-12-20 11:14:00 +1200
commit7e1af8bca57f405a8444b575a870918a6d88fc5c (patch)
treeb443adc34d8d77831bf947076abd5770335592cf /lib/Net/FTP.pm
parent7f3dfc00eaef7e421633b2b47af9963dbc626e75 (diff)
downloadperl-7e1af8bca57f405a8444b575a870918a6d88fc5c.tar.gz
[inseparable changes from patch from perl5.003_12 to perl5.003_13]
DOCUMENTATION Subject: small doc tweaks for _12 Date: Thu, 19 Dec 1996 11:05:57 -0500 From: Roderick Schertler <roderick@gate.net> Files: lib/UNIVERSAL.pm pod/perldiag.pod pod/perltie.pod Msg-ID: <1826.851011557@eeyore.ibcinc.com> (applied based on p5p patch as commit 3314ffc68a11690bd9977cbdd7ea0601ad6ced13) PORTABILITY Subject: Add missing backslash in Configure From: Chip Salzenberg <chip@atlantic.net> Files: Configure UTILITIES, LIBRARY, AND EXTENSIONS Subject: Include libnet-1.01 instead of old Net::FTP From: Graham Barr <Graham.Barr@tiuk.ti.com> Files: MANIFEST lib/Net/Cmd.pm lib/Net/Domain.pm lib/Net/DummyInetd.pm lib/Net/FTP.pm lib/Net/NNTP.pm lib/Net/Netrc.pm lib/Net/POP3.pm lib/Net/SMTP.pm lib/Net/SNPP.pm lib/Net/Socket.pm lib/Net/Telnet.pm lib/Net/Time.pm pod/perlmod.pod Subject: Use binmode when doing binary FTP From: Ilya Zakharevich <ilya@math.ohio-state.edu> Files: lib/Net/FTP.pm Subject: Re: Open3.pm tries to close unopened file handle Date: 18 Dec 1996 22:19:54 -0500 From: Roderick Schertler <roderick@gate.net> Files: MANIFEST lib/IPC/Open2.pm lib/IPC/Open3.pm lib/open2.pl lib/open3.pl pod/perldiag.pod pod/perlfunc.pod t/lib/open2.t t/lib/open3.t Msg-ID: <pzloavmd9h.fsf@eeyore.ibcinc.com> (applied based on p5p patch as commit 982b4e8fc47473059e209787b589853f4c8f8f9e) Subject: Long-standing problem in Socket module Date: Wed, 18 Dec 1996 23:18:14 -0500 From: Spider Boardman <spider@orb.nashua.nh.us> Files: Configure Porting/Glossary config_H config_h.SH ext/Socket/Socket.pm ext/Socket/Socket.xs Msg-ID: <199612190418.XAA07291@Orb.Nashua.NH.US> (applied based on p5p patch as commit 3e6a22d2723daf415793f9a4fc1b57f4d8a576fd) Subject: flock() constants Date: Thu, 19 Dec 1996 01:37:17 -0500 From: Roderick Schertler <roderick@gate.net> Files: ext/Fcntl/Fcntl.pm ext/Fcntl/Fcntl.xs pod/perlfunc.pod Msg-ID: <26669.850977437@eeyore.ibcinc.com> (applied based on p5p patch as commit 3dea0e15e4684f6defe2f25a16bc696b96697ac2)
Diffstat (limited to 'lib/Net/FTP.pm')
-rw-r--r--lib/Net/FTP.pm1538
1 files changed, 993 insertions, 545 deletions
diff --git a/lib/Net/FTP.pm b/lib/Net/FTP.pm
index 64b21fe751..d635f000bc 100644
--- a/lib/Net/FTP.pm
+++ b/lib/Net/FTP.pm
@@ -1,16 +1,8 @@
-;# Net::FTP.pm
-;#
-;# Copyright (c) 1995 Graham Barr <Graham.Barr@tiuk.ti.com>. All rights
-;# reserved. This program is free software; you can redistribute it and/or
-;# modify it under the same terms as Perl itself.
-
-;#Notes
-;# should I have a dataconn::close sub which calls response ??
-;# FTP should hold state reguarding cmds sent
-;# A::read needs some more thought
-;# A::write What is previous pkt ended in \r or not ??
-;# need to do some heavy tidy-ing up !!!!
-;# need some documentation
+# Net::FTP.pm
+#
+# Copyright (c) 1995 Graham Barr <Graham.Barr@tiuk.ti.com>. All rights
+# reserved. This program is free software; you can redistribute it and/or
+# modify it under the same terms as Perl itself.
package Net::FTP;
@@ -20,277 +12,649 @@ Net::FTP - FTP Client class
=head1 SYNOPSIS
- require Net::FTP;
-
- $ftp = Net::FTP->new("some.host.name");
- $ftp->login("anonymous","me@here.there");
- $ftp->cwd("/pub");
- $ftp->get("that.file");
- $ftp->quit;
+ use Net::FTP;
+
+ $ftp = Net::FTP->new("some.host.name");
+ $ftp->login("anonymous","me@here.there");
+ $ftp->cwd("/pub");
+ $ftp->get("that.file");
+ $ftp->quit;
=head1 DESCRIPTION
C<Net::FTP> is a class implementing a simple FTP client in Perl as described
in RFC959
-=head2 TO BE CONTINUED ...
+C<Net::FTP> provides methods that will perform various operations. These methods
+could be split into groups depending the level of interface the user requires.
-=cut
+=head1 CONSTRUCTOR
-require 5.001;
-use Socket 1.3;
-use Carp;
-use Net::Socket;
+=over 4
-@ISA = qw(Net::Socket);
+=item new (HOST [,OPTIONS])
-$VERSION = sprintf("%d.%02d", q$Revision: 1.18 $ =~ /(\d+)\.(\d+)/);
-sub Version { $VERSION }
+This is the constructor for a new Net::SMTP object. C<HOST> is the
+name of the remote host to which a FTP connection is required.
-use strict;
+C<OPTIONS> are passed in a hash like fasion, using key and value pairs.
+Possible options are:
+
+B<Firewall> - The name of a machine which acts as a FTP firewall. This can be
+overridden by an environment variable C<FTP_FIREWALL>. If specified, and the
+given host cannot be directly connected to, then the
+connection is made to the firwall machine and the string C<@hostname> is
+appended to the login identifier.
+
+B<Port> - The port number to connect to on the remote machine for the
+FTP connection
+
+B<Timeout> - Set a timeout value (defaults to 120)
+
+B<Debug> - Debug level
+
+B<Passive> - If set to I<true> then all data transfers will be done using
+passive mode. This is required for some I<dumb> servers.
+
+=back
=head1 METHODS
-All methods return 0 or undef upon failure
+Unless otherwise stated all methods return either a I<true> or I<false>
+value, with I<true> meaning that the operation was a success. When a method
+states that it returns a value, falure will be returned as I<undef> or an
+empty list.
+
+=over 4
+
+=item login ([LOGIN [,PASSWORD [, ACCOUNT] ] ])
+
+Log into the remote FTP server with the given login information. If
+no arguments are given then the C<Net::FTP> uses the C<Net::Netrc>
+package to lookup the login information for the connected host.
+If no information is found then a login of I<anonymous> is used.
+If no password is given and the login is I<anonymous> then the users
+Email address will be used for a password.
+
+If the connection is via a firewall then the C<authorize> method will
+be called with no arguments.
+
+=item authorize ( [AUTH [, RESP]])
+
+This is a protocol used by some firewall ftp proxies. It is used
+to authorise the user to send data out. If both arguments are not specified
+then C<authorize> uses C<Net::Netrc> to do a lookup.
+
+=item type (TYPE [, ARGS])
+
+This method will send the TYPE command to the remote FTP server
+to change the type of data transfer. The return value is the previous
+value.
+
+=item ascii ([ARGS]) binary([ARGS]) ebcdic([ARGS]) byte([ARGS])
+
+Synonyms for C<type> with the first arguments set correctly
+
+B<NOTE> ebcdic and byte are not fully supported.
+
+=item rename ( OLDNAME, NEWNAME )
+
+Rename a file on the remote FTP server from C<OLDNAME> to C<NEWNAME>. This
+is done by sending the RNFR and RNTO commands.
+
+=item delete ( FILENAME )
+
+Send a request to the server to delete C<FILENAME>.
+
+=item cwd ( [ DIR ] )
+
+Change the current working directory to C<DIR>, or / if not given.
+
+=item cdup ()
+
+Change directory to the parent of the current directory.
+
+=item pwd ()
+
+Returns the full pathname of the current directory.
+
+=item rmdir ( DIR )
+
+Remove the directory with the name C<DIR>.
+
+=item mkdir ( DIR [, RECURSE ])
+
+Create a new directory with the name C<DIR>. If C<RECURSE> is I<true> then
+C<mkdir> will attempt to create all the directories in the given path.
+
+Returns the full pathname to the new directory.
+
+=item ls ( [ DIR ] )
+
+Get a directory listing of C<DIR>, or the current directory.
+
+Returns a reference to a list of lines returned from the server.
+
+=item dir ( [ DIR ] )
+
+Get a directory listing of C<DIR>, or the current directory in long format.
+
+Returns a reference to a list of lines returned from the server.
+
+=item get ( REMOTE_FILE [, LOCAL_FILE ] )
+
+Get C<REMOTE_FILE> from the server and store locally. C<LOCAL_FILE> may be
+a filename or a filehandle. If not specified the the file will be stored in
+the current directory with the same leafname as the remote file.
+
+Returns C<LOCAL_FILE>, or the generated local file name if C<LOCAL_FILE>
+is not given.
+
+=item put ( LOCAL_FILE [, REMOTE_FILE ] )
+
+Put a file on the remote server. C<LOCAL_FILE> may be a name or a filehandle.
+If C<LOCAL_FILE> is a filehandle then C<REMOTE_FILE> must be specified. If
+C<REMOTE_FILE> is not specified then the file will be stored in the current
+directory with the same leafname as C<LOCAL_FILE>.
+
+Returns C<REMOTE_FILE>, or the generated remote filename if C<REMOTE_FILE>
+is not given.
+
+=item put_unique ( LOCAL_FILE [, REMOTE_FILE ] )
+
+Same as put but uses the C<STOU> command.
+
+Returns the name of the file on the server.
+
+=item append ( LOCAL_FILE [, REMOTE_FILE ] )
+
+Same as put but appends to the file on the remote server.
+
+Returns C<REMOTE_FILE>, or the generated remote filename if C<REMOTE_FILE>
+is not given.
+
+=item unique_name ()
+
+Returns the name of the last file stored on the server using the
+C<STOU> command.
+
+=item mdtm ( FILE )
+
+Returns the I<modification time> of the given file
+
+=item size ( FILE )
+
+Returns the size in bytes for the given file.
+
+=back
+
+The following methods can return different results depending on
+how they are called. If the user explicitly calls either
+of the C<pasv> or C<port> methods then these methods will
+return a I<true> or I<false> value. If the user does not
+call either of these methods then the result will be a
+reference to a C<Net::FTP::dataconn> based object.
+
+=over 4
+
+=item nlst ( [ DIR ] )
+
+Send a C<NLST> command to the server, with an optional parameter.
+
+=item list ( [ DIR ] )
+
+Same as C<nlst> but using the C<LIST> command
+
+=item retr ( FILE )
+
+Begin the retrieval of a file called C<FILE> from the remote server.
+
+=item stor ( FILE )
+
+Tell the server that you wish to store a file. C<FILE> is the
+name of the new file that should be created.
+
+=item stou ( FILE )
+
+Same as C<stor> but using the C<STOU> command. The name of the unique
+file which was created on the server will be avalaliable via the C<unique_name>
+method after the data connection has been closed.
+
+=item appe ( FILE )
+
+Tell the server that we want to append some data to the end of a file
+called C<FILE>. If this file does not exist then create it.
+
+=back
+
+If for some reason you want to have complete control over the data connection,
+this includes generating it and calling the C<response> method when required,
+then the user can use these methods to do so.
+
+However calling these methods only affects the use of the methods above that
+can return a data connection. They have no effect on methods C<get>, C<put>,
+C<put_unique> and those that do not require data connections.
+
+=over 4
+
+=item port ( [ PORT ] )
+
+Send a C<PORT> command to the server. If C<PORT> is specified then it is sent
+to the server. If not the a listen socket is created and the correct information
+sent to the server.
+
+=item pasv ()
+
+Tell the server to go into passive mode. Returns the text that represents the
+port on which the server is listening, this text is in a suitable form to
+sent to another ftp server using the C<port> method.
+
+=back
-=head2 * new($host [, option => value [,...]] )
+The following methods can be used to transfer files between two remote
+servers, providing that these two servers can connect directly to each other.
-Constructor for the FTP client. It will create the connection to the
-remote host. Possible options are:
+=over 4
- Port => port to use for FTP connection
- Timeout => set timeout value (defaults to 120)
- Debug => debug level
+=item pasv_xfer ( SRC_FILE, DEST_SERVER [, DEST_FILE ] )
+
+This method will do a file transfer between two remote ftp servers. If
+C<DEST_FILE> is omitted then the leaf name of C<SRC_FILE> will be used.
+
+=item pasv_wait ( NON_PASV_SERVER )
+
+This method can be used to wait for a transfer to complete between a passive
+server and a non-passive server. The method should be called on the passive
+server with the C<Net::FTP> object for the non-passive server passed as an
+argument.
+
+=item abort ()
+
+Abort the current data transfer.
+
+=item quit ()
+
+Send the QUIT command to the remote FTP server and close the socket connection.
+
+=back
+
+=head2 Methods for the adventurous
+
+C<Net::FTP> inherits from C<Net::Cmd> so methods defined in C<Net::Cmd> may
+be used to send commands to the remote FTP server.
+
+=over 4
+
+=item quot (CMD [,ARGS])
+
+Send a command, that Net::FTP does not directly support, to the remote
+server and wait for a response.
+
+Returns most significant digit of the response code.
+
+B<WARNING> This call should only be used on commands that do not require
+data connections. Misuse of this method can hang the connection.
+
+=back
+
+=head1 THE dataconn CLASS
+
+Some of the methods defined in C<Net::FTP> return an object which will
+be derived from this class.The dataconn class itself is derived from
+the C<IO::Socket::INET> class, so any normal IO operations can be performed.
+However the following methods are defined in the dataconn class and IO should
+be performed using these.
+
+=over 4
+
+=item read ( BUFFER, SIZE [, TIMEOUT ] )
+
+Read C<SIZE> bytes of data from the server and place it into C<BUFFER>, also
+performing any <CRLF> translation necessary. C<TIMEOUT> is optional, if not
+given the the timeout value from the command connection will be used.
+
+Returns the number of bytes read before any <CRLF> translation.
+
+=item write ( BUFFER, SIZE [, TIMEOUT ] )
+
+Write C<SIZE> bytes of data from C<BUFFER> to the server, also
+performing any <CRLF> translation necessary. C<TIMEOUT> is optional, if not
+given the the timeout value from the command connection will be used.
+
+Returns the number of bytes written before any <CRLF> translation.
+
+=item abort ()
+
+Abort the current data transfer.
+
+=item close ()
+
+Close the data connection and get a response from the FTP server. Returns
+I<true> if the connection was closed sucessfully and the first digit of
+the response from the server was a '2'.
+
+=back
+
+=head1 AUTHOR
+
+Graham Barr <Graham.Barr@tiuk.ti.com>
+
+=head1 REVISION
+
+$Revision: 2.8 $
+$Date: 1996/09/05 06:53:58 $
+
+The VERSION is derived from the revision by changing each number after the
+first dot into a 2 digit number so
+
+ Revision 1.8 => VERSION 1.08
+ Revision 1.2.3 => VERSION 1.0203
+
+=head1 SEE ALSO
+
+L<Net::Netrc>
+L<Net::Cmd>
+
+=head1 CREDITS
+
+Henry Gabryjelski <henryg@WPI.EDU> - for the suggestion of creating directories
+recursively.
+
+=head1 COPYRIGHT
+
+Copyright (c) 1995 Graham Barr. All rights reserved. This program is free
+software; you can redistribute it and/or modify it under the same terms
+as Perl itself.
=cut
-sub FTP_READY { 0 } # Ready
-sub FTP_RESPONSE { 1 } # Waiting for a response
-sub FTP_XFER { 2 } # Doing data xfer
+require 5.001;
-sub new {
+use strict;
+use vars qw(@ISA $VERSION);
+use Carp;
+
+use Socket 1.3;
+use IO::Socket;
+use Time::Local;
+use Net::Cmd;
+use Net::Telnet qw(TELNET_IAC TELNET_IP TELNET_DM);
+
+$VERSION = do{my @r=(q$Revision: 2.8 $=~/(\d+)/g);sprintf "%d."."%02d"x$#r,@r};
+@ISA = qw(Exporter Net::Cmd IO::Socket::INET);
+
+sub new
+{
my $pkg = shift;
- my $host = shift;
+ my $peer = shift;
my %arg = @_;
- my $me = bless Net::Socket->new(Peer => $host,
- Service => 'ftp',
- Port => $arg{Port} || 'ftp'
- ), $pkg;
-
- ${*$me} = ""; # partial response text
- @{*$me} = (); # Last response text
-
- %{*$me} = (%{*$me}, # Copy current values
- Code => 0, # Last response code
- Type => 'A', # Ascii/Binary/etc mode
- Timeout => $arg{Timeout} || 120, # Timeout value
- Debug => $arg{Debug} || 0, # Output debug information
- FtpHost => $host, # Remote hostname
- State => FTP_RESPONSE, # Current state
-
- ##############################################################
- # Other elements used during the lifetime of the object are
- #
- # LISTEN Listen socket
- # DATA Data socket
- );
-
- $me->autoflush(1);
-
- $me->debug($arg{Debug})
- if(exists $arg{Debug});
-
- unless(2 == $me->response())
+
+ my $host = $peer;
+ my $fire = undef;
+
+ unless(defined inet_aton($peer))
{
- $me->close();
- undef $me;
+ $fire = $ENV{FTP_FIREWALL} || $arg{Firewall} || undef;
+ if(defined $fire)
+ {
+ $peer = $fire;
+ delete $arg{Port};
+ }
}
- $me;
-}
+ my $ftp = $pkg->SUPER::new(PeerAddr => $peer,
+ PeerPort => $arg{Port} || 'ftp(21)',
+ Proto => 'tcp',
+ Timeout => defined $arg{Timeout}
+ ? $arg{Timeout}
+ : 120
+ ) or return undef;
-##
-## User interface methods
-##
+ ${*$ftp}{'net_ftp_passive'} = $arg{Passive} || 0; # Always use pasv mode
+ ${*$ftp}{'net_ftp_host'} = $host; # Remote hostname
+ ${*$ftp}{'net_ftp_type'} = 'A'; # ASCII/binary/etc mode
-=head2 * debug( $value )
+ ${*$ftp}{'net_ftp_firewall'} = $fire
+ if defined $fire;
-Set the level of debug information for this object. If no argument is given
-then the current state is returned. Otherwise the state is changed to
-C<$value>and the previous state returned.
+ $ftp->autoflush(1);
-=cut
+ $ftp->debug(exists $arg{Debug} ? $arg{Debug} : undef);
-sub debug {
- my $me = shift;
- my $debug = ${*$me}{Debug};
-
- if(@_)
+ unless ($ftp->response() == CMD_OK)
{
- ${*$me}{Debug} = 0 + shift;
-
- printf STDERR "\n$me VERSION %s\n", $Net::FTP::VERSION
- if(${*$me}{Debug});
+ $ftp->SUPER::close();
+ undef $ftp;
}
- $debug;
+ $ftp;
}
-=head2 quit
-
-Send the QUIT command to the remote FTP server and close the socket connection.
-
-=cut
-
-sub quit {
- my $me = shift;
-
- return undef
- unless $me->QUIT;
+##
+## User interface methods
+##
- close($me);
+sub quit
+{
+ my $ftp = shift;
- return 1;
+ $ftp->_QUIT
+ && $ftp->SUPER::close;
}
-=head2 ascii/ebcdic/binary/byte
+sub close
+{
+ my $ftp = shift;
-Put the remote FTP server ant the FTP package into the given mode
-of data transfer.
+ ref($ftp)
+ && defined fileno($ftp)
+ && $ftp->quit;
+}
-=cut
+sub DESTROY { shift->close }
sub ascii { shift->type('A',@_); }
-sub ebcdic { shift->type('E',@_); }
sub binary { shift->type('I',@_); }
-sub byte { shift->type('L',@_); }
+
+sub ebcdic
+{
+ carp "TYPE E is unsupported, shall default to I";
+ shift->type('E',@_);
+}
+
+sub byte
+{
+ carp "TYPE L is unsupported, shall default to I";
+ shift->type('L',@_);
+}
# Allow the user to send a command directly, BE CAREFUL !!
-sub quot {
- my $me = shift;
+sub quot
+{
+ my $ftp = shift;
my $cmd = shift;
- $me->send_cmd( uc $cmd, @_);
-
- $me->response();
+ $ftp->command( uc $cmd, @_);
+ $ftp->response();
}
-=head2 login([$login [, $password [, $account]]])
+sub mdtm
+{
+ my $ftp = shift;
+ my $file = shift;
-Log into the remote FTP server with the given login information. If
-no arguments are given then the users $HOME/.netrc file is searched
-for the remote server's hostname. If no information is found then
-a login of I<anonymous> is used. If no password is given and the login
-is anonymous then the users Email address will be used for a password
+ return undef
+ unless $ftp->_MDTM($file);
-=cut
+ my @gt = reverse ($ftp->message =~ /(\d{4})(\d\d)(\d\d)(\d\d)(\d\d)(\d\d)/);
+ $gt[5] -= 1;
+ timegm(@gt);
+}
-sub login {
- my $me = shift;
- my $user = shift;
- my $pass = shift if(defined $user);
- my $acct = shift if(defined $pass);
- my $ok;
+sub size
+{
+ my $ftp = shift;
+ my $file = shift;
+
+ $ftp->_SIZE($file)
+ ? ($ftp->message =~ /(\d+)/)[0]
+ : undef;
+}
+
+sub login
+{
+ my($ftp,$user,$pass,$acct) = @_;
+ my($ok,$ruser);
- unless(defined $user)
+ unless (defined $user)
{
require Net::Netrc;
- my $rc = Net::Netrc->lookup(${*$me}{FtpHost});
+
+ my $rc = Net::Netrc->lookup(${*$ftp}{'net_ftp_host'});
($user,$pass,$acct) = $rc->lpa()
- if $rc;
+ if ($rc);
}
- $user = "anonymous"
- unless defined $user;
+ $user ||= "anonymous";
+ $ruser = $user;
- $pass = "-" . (getpwuid($>))[0] . "@"
- if !defined $pass && $user eq "anonymous";
+ if(defined ${*$ftp}{'net_ftp_firewall'})
+ {
+ $user .= "@" . ${*$ftp}{'net_ftp_host'};
+ }
- $ok = $me->USER($user);
+ $ok = $ftp->_USER($user);
- $ok = $me->PASS($pass)
- if $ok == 3;
+ # Some dumb firewall's don't prefix the connection messages
+ $ok = $ftp->response()
+ if($ok == CMD_OK && $ftp->code == 220 && $user =~ /\@/);
- $ok = $me->ACCT($acct || "")
- if $ok == 3;
+ if ($ok == CMD_MORE)
+ {
+ unless(defined $pass)
+ {
+ require Net::Netrc;
- $ok == 2;
-}
+ my $rc = Net::Netrc->lookup(${*$ftp}{'net_ftp_host'}, $ruser);
-=head2 authorise($auth, $resp)
+ ($ruser,$pass,$acct) = $rc->lpa()
+ if ($rc);
-This is a protocol used by some firewall ftp proxies. It is used
-to authorise the user to send data out.
+ $pass = "-" . (getpwuid($>))[0] . "@"
+ if (!defined $pass && $ruser =~ /^anonymous/o);
+ }
-=cut
+ $ok = $ftp->_PASS($pass || "");
+ }
-sub authorise {
- my($me,$auth,$resp) = @_;
- my $ok;
+ $ok = $ftp->_ACCT($acct || "")
+ if ($ok == CMD_MORE);
- carp "Net::FTP::authorise <auth> <resp>\n"
- unless defined $auth && defined $resp;
+ $ftp->authorize()
+ if($ok == CMD_OK && defined ${*$ftp}{'net_ftp_firewall'});
- $ok = $me->AUTH($auth);
+ $ok == CMD_OK;
+}
- $ok = $me->RESP($resp)
- if $ok == 3;
+sub authorize
+{
+ @_ >= 1 || @_ <= 3 or croak 'usage: $ftp->authorize( [AUTH [, RESP]])';
- $ok == 2;
-}
+ my($ftp,$auth,$resp) = @_;
-=head2 rename( $oldname, $newname)
+ unless(defined $resp)
+ {
+ require Net::Netrc;
-Rename a file on the remote FTP server from C<$oldname> to C<$newname>
+ $auth ||= (getpwuid($>))[0];
-=cut
+ my $rc = Net::Netrc->lookup(${*$ftp}{'net_ftp_firewall'}, $auth)
+ || Net::Netrc->lookup(${*$ftp}{'net_ftp_firewall'});
-sub rename {
- my($me,$from,$to) = @_;
+ ($auth,$resp) = $rc->lpa()
+ if($rc);
+ }
+
+ my $ok = $ftp->_AUTH($auth || "");
+
+ $ok = $ftp->_RESP($resp || "")
+ if ($ok == CMD_MORE);
+
+ $ok == CMD_OK;
+}
+
+sub rename
+{
+ @_ == 3 or croak 'usage: $ftp->rename(FROM, TO)';
- croak "Net::FTP:rename <from> <to>\n"
- unless defined $from && defined $to;
+ my($ftp,$from,$to) = @_;
- $me->RNFR($from) and $me->RNTO($to);
+ $ftp->_RNFR($from)
+ && $ftp->_RNTO($to);
}
-sub type {
- my $me = shift;
+sub type
+{
+ my $ftp = shift;
my $type = shift;
- my $ok = 0;
+ my $oldval = ${*$ftp}{'net_ftp_type'};
- return ${*$me}{Type}
- unless defined $type;
+ return $oldval
+ unless (defined $type);
return undef
- unless($me->TYPE($type,@_));
+ unless ($ftp->_TYPE($type,@_));
- ${*$me}{Type} = join(" ",$type,@_);
+ ${*$ftp}{'net_ftp_type'} = join(" ",$type,@_);
+
+ $oldval;
}
-sub abort {
- my $me = shift;
+sub abort
+{
+ my $ftp = shift;
+
+ send($ftp,pack("CC",TELNET_IAC,TELNET_IP),0);
+ send($ftp,pack("C", TELNET_IAC),MSG_OOB);
+ send($ftp,pack("C", TELNET_DM),0);
+
+ $ftp->command("ABOR");
+
+ defined ${*$ftp}{'net_ftp_dataconn'}
+ ? ${*$ftp}{'net_ftp_dataconn'}->close()
+ : $ftp->response();
+
+ $ftp->response()
+ if $ftp->status == CMD_REJECT;
- ${*$me}{DATA}->abort()
- if defined ${*$me}{DATA};
+ $ftp->status == CMD_OK;
}
-sub get {
- my $me = shift;
- my $remote = shift;
- my $local = shift;
- my $where = shift || 0;
+sub get
+{
+ my($ftp,$remote,$local,$where) = @_;
+
my($loc,$len,$buf,$resp,$localfd,$data);
local *FD;
$localfd = ref($local) ? fileno($local)
- : 0;
+ : undef;
+
+ ($local = $remote) =~ s#^.*/##
+ unless(defined $local);
+
+ ${*$ftp}{'net_ftp_rest'} = $where
+ if ($where);
- ($local = $remote) =~ s#^.*/## unless(defined $local);
+ delete ${*$ftp}{'net_ftp_port'};
+ delete ${*$ftp}{'net_ftp_pasv'};
- if($localfd)
+ $data = $ftp->retr($remote) or
+ return undef;
+
+ if(defined $localfd)
{
$loc = $local;
}
@@ -301,18 +665,15 @@ sub get {
unless(($where) ? open($loc,">>$local") : open($loc,">$local"))
{
carp "Cannot open Local file $local: $!\n";
+ $data->abort;
return undef;
}
}
-
- if ($where) {
- $data = $me->rest_cmd($where,$remote) or
- return undef;
- }
- else {
- $data = $me->retr($remote) or
- return undef;
- }
+ if ($ftp->binary && !binmode($loc))
+ {
+ carp "Cannot binmode Local file $local: $!\n";
+ return undef;
+ }
$buf = '';
@@ -323,57 +684,116 @@ sub get {
while($len > 0 && syswrite($loc,$buf,$len) == $len);
close($loc)
- unless $localfd;
+ unless defined $localfd;
- $data->close() == 2; # implied $me->response
+ $data->close(); # implied $ftp->response
+
+ return $local;
+}
+
+sub cwd
+{
+ @_ == 2 || @_ == 3 or croak 'usage: $ftp->cwd( [ DIR ] )';
+
+ my($ftp,$dir) = @_;
+
+ $dir ||= "/";
+
+ $dir eq ".."
+ ? $ftp->_CDUP()
+ : $ftp->_CWD($dir);
+}
+
+sub cdup
+{
+ @_ == 1 or croak 'usage: $ftp->cdup()';
+ $_[0]->_CDUP;
}
-sub cwd {
- my $me = shift;
- my $dir = shift || "/";
+sub pwd
+{
+ @_ == 1 || croak 'usage: $ftp->pwd()';
+ my $ftp = shift;
+
+ $ftp->_PWD();
+ $ftp->_extract_path;
+}
+
+sub rmdir
+{
+ @_ == 2 || croak 'usage: $ftp->rmdir( DIR )';
+
+ $_[0]->_RMD($_[1]);
+}
+
+sub mkdir
+{
+ @_ == 2 || @_ == 3 or croak 'usage: $ftp->mkdir( DIR [, RECURSE ] )';
+
+ my($ftp,$dir,$recurse) = @_;
- return $dir eq ".." ? $me->CDUP()
- : $me->CWD($dir);
+ $ftp->_MKD($dir) || $recurse or
+ return undef;
+
+ my $path = undef;
+ unless($ftp->ok)
+ {
+ my @path = split(m#(?=/+)#, $dir);
+
+ $path = "";
+
+ while(@path)
+ {
+ $path .= shift @path;
+
+ $ftp->_MKD($path);
+ $path = $ftp->_extract_path($path);
+
+ # 521 means directory already exists
+ last
+ unless $ftp->ok || $ftp->code == 521;
+ }
+ }
+
+ $ftp->_extract_path($path);
}
-sub pwd {
- my $me = shift;
+sub delete
+{
+ @_ == 2 || croak 'usage: $ftp->delete( FILENAME )';
- $me->PWD() ? ($me->message =~ /\"([^\"]+)/)[0]
- : undef;
+ $_[0]->_DELE($_[1]);
}
-sub put { shift->send("stor",@_) }
-sub put_unique { shift->send("stou",@_) }
-sub append { shift->send("appe",@_) }
+sub put { shift->_store_cmd("stor",@_) }
+sub put_unique { shift->_store_cmd("stou",@_) }
+sub append { shift->_store_cmd("appe",@_) }
-sub nlst { shift->data_cmd("NLST",@_) }
-sub list { shift->data_cmd("LIST",@_) }
-sub retr { shift->data_cmd("RETR",@_) }
-sub stor { shift->data_cmd("STOR",@_) }
-sub stou { shift->data_cmd("STOU",@_) }
-sub appe { shift->data_cmd("APPE",@_) }
+sub nlst { shift->_data_cmd("NLST",@_) }
+sub list { shift->_data_cmd("LIST",@_) }
+sub retr { shift->_data_cmd("RETR",@_) }
+sub stor { shift->_data_cmd("STOR",@_) }
+sub stou { shift->_data_cmd("STOU",@_) }
+sub appe { shift->_data_cmd("APPE",@_) }
-sub send {
- my $me = shift;
- my $cmd = shift;
- my $local = shift;
- my $remote = shift;
+sub _store_cmd
+{
+ my($ftp,$cmd,$local,$remote) = @_;
my($loc,$sock,$len,$buf,$localfd);
local *FD;
$localfd = ref($local) ? fileno($local)
- : 0;
+ : undef;
unless(defined $remote)
{
- croak "Must specify remote filename with stream input\n"
- if $localfd;
+ croak 'Must specify remote filename with stream input'
+ if defined $localfd;
($remote = $local) =~ s%.*/%%;
}
- if($localfd)
+ if(defined $localfd)
{
$loc = $local;
}
@@ -386,134 +806,175 @@ sub send {
carp "Cannot open Local file $local: $!\n";
return undef;
}
+ if ($ftp->binary && !binmode($loc))
+ {
+ carp "Cannot binmode Local file $local: $!\n";
+ return undef;
+ }
}
- $cmd = lc $cmd;
+ delete ${*$ftp}{'net_ftp_port'};
+ delete ${*$ftp}{'net_ftp_pasv'};
- $sock = $me->$cmd($remote) or
+ $sock = $ftp->_data_cmd($cmd, $remote) or
return undef;
do
{
- $len = sysread($loc,$buf,1024);
+ $len = sysread($loc,$buf="",1024);
}
while($len && $sock->write($buf,$len) == $len);
close($loc)
- unless $localfd;
+ unless defined $localfd;
$sock->close();
- ($remote) = $me->message =~ /unique file name:\s*(\S*)\s*\)/
- if $cmd eq 'stou' ;
+ ($remote) = $ftp->message =~ /unique file name:\s*(\S*)\s*\)/
+ if ('STOU' eq uc $cmd);
return $remote;
}
-sub port {
- my $me = shift;
- my $port = shift;
+sub port
+{
+ @_ == 1 || @_ == 2 or croak 'usage: $ftp->port([PORT])';
+
+ my($ftp,$port) = @_;
my $ok;
+ delete ${*$ftp}{'net_ftp_intern_port'};
+
unless(defined $port)
{
- my $listen;
-
- if(defined ${*$me}{LISTEN})
- {
- ${*$me}{LISTEN}->close();
- delete ${*$me}{LISTEN};
- }
-
# create a Listen socket at same address as the command socket
- $listen = Net::Socket->new(Listen => 5,
- Service => 'ftp',
- Addr => $me->sockhost,
- );
+ ${*$ftp}{'net_ftp_listen'} ||= IO::Socket::INET->new(Listen => 5,
+ Proto => 'tcp',
+ LocalAddr => $ftp->sockhost,
+ );
- ${*$me}{LISTEN} = $listen;
+ my $listen = ${*$ftp}{'net_ftp_listen'};
my($myport, @myaddr) = ($listen->sockport, split(/\./,$listen->sockhost));
$port = join(',', @myaddr, $myport >> 8, $myport & 0xff);
+
+ ${*$ftp}{'net_ftp_intern_port'} = 1;
}
- $ok = $me->PORT($port);
+ $ok = $ftp->_PORT($port);
- ${*$me}{Port} = $port;
+ ${*$ftp}{'net_ftp_port'} = $port;
$ok;
}
-sub ls { shift->list_cmd("NLST",@_); }
-sub lsl { shift->list_cmd("LIST",@_); }
+sub ls { shift->_list_cmd("NLST",@_); }
+sub dir { shift->_list_cmd("LIST",@_); }
-sub pasv {
- my $me = shift;
- my $hostport;
+sub pasv
+{
+ @_ == 1 or croak 'usage: $ftp->pasv()';
- return undef
- unless $me->PASV();
+ my $ftp = shift;
+
+ delete ${*$ftp}{'net_ftp_intern_port'};
- ($hostport) = $me->message =~ /(\d+(,\d+)+)/;
+ $ftp->_PASV && $ftp->message =~ /(\d+(,\d+)+)/
+ ? ${*$ftp}{'net_ftp_pasv'} = $1
+ : undef;
+}
- ${*$me}{Pasv} = $hostport;
+sub unique_name
+{
+ my $ftp = shift;
+ ${*$ftp}{'net_ftp_unique'} || undef;
}
##
-## Communication methods
+## Depreciated methods
##
-sub timeout {
- my $me = shift;
- my $timeout = ${*$me}{Timeout};
-
- ${*$me}{Timeout} = 0 + shift if(@_);
-
- $timeout;
+sub lsl
+{
+ carp "Use of Net::FTP::lsl depreciated, use 'dir'"
+ if $^W;
+ goto &dir;
}
-sub accept {
- my $me = shift;
+sub authorise
+{
+ carp "Use of Net::FTP::authorise depreciated, use 'authorize'"
+ if $^W;
+ goto &authorize;
+}
- return undef unless defined ${*$me}{LISTEN};
- my $data = ${*$me}{LISTEN}->accept;
+##
+## Private methods
+##
- ${*$me}{LISTEN}->close();
- delete ${*$me}{LISTEN};
+sub _extract_path
+{
+ my($ftp, $path) = @_;
- ${*$data}{Timeout} = ${*$me}{Timeout};
- ${*$data}{Cmd} = $me;
- ${*$data} = "";
+ $ftp->ok &&
+ $ftp->message =~ /\s\"(.*)\"\s/o &&
+ ($path = $1) =~ s/\"\"/\"/g;
- ${*$me}{State} = FTP_XFER;
- ${*$me}{DATA} = bless $data, "Net::FTP::" . ${*$me}{Type};
+ $path;
}
-sub message {
- my $me = shift;
- join("\n", @{*$me});
-}
+##
+## Communication methods
+##
-sub ok {
- my $me = shift;
- my $code = ${*$me}{Code} || 0;
+sub _dataconn
+{
+ my $ftp = shift;
+ my $data = undef;
+ my $pkg = "Net::FTP::" . $ftp->type;
- 0 < $code && $code < 400;
-}
+ $pkg =~ s/ /_/g;
+
+ delete ${*$ftp}{'net_ftp_dataconn'};
-sub code {
- my $me = shift;
+ if(defined ${*$ftp}{'net_ftp_pasv'})
+ {
+ my @port = split(/,/,${*$ftp}{'net_ftp_pasv'});
- ${*$me}{Code};
+ $data = $pkg->new(PeerAddr => join(".",@port[0..3]),
+ PeerPort => $port[4] * 256 + $port[5],
+ Proto => 'tcp'
+ );
+ }
+ elsif(defined ${*$ftp}{'net_ftp_listen'})
+ {
+ $data = ${*$ftp}{'net_ftp_listen'}->accept($pkg);
+ close(delete ${*$ftp}{'net_ftp_listen'});
+ }
+
+ if($data)
+ {
+ ${*$data} = "";
+ $data->timeout($ftp->timeout);
+ ${*$ftp}{'net_ftp_dataconn'} = $data;
+ ${*$data}{'net_ftp_cmd'} = $ftp;
+ }
+
+ $data;
}
-sub list_cmd {
- my $me = shift;
- my $cmd = lc shift;
- my $data = $me->$cmd(@_);
+sub _list_cmd
+{
+ my $ftp = shift;
+ my $cmd = uc shift;
+
+ delete ${*$ftp}{'net_ftp_port'};
+ delete ${*$ftp}{'net_ftp_pasv'};
+
+ my $data = $ftp->_data_cmd($cmd,@_);
return undef
unless(defined $data);
@@ -523,99 +984,137 @@ sub list_cmd {
my $databuf = '';
my $buf = '';
- while($data->read($databuf,1024)) {
+ while($data->read($databuf,1024))
+ {
$buf .= $databuf;
- }
+ }
my $list = [ split(/\n/,$buf) ];
$data->close();
- wantarray ? @{$list} : $list;
+ wantarray ? @{$list}
+ : $list;
}
-sub data_cmd {
- my $me = shift;
+sub _data_cmd
+{
+ my $ftp = shift;
my $cmd = uc shift;
my $ok = 1;
- my $pasv = defined ${*$me}{Pasv} ? 1 : 0;
+ my $where = delete ${*$ftp}{'net_ftp_rest'} || 0;
- $ok = $me->port
- unless $pasv || defined ${*$me}{Port};
+ if(${*$ftp}{'net_ftp_passive'} &&
+ !defined ${*$ftp}{'net_ftp_pasv'} &&
+ !defined ${*$ftp}{'net_ftp_port'})
+ {
+ my $data = undef;
- $ok = $me->$cmd(@_)
- if $ok;
+ $ok = defined $ftp->pasv;
+ $ok = $ftp->_REST($where)
+ if $ok && $where;
- return $pasv ? $ok
- : $ok ? $me->accept()
- : undef;
-}
+ if($ok)
+ {
+ $ftp->command($cmd,@_);
+ $data = $ftp->_dataconn();
+ $ok = CMD_INFO == $ftp->response();
+ }
+ return $ok ? $data
+ : undef;
+ }
-sub rest_cmd {
- my $me = shift;
- my $ok = 1;
- my $pasv = defined ${*$me}{Pasv} ? 1 : 0;
- my $where = shift;
- my $file = shift;
+ $ok = $ftp->port
+ unless (defined ${*$ftp}{'net_ftp_port'} ||
+ defined ${*$ftp}{'net_ftp_pasv'});
- $ok = $me->port
- unless $pasv || defined ${*$me}{Port};
+ $ok = $ftp->_REST($where)
+ if $ok && $where;
- $ok = $me->REST($where)
- if $ok;
+ return undef
+ unless $ok;
+
+ $ftp->command($cmd,@_);
+
+ return 1
+ if(defined ${*$ftp}{'net_ftp_pasv'});
- $ok = $me->RETR($file)
- if $ok;
+ $ok = CMD_INFO == $ftp->response();
- return $pasv ? $ok
- : $ok ? $me->accept()
- : undef;
+ return $ok
+ unless exists ${*$ftp}{'net_ftp_intern_port'};
+
+ $ok ? $ftp->_dataconn()
+ : undef;
}
-sub cmd {
- my $me = shift;
+##
+## Over-ride methods (Net::Cmd)
+##
- $me->send_cmd(@_);
- $me->response();
+sub debug_text { $_[2] =~ /^(pass|resp)/i ? "$1 ....\n" : $_[2]; }
+
+sub command
+{
+ my $ftp = shift;
+
+ delete ${*$ftp}{'net_ftp_port'};
+ $ftp->SUPER::command(@_);
}
-sub send_cmd {
- my $me = shift;
+sub response
+{
+ my $ftp = shift;
+ my $code = $ftp->SUPER::response();
+
+ delete ${*$ftp}{'net_ftp_pasv'}
+ if ($code != CMD_MORE && $code != CMD_INFO);
+
+ $code;
+}
- if(scalar(@_)) {
- my $cmd = join(" ", @_) . "\r\n";
+##
+## Allow 2 servers to talk directly
+##
- delete ${*$me}{Pasv};
- delete ${*$me}{Port};
+sub pasv_xfer
+{
+ my($sftp,$sfile,$dftp,$dfile) = @_;
- syswrite($me,$cmd,length $cmd);
+ ($dfile = $sfile) =~ s#.*/##
+ unless(defined $dfile);
- ${*$me}{State} = FTP_RESPONSE;
+ my $port = $sftp->pasv or
+ return undef;
- printf STDERR "\n$me>> %s", $cmd=~/^(pass|resp)/i ? "$1 ....\n" : $cmd
- if $me->debug;
- }
+ unless($dftp->port($port) && $sftp->retr($sfile) && $dftp->stou($dfile))
+ {
+ $sftp->abort;
+ $dftp->abort;
+ return undef;
+ }
- $me;
+ $dftp->pasv_wait($sftp);
}
-sub pasv_wait {
- my $me = shift;
- my $non_pasv = shift;
- my $file;
+sub pasv_wait
+{
+ @_ == 2 or croak 'usage: $ftp->pasv_wait(NON_PASV_FTP)';
+
+ my($ftp, $non_pasv) = @_;
+ my($file,$rin,$rout);
- my($rin,$rout);
- vec($rin,fileno($me),1) = 1;
+ vec($rin,fileno($ftp),1) = 1;
select($rout=$rin, undef, undef, undef);
- $me->response();
+ $ftp->response();
$non_pasv->response();
return undef
- unless $me->ok() && $non_pasv->ok();
+ unless $ftp->ok() && $non_pasv->ok();
return $1
- if $me->message =~ /unique file name:\s*(\S*)\s*\)/;
+ if $ftp->message =~ /unique file name:\s*(\S*)\s*\)/;
return $1
if $non_pasv->message =~ /unique file name:\s*(\S*)\s*\)/;
@@ -623,152 +1122,102 @@ sub pasv_wait {
return 1;
}
-sub response {
- my $me = shift;
- my $timeout = ${*$me}{Timeout};
- my($code,$more,$rin,$rout,$partial,$buf) = (undef,0,'','','','');
+sub cmd { shift->command(@_)->responce() }
+
+########################################
+#
+# RFC959 commands
+#
+
+sub _ABOR { shift->command("ABOR")->response() == CMD_OK }
+sub _CDUP { shift->command("CDUP")->response() == CMD_OK }
+sub _NOOP { shift->command("NOOP")->response() == CMD_OK }
+sub _PASV { shift->command("PASV")->response() == CMD_OK }
+sub _QUIT { shift->command("QUIT")->response() == CMD_OK }
+sub _DELE { shift->command("DELE",@_)->response() == CMD_OK }
+sub _CWD { shift->command("CWD", @_)->response() == CMD_OK }
+sub _PORT { shift->command("PORT",@_)->response() == CMD_OK }
+sub _RMD { shift->command("RMD", @_)->response() == CMD_OK }
+sub _MKD { shift->command("MKD", @_)->response() == CMD_OK }
+sub _PWD { shift->command("PWD", @_)->response() == CMD_OK }
+sub _TYPE { shift->command("TYPE",@_)->response() == CMD_OK }
+sub _RNTO { shift->command("RNTO",@_)->response() == CMD_OK }
+sub _ACCT { shift->command("ACCT",@_)->response() == CMD_OK }
+sub _RESP { shift->command("RESP",@_)->response() == CMD_OK }
+sub _MDTM { shift->command("MDTM",@_)->response() == CMD_OK }
+sub _SIZE { shift->command("SIZE",@_)->response() == CMD_OK }
+sub _APPE { shift->command("APPE",@_)->response() == CMD_INFO }
+sub _LIST { shift->command("LIST",@_)->response() == CMD_INFO }
+sub _NLST { shift->command("NLST",@_)->response() == CMD_INFO }
+sub _RETR { shift->command("RETR",@_)->response() == CMD_INFO }
+sub _STOR { shift->command("STOR",@_)->response() == CMD_INFO }
+sub _STOU { shift->command("STOU",@_)->response() == CMD_INFO }
+sub _RNFR { shift->command("RNFR",@_)->response() == CMD_MORE }
+sub _REST { shift->command("REST",@_)->response() == CMD_MORE }
+sub _USER { shift->command("user",@_)->response() } # A certain brain dead firewall :-)
+sub _PASS { shift->command("PASS",@_)->response() }
+sub _AUTH { shift->command("AUTH",@_)->response() }
+
+sub _ALLO { shift->unsupported(@_) }
+sub _SMNT { shift->unsupported(@_) }
+sub _HELP { shift->unsupported(@_) }
+sub _MODE { shift->unsupported(@_) }
+sub _SITE { shift->unsupported(@_) }
+sub _SYST { shift->unsupported(@_) }
+sub _STAT { shift->unsupported(@_) }
+sub _STRU { shift->unsupported(@_) }
+sub _REIN { shift->unsupported(@_) }
- @{*$me} = (); # the responce
- $buf = ${*$me};
- my @buf = ();
-
- vec($rin,fileno($me),1) = 1;
-
- do
- {
- if(length($buf) || ($timeout==0) || select($rout=$rin, undef, undef, $timeout))
- {
- unless(length($buf) || sysread($me, $buf, 1024))
- {
- carp "Unexpected EOF on command channel";
- return undef;
- }
-
- substr($buf,0,0) = $partial; ## prepend from last sysread
-
- @buf = split(/\r?\n/, $buf); ## break into lines
-
- $partial = (substr($buf, -1, 1) eq "\n") ? ''
- : pop(@buf);
-
- $buf = "";
-
- while (@buf)
- {
- my $cmd = shift @buf;
- print STDERR "$me<< $cmd\n"
- if $me->debug;
-
- ($code,$more) = ($1,$2)
- if $cmd =~ /^(\d\d\d)(.)/;
-
- push(@{*$me},$');
-
- last unless(defined $more && $more eq "-");
- }
- }
- else
- {
- carp "$me: Timeout" if($me->debug);
- return undef;
- }
- }
- while((scalar(@{*$me}) == 0) || (defined $more && $more eq "-"));
-
- ${*$me} = @buf ? join("\n",@buf,"") : "";
- ${*$me} .= $partial;
-
- ${*$me}{Code} = $code;
- ${*$me}{State} = FTP_READY;
-
- substr($code,0,1);
-}
-
-;########################################
-;#
-;# RFC959 commands
-;#
-
-sub no_imp { croak "Not implemented\n"; }
-
-sub ABOR { shift->send_cmd("ABOR")->response() == 2}
-sub CDUP { shift->send_cmd("CDUP")->response() == 2}
-sub NOOP { shift->send_cmd("NOOP")->response() == 2}
-sub PASV { shift->send_cmd("PASV")->response() == 2}
-sub QUIT { shift->send_cmd("QUIT")->response() == 2}
-sub DELE { shift->send_cmd("DELE",@_)->response() == 2}
-sub CWD { shift->send_cmd("CWD", @_)->response() == 2}
-sub PORT { shift->send_cmd("PORT",@_)->response() == 2}
-sub RMD { shift->send_cmd("RMD", @_)->response() == 2}
-sub MKD { shift->send_cmd("MKD", @_)->response() == 2}
-sub PWD { shift->send_cmd("PWD", @_)->response() == 2}
-sub TYPE { shift->send_cmd("TYPE",@_)->response() == 2}
-sub APPE { shift->send_cmd("APPE",@_)->response() == 1}
-sub LIST { shift->send_cmd("LIST",@_)->response() == 1}
-sub NLST { shift->send_cmd("NLST",@_)->response() == 1}
-sub RETR { shift->send_cmd("RETR",@_)->response() == 1}
-sub STOR { shift->send_cmd("STOR",@_)->response() == 1}
-sub STOU { shift->send_cmd("STOU",@_)->response() == 1}
-sub RNFR { shift->send_cmd("RNFR",@_)->response() == 3}
-sub RNTO { shift->send_cmd("RNTO",@_)->response() == 2}
-sub ACCT { shift->send_cmd("ACCT",@_)->response() == 2}
-sub RESP { shift->send_cmd("RESP",@_)->response() == 2}
-sub REST { shift->send_cmd("REST",@_)->response() == 3}
-sub USER { my $ok = shift->send_cmd("USER",@_)->response();($ok == 2 || $ok == 3) ? $ok : 0;}
-sub PASS { my $ok = shift->send_cmd("PASS",@_)->response();($ok == 2 || $ok == 3) ? $ok : 0;}
-sub AUTH { my $ok = shift->send_cmd("AUTH",@_)->response();($ok == 2 || $ok == 3) ? $ok : 0;}
-
-sub ALLO { no_imp; }
-sub SMNT { no_imp; }
-sub HELP { no_imp; }
-sub MODE { no_imp; }
-sub SITE { no_imp; }
-sub SYST { no_imp; }
-sub STAT { no_imp; }
-sub STRU { no_imp; }
-sub REIN { no_imp; }
+##
+## Generic data connection package
+##
package Net::FTP::dataconn;
+
use Carp;
-no strict 'vars';
+use vars qw(@ISA $timeout);
+use Net::Cmd;
-sub abort {
- my $fd = shift;
- my $ftp = ${*$fd}{Cmd};
+@ISA = qw(IO::Socket::INET);
- $ftp->send_cmd("ABOR");
- $fd->close();
-}
+sub abort
+{
+ my $data = shift;
+ my $ftp = ${*$data}{'net_ftp_cmd'};
-sub close {
- my $fd = shift;
- my $ftp = ${*$fd}{Cmd};
+ $ftp->abort; # this will close me
+}
- $fd->Net::Socket::close();
- delete ${*$ftp}{DATA};
+sub close
+{
+ my $data = shift;
+ my $ftp = ${*$data}{'net_ftp_cmd'};
- $ftp->response();
-}
+ $data->SUPER::close();
-sub timeout {
- my $me = shift;
- my $timeout = ${*$me}{Timeout};
+ delete ${*$ftp}{'net_ftp_dataconn'}
+ if exists ${*$ftp}{'net_ftp_dataconn'} &&
+ $data == ${*$ftp}{'net_ftp_dataconn'};
- ${*$me}{Timeout} = 0 + shift if(@_);
+ $ftp->response() == CMD_OK &&
+ $ftp->message =~ /unique file name:\s*(\S*)\s*\)/ &&
+ (${*$ftp}{'net_ftp_unique'} = $1);
- $timeout;
+ $ftp->status == CMD_OK;
}
-sub _select {
- my $fd = shift;
+sub _select
+{
+ my $data = shift;
local *timeout = \$_[0]; shift;
- my $rw = shift;
+ my $rw = shift;
+
my($rin,$win);
return 1 unless $timeout;
$rin = '';
- vec($rin,fileno($fd),1) = 1;
+ vec($rin,fileno($data),1) = 1;
$win = $rw ? undef : $rin;
$rin = undef unless $rw;
@@ -781,43 +1230,51 @@ sub _select {
return $nfound;
}
-sub can_read {
- my $fd = shift;
+sub can_read
+{
+ my $data = shift;
local *timeout = \$_[0];
- $fd->_select($timeout,1);
+ $data->_select($timeout,1);
}
-sub can_write {
- my $fd = shift;
+sub can_write
+{
+ my $data = shift;
local *timeout = \$_[0];
- $fd->_select($timeout,0);
+ $data->_select($timeout,0);
}
-sub cmd {
- my $me = shift;
+sub cmd
+{
+ my $ftp = shift;
- ${*$me}{Cmd};
+ ${*$ftp}{'net_ftp_cmd'};
}
@Net::FTP::L::ISA = qw(Net::FTP::I);
@Net::FTP::E::ISA = qw(Net::FTP::I);
+##
+## Package to read/write on ASCII data connections
+##
+
package Net::FTP::A;
-@Net::FTP::A::ISA = qw(Net::FTP::dataconn);
+
+use vars qw(@ISA $buf);
use Carp;
-no strict 'vars';
+@ISA = qw(Net::FTP::dataconn);
-sub read {
- my $fd = shift;
- local *buf = \$_[0]; shift;
- my $size = shift || croak 'read($buf,$size,[$offset])';
- my $offset = shift || 0;
- my $timeout = ${*$fd}{Timeout};
- my $l;
+sub read
+{
+ my $data = shift;
+ local *buf = \$_[0]; shift;
+ my $size = shift || croak 'read($buf,$size,[$offset])';
+ my $offset = shift || 0;
+ my $timeout = $data->timeout;
croak "Bad offset"
if($offset < 0);
@@ -825,61 +1282,61 @@ sub read {
$offset = length $buf
if($offset > length $buf);
- $l = 0;
+ ${*$data} ||= "";
+ my $l = 0;
+
READ:
{
- $fd->can_read($timeout) or
+ $data->can_read($timeout) or
croak "Timeout";
- my $n = sysread($fd, ${*$fd}, $size, length ${*$fd});
+ my $n = sysread($data, ${*$data}, $size, length ${*$data});
return $n
unless($n >= 0);
-# my $lf = substr(${*$fd},-1,1) eq "\r" ? chop(${*$fd})
-# : "";
-
- my $lf = (length ${*$fd} > 0 && substr(${*$fd},-1,1) eq "\r") ? chop(${*$fd})
- : "";
+ ${*$data} =~ s/(\015)?(?!\012)\Z//so;
+ my $lf = $1 || "";
- ${*$fd} =~ s/\r\n/\n/go;
+ ${*$data} =~ s/\015\012/\n/sgo;
- substr($buf,$offset) = ${*$fd};
+ substr($buf,$offset) = ${*$data};
- $l += length(${*$fd});
- $offset += length(${*$fd});
+ $l += length(${*$data});
+ $offset += length(${*$data});
- ${*$fd} = $lf;
+ ${*$data} = $lf;
redo READ
if($l == 0 && $n > 0);
if($n == 0 && $l == 0)
{
- substr($buf,$offset) = ${*$fd};
- ${*$fd} = "";
+ substr($buf,$offset) = ${*$data};
+ ${*$data} = "";
}
}
return $l;
}
-sub write {
- my $fd = shift;
- local *buf = \$_[0]; shift;
- my $size = shift || croak 'write($buf,$size,[$timeout])';
- my $timeout = @_ ? shift : ${*$fd}{Timeout};
+sub write
+{
+ my $data = shift;
+ local *buf = \$_[0]; shift;
+ my $size = shift || croak 'write($buf,$size,[$timeout])';
+ my $timeout = @_ ? shift : $data->timeout;
- $fd->can_write($timeout) or
+ $data->can_write($timeout) or
croak "Timeout";
- # What is previous pkt ended in \r or not ??
+ # What is previous pkt ended in \015 or not ??
my $tmp;
- ($tmp = $buf) =~ s/(?!\r)\n/\r\n/g;
+ ($tmp = $buf) =~ s/(?!\015)\012/\015\012/sg;
my $len = $size + length($tmp) - length($buf);
- my $wrote = syswrite($fd, $tmp, $len);
+ my $wrote = syswrite($data, $tmp, $len);
if($wrote >= 0)
{
@@ -890,54 +1347,45 @@ sub write {
return $wrote;
}
+##
+## Package to read/write on BINARY data connections
+##
+
package Net::FTP::I;
-@Net::FTP::I::ISA = qw(Net::FTP::dataconn);
+
+use vars qw(@ISA $buf);
use Carp;
-no strict 'vars';
+@ISA = qw(Net::FTP::dataconn);
-sub read {
- my $fd = shift;
- local *buf = \$_[0]; shift;
- my $size = shift || croak 'read($buf,$size,[$timeout])';
- my $timeout = @_ ? shift : ${*$fd}{Timeout};
+sub read
+{
+ my $data = shift;
+ local *buf = \$_[0]; shift;
+ my $size = shift || croak 'read($buf,$size,[$timeout])';
+ my $timeout = @_ ? shift : $data->timeout;
- $fd->can_read($timeout) or
+ $data->can_read($timeout) or
croak "Timeout";
- my $n = sysread($fd, $buf, $size);
+ my $n = sysread($data, $buf, $size);
$n;
}
-sub write {
- my $fd = shift;
- local *buf = \$_[0]; shift;
- my $size = shift || croak 'write($buf,$size,[$timeout])';
- my $timeout = @_ ? shift : ${*$fd}{Timeout};
+sub write
+{
+ my $data = shift;
+ local *buf = \$_[0]; shift;
+ my $size = shift || croak 'write($buf,$size,[$timeout])';
+ my $timeout = @_ ? shift : $data->timeout;
- $fd->can_write($timeout) or
+ $data->can_write($timeout) or
croak "Timeout";
- syswrite($fd, $buf, $size);
+ syswrite($data, $buf, $size);
}
-=head2 AUTHOR
-
-Graham Barr <Graham.Barr@tiuk.ti.com>
-
-=head2 REVISION
-
-$Revision: 1.17 $
-
-=head2 COPYRIGHT
-
-Copyright (c) 1995 Graham Barr. All rights reserved. This program is free
-software; you can redistribute it and/or modify it under the same terms
-as Perl itself.
-
-=cut
-
1;