use the directory of the most recently opened file in wxDocManager if we have any
[wxWidgets.git] / src / common / url.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/url.cpp
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
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // For compilers that support precompilation, includes "wx.h".
13 #include "wx/wxprec.h"
14
15 #ifdef __BORLANDC__
16 #pragma hdrstop
17 #endif
18
19 #if wxUSE_URL
20
21 #include "wx/url.h"
22
23 #ifndef WX_PRECOMP
24 #include "wx/list.h"
25 #include "wx/string.h"
26 #include "wx/utils.h"
27 #include "wx/module.h"
28 #endif
29
30 #include <string.h>
31 #include <ctype.h>
32
33 IMPLEMENT_CLASS(wxProtoInfo, wxObject)
34 IMPLEMENT_CLASS(wxURL, wxURI)
35
36 // Protocols list
37 wxProtoInfo *wxURL::ms_protocols = NULL;
38
39 // Enforce linking of protocol classes:
40 USE_PROTOCOL(wxFileProto)
41
42 #if wxUSE_PROTOCOL_HTTP
43 USE_PROTOCOL(wxHTTP)
44
45 wxHTTP *wxURL::ms_proxyDefault = NULL;
46 bool wxURL::ms_useDefaultProxy = false;
47 #endif
48
49 #if wxUSE_PROTOCOL_FTP
50 USE_PROTOCOL(wxFTP)
51 #endif
52
53 // --------------------------------------------------------------
54 //
55 // wxURL
56 //
57 // --------------------------------------------------------------
58
59 // --------------------------------------------------------------
60 // Construction
61 // --------------------------------------------------------------
62
63 wxURL::wxURL(const wxString& url) : wxURI(url)
64 {
65 Init(url);
66 ParseURL();
67 }
68
69 wxURL::wxURL(const wxURI& url) : wxURI(url)
70 {
71 Init(url.BuildURI());
72 ParseURL();
73 }
74
75 void wxURL::Init(const wxString& url)
76 {
77 m_protocol = NULL;
78 m_error = wxURL_NOERR;
79 m_url = url;
80 #if wxUSE_URL_NATIVE
81 m_nativeImp = CreateNativeImpObject();
82 #endif
83
84 #if wxUSE_PROTOCOL_HTTP
85 if ( ms_useDefaultProxy && !ms_proxyDefault )
86 {
87 SetDefaultProxy( wxGetenv(wxT("HTTP_PROXY")) );
88
89 if ( !ms_proxyDefault )
90 {
91 // don't try again
92 ms_useDefaultProxy = false;
93 }
94 }
95
96 m_useProxy = ms_proxyDefault != NULL;
97 m_proxy = ms_proxyDefault;
98 #endif // wxUSE_PROTOCOL_HTTP
99
100 }
101
102 // --------------------------------------------------------------
103 // Assignment
104 // --------------------------------------------------------------
105
106 wxURL& wxURL::operator = (const wxURI& url)
107 {
108 wxURI::operator = (url);
109 Init(url.BuildURI());
110 ParseURL();
111 return *this;
112 }
113 wxURL& wxURL::operator = (const wxString& url)
114 {
115 wxURI::operator = (url);
116 Init(url);
117 ParseURL();
118 return *this;
119 }
120
121 // --------------------------------------------------------------
122 // ParseURL
123 //
124 // Builds the URL and takes care of the old protocol stuff
125 // --------------------------------------------------------------
126
127 bool wxURL::ParseURL()
128 {
129 // If the URL was already parsed (m_protocol != NULL), pass this section.
130 if (!m_protocol)
131 {
132 // Clean up
133 CleanData();
134
135 // Make sure we have a protocol/scheme
136 if (!HasScheme())
137 {
138 m_error = wxURL_SNTXERR;
139 return false;
140 }
141
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 }
159 }
160
161 #if wxUSE_PROTOCOL_HTTP
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 }
172 #endif // wxUSE_PROTOCOL_HTTP
173
174 m_error = wxURL_NOERR;
175 return true;
176 }
177
178 // --------------------------------------------------------------
179 // Destruction/Cleanup
180 // --------------------------------------------------------------
181
182 void wxURL::CleanData()
183 {
184 #if wxUSE_PROTOCOL_HTTP
185 if (!m_useProxy)
186 #endif // wxUSE_PROTOCOL_HTTP
187 if (m_protocol)
188 // Need to safely delete the socket (pending events)
189 m_protocol->Destroy();
190 }
191
192 wxURL::~wxURL()
193 {
194 CleanData();
195 #if wxUSE_PROTOCOL_HTTP
196 if (m_proxy && m_proxy != ms_proxyDefault)
197 delete m_proxy;
198 #endif // wxUSE_PROTOCOL_HTTP
199 #if wxUSE_URL_NATIVE
200 delete m_nativeImp;
201 #endif
202 }
203
204 // --------------------------------------------------------------
205 // FetchProtocol
206 // --------------------------------------------------------------
207
208 bool wxURL::FetchProtocol()
209 {
210 wxProtoInfo *info = ms_protocols;
211
212 while (info)
213 {
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;
223 }
224 return false;
225 }
226
227 // --------------------------------------------------------------
228 // GetInputStream
229 // --------------------------------------------------------------
230
231 wxInputStream *wxURL::GetInputStream()
232 {
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(Unescape(m_userinfo));
246 else
247 {
248 m_protocol->SetUser(Unescape(m_userinfo(0, dwPasswordPos)));
249 m_protocol->SetPassword(Unescape(m_userinfo(dwPasswordPos+1, m_userinfo.length() + 1)));
250 }
251 }
252
253 #if wxUSE_URL_NATIVE
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
264 #endif // wxUSE_URL_NATIVE
265
266 #if wxUSE_SOCKETS
267 wxIPV4address addr;
268
269 // m_protoinfo is NULL when we use a proxy
270 if (
271 #if wxUSE_PROTOCOL_HTTP
272 !m_useProxy &&
273 #endif // wxUSE_PROTOCOL_HTTP
274 m_protoinfo->m_needhost )
275 {
276 if (!addr.Hostname(m_server))
277 {
278 m_error = wxURL_NOHOST;
279 return NULL;
280 }
281
282 addr.Service(m_port);
283
284 if (!m_protocol->Connect(addr, true)) // Watcom needs the 2nd arg for some reason
285 {
286 m_error = wxURL_CONNERR;
287 return NULL;
288 }
289 }
290 #endif // wxUSE_SOCKETS
291
292 wxString fullPath;
293
294 #if wxUSE_PROTOCOL_HTTP
295 // When we use a proxy, we have to pass the whole URL to it.
296 if (m_useProxy)
297 fullPath += m_url;
298 #endif // wxUSE_PROTOCOL_HTTP
299
300 if(m_path.empty())
301 fullPath += wxT("/");
302 else
303 fullPath += m_path;
304
305 if (HasQuery())
306 fullPath += wxT("?") + m_query;
307
308 if (HasFragment())
309 fullPath += wxT("#") + m_fragment;
310
311 wxInputStream *the_i_stream = m_protocol->GetInputStream(fullPath);
312
313 if (!the_i_stream)
314 {
315 m_error = wxURL_PROTOERR;
316 return NULL;
317 }
318
319 return the_i_stream;
320 }
321
322 #if wxUSE_PROTOCOL_HTTP
323 void wxURL::SetDefaultProxy(const wxString& url_proxy)
324 {
325 if ( !url_proxy )
326 {
327 if ( ms_proxyDefault )
328 {
329 ms_proxyDefault->Close();
330 delete ms_proxyDefault;
331 ms_proxyDefault = NULL;
332 }
333 }
334 else
335 {
336 wxString tmp_str = url_proxy;
337 int pos = tmp_str.Find(wxT(':'));
338 if (pos == wxNOT_FOUND)
339 return;
340
341 wxString hostname = tmp_str(0, pos),
342 port = tmp_str(pos+1, tmp_str.length()-pos);
343 wxIPV4address addr;
344
345 if (!addr.Hostname(hostname))
346 return;
347 if (!addr.Service(port))
348 return;
349
350 if (ms_proxyDefault)
351 // Finally, when all is right, we connect the new proxy.
352 ms_proxyDefault->Close();
353 else
354 ms_proxyDefault = new wxHTTP();
355 ms_proxyDefault->Connect(addr, true); // Watcom needs the 2nd arg for some reason
356 }
357 }
358
359 void wxURL::SetProxy(const wxString& url_proxy)
360 {
361 if ( !url_proxy )
362 {
363 if ( m_proxy && m_proxy != ms_proxyDefault )
364 {
365 m_proxy->Close();
366 delete m_proxy;
367 }
368
369 m_useProxy = false;
370 }
371 else
372 {
373 wxString tmp_str;
374 wxString hostname, port;
375 int pos;
376 wxIPV4address addr;
377
378 tmp_str = url_proxy;
379 pos = tmp_str.Find(wxT(':'));
380 // This is an invalid proxy name.
381 if (pos == wxNOT_FOUND)
382 return;
383
384 hostname = tmp_str(0, pos);
385 port = tmp_str(pos+1, tmp_str.length()-pos);
386
387 addr.Hostname(hostname);
388 addr.Service(port);
389
390 // Finally, create the whole stuff.
391 if (m_proxy && m_proxy != ms_proxyDefault)
392 delete m_proxy;
393 m_proxy = new wxHTTP();
394 m_proxy->Connect(addr, true); // Watcom needs the 2nd arg for some reason
395
396 CleanData();
397 // Reparse url.
398 m_useProxy = true;
399 ParseURL();
400 }
401 }
402 #endif // wxUSE_PROTOCOL_HTTP
403
404 // ----------------------------------------------------------------------
405 // wxURLModule
406 //
407 // A module which deletes the default proxy if we created it
408 // ----------------------------------------------------------------------
409
410 #if wxUSE_SOCKETS
411
412 class wxURLModule : public wxModule
413 {
414 public:
415 wxURLModule();
416
417 virtual bool OnInit();
418 virtual void OnExit();
419
420 private:
421 DECLARE_DYNAMIC_CLASS(wxURLModule)
422 };
423
424 IMPLEMENT_DYNAMIC_CLASS(wxURLModule, wxModule)
425
426 wxURLModule::wxURLModule()
427 {
428 // we must be cleaned up before wxSocketModule as otherwise deleting
429 // ms_proxyDefault from our OnExit() won't work (and can actually crash)
430 AddDependency(wxClassInfo::FindClass(_T("wxSocketModule")));
431 }
432
433 bool wxURLModule::OnInit()
434 {
435 #if wxUSE_PROTOCOL_HTTP
436 // env var HTTP_PROXY contains the address of the default proxy to use if
437 // set, but don't try to create this proxy right now because it will slow
438 // down the program startup (especially if there is no DNS server
439 // available, in which case it may take up to 1 minute)
440
441 if ( wxGetenv(_T("HTTP_PROXY")) )
442 {
443 wxURL::ms_useDefaultProxy = true;
444 }
445 #endif // wxUSE_PROTOCOL_HTTP
446 return true;
447 }
448
449 void wxURLModule::OnExit()
450 {
451 #if wxUSE_PROTOCOL_HTTP
452 delete wxURL::ms_proxyDefault;
453 wxURL::ms_proxyDefault = NULL;
454 #endif // wxUSE_PROTOCOL_HTTP
455 }
456
457 #endif // wxUSE_SOCKETS
458
459
460 #endif // wxUSE_URL