]> git.saurik.com Git - wxWidgets.git/blame - src/common/url.cpp
Minor header cleaning.
[wxWidgets.git] / src / common / url.cpp
CommitLineData
f4ada568 1/////////////////////////////////////////////////////////////////////////////
8ecff181 2// Name: src/common/url.cpp
f4ada568
GL
3// Purpose: URL parser
4// Author: Guilhem Lavaux
5// Modified by:
6// Created: 20/07/1997
7// RCS-ID: $Id$
8// Copyright: (c) 1997, 1998 Guilhem Lavaux
65571936 9// Licence: wxWindows licence
f4ada568
GL
10/////////////////////////////////////////////////////////////////////////////
11
fcc6dddd
JS
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
8ecff181 16 #pragma hdrstop
fcc6dddd
JS
17#endif
18
a5d46b73 19#if wxUSE_URL
f4ada568 20
df91131c
WS
21#include "wx/url.h"
22
8ecff181
WS
23#ifndef WX_PRECOMP
24 #include "wx/list.h"
df91131c 25 #include "wx/string.h"
de6185e2 26 #include "wx/utils.h"
02761f6c 27 #include "wx/module.h"
8ecff181
WS
28#endif
29
a5d46b73
VZ
30#include <string.h>
31#include <ctype.h>
32
f4ada568 33IMPLEMENT_CLASS(wxProtoInfo, wxObject)
b60b2ec8 34IMPLEMENT_CLASS(wxURL, wxURI)
f4ada568
GL
35
36// Protocols list
b2b35524 37wxProtoInfo *wxURL::ms_protocols = NULL;
8a4df159 38
f92f546c 39// Enforce linking of protocol classes:
f92f546c
VS
40USE_PROTOCOL(wxFileProto)
41
ce195ee6 42#if wxUSE_PROTOCOL_HTTP
f80eabe5 43USE_PROTOCOL(wxHTTP)
f80eabe5 44
b2b35524 45 wxHTTP *wxURL::ms_proxyDefault = NULL;
cb719f2e 46 bool wxURL::ms_useDefaultProxy = false;
8a4df159 47#endif
f4ada568 48
ce195ee6
MB
49#if wxUSE_PROTOCOL_FTP
50USE_PROTOCOL(wxFTP)
51#endif
52
fae05df5 53// --------------------------------------------------------------
b60b2ec8
RN
54//
55// wxURL
56//
fae05df5 57// --------------------------------------------------------------
f4ada568 58
ce321570
RN
59// --------------------------------------------------------------
60// Construction
61// --------------------------------------------------------------
62
b60b2ec8
RN
63wxURL::wxURL(const wxString& url) : wxURI(url)
64{
65 Init(url);
66 ParseURL();
67}
68
69wxURL::wxURL(const wxURI& url) : wxURI(url)
70{
86470d43 71 Init(url.BuildURI());
b60b2ec8
RN
72 ParseURL();
73}
74
b60b2ec8 75void wxURL::Init(const wxString& url)
f4ada568 76{
b2b35524
VZ
77 m_protocol = NULL;
78 m_error = wxURL_NOERR;
79 m_url = url;
25959b95
VZ
80#if wxUSE_URL_NATIVE
81 m_nativeImp = CreateNativeImpObject();
82#endif
b2b35524 83
ce195ee6 84#if wxUSE_PROTOCOL_HTTP
b2b35524
VZ
85 if ( ms_useDefaultProxy && !ms_proxyDefault )
86 {
2b5f62a0 87 SetDefaultProxy( wxGetenv(wxT("HTTP_PROXY")) );
b2b35524
VZ
88
89 if ( !ms_proxyDefault )
90 {
91 // don't try again
cb719f2e 92 ms_useDefaultProxy = false;
b2b35524
VZ
93 }
94 }
95
96 m_useProxy = ms_proxyDefault != NULL;
97 m_proxy = ms_proxyDefault;
ce195ee6 98#endif // wxUSE_PROTOCOL_HTTP
b2b35524 99
f4ada568 100}
ce321570
RN
101
102// --------------------------------------------------------------
103// Assignment
104// --------------------------------------------------------------
105
106wxURL& wxURL::operator = (const wxURI& url)
107{
108 wxURI::operator = (url);
109 Init(url.BuildURI());
110 ParseURL();
111 return *this;
112}
113wxURL& wxURL::operator = (const wxString& url)
114{
115 wxURI::operator = (url);
116 Init(url);
117 ParseURL();
118 return *this;
119}
120
b60b2ec8
RN
121// --------------------------------------------------------------
122// ParseURL
123//
124// Builds the URL and takes care of the old protocol stuff
125// --------------------------------------------------------------
f4ada568
GL
126
127bool wxURL::ParseURL()
128{
fd725bce
WS
129 // If the URL was already parsed (m_protocol != NULL), pass this section.
130 if (!m_protocol)
f6bcfd97 131 {
fd725bce
WS
132 // Clean up
133 CleanData();
f61815af 134
fd725bce
WS
135 // Make sure we have a protocol/scheme
136 if (!HasScheme())
137 {
138 m_error = wxURL_SNTXERR;
139 return false;
140 }
f61815af 141
fd725bce
WS
142 // Find and create the protocol object
143 if (!FetchProtocol())
144 {
145 m_error = wxURL_NOPROTO;
146 return false;
147 }
148
149 // Do we need a host name ?
150 if (m_protoinfo->m_needhost)
151 {
152 // Make sure we have one, then
153 if (!HasServer())
154 {
155 m_error = wxURL_SNTXERR;
156 return false;
157 }
158 }
f61815af 159 }
f4ada568 160
ce195ee6 161#if wxUSE_PROTOCOL_HTTP
fd725bce
WS
162 if (m_useProxy)
163 {
164 // Third, we rebuild the URL.
165 m_url = m_scheme + wxT(":");
166 if (m_protoinfo->m_needhost)
167 m_url = m_url + wxT("//") + m_server;
168
169 // We initialize specific variables.
170 m_protocol = m_proxy; // FIXME: we should clone the protocol
171 }
ce195ee6 172#endif // wxUSE_PROTOCOL_HTTP
f4ada568 173
fd725bce
WS
174 m_error = wxURL_NOERR;
175 return true;
f4ada568
GL
176}
177
ce321570
RN
178// --------------------------------------------------------------
179// Destruction/Cleanup
180// --------------------------------------------------------------
181
f4ada568
GL
182void wxURL::CleanData()
183{
ce195ee6 184#if wxUSE_PROTOCOL_HTTP
fd725bce 185 if (!m_useProxy)
ce195ee6 186#endif // wxUSE_PROTOCOL_HTTP
5f3742c2
KH
187 if (m_protocol)
188 // Need to safely delete the socket (pending events)
189 m_protocol->Destroy();
f4ada568
GL
190}
191
192wxURL::~wxURL()
193{
25959b95 194 CleanData();
ce195ee6 195#if wxUSE_PROTOCOL_HTTP
25959b95
VZ
196 if (m_proxy && m_proxy != ms_proxyDefault)
197 delete m_proxy;
ce195ee6 198#endif // wxUSE_PROTOCOL_HTTP
25959b95
VZ
199#if wxUSE_URL_NATIVE
200 delete m_nativeImp;
8a4df159 201#endif
f4ada568
GL
202}
203
ce321570
RN
204// --------------------------------------------------------------
205// FetchProtocol
206// --------------------------------------------------------------
f4ada568
GL
207
208bool wxURL::FetchProtocol()
209{
fd725bce 210 wxProtoInfo *info = ms_protocols;
f4ada568 211
fd725bce 212 while (info)
f6bcfd97 213 {
fd725bce
WS
214 if (m_scheme == info->m_protoname)
215 {
216 if (m_port.IsNull())
217 m_port = info->m_servname;
218 m_protoinfo = info;
219 m_protocol = (wxProtocol *)m_protoinfo->m_cinfo->CreateObject();
220 return true;
221 }
222 info = info->next;
f4ada568 223 }
fd725bce 224 return false;
f4ada568
GL
225}
226
fae05df5 227// --------------------------------------------------------------
ce321570 228// GetInputStream
fae05df5
GL
229// --------------------------------------------------------------
230
58c837a4 231wxInputStream *wxURL::GetInputStream()
f4ada568 232{
fd725bce
WS
233 if (!m_protocol)
234 {
235 m_error = wxURL_NOPROTO;
236 return NULL;
237 }
238
239 m_error = wxURL_NOERR;
240 if (HasUserInfo())
241 {
242 size_t dwPasswordPos = m_userinfo.find(':');
243
244 if (dwPasswordPos == wxString::npos)
245 m_protocol->SetUser(m_userinfo);
246 else
247 {
248 m_protocol->SetUser(m_userinfo(0, dwPasswordPos));
249 m_protocol->SetPassword(m_userinfo(dwPasswordPos+1, m_userinfo.length() + 1));
250 }
251 }
856d2e52 252
25959b95 253#if wxUSE_URL_NATIVE
fd725bce
WS
254 // give the native implementation to return a better stream
255 // such as the native WinINet functionality under MS-Windows
256 if (m_nativeImp)
257 {
258 wxInputStream *rc;
259 rc = m_nativeImp->GetInputStream(this);
260 if (rc != 0)
261 return rc;
262 }
263 // else use the standard behaviour
25959b95
VZ
264#endif // wxUSE_URL_NATIVE
265
8a4df159 266#if wxUSE_SOCKETS
19e0e04b
RD
267 wxIPV4address addr;
268
fd725bce
WS
269 // m_protoinfo is NULL when we use a proxy
270 if (!m_useProxy && m_protoinfo->m_needhost)
f6bcfd97 271 {
fd725bce
WS
272 if (!addr.Hostname(m_server))
273 {
274 m_error = wxURL_NOHOST;
275 return NULL;
276 }
277
278 addr.Service(m_port);
279
280 if (!m_protocol->Connect(addr, true)) // Watcom needs the 2nd arg for some reason
281 {
282 m_error = wxURL_CONNERR;
283 return NULL;
284 }
f4ada568 285 }
fd725bce
WS
286#endif
287
288 wxString fullPath;
f4ada568 289
fd725bce
WS
290 // When we use a proxy, we have to pass the whole URL to it.
291 if (m_useProxy)
292 fullPath += m_url;
f4ada568 293
fd725bce
WS
294 if(m_path.empty())
295 fullPath += wxT("/");
296 else
297 fullPath += m_path;
298
299 if (HasQuery())
300 fullPath += wxT("?") + m_query;
301
302 if (HasFragment())
303 fullPath += wxT("#") + m_fragment;
304
305 wxInputStream *the_i_stream = m_protocol->GetInputStream(fullPath);
306
307 if (!the_i_stream)
8a2c6ef8 308 {
fd725bce
WS
309 m_error = wxURL_PROTOERR;
310 return NULL;
f4ada568 311 }
f4ada568 312
fd725bce 313 return the_i_stream;
f4ada568
GL
314}
315
ce195ee6 316#if wxUSE_PROTOCOL_HTTP
f4ada568
GL
317void wxURL::SetDefaultProxy(const wxString& url_proxy)
318{
fd725bce
WS
319 if ( !url_proxy )
320 {
321 if ( ms_proxyDefault )
322 {
323 ms_proxyDefault->Close();
324 delete ms_proxyDefault;
325 ms_proxyDefault = NULL;
326 }
327 }
328 else
329 {
330 wxString tmp_str = url_proxy;
331 int pos = tmp_str.Find(wxT(':'));
332 if (pos == wxNOT_FOUND)
333 return;
334
335 wxString hostname = tmp_str(0, pos),
8ecff181 336 port = tmp_str(pos+1, tmp_str.length()-pos);
fd725bce
WS
337 wxIPV4address addr;
338
339 if (!addr.Hostname(hostname))
340 return;
341 if (!addr.Service(port))
342 return;
343
344 if (ms_proxyDefault)
345 // Finally, when all is right, we connect the new proxy.
346 ms_proxyDefault->Close();
347 else
348 ms_proxyDefault = new wxHTTP();
349 ms_proxyDefault->Connect(addr, true); // Watcom needs the 2nd arg for some reason
350 }
f4ada568
GL
351}
352
353void wxURL::SetProxy(const wxString& url_proxy)
354{
b2b35524
VZ
355 if ( !url_proxy )
356 {
357 if ( m_proxy && m_proxy != ms_proxyDefault )
358 {
359 m_proxy->Close();
360 delete m_proxy;
361 }
f4ada568 362
cb719f2e 363 m_useProxy = false;
b2b35524
VZ
364 }
365 else
366 {
367 wxString tmp_str;
368 wxString hostname, port;
369 int pos;
370 wxIPV4address addr;
371
372 tmp_str = url_proxy;
373 pos = tmp_str.Find(wxT(':'));
374 // This is an invalid proxy name.
cb719f2e 375 if (pos == wxNOT_FOUND)
b2b35524
VZ
376 return;
377
378 hostname = tmp_str(0, pos);
8ecff181 379 port = tmp_str(pos+1, tmp_str.length()-pos);
b2b35524
VZ
380
381 addr.Hostname(hostname);
382 addr.Service(port);
383
384 // Finally, create the whole stuff.
385 if (m_proxy && m_proxy != ms_proxyDefault)
386 delete m_proxy;
387 m_proxy = new wxHTTP();
cb719f2e 388 m_proxy->Connect(addr, true); // Watcom needs the 2nd arg for some reason
b2b35524
VZ
389
390 CleanData();
391 // Reparse url.
cb719f2e 392 m_useProxy = true;
b2b35524
VZ
393 ParseURL();
394 }
f4ada568 395}
ce195ee6 396#endif // wxUSE_PROTOCOL_HTTP
35a4dab7 397
b2b35524 398// ----------------------------------------------------------------------
ce321570
RN
399// wxURLModule
400//
b2b35524
VZ
401// A module which deletes the default proxy if we created it
402// ----------------------------------------------------------------------
403
404#if wxUSE_SOCKETS
405
406class wxURLModule : public wxModule
407{
408public:
409 virtual bool OnInit();
410 virtual void OnExit();
411
412private:
413 DECLARE_DYNAMIC_CLASS(wxURLModule)
414};
415
416IMPLEMENT_DYNAMIC_CLASS(wxURLModule, wxModule)
417
418bool wxURLModule::OnInit()
419{
ce195ee6 420#if wxUSE_PROTOCOL_HTTP
b2b35524
VZ
421 // env var HTTP_PROXY contains the address of the default proxy to use if
422 // set, but don't try to create this proxy right now because it will slow
423 // down the program startup (especially if there is no DNS server
424 // available, in which case it may take up to 1 minute)
f6bcfd97 425
67acaf80 426 if ( wxGetenv(_T("HTTP_PROXY")) )
b2b35524 427 {
cb719f2e 428 wxURL::ms_useDefaultProxy = true;
b2b35524 429 }
ce195ee6 430#endif // wxUSE_PROTOCOL_HTTP
cb719f2e 431 return true;
b2b35524
VZ
432}
433
434void wxURLModule::OnExit()
435{
ce195ee6 436#if wxUSE_PROTOCOL_HTTP
b2b35524
VZ
437 delete wxURL::ms_proxyDefault;
438 wxURL::ms_proxyDefault = NULL;
ce195ee6 439#endif // wxUSE_PROTOCOL_HTTP
b2b35524
VZ
440}
441
442#endif // wxUSE_SOCKETS
a5d46b73 443
97d8fe87
VZ
444// ---------------------------------------------------------------------------
445//
446// wxURL Compatibility
447//
448// ---------------------------------------------------------------------------
449
450#if WXWIN_COMPATIBILITY_2_4
451
452#include "wx/url.h"
453
454wxString wxURL::GetProtocolName() const
455{
456 return m_scheme;
457}
458
459wxString wxURL::GetHostName() const
460{
461 return m_server;
462}
463
464wxString wxURL::GetPath() const
465{
466 return m_path;
467}
468
469//Note that this old code really doesn't convert to a URI that well and looks
470//more like a dirty hack than anything else...
471
472wxString wxURL::ConvertToValidURI(const wxString& uri, const wxChar* delims)
473{
474 wxString out_str;
475 wxString hexa_code;
476 size_t i;
477
478 for (i = 0; i < uri.Len(); i++)
479 {
480 wxChar c = uri.GetChar(i);
481
482 if (c == wxT(' '))
483 {
484 // GRG, Apr/2000: changed to "%20" instead of '+'
485
486 out_str += wxT("%20");
487 }
488 else
489 {
490 // GRG, Apr/2000: modified according to the URI definition (RFC 2396)
491 //
492 // - Alphanumeric characters are never escaped
493 // - Unreserved marks are never escaped
494 // - Delimiters must be escaped if they appear within a component
495 // but not if they are used to separate components. Here we have
496 // no clear way to distinguish between these two cases, so they
497 // are escaped unless they are passed in the 'delims' parameter
498 // (allowed delimiters).
499
500 static const wxChar marks[] = wxT("-_.!~*()'");
501
502 if ( !wxIsalnum(c) && !wxStrchr(marks, c) && !wxStrchr(delims, c) )
503 {
504 hexa_code.Printf(wxT("%%%02X"), c);
505 out_str += hexa_code;
506 }
507 else
508 {
509 out_str += c;
510 }
511 }
512 }
513
514 return out_str;
515}
516
517wxString wxURL::ConvertFromURI(const wxString& uri)
518{
519 return wxURI::Unescape(uri);
520}
521
522#endif //WXWIN_COMPATIBILITY_2_4
523
a5d46b73 524#endif // wxUSE_URL