1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        common/ftp.cpp 
   3 // Purpose:     FTP protocol 
   4 // Author:      Guilhem Lavaux 
   5 // Modified by: Mark Johnson, wxWindows@mj10777.de 
   6 //              20000917 : RmDir, GetLastResult, GetList 
   7 //              Vadim Zeitlin (numerous fixes and rewrites to all part of the 
   8 //              code, support ASCII/Binary modes, better error reporting, more 
   9 //              robust Abort(), support for arbitrary FTP commands, ...) 
  10 //              Randall Fox (support for active mode) 
  11 // Created:     07/07/1997 
  13 // Copyright:   (c) 1997, 1998 Guilhem Lavaux 
  14 //              (c) 1998-2004 wxWidgets team 
  15 // Licence:     wxWindows licence 
  16 ///////////////////////////////////////////////////////////////////////////// 
  18 // ============================================================================ 
  20 // ============================================================================ 
  22 // ---------------------------------------------------------------------------- 
  24 // ---------------------------------------------------------------------------- 
  26 // For compilers that support precompilation, includes "wx.h". 
  27 #include "wx/wxprec.h" 
  33 #if wxUSE_PROTOCOL_FTP 
  37     #include "wx/string.h" 
  43 #include "wx/sckaddr.h" 
  44 #include "wx/socket.h" 
  46 #include "wx/sckstrm.h" 
  47 #include "wx/protocol/protocol.h" 
  48 #include "wx/protocol/ftp.h" 
  50 #if defined(__WXMAC__) 
  51     #include "wx/mac/macsock.h" 
  58 // ---------------------------------------------------------------------------- 
  60 // ---------------------------------------------------------------------------- 
  62 // the length of FTP status code (3 digits) 
  63 static const size_t LEN_CODE 
= 3; 
  65 // ---------------------------------------------------------------------------- 
  67 // ---------------------------------------------------------------------------- 
  69 IMPLEMENT_DYNAMIC_CLASS(wxFTP
, wxProtocol
) 
  70 IMPLEMENT_PROTOCOL(wxFTP
, wxT("ftp"), wxT("ftp"), true) 
  72 // ============================================================================ 
  74 // ============================================================================ 
  76 // ---------------------------------------------------------------------------- 
  77 // wxFTP constructor and destructor 
  78 // ---------------------------------------------------------------------------- 
  82     m_lastError 
= wxPROTO_NOERR
; 
  84     m_currentTransfermode 
= NONE
; 
  86     m_user 
= wxT("anonymous"); 
  87     m_passwd 
<< wxGetUserId() << wxT('@') << wxGetFullHostName(); 
  90     SetFlags(wxSOCKET_NONE
); 
  92     SetDefaultTimeout(60); // Default is Sixty Seconds 
  93     m_bEncounteredError 
= false; 
 100         // if we are streaming, this will issue 
 101         // an FTP ABORT command, to tell the server we are aborting 
 105     // now this issues a "QUIT" command to tell the server we are 
 109 // ---------------------------------------------------------------------------- 
 110 // wxFTP connect and login methods 
 111 // ---------------------------------------------------------------------------- 
 113 bool wxFTP::Connect(wxSockAddress
& addr
, bool WXUNUSED(wait
)) 
 115     if ( !wxProtocol::Connect(addr
) ) 
 117         m_lastError 
= wxPROTO_NETERR
; 
 123         m_lastError 
= wxPROTO_CONNERR
; 
 127     // we should have 220 welcome message 
 128     if ( !CheckResult('2') ) 
 135     command
.Printf(wxT("USER %s"), m_user
.c_str()); 
 136     char rc 
= SendCommand(command
); 
 139         // 230 return: user accepted without password 
 149     command
.Printf(wxT("PASS %s"), m_passwd
.c_str()); 
 150     if ( !CheckCommand(command
, '2') ) 
 159 bool wxFTP::Connect(const wxString
& host
) 
 163     addr
.Service(wxT("ftp")); 
 165     return Connect(addr
); 
 172         m_lastError 
