]> git.saurik.com Git - wxWidgets.git/blame - src/common/ftp.cpp
Couple of unicode fixes (the #if should be wxUSE_UNICODE, not wxUNICODE,
[wxWidgets.git] / src / common / ftp.cpp
CommitLineData
f4ada568
GL
1/////////////////////////////////////////////////////////////////////////////
2// Name: ftp.cpp
3// Purpose: FTP protocol
4// Author: Guilhem Lavaux
5// Modified by:
6// Created: 07/07/1997
7// RCS-ID: $Id$
8// Copyright: (c) 1997, 1998 Guilhem Lavaux
9// Licence: wxWindows license
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
2df7be7f 13 #pragma implementation "ftp.h"
f4ada568 14#endif
ec45f8ee
UU
15
16// For compilers that support precompilation, includes "wx.h".
17#include "wx/wxprec.h"
18
19ae5cf0 19#ifdef __BORLANDC__
2df7be7f 20 #pragma hdrstop
19ae5cf0
UU
21#endif
22
35a4dab7
GL
23#if wxUSE_SOCKETS
24
469e1e5c 25#ifndef __MWERKS__
f4ada568 26#include <memory.h>
469e1e5c 27#endif
17dff81c
SC
28#if defined(__WXMAC__)
29#include "/wx/mac/macsock.h"
30#endif
31
f4ada568
GL
32#include <stdlib.h>
33#include "wx/string.h"
34#include "wx/utils.h"
35// #include "wx/data.h"
36#define WXSOCK_INTERNAL
37#include "wx/sckaddr.h"
38#undef WXSOCK_INTERNAL
39#include "wx/socket.h"
40#include "wx/url.h"
41#include "wx/sckstrm.h"
42#include "wx/protocol/protocol.h"
43#include "wx/protocol/ftp.h"
44
45#ifdef __BORLANDC__
46#pragma hdrstop
47#endif
48
49#define FTP_BSIZE 1024
50
51#if !USE_SHARED_LIBRARY
52IMPLEMENT_DYNAMIC_CLASS(wxFTP, wxProtocol)
4846abaf 53IMPLEMENT_PROTOCOL(wxFTP, _T("ftp"), _T("ftp"), TRUE)
f4ada568
GL
54#endif
55
56////////////////////////////////////////////////////////////////
57////// wxFTP constructor and destructor ////////////////////////
58////////////////////////////////////////////////////////////////
59
60wxFTP::wxFTP()
61 : wxProtocol()
62{
f4ada568
GL
63 m_lastError = wxPROTO_NOERR;
64 m_streaming = FALSE;
65
4846abaf 66 m_user = _T("anonymous");
a737331d
GL
67 m_passwd = wxGetUserId();
68 m_passwd += '@';
69 m_passwd += wxGetHostName();
f4ada568
GL
70
71 SetNotify(0);
a324a7bc 72 SetFlags(NONE);
f4ada568
GL
73}
74
75wxFTP::~wxFTP()
76{
77 SendCommand("QUIT", '2');
78}
79
80////////////////////////////////////////////////////////////////
81////// wxFTP connect and login methods /////////////////////////
82////////////////////////////////////////////////////////////////
8a2c6ef8 83bool wxFTP::Connect(wxSockAddress& addr, bool WXUNUSED(wait))
f4ada568 84{
f4ada568
GL
85 if (!wxProtocol::Connect(addr)) {
86 m_lastError = wxPROTO_NETERR;
87 return FALSE;
88 }
89
90 if (!m_user || !m_passwd) {
91 m_lastError = wxPROTO_CONNERR;
92 return FALSE;
93 }
94
95 wxString command;
96
97 if (!GetResult('2')) {
98 Close();
99 return FALSE;
100 }
101
4846abaf 102 command.sprintf(_T("USER %s"), (const wxChar *)m_user);
f4ada568
GL
103 if (!SendCommand(command, '3')) {
104 Close();
105 return FALSE;
106 }
107
4846abaf 108 command.sprintf(_T("PASS %s"), (const wxChar *)m_passwd);
f4ada568
GL
109 if (!SendCommand(command, '2')) {
110 Close();
111 return FALSE;
112 }
113
114 return TRUE;
115}
116
117bool wxFTP::Connect(const wxString& host)
118{
119 wxIPV4address addr;
120 wxString my_host = host;
121
122 addr.Hostname(my_host);
4846abaf 123 addr.Service(_T("ftp"));
f4ada568
GL
124
125 return Connect(addr);
126}
127
128bool wxFTP::Close()
129{
130 if (m_streaming) {
131 m_lastError = wxPROTO_STREAMING;
132 return FALSE;
133 }
134 if (m_connected)
4846abaf 135 SendCommand(wxString(_T("QUIT")), '2');
f4ada568
GL
136 return wxSocketClient::Close();
137}
138
139////////////////////////////////////////////////////////////////
140////// wxFTP low-level methods /////////////////////////////////
141////////////////////////////////////////////////////////////////
142bool wxFTP::SendCommand(const wxString& command, char exp_ret)
143{
144 wxString tmp_str;
145
146 if (m_streaming) {
147 m_lastError = wxPROTO_STREAMING;
148 return FALSE;
149 }
4846abaf 150 tmp_str = command + _T("\r\n");
fde7d98e 151 const wxWX2MBbuf tmp_buf = tmp_str.mb_str();
4846abaf 152 if (Write(MBSTRINGCAST tmp_buf, strlen(tmp_buf)).Error()) {
f4ada568
GL
153 m_lastError = wxPROTO_NETERR;
154 return FALSE;
155 }
156 return GetResult(exp_ret);
157}
158
159bool wxFTP::GetResult(char exp)
160{
161 if ((m_lastError = GetLine(this, m_lastResult)))
162 return FALSE;
4b5f3fe6 163 if (m_lastResult.GetChar(0) != exp) {
f4ada568
GL
164 m_lastError = wxPROTO_PROTERR;
165 return FALSE;
166 }
167
4b5f3fe6 168 if (m_lastResult.GetChar(3) == '-') {
f4ada568
GL
169 wxString key = m_lastResult.Left((size_t)3);
170
4846abaf 171 key += _T(' ');
f4ada568
GL
172
173 while (m_lastResult.Index(key) != 0) {
174 if ((m_lastError = GetLine(this, m_lastResult)))
175 return FALSE;
176 }
177 }
178 return TRUE;
179}
180
181////////////////////////////////////////////////////////////////
182////// wxFTP low-level methods /////////////////////////////////
183////////////////////////////////////////////////////////////////
184bool wxFTP::ChDir(const wxString& dir)
185{
186 wxString str = dir;
187
4846abaf 188 str.Prepend(_T("CWD "));
f4ada568
GL
189 return SendCommand(str, '2');
190}
191
192bool wxFTP::MkDir(const wxString& dir)
193{
194 wxString str = dir;
4846abaf 195 str.Prepend(_T("MKD "));
f4ada568
GL
196 return SendCommand(str, '2');
197}
198
199bool wxFTP::RmDir(const wxString& dir)
200{
201 wxString str = dir;
202
4846abaf 203 str.Prepend(_T("PWD "));
f4ada568
GL
204 return SendCommand(str, '2');
205}
206
207wxString wxFTP::Pwd()
208{
209 int beg, end;
210
4846abaf 211 if (!SendCommand(_T("PWD"), '2'))
f4ada568
GL
212 return wxString((char *)NULL);
213
4846abaf
OK
214 beg = m_lastResult.Find(_T('\"'),FALSE);
215 end = m_lastResult.Find(_T('\"'),TRUE);
f4ada568
GL
216
217 return wxString(beg+1, end);
218}
219
220bool wxFTP::Rename(const wxString& src, const wxString& dst)
221{
222 wxString str;
223
4846abaf 224 str = _T("RNFR ") + src;
f4ada568
GL
225 if (!SendCommand(str, '3'))
226 return FALSE;
227
4846abaf 228 str = _T("RNTO ") + dst;
f4ada568
GL
229 return SendCommand(str, '2');
230}
231
232bool wxFTP::RmFile(const wxString& path)
233{
234 wxString str;
235
4846abaf 236 str = _T("DELE ");
f4ada568
GL
237 str += path;
238 return SendCommand(str, '2');
239}
240
241////////////////////////////////////////////////////////////////
242////// wxFTP download*upload ///////////////////////////////////
243////////////////////////////////////////////////////////////////
244
245class wxInputFTPStream : public wxSocketInputStream {
246public:
247 wxFTP *m_ftp;
9a1b2c28 248 size_t m_ftpsize;
f4ada568
GL
249
250 wxInputFTPStream(wxFTP *ftp_clt, wxSocketBase *sock)
251 : wxSocketInputStream(*sock), m_ftp(ftp_clt) {}
375abe3d 252 size_t StreamSize() const { return m_ftpsize; }
f4ada568
GL
253 virtual ~wxInputFTPStream(void)
254 {
a324a7bc 255 if (LastError() == wxStream_NOERROR)
f4ada568
GL
256 m_ftp->GetResult('2');
257 else
258 m_ftp->Abort();
259 delete m_i_socket;
260 }
261};
262
263class wxOutputFTPStream : public wxSocketOutputStream {
264public:
265 wxFTP *m_ftp;
266
267 wxOutputFTPStream(wxFTP *ftp_clt, wxSocketBase *sock)
268 : wxSocketOutputStream(*sock), m_ftp(ftp_clt) {}
269 virtual ~wxOutputFTPStream(void)
270 {
75ed1d15 271 if (LastError() != wxStream_NOERROR)
f4ada568
GL
272 m_ftp->GetResult('2');
273 else
274 m_ftp->Abort();
275 delete m_o_socket;
276 }
277};
278
279wxSocketClient *wxFTP::GetPort()
280{
281 wxIPV4address addr;
282 wxSocketClient *client;
f4ada568
GL
283 int a[6];
284 wxString straddr;
285 int addr_pos;
a324a7bc
GL
286 wxUint16 port;
287 wxUint32 hostaddr;
f4ada568 288
4846abaf 289 if (!SendCommand(_T("PASV"), '2'))
f4ada568
GL
290 return NULL;
291
4846abaf 292 addr_pos = m_lastResult.Find(_T('('));
f4ada568
GL
293 if (addr_pos == -1) {
294 m_lastError = wxPROTO_PROTERR;
295 return NULL;
296 }
297 straddr = m_lastResult(addr_pos+1, m_lastResult.Length());
4846abaf 298 wxSscanf((const wxChar *)straddr,_T("%d,%d,%d,%d,%d,%d"),&a[2],&a[3],&a[4],&a[5],&a[0],&a[1]);
f4ada568 299
a324a7bc
GL
300 hostaddr = (wxUint16)a[5] << 24 | (wxUint16)a[4] << 16 |
301 (wxUint16)a[3] << 8 | a[2];
302 addr.Hostname(hostaddr);
303
304 port = (wxUint16)a[0] << 8 | a[1];
305 addr.Service(port);
f4ada568 306
a324a7bc 307 client = new wxSocketClient();
f4ada568
GL
308 if (!client->Connect(addr)) {
309 delete client;
310 return NULL;
311 }
312 client->Notify(FALSE);
313
314 return client;
315}
316
317bool wxFTP::Abort(void)
318{
319 m_streaming = FALSE;
4846abaf 320 if (!SendCommand(_T("ABOR"), '4'))
f4ada568
GL
321 return FALSE;
322 return GetResult('2');
323}
324
325wxInputStream *wxFTP::GetInputStream(const wxString& path)
326{
327 wxString tmp_str;
9a1b2c28
GL
328 int pos_size;
329 wxInputFTPStream *in_stream;
f4ada568 330
4846abaf 331 if (!SendCommand(_T("TYPE I"), '2'))
f4ada568
GL
332 return NULL;
333
334 wxSocketClient *sock = GetPort();
335
336 if (!sock) {
337 m_lastError = wxPROTO_NETERR;
338 return NULL;
339 }
340
4846abaf 341 tmp_str = _T("RETR ") + path;
f4ada568
GL
342 if (!SendCommand(tmp_str, '1'))
343 return NULL;
344
9a1b2c28
GL
345 in_stream = new wxInputFTPStream(this, sock);
346
4846abaf 347 pos_size = m_lastResult.Index(_T('('));
9a1b2c28 348 if (pos_size != wxNOT_FOUND) {
4846abaf 349 wxString str_size = m_lastResult(pos_size+1, m_lastResult.Index(_T(')'))-1);
9a1b2c28 350
4846abaf 351 in_stream->m_ftpsize = wxAtoi(WXSTRINGCAST str_size);
9a1b2c28 352 }
062c4861 353 sock->SetFlags(WAITALL);
9a1b2c28
GL
354
355 return in_stream;
f4ada568
GL
356}
357
358wxOutputStream *wxFTP::GetOutputStream(const wxString& path)
359{
360 wxString tmp_str;
361
4846abaf 362 if (!SendCommand(_T("TYPE I"), '2'))
f4ada568
GL
363 return NULL;
364
365 wxSocketClient *sock = GetPort();
366
4846abaf 367 tmp_str = _T("STOR ") + path;
f4ada568
GL
368 if (!SendCommand(tmp_str, '1'))
369 return FALSE;
370
371 return new wxOutputFTPStream(this, sock);
372}
373
374wxList *wxFTP::GetList(const wxString& wildcard)
375{
376 wxList *file_list = new wxList;
377 wxSocketBase *sock = GetPort();
4846abaf 378 wxString tmp_str = _T("NLST");
f4ada568
GL
379
380 if (!wildcard.IsNull())
381 tmp_str += wildcard;
382
383 if (!SendCommand(tmp_str, '1')) {
384 delete sock;
385 delete file_list;
386 return NULL;
387 }
388
389 while (GetLine(sock, tmp_str) == wxPROTO_NOERR) {
390 file_list->Append((wxObject *)(new wxString(tmp_str)));
391 }
392
393 if (!GetResult('2')) {
394 delete sock;
395 file_list->DeleteContents(TRUE);
396 delete file_list;
397 return NULL;
398 }
399
400 return file_list;
401}
35a4dab7
GL
402#endif
403 // wxUSE_SOCKETS