1 // -*- c++ -*- ///////////////////////////////////////////////////////////////
2 // Name: unix/dialup.cpp
3 // Purpose: Network related wxWindows classes and functions
4 // Author: Karsten Ballüder
8 // Copyright: (c) Karsten Ballüder
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
15 # pragma implementation "dialup.h"
18 #if wxUSE_DIALUP_MANAGER
24 #include "wx/string.h"
26 #include "wx/dialup.h"
28 #include "wx/filefn.h"
32 #include "wx/process.h"
35 #include "wx/wxchar.h"
42 #define __STRICT_ANSI__
43 #include <sys/socket.h>
45 #include <netinet/in.h>
46 #include <arpa/inet.h>
48 // ----------------------------------------------------------------------------
49 // A class which groups functions dealing with connecting to the network from a
50 // workstation using dial-up access to the net. There is at most one instance
51 // of this class in the program accessed via GetDialUpManager().
52 // ----------------------------------------------------------------------------
56 * 1. more configurability for Unix: i.e. how to initiate the connection, how
57 * to check for online status, &c.
58 * 2. add a "long Dial(long connectionId = -1)" function which asks the user
59 * about which connection to dial (this may be done using native dialogs
60 * under NT, need generic dialogs for all others) and returns the identifier
61 * of the selected connection (it's opaque to the application) - it may be
62 * reused later to dial the same connection later (or use strings instead of
64 * 3. add an async version of dialing functions which notify the caller about
65 * the progress (or may be even start another thread to monitor it)
66 * 4. the static creation/accessor functions are not MT-safe - but is this
67 * really crucial? I think we may suppose they're always called from the
71 class WXDLLEXPORT wxDialUpManagerImpl
: public wxDialUpManager
74 wxDialUpManagerImpl();
75 ~wxDialUpManagerImpl();
77 /** Could the dialup manager be initialized correctly? If this function
78 returns FALSE, no other functions will work neither, so it's a good idea
79 to call this function and check its result before calling any other
80 wxDialUpManager methods.
82 virtual bool IsOk() const
85 /** The simplest way to initiate a dial up: this function dials the given
86 ISP (exact meaning of the parameter depends on the platform), returns
87 TRUE on success or FALSE on failure and logs the appropriate error
88 message in the latter case.
89 @param nameOfISP optional paramater for dial program
90 @param username unused
91 @param password unused
93 virtual bool Dial(const wxString
& nameOfISP
,
94 const wxString
& WXUNUSED(username
),
95 const wxString
& WXUNUSED(password
),
98 /// Hang up the currently active dial up connection.
99 virtual bool HangUp();
101 // returns TRUE if the computer is connected to the network: under Windows,
102 // this just means that a RAS connection exists, under Unix we check that
103 // the "well-known host" (as specified by SetWellKnownHost) is reachable
104 virtual bool IsOnline() const
106 if( (! m_timer
) // we are not polling, so test now:
110 return m_IsOnline
!= 0;
113 /// do we have a constant net connection? -- GUESS!
114 bool IsAlwaysOnline() const
116 ((wxDialUpManagerImpl
*) this)->HangUp(); // brutal but necessary
119 /// returns TRUE if (async) dialing is in progress
120 inline virtual bool IsDialing() const
121 { return m_DialProcess
!= NULL
; }
123 // cancel dialing the number initiated with Dial(async = TRUE)
124 // NB: this won't result in DISCONNECTED event being sent
125 virtual bool CancelDialing();
127 unsigned int GetISPNames(class wxArrayString
&) const
130 // sometimes the built-in logic for determining the online status may fail,
131 // so, in general, the user should be allowed to override it. This function
132 // allows to forcefully set the online status - whatever our internal
133 // algorithm may think about it.
134 virtual void SetOnlineStatus(bool isOnline
= TRUE
)
135 { m_IsOnline
= isOnline
; }
137 // set misc wxDialUpManager options
138 // --------------------------------
140 // enable automatical checks for the connection status and sending of
141 // wxEVT_DIALUP_CONNECTED/wxEVT_DIALUP_DISCONNECTED events. The interval
142 // parameter is only for Unix where we do the check manually: under
143 // Windows, the notification about the change of connection status is
146 // Returns FALSE if couldn't set up automatic check for online status.
147 virtual bool EnableAutoCheckOnlineStatus(size_t nSeconds
);
149 // disable automatic check for connection status change - notice that the
150 // wxEVT_DIALUP_XXX events won't be sent any more neither.
151 virtual void DisableAutoCheckOnlineStatus();
153 // under Unix, the value of well-known host is used to check whether we're
154 // connected to the internet. It's unused under Windows, but this function
155 // is always safe to call. The default value is www.yahoo.com.
156 virtual void SetWellKnownHost(const wxString
& hostname
,
158 /** Sets the commands to start up the network and to hang up
159 again. Used by the Unix implementations only.
161 virtual void SetConnectCommand(const wxString
&command
, const wxString
&hupcmd
)
162 { m_ConnectCommand
= command
; m_HangUpCommand
= hupcmd
; }
165 /// -1: don´t know, 0 = no, 1 = yes
168 /// Can we use ifconfig to list active devices?
169 int m_CanUseIfconfig
;
170 /// The path to ifconfig
171 wxString m_IfconfigPath
;
174 wxString m_BeaconHost
;
175 /// beacon host portnumber for connect:
178 /// command to connect to network
179 wxString m_ConnectCommand
;
180 /// command to hang up
181 wxString m_HangUpCommand
;
184 /// a timer for regular testing
185 class AutoCheckTimer
*m_timer
;
186 friend class AutoCheckTimer
;
188 /// a wxProcess for dialling in background
189 class wxDialProcess
*m_DialProcess
;
190 /// pid of dial process
192 friend class wxDialProcess
;
195 void CheckStatus(bool fromAsync
= FALSE
) const;
197 /// real status check
198 void CheckStatusInternal(void);
202 class AutoCheckTimer
: public wxTimer
205 AutoCheckTimer(wxDialUpManagerImpl
*dupman
)
211 virtual bool Start( int millisecs
= -1 )
212 { m_started
= TRUE
; return wxTimer::Start(millisecs
, FALSE
); }
214 virtual void Notify()
215 { wxLogTrace("Checking dial up network status."); m_dupman
->CheckStatus(); }
218 { if ( m_started
) wxTimer::Stop(); }
221 wxDialUpManagerImpl
*m_dupman
;
224 class wxDialProcess
: public wxProcess
227 wxDialProcess(wxDialUpManagerImpl
*dupman
)
231 void Disconnect(void) { m_DupMan
= NULL
; }
232 void OnTerminate(int WXUNUSED(pid
), int WXUNUSED(status
)) const
236 m_DupMan
->m_DialProcess
= NULL
;
237 m_DupMan
->CheckStatus(TRUE
);
241 wxDialUpManagerImpl
*m_DupMan
;
245 wxDialUpManagerImpl::wxDialUpManagerImpl()
247 m_IsOnline
= -2; // -1 or -2, unknown
248 m_DialProcess
= NULL
;
250 m_CanUseIfconfig
= -1; // unknown
251 m_BeaconHost
= WXDIALUP_MANAGER_DEFAULT_BEACONHOST
;
253 SetConnectCommand("pon", "poff"); // default values for Debian/GNU linux
254 wxChar
* dial
= wxGetenv(_T("WXDIALUP_DIALCMD"));
255 wxChar
* hup
= wxGetenv(_T("WXDIALUP_HUPCMD"));
257 SetConnectCommand(dial
? wxString(dial
) : m_ConnectCommand
,
258 hup
? wxString(hup
) : m_HangUpCommand
);
261 wxDialUpManagerImpl::~wxDialUpManagerImpl()
263 if(m_timer
) delete m_timer
;
266 m_DialProcess
->Disconnect();
267 m_DialProcess
->Detach();
272 wxDialUpManagerImpl::Dial(const wxString
&isp
,
273 const wxString
& WXUNUSED(username
),
274 const wxString
& WXUNUSED(password
),
282 if(m_ConnectCommand
.Find("%s"))
283 cmd
.Printf(m_ConnectCommand
,m_ISPname
.c_str());
285 cmd
= m_ConnectCommand
;
289 m_DialProcess
= new wxDialProcess(this);
290 m_DialPId
= wxExecute(cmd
, FALSE
, m_DialProcess
);
293 delete m_DialProcess
;
294 m_DialProcess
= NULL
;
301 return wxExecute(cmd
, /* sync */ TRUE
) == 0;
305 wxDialUpManagerImpl::HangUp(void)
311 wxLogError(_("Already dialling ISP."));
316 if(m_HangUpCommand
.Find("%s"))
317 cmd
.Printf(m_HangUpCommand
,m_ISPname
.c_str(), m_DialProcess
);
319 cmd
= m_HangUpCommand
;
320 return wxExecute(cmd
, /* sync */ TRUE
) == 0;
325 wxDialUpManagerImpl::CancelDialing()
329 return kill(m_DialPId
, SIGTERM
) > 0;
333 wxDialUpManagerImpl::EnableAutoCheckOnlineStatus(size_t nSeconds
)
335 wxASSERT(m_timer
== NULL
);
336 m_timer
= new AutoCheckTimer(this);
337 bool rc
= m_timer
->Start(nSeconds
*1000);
347 wxDialUpManagerImpl::DisableAutoCheckOnlineStatus()
349 wxASSERT(m_timer
!= NULL
);
357 wxDialUpManagerImpl::SetWellKnownHost(const wxString
& hostname
, int portno
)
359 /// does hostname contain a port number?
360 wxString port
= hostname
.After(':');
363 m_BeaconHost
= hostname
.Before(':');
364 m_BeaconPort
= atoi(port
);
368 m_BeaconHost
= hostname
;
369 m_BeaconPort
= portno
;
375 wxDialUpManagerImpl::CheckStatus(bool fromAsync
) const
377 // This function calls the CheckStatusInternal() helper function
378 // which is OS - specific and then sends the events.
380 int oldIsOnline
= m_IsOnline
;
381 ( /* non-const */ (wxDialUpManagerImpl
*)this)->CheckStatusInternal();
383 // now send the events as appropriate:
384 if(m_IsOnline
!= oldIsOnline
&& m_IsOnline
!= -1 && oldIsOnline
!= -2) // -2: first time!
386 wxDialUpEvent
event(m_IsOnline
, ! fromAsync
);
387 (void)wxTheApp
->ProcessEvent(event
);
392 We have three methods that we can use:
394 1. test via /sbin/ifconfig and grep for "sl", "ppp", "pl"
395 --> should be fast enough for regular polling
396 2. test if we can reach the well known beacon host
397 --> too slow for polling
398 3. check /proc/net/dev on linux??
399 This method should be preferred, if possible. Need to do more
405 wxDialUpManagerImpl::CheckStatusInternal(void)
409 // First time check for ifconfig location. We only use the variant
410 // which does not take arguments, a la GNU.
411 if(m_CanUseIfconfig
== -1) // unknown
413 if(wxFileExists("/sbin/ifconfig"))
414 m_IfconfigPath
= "/sbin/ifconfig";
415 else if(wxFileExists("/usr/sbin/ifconfig"))
416 m_IfconfigPath
= "/usr/sbin/ifconfig";
419 wxLogNull ln
; // suppress all error messages
420 // Let´s try the ifconfig method first, should be fastest:
421 if(m_CanUseIfconfig
!= 0) // unknown or yes
423 wxASSERT(m_IfconfigPath
.length());
425 wxString tmpfile
= wxGetTempFileName("_wxdialuptest");
426 wxString cmd
= "/bin/sh -c \'";
427 cmd
<< m_IfconfigPath
<< " >" << tmpfile
<< '\'';
428 /* I tried to add an option to wxExecute() to not close stdout,
429 so we could let ifconfig write directly to the tmpfile, but
430 this does not work. That should be faster, as it doesn´t call
431 the shell first. I have no idea why. :-( (KB) */
433 // temporarily redirect stdout/stderr:
435 new_stdout
= dup(STDOUT_FILENO
),
436 new_stderr
= dup(STDERR_FILENO
);
437 close(STDOUT_FILENO
);
438 close(STDERR_FILENO
);
442 output_fd
= open(tmpfile
, O_CREAT
|O_TRUNC
, S_IRUSR
|S_IWUSR
),
444 null_fd
= open("/dev/null", O_CREAT
, S_IRUSR
|S_IWUSR
);
445 // verify well behaved unix behaviour:
446 wxASSERT(output_fd
== STDOUT_FILENO
);
447 wxASSERT(null_fd
== STDERR_FILENO
);
448 int rc
= wxExecute(m_IfconfigPath
,TRUE
/* sync */,NULL
,wxEXECUTE_DONT_CLOSE_FDS
);
449 close(null_fd
); close(output_fd
);
450 // restore old stdout, stderr:
452 test
= dup(new_stdout
); close(new_stdout
); wxASSERT(test
== STDOUT_FILENO
);
453 test
= dup(new_stderr
); close(new_stderr
); wxASSERT(test
== STDERR_FILENO
);
456 if(wxExecute(cmd
,TRUE
/* sync */) == 0)
458 m_CanUseIfconfig
= 1;
460 if( file
.Open(tmpfile
) )
462 char *output
= new char [file
.Length()+1];
463 output
[file
.Length()] = '\0';
464 if(file
.Read(output
,file
.Length()) == file
.Length())
466 if(strstr(output
,"ppp") // ppp
467 || strstr(output
,"sl") // slip
468 || strstr(output
,"pl") // plip
477 // else m_IsOnline remains -1 as we don't know for sure
479 else // could not run ifconfig correctly
480 m_CanUseIfconfig
= 0; // don´t try again
481 (void) wxRemoveFile(tmpfile
);
482 if(m_IsOnline
!= -1) // we are done
486 // second method: try to connect to well known host:
487 // This can be used under Win 9x, too!
489 struct sockaddr_in serv_addr
;
491 m_IsOnline
= 0; // assume false
492 if((hp
= gethostbyname(m_BeaconHost
)) == NULL
)
493 return; // no DNS no net
495 serv_addr
.sin_family
= hp
->h_addrtype
;
496 memcpy(&serv_addr
.sin_addr
,hp
->h_addr
, hp
->h_length
);
497 serv_addr
.sin_port
= htons(m_BeaconPort
);
502 if( ( sockfd
= socket(hp
->h_addrtype
, SOCK_STREAM
, 0)) < 0)
504 // sys_error("cannot create socket for gw");
508 if(sendto(sockfd
, "hello",
509 strlen("hello"), /* flags */ 0,
510 (struct sockaddr
*) &serv_addr
,
511 sizeof(serv_addr
)) == -1)
518 if( connect(sockfd
, (struct sockaddr
*) &serv_addr
, sizeof(serv_addr
)) < 0)
520 //sys_error("cannot connect to server");
532 wxDialUpManager::Create(void)
534 return new wxDialUpManagerImpl
;
537 #endif // wxUSE_DIALUP_MANAGER