= wxPROTO_STREAMING
; 
 178         if ( !CheckCommand(wxT("QUIT"), '2') ) 
 180             wxLogDebug(_T("Failed to close connection gracefully.")); 
 184     return wxSocketClient::Close(); 
 187 // ============================================================================ 
 189 // ============================================================================ 
 191 // ---------------------------------------------------------------------------- 
 192 // Send command to FTP server 
 193 // ---------------------------------------------------------------------------- 
 195 char wxFTP::SendCommand(const wxString
& command
) 
 199         m_lastError 
= wxPROTO_STREAMING
; 
 203     wxString tmp_str 
= command 
+ wxT("\r\n"); 
 204     const wxWX2MBbuf tmp_buf 
= tmp_str
.mb_str(); 
 205     if ( Write(wxMBSTRINGCAST tmp_buf
, strlen(tmp_buf
)).Error()) 
 207         m_lastError 
= wxPROTO_NETERR
; 
 212     // don't show the passwords in the logs (even in debug ones) 
 213     wxString cmd
, password
; 
 214     if ( command
.Upper().StartsWith(_T("PASS "), &password
) ) 
 216         cmd 
<< _T("PASS ") << wxString(_T('*'), password
.length()); 
 223     wxLogTrace(FTP_TRACE_MASK
, _T("==> %s"), cmd
.c_str()); 
 224 #endif // __WXDEBUG__ 
 229 // ---------------------------------------------------------------------------- 
 230 // Recieve servers reply 
 231 // ---------------------------------------------------------------------------- 
 233 char wxFTP::GetResult() 
 235     // if we've already had a read or write timeout error, the connection is 
 236     // probably toast, so don't bother, it just wastes the users time 
 237     if ( m_bEncounteredError 
) 
 242     // m_lastResult will contain the entire server response, possibly on 
 244     m_lastResult
.clear(); 
 246     // we handle multiline replies here according to RFC 959: it says that a 
 247     // reply may either be on 1 line of the form "xyz ..." or on several lines 
 248     // in whuch case it looks like 
 252     // and the intermeidate lines may start with xyz or not 
 253     bool badReply 
= false; 
 254     bool firstLine 
= true; 
 255     bool endOfReply 
= false; 
 256     while ( !endOfReply 
&& !badReply 
) 
 259         m_lastError 
= ReadLine(this,line
); 
 262             m_bEncounteredError 
= true; 
 266         if ( !m_lastResult
.empty() ) 
 268             // separate from last line 
 269             m_lastResult 
+= _T('\n'); 
 272         m_lastResult 
+= line
; 
 274         // unless this is an intermediate line of a multiline reply, it must 
 275         // contain the code in the beginning and '-' or ' ' following it 
 276         if ( line
.Len() < LEN_CODE 
+ 1 ) 
 284                 wxLogTrace(FTP_TRACE_MASK
, _T("<== %s %s"), 
 285                            code
.c_str(), line
.c_str()); 
 288         else // line has at least 4 chars 
 290             // this is the char which tells us what we're dealing with 
 291             wxChar chMarker 
= line
.GetChar(LEN_CODE
); 
 295                 code 
= wxString(line
, LEN_CODE
); 
 296                 wxLogTrace(FTP_TRACE_MASK
, _T("<== %s %s"), 
 297                            code
.c_str(), line
.c_str() + LEN_CODE 
+ 1); 
 314             else // subsequent line of multiline reply 
 316                 if ( wxStrncmp(line
, code
, LEN_CODE
) == 0 ) 
 318                     if ( chMarker 
== _T(' ') ) 
 323                     wxLogTrace(FTP_TRACE_MASK
, _T("<== %s %s"), 
 324                                code
.c_str(), line
.c_str() + LEN_CODE 
+ 1); 
 328                     // just part of reply 
 329                     wxLogTrace(FTP_TRACE_MASK
, _T("<== %s %s"), 
 330                                code
.c_str(), line
.c_str()); 
 338         wxLogDebug(_T("Broken FTP server: '%s' is not a valid reply."), 
 339                    m_lastResult
.c_str()); 
 341         m_lastError 
= wxPROTO_PROTERR
; 
 346     // if we got here we must have a non empty code string 
 347     return (char)code
[0u]; 
 350 // ---------------------------------------------------------------------------- 
 351 // wxFTP simple commands 
 352 // ---------------------------------------------------------------------------- 
 354 bool wxFTP::SetTransferMode(TransferMode transferMode
) 
 356     if ( transferMode 
== m_currentTransfermode 
) 
 363     switch ( transferMode 
) 
 366             wxFAIL_MSG(_T("unknown FTP transfer mode")); 
 378     if ( !DoSimpleCommand(_T("TYPE"), mode
) ) 
 380         wxLogError(_("Failed to set FTP transfer mode to %s."), (const wxChar
*) 
 381                    (transferMode 
== ASCII 
? _("ASCII") : _("binary"))); 
 386     // If we get here the operation has been successfully completed 
 387     // Set the status-member 
 388     m_currentTransfermode 
= transferMode
; 
 393 bool wxFTP::DoSimpleCommand(const wxChar 
*command
, const wxString
& arg
) 
 395     wxString fullcmd 
= command
; 
 398         fullcmd 
<< _T(' ') << arg
; 
 401     if ( !CheckCommand(fullcmd
, '2') ) 
 403         wxLogDebug(_T("FTP command '%s' failed."), fullcmd
.c_str()); 
 411 bool wxFTP::ChDir(const wxString
& dir
) 
 413     // some servers might not understand ".." if they use different directory 
 414     // tree conventions, but they always understand CDUP - should we use it if 
 415     // dir == ".."? OTOH, do such servers (still) exist? 
 417     return DoSimpleCommand(_T("CWD"), dir
); 
 420 bool wxFTP::MkDir(const wxString
& dir
) 
 422     return DoSimpleCommand(_T("MKD"), dir
); 
 425 bool wxFTP::RmDir(const wxString
& dir
) 
 427     return DoSimpleCommand(_T("RMD"), dir
); 
 430 wxString 
wxFTP::Pwd() 
 434     if ( CheckCommand(wxT("PWD"), '2') ) 
 436         // the result is at least that long if CheckCommand() succeeded 
 437         const wxChar 
*p 
= m_lastResult
.c_str() + LEN_CODE 
+ 1; 
 440             wxLogDebug(_T("Missing starting quote in reply for PWD: %s"), p
); 
 448                     // check if the quote is doubled 
 450                     if ( !*p 
|| *p 
!= _T('"') ) 
 452                         // no, this is the end 
 455                     //else: yes, it is: this is an embedded quote in the 
 456                     //      filename, treat as normal char 
 464                 wxLogDebug(_T("Missing ending quote in reply for PWD: %s"), 
 465                            m_lastResult
.c_str() + LEN_CODE 
+ 1); 
 471         wxLogDebug(_T("FTP PWD command failed.")); 
 477 bool wxFTP::Rename(const wxString
& src
, const wxString
& dst
) 
 481     str 
= wxT("RNFR ") + src
; 
 482     if ( !CheckCommand(str
, '3') ) 
 485     str 
= wxT("RNTO ") + dst
; 
 487     return CheckCommand(str
, '2'); 
 490 bool wxFTP::RmFile(const wxString
& path
) 
 493     str 
= wxT("DELE ") + path
; 
 495     return CheckCommand(str
, '2'); 
 498 // ---------------------------------------------------------------------------- 
 499 // wxFTP download and upload 
 500 // ---------------------------------------------------------------------------- 
 502 class wxInputFTPStream 
: public wxSocketInputStream
 
 505     wxInputFTPStream(wxFTP 
*ftp
, wxSocketBase 
*sock
) 
 506         : wxSocketInputStream(*sock
) 
 509         // socket timeout automatically set in GetPort function 
 512     virtual ~wxInputFTPStream() 
 514         delete m_i_socket
;   // keep at top 
 516         // when checking the result, the stream will 
 517         // almost always show an error, even if the file was 
 518         // properly transfered, thus, lets just grab the result 
 520         // we are looking for "226 transfer completed" 
 521         char code 
= m_ftp
->GetResult(); 
 524             // it was a good transfer. 
 526              m_ftp
->m_streaming 
= false; 
 532             // the connection is probably toast. issue an abort, and 
 533             // then a close. there won't be any more waiting 
 534             // for this connection 
 539         // There was a problem with the transfer and the server 
 540         // has acknowledged it.  If we issue an "ABORT" now, the user 
 541         // would get the "226" for the abort and think the xfer was 
 542         // complete, thus, don't do anything here, just return 
 547     DECLARE_NO_COPY_CLASS(wxInputFTPStream
) 
 550 class wxOutputFTPStream 
: public wxSocketOutputStream
 
 553     wxOutputFTPStream(wxFTP 
*ftp_clt
, wxSocketBase 
*sock
) 
 554         : wxSocketOutputStream(*sock
), m_ftp(ftp_clt
) 
 558     virtual ~wxOutputFTPStream(void) 
 562             // close data connection first, this will generate "transfer 
 567             m_ftp
->GetResult(); // save result so user can get to it 
 569             m_ftp
->m_streaming 
= false; 
 573             // abort data connection first 
 576             // and close it after 
 583     DECLARE_NO_COPY_CLASS(wxOutputFTPStream
) 
 586 void wxFTP::SetDefaultTimeout(wxUint32 Value
) 
 588     m_uiDefaultTimeout 
= Value
; 
 589     SetTimeout(Value
); // sets it for this socket 
 593 wxSocketBase 
*wxFTP::GetPort() 
 596     PASSIVE:    Client sends a "PASV" to the server.  The server responds with 
 597                 an address and port number which it will be listening on. Then 
 598                 the client connects to the server at the specified address and 
 601     ACTIVE:     Client sends the server a PORT command which includes an 
 602                 address and port number which the client will be listening on. 
 603                 The server then connects to the client at that address and 
 607     wxSocketBase 
*socket 
= m_bPassive 
? GetPassivePort() : GetActivePort(); 
 610         m_bEncounteredError 
= true; 
 614     // Now set the time for the new socket to the default or user selected 
 616     socket
->SetTimeout(m_uiDefaultTimeout
); 
 621 wxSocketBase 
*wxFTP::AcceptIfActive(wxSocketBase 
*sock
) 
 626     // now wait for a connection from server 
 627     wxSocketServer 
*sockSrv 
= (wxSocketServer 
*)sock
; 
 628     if ( !sockSrv
->WaitForAccept() ) 
 630         m_lastError 
= wxPROTO_CONNERR
; 
 631         wxLogError(_("Timeout while waiting for FTP server to connect, try passive mode.")); 
 637         sock 
= sockSrv
->Accept(true); 
 644 wxString 
wxFTP::GetPortCmdArgument(const wxIPV4address
& addrLocal
, 
 645                                    const wxIPV4address
& addrNew
) 
 647     // Just fills in the return value with the local IP 
 648     // address of the current socket.  Also it fill in the 
 649     // PORT which the client will be listening on 
 651     wxString addrIP 
= addrLocal
.IPAddress(); 
 652     int portNew 
= addrNew
.Service(); 
 654     // We need to break the PORT number in bytes 
 655     addrIP
.Replace(_T("."), _T(",")); 
 657            << wxString::Format(_T("%d"), portNew 
>> 8) << _T(',') 
 658            << wxString::Format(_T("%d"), portNew 
& 0xff); 
 660     // Now we have a value like "10,0,0,1,5,23" 
 664 wxSocketBase 
*wxFTP::GetActivePort() 
 666     // we need an address to listen on 
 667     wxIPV4address addrNew
, addrLocal
; 
 669     addrNew
.AnyAddress(); 
 670     addrNew
.Service(0); // pick an open port number. 
 672     wxSocketServer 
*sockSrv 
= new wxSocketServer(addrNew
); 
 675         // We use Ok() here to see if everything is ok 
 676         m_lastError 
= wxPROTO_PROTERR
; 
 681     //gets the new address, actually it is just the port number 
 682     sockSrv
->GetLocal(addrNew
); 
 684     // Now we create the argument of the PORT command, we send in both 
 685     // addresses because the addrNew has an IP of "0.0.0.0", so we need the 
 686     // value in addrLocal 
 687     wxString port 
= GetPortCmdArgument(addrLocal
, addrNew
); 
 688     if ( !DoSimpleCommand(_T("PORT "), port
) ) 
 690         m_lastError 
= wxPROTO_PROTERR
; 
 692         wxLogError(_("The FTP server doesn't support the PORT command.")); 
 696     sockSrv
->Notify(false); // Don't send any events 
 700 wxSocketBase 
*wxFTP::GetPassivePort() 
 702     if ( !DoSimpleCommand(_T("PASV")) ) 
 704         wxLogError(_("The FTP server doesn't support passive mode.")); 
 708     const wxChar 
*addrStart 
= wxStrchr(m_lastResult
, _T('(')); 
 709     const wxChar 
*addrEnd 
= addrStart 
? wxStrchr(addrStart
, _T(')')) : NULL
; 
 712         m_lastError 
= wxPROTO_PROTERR
; 
 717     // get the port number and address 
 719     wxString 
straddr(addrStart 
+ 1, addrEnd
); 
 720     wxSscanf(straddr
, wxT("%d,%d,%d,%d,%d,%d"), 
 721              &a
[2],&a
[3],&a
[4],&a
[5],&a
[0],&a
[1]); 
 723     wxUint32 hostaddr 
= (wxUint16
)a
[2] << 24 | 
 724                         (wxUint16
)a
[3] << 16 | 
 725                         (wxUint16
)a
[4] << 8 | 
 727     wxUint16 port 
= (wxUint16
)(a
[0] << 8 | a
[1]); 
 730     addr
.Hostname(hostaddr
); 
 733     wxSocketClient 
*client 
= new wxSocketClient(); 
 734     if ( !client
->Connect(addr
) ) 
 740     client
->Notify(false); 
 751     if ( !CheckCommand(wxT("ABOR"), '4') ) 
 754     return CheckResult('2'); 
 757 wxInputStream 
*wxFTP::GetInputStream(const wxString
& path
) 
 759     if ( ( m_currentTransfermode 
== NONE 
) && !SetTransferMode(BINARY
) ) 
 762     wxSocketBase 
*sock 
= GetPort(); 
 766         m_lastError 
= wxPROTO_NETERR
; 
 770     wxString tmp_str 
= wxT("RETR ") + wxURI::Unescape(path
); 
 771     if ( !CheckCommand(tmp_str
, '1') ) 
 774     sock 
= AcceptIfActive(sock
); 
 778     sock
->SetFlags(wxSOCKET_WAITALL
); 
 782     wxInputFTPStream 
*in_stream 
= new wxInputFTPStream(this, sock
); 
 787 wxOutputStream 
*wxFTP::GetOutputStream(const wxString
& path
) 
 789     if ( ( m_currentTransfermode 
== NONE 
) && !SetTransferMode(BINARY
) ) 
 792     wxSocketBase 
*sock 
= GetPort(); 
 794     wxString tmp_str 
= wxT("STOR ") + path
; 
 795     if ( !CheckCommand(tmp_str
, '1') ) 
 798     sock 
= AcceptIfActive(sock
); 
 802     return new wxOutputFTPStream(this, sock
); 
 805 // ---------------------------------------------------------------------------- 
 806 // FTP directory listing 
 807 // ---------------------------------------------------------------------------- 
 809 bool wxFTP::GetList(wxArrayString
& files
, 
 810                     const wxString
& wildcard
, 
 813     wxSocketBase 
*sock 
= GetPort(); 
 817     // NLST : List of Filenames (including Directory's !) 
 818     // LIST : depending on BS of FTP-Server 
 819     //        - Unix    : result like "ls" command 
 820     //        - Windows : like "dir" command 
 822     wxString 
line(details 
? _T("LIST") : _T("NLST")); 
 823     if ( !wildcard
.empty() ) 
 825         line 
<< _T(' ') << wildcard
; 
 828     if ( !CheckCommand(line
, '1') ) 
 830         m_lastError 
= wxPROTO_PROTERR
; 
 831         wxLogDebug(_T("FTP 'LIST' command returned unexpected result from server")); 
 836     sock 
= AcceptIfActive(sock
); 
 841     while (ReadLine(sock
, line
) == wxPROTO_NOERR 
) 
 848     // the file list should be terminated by "226 Transfer complete"" 
 849     return CheckResult('2'); 
 852 bool wxFTP::FileExists(const wxString
& fileName
) 
 854     // This function checks if the file specified in fileName exists in the 
 855     // current dir. It does so by simply doing an NLST (via GetList). 
 856     // If this succeeds (and the list is not empty) the file exists. 
 859     wxArrayString fileList
; 
 861     if ( GetList(fileList
, fileName
, false) ) 
 863         // Some ftp-servers (Ipswitch WS_FTP Server 1.0.5 does this) 
 864         // displays this behaviour when queried on a nonexistent file: 
 865         // NLST this_file_does_not_exist 
 866         // 150 Opening ASCII data connection for directory listing 
 867         // (no data transferred) 
 868         // 226 Transfer complete 
 869         // Here wxFTP::GetList(...) will succeed but it will return an empty 
 871         retval 
= !fileList
.IsEmpty(); 
 877 // ---------------------------------------------------------------------------- 
 879 // ---------------------------------------------------------------------------- 
 881 int wxFTP::GetFileSize(const wxString
& fileName
) 
 883     // return the filesize of the given file if possible 
 884     // return -1 otherwise (predominantly if file doesn't exist 
 889     // Check for existance of file via wxFTP::FileExists(...) 
 890     if ( FileExists(fileName
) ) 
 894         // First try "SIZE" command using BINARY(IMAGE) transfermode 
 895         // Especially UNIX ftp-servers distinguish between the different 
 896         // transfermodes and reports different filesizes accordingly. 
 897         // The BINARY size is the interesting one: How much memory 
 898         // will we need to hold this file? 
 899         TransferMode oldTransfermode 
= m_currentTransfermode
; 
 900         SetTransferMode(BINARY
); 
 901         command 
<< _T("SIZE ") << fileName
; 
 903         bool ok 
= CheckCommand(command
, '2'); 
 907             // The answer should be one line: "213 <filesize>\n" 
 908             // 213 is File Status (STD9) 
 909             // "SIZE" is not described anywhere..? It works on most servers 
 911             if ( wxSscanf(GetLastResult().c_str(), _T("%i %i"), 
 912                           &statuscode
, &filesize
) == 2 ) 
 914                 // We've gotten a good reply. 
 919                 // Something bad happened.. A "2yz" reply with no size 
 925         // Set transfermode back to the original. Only the "SIZE"-command 
 926         // is dependant on transfermode 
 927         if ( oldTransfermode 
!= NONE 
) 
 929             SetTransferMode(oldTransfermode
); 
 932         // this is not a direct else clause.. The size command might return an 
 933         // invalid "2yz" reply 
 936             // The server didn't understand the "SIZE"-command or it 
 937             // returned an invalid reply. 
 938             // We now try to get details for the file with a "LIST"-command 
 939             // and then parse the output from there.. 
 940             wxArrayString fileList
; 
 941             if ( GetList(fileList
, fileName
, true) ) 
 943                 if ( !fileList
.IsEmpty() ) 
 945                     // We _should_ only get one line in return, but just to be 
 946                     // safe we run through the line(s) returned and look for a 
 947                     // substring containing the name we are looking for. We 
 948                     // stop the iteration at the first occurrence of the 
 949                     // filename. The search is not case-sensitive. 
 950                     bool foundIt 
= false; 
 953                     for ( i 
= 0; !foundIt 
&& i 
< fileList
.Count(); i
++ ) 
 955                         foundIt 
= fileList
[i
].Upper().Contains(fileName
.Upper()); 
 960                         // The index i points to the first occurrence of 
 961                         // fileName in the array Now we have to find out what 
 962                         // format the LIST has returned. There are two 
 963                         // "schools": Unix-like 
 965                         // '-rw-rw-rw- owner group size month day time filename' 
 969                         // 'date size filename' 
 971                         // check if the first character is '-'. This would 
 972                         // indicate Unix-style (this also limits this function 
 973                         // to searching for files, not directories) 
 974                         if ( fileList
[i
].Mid(0, 1) == _T("-") ) 
 977                             if ( wxSscanf(fileList
[i
].c_str(), 
 978                                           _T("%*s %*s %*s %*s %i %*s %*s %*s %*s"), 
 981                                 // Hmm... Invalid response 
 982                                 wxLogTrace(FTP_TRACE_MASK
, 
 983                                            _T("Invalid LIST response")); 
 986                         else // Windows-style response (?) 
 988                             if ( wxSscanf(fileList
[i
].c_str(), 
 989                                           _T("%*s %*s %i %*s"), 
 992                                 // something bad happened..? 
 993                                 wxLogTrace(FTP_TRACE_MASK
, 
 994                                            _T("Invalid or unknown LIST response")); 
1003     // filesize might still be -1 when exiting 
1007 #endif // wxUSE_PROTOCOL_FTP