MagAO-X
Operations Applications Utilities Source
telnetConn.hpp
Go to the documentation of this file.
1 /** \file telnetConn.hpp
2  * \brief Managing a connection to a telnet device.
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * \ingroup tty_files
6  * History:
7  * - 2018-08-24 created by JRM
8  */
9 #ifndef telnet_telnetConn_hpp
10 #define telnet_telnetConn_hpp
11 
12 
13 /* Much of the code in this file was taken from telnet-client.c in
14  * libtelnet (https://github.com/seanmiddleditch/libtelnet), with modifications for our needs.
15  *
16  * That code was placed in the public domain:
17  *
18  * libtelnet - TELNET protocol handling library
19  *
20  * Sean Middleditch
21  * sean@sourcemud.org
22  *
23  * The author or authors of [the libtelnet] code dedicate any and all copyright interest
24  * in [the libtelnet] to the public domain. We make this dedication for the benefit
25  * of the public at large and to the detriment of our heirs and successors. We
26  * intend this dedication to be an overt act of relinquishment in perpetuity of
27  * all present and future rights to this code under copyright law.
28  */
29 
30 
31 #include <mx/timeUtils.hpp>
32 
33 #include "../../libs/libtelnet/libtelnet.h"
34 #include "ttyErrors.hpp"
35 #include "ttyIOUtils.hpp"
36 
37 namespace MagAOX
38 {
39 namespace tty
40 {
41 
42 #ifndef TELNET_BUFFSIZE
43  #define TELNET_BUFFSIZE (1024)
44 #endif
45 
46 /// libtelnet option table.
47 /** \ingroup tty
48  */
49 static const telnet_telopt_t telopts[] = {
50  { TELNET_TELOPT_ECHO, TELNET_WONT, TELNET_DO },
51  { TELNET_TELOPT_TTYPE, TELNET_WILL, TELNET_DONT },
52  { TELNET_TELOPT_COMPRESS2, TELNET_WONT, TELNET_DO },
53  { TELNET_TELOPT_MSSP, TELNET_WONT, TELNET_DO },
54  { -1, 0, 0 } };
55 
56 #define TELNET_WAITING_USER (0)
57 #define TELNET_GOT_USER (1)
58 #define TELNET_WAITING_PASS (2)
59 #define TELNET_GOT_PASS (3)
60 #define TELNET_WAITING_PROMPT (4)
61 #define TELNET_LOGGED_IN (5)
62 
63 /// A Telnet connection manager, wrapping \p libtelnet.
64 /**
65  * Establishes the connection to the server, and initializes the
66  * \p libtelnet structure, including registering the event handler callback.
67  *
68  * Errors encountered during telnet event handling are indicated by an internal flag,
69  * which must be checked each time a libtelnet function is called. If it is nonzero an
70  * error has occurred.
71  *
72  * Responses from the server are accumulated in the \p m_strRead member. It is typically
73  * cleared before reading, but this can be suppressed when desired.
74  *
75  * Because of the way event handling is managed, and the class-global error and response accumulation
76  * this is not thread-safe. Any calls to this class methods should be mutex-ed.
77  *
78  * \ingroup tty
79  */
80 struct telnetConn
81 {
82  int m_sock {0}; ///< The socket file descriptor.
83 
84  telnet_t * m_telnet {nullptr}; ///< libtelnet telnet_t structure
85 
86  ///The device's username entry prompt, used for managing login.
87  std::string m_usernamePrompt {"Username:"};
88 
89  ///The device's password entry prompt, used for managing login.
90  std::string m_passwordPrompt {"Password:"};
91 
92  std::string m_prompt {"$> "}; ///< The device's prompt, used for detecting end of transmission.
93 
94  ///Flag denoting the login state.
95  /** Used to manage different behaviors in the libtelnet event handler.
96  *
97  * - TELNET_WAITING_USER: waiting on m_usernamePrompt
98  * - TELNET_GOT_USER: got m_usernamePrompt
99  * - TELNET_WAITING_PASS: waiting on m_passwordPrompt
100  * - TELNET_GOT_PASS: got m_passwordPrompt
101  * - TELNET_WAITING_PROMPT: waiting on m_prompt
102  * - TELNET_LOGGED_IN: logged in
103  */
104  int m_loggedin {0};
105 
106  /// Used to indicate an error occurred in the event handler callback.
107  int m_EHError {0};
108 
109  /// The accumulated string read from the device.
110  /** This needs to be clear()-ed when expecting a new response to start.
111  * \warning This makes telnetConn NOT threadsafe.
112  */
113  std::string m_strRead;
114 
115  /// D'tor, conducts connection cleanup.
116  ~telnetConn();
117 
118  /// Connect to the device
119  int connect( const std::string & host, ///< [in] The host specification (i.p. address)
120  const std::string & port ///< [in] the port on the host.
121  );
122 
123  /// Manage the login process on this device.
124  int login( const std::string & username, /// [in] The username
125  const std::string & password /// [in] The password.
126  );
127 
128  /// Set flags as if we're logged in, used when device doesn't require it.
129  int noLogin();
130 
131  /// Write to a telnet connection
132  /**
133  *
134  * \returns TTY_E_NOERROR on success
135  * \returns TTY_E_TIMEOUTONWRITEPOLL if the poll times out.
136  * \returns TTY_E_ERRORONWRITEPOLL if an error is returned by poll.
137  * \returns TTY_E_TIMEOUTONWRITE if a timeout occurs during the write.
138  * \returns TTY_E_ERRORONWRITE if an error occurs writing to the file.
139  */
140  int write( const std::string & buffWrite, ///< [in] The characters to write to the telnet.
141  int timeoutWrite ///< [in] The timeout in milliseconds.
142  );
143 
144  /// Read from a telnet connection, until end-of-transmission string is read.
145  /**
146  * \returns TTY_E_NOERROR on success
147  * \returns TTY_E_TIMEOUTONREADPOLL if the poll times out.
148  * \returns TTY_E_ERRORONREADPOLL if an error is returned by poll.
149  * \returns TTY_E_TIMEOUTONREAD if a timeout occurs during the read.
150  * \returns TTY_E_ERRORONREAD if an error occurs reading from the file.
151  */
152  int read( const std::string & eot, ///< [in] the end-of-transmission indicator
153  int timeoutRead, ///< [in] The timeout in milliseconds.
154  bool clear=true ///< [in] [optional] whether or not to clear the strRead buffer
155  );
156 
157  /// Read from a telnet connection, until m_prompt is read.
158  /**
159  * \returns TTY_E_NOERROR on success
160  * \returns TTY_E_TIMEOUTONREADPOLL if the poll times out.
161  * \returns TTY_E_ERRORONREADPOLL if an error is returned by poll.
162  * \returns TTY_E_TIMEOUTONREAD if a timeout occurs during the read.
163  * \returns TTY_E_ERRORONREAD if an error occurs reading from the file.
164  */
165  int read( int timeoutRead, ///< [in] The timeout in milliseconds.
166  bool clear=true ///< [in] [optional] whether or not to clear the strRead buffer
167  );
168 
169  /// Write to a telnet connection, then get the reply.
170  /** The read is conducted until the m_prompt string is received.
171  * Echo characters are swallowed if desired.
172  *
173  * \returns TTY_E_NOERROR on success
174  * \returns TTY_E_TIMEOUTONWRITEPOLL if the poll times out.
175  * \returns TTY_E_ERRORONWRITEPOLL if an error is returned by poll.
176  * \returns TTY_E_TIMEOUTONWRITE if a timeout occurs during the write.
177  * \returns TTY_E_ERRORONWRITE if an error occurs writing to the file.
178  * \returns TTY_E_TIMEOUTONREADPOLL if the poll times out.
179  * \returns TTY_E_ERRORONREADPOLL if an error is returned by poll.
180  * \returns TTY_E_TIMEOUTONREAD if a timeout occurs during the read.
181  * \returns TTY_E_ERRORONREAD if an error occurs reading from the file.
182  */
183  int writeRead( const std::string & strWrite, ///< [in] The characters to write to the telnet.
184  bool swallowEcho, ///< [in] If true, strWrite.size() characters are read after the write
185  int timeoutWrite, ///< [in] The write timeout in milliseconds.
186  int timeoutRead ///< [in] The read timeout in milliseconds.
187  );
188 
189  /// Internal send for use by event_handler.
190  static int send( int sock,
191  const char *buffer,
192  size_t size
193  );
194 
195  /// Event handler callback for libtelnet processing.
196  /** Resets the internal m_EHError value to TTY_E_NOERROR on entry.
197  * Will set it to an error flag if an error is encountered, so this
198  * flag should be checked after any call to a libtelnet function.
199  * \warning this makes telnetConn not thread safe
200  */
201  static void event_handler( telnet_t *telnet,
202  telnet_event_t *ev,
203  void *user_data
204  );
205 };
206 
207 inline
209 {
210  /* clean up */
211  if(m_telnet) telnet_free(m_telnet);
212  if(m_sock) close(m_sock);
213 }
214 
215 inline
216 int telnetConn::connect( const std::string & host,
217  const std::string & port
218  )
219 {
220  //First cleanup any previous connections.
221  if(m_telnet) telnet_free(m_telnet);
222  m_telnet = 0;
223  if(m_sock) close(m_sock);
224  m_sock = 0;
225 
226  int rs;
227 
228  struct sockaddr_in addr;
229  struct addrinfo *ai;
230  struct addrinfo hints;
231 
232  /* look up server host */
233  memset(&hints, 0, sizeof(hints));
234  hints.ai_family = AF_UNSPEC;
235  hints.ai_socktype = SOCK_STREAM;
236  if ((rs = getaddrinfo(host.c_str(), port.c_str(), &hints, &ai)) != 0)
237  {
238  fprintf(stderr, "getaddrinfo() failed for %s: %s\n", host.c_str(),
239  gai_strerror(rs));
240  return TELNET_E_GETADDR;
241  }
242 
243  /* create server m_socket */
244  if ((m_sock = socket(AF_INET, SOCK_STREAM, 0)) == -1)
245  {
246  fprintf(stderr, "socket() failed: %s\n", strerror(errno));
247  return TELNET_E_SOCKET;
248  }
249 
250  /* bind server socket */
251  memset(&addr, 0, sizeof(addr));
252  addr.sin_family = AF_INET;
253  if (bind(m_sock, (struct sockaddr *)&addr, sizeof(addr)) == -1)
254  {
255  fprintf(stderr, "bind() failed: %s\n", strerror(errno));
256  return TELNET_E_BIND;
257  }
258 
259  /* connect */
260  if (::connect(m_sock, ai->ai_addr, ai->ai_addrlen) == -1)
261  {
262  fprintf(stderr, "connect() failed: %s\n", strerror(errno));
263  return TELNET_E_CONNECT;
264  }
265 
266  /* free address lookup info */
267  freeaddrinfo(ai);
268 
269  /* initialize the telnet box */
270  m_telnet = telnet_init(telopts, telnetConn::event_handler, 0, this);
271 
272  if(m_telnet == nullptr)
273  {
274  fprintf(stderr, "error initializing telnet");
275  return TELNET_E_TELNETINIT;
276  }
277 
278  return TTY_E_NOERROR;
279 }
280 
281 inline
282 int telnetConn::login( const std::string & username,
283  const std::string & password
284  )
285 {
286  char buffer[512];
287  int rs;
288 
289  struct pollfd pfd[1];
290 
291  /* initialize poll descriptors */
292  memset(pfd, 0, sizeof(pfd));
293  pfd[0].fd = m_sock;
294  pfd[0].events = POLLIN;
295 
296  //Loop while waiting on the login process to complete.
297  while (poll(pfd, 1, -1) != -1)
298  {
299  /* read from client */
300  if (pfd[0].revents & POLLIN)
301  {
302  if ((rs = recv(m_sock, buffer, sizeof(buffer), 0)) > 0)
303  {
304  telnet_recv(m_telnet, buffer, rs);
305  if(m_EHError != TTY_E_NOERROR) return m_EHError;
306  }
307  else if (rs == 0)
308  {
309  break;
310  }
311  else
312  {
313  fprintf(stderr, "recv(client) failed: %s\n",
314  strerror(errno));
315  return TTY_E_ERRORONREAD;
316  }
317  }
318 
320  {
321  int rv = write(username + "\n", 1000);
322  if(rv != TTY_E_NOERROR) return rv;
323 
325  }
326 
328  {
329  int rv = write(password + "\n", 1000);
330  if(rv != TTY_E_NOERROR) return rv;
331 
333  }
334 
336  {
337  break;
338  }
339  }
340 
341  return TTY_E_NOERROR;
342 }
343 
344 inline
346 {
348  return TTY_E_NOERROR;
349 }
350 
351 inline
352 int telnetConn::write( const std::string & buffWrite,
353  int timeoutWrite
354  )
355 {
356  double t0;
357  struct pollfd pfd;
358 
359  errno = 0;
360  pfd.fd = m_sock;
361  pfd.events = POLLOUT;
362 
363  std::string _buffWrite;
364  telnetCRLF(_buffWrite, buffWrite);
365 
366 
367  t0 = mx::get_curr_time();
368 
369  size_t totWritten = 0;
370  while( totWritten < _buffWrite.size())
371  {
372  int timeoutCurrent = timeoutWrite - (mx::get_curr_time()-t0)*1000;
373  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONWRITE;
374 
375  int rv = poll( &pfd, 1, timeoutCurrent);
376  if( rv == 0 ) return TTY_E_TIMEOUTONWRITEPOLL;
377  else if( rv < 0 ) return TTY_E_ERRORONWRITEPOLL;
378 
379  telnet_send(m_telnet, _buffWrite.c_str(), _buffWrite.size());
380  totWritten = _buffWrite.size();
381 
382  #ifdef TELNET_DEBUG
383  std::cerr << "Wrote " << totWritten << " chars of " << buffWrite.size() << "\n";
384  #endif
385 
386 
387  if( ( mx::get_curr_time()-t0)*1000 > timeoutWrite ) return TTY_E_TIMEOUTONWRITE;
388  }
389 
390  return TTY_E_NOERROR;
391 }
392 
393 inline
394 int telnetConn::read( const std::string & eot,
395  int timeoutRead,
396  bool clear
397  )
398 {
399  int rv;
400  int timeoutCurrent;
401  double t0;
402 
403  struct pollfd pfd;
404 
405  errno = 0;
406 
407  pfd.fd = m_sock;
408  pfd.events = POLLIN;
409 
410  char buffRead[TELNET_BUFFSIZE];
411 
412  //Start timeout clock for reading.
413  t0 = mx::get_curr_time();
414  timeoutCurrent = timeoutRead;
415 
416  //Now read the response up to the eot.
417  if(clear) m_strRead = "";
418 
419 
420  rv = poll( &pfd, 1, timeoutCurrent);
421  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
422  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
423 
424  rv = ::read(m_sock, buffRead, TELNET_BUFFSIZE);
425  if( rv < 0 ) return TTY_E_ERRORONREAD;
426  buffRead[rv] = '\0';
427 
428  telnet_recv(m_telnet, buffRead, rv);
429  if(m_EHError != TTY_E_NOERROR) return m_EHError;
430 
431  while( !isEndOfTrans(m_strRead, eot) )
432  {
433  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
434  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
435 
436  rv = poll( &pfd, 1, timeoutCurrent);
437  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
438  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
439 
440  rv = ::read(m_sock, buffRead, TELNET_BUFFSIZE);
441  if( rv < 0 ) return TTY_E_ERRORONREAD;
442  buffRead[rv] ='\0';
443 
444  telnet_recv(m_telnet, buffRead, rv);
445  if(m_EHError != TTY_E_NOERROR) return m_EHError;
446 
447  #ifdef TELNET_DEBUG
448  std::cerr << "telnetRead: read " << rv << " bytes. buffRead=" << buffRead << "\n";
449  #endif
450  }
451 
452 
453  return TTY_E_NOERROR;
454 
455 
456 }
457 
458 inline
459 int telnetConn::read( int timeoutRead,
460  bool clear
461  )
462 {
463  return read(m_prompt, timeoutRead, clear);
464 }
465 
466 inline
467 int telnetConn::writeRead( const std::string & strWrite,
468  bool swallowEcho,
469  int timeoutWrite,
470  int timeoutRead
471  )
472 {
473  m_strRead.clear();
474 
475  int rv;
476 
477  //Write First
478  rv = write( strWrite, timeoutWrite);
479  if(rv != TTY_E_NOERROR) return rv;
480 
481  //Now read response from console
482  int timeoutCurrent;
483  double t0;
484 
485  struct pollfd pfd;
486  pfd.fd = m_sock;
487  pfd.events = POLLIN;
488 
489 
490  //Start timeout clock for reading.
491  t0 = mx::get_curr_time();;
492 
493  if(swallowEcho)
494  {
495  char buffRead[TELNET_BUFFSIZE];
496 
497  //First swallow the echo.
498  while( m_strRead.size() <= strWrite.size() )
499  {
500  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
501  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
502 
503  rv = poll( &pfd, 1, timeoutCurrent);
504  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
505  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
506 
507  rv = ::read(m_sock, buffRead, TELNET_BUFFSIZE);
508  if( rv < 0 ) return TTY_E_ERRORONREAD;
509 
510  telnet_recv(m_telnet, buffRead, rv);
511  if(m_EHError != TTY_E_NOERROR) return m_EHError;
512  }
513 
514  m_strRead.erase(0, strWrite.size());
515  }
516 
518 
519  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
520  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
521 
522  //Now read the response up to the eot.
523  return read(timeoutCurrent, false);
524 }
525 
526 inline
527 int telnetConn::send(int sock, const char *buffer, size_t size)
528 {
529  /* send data */
530  while (size > 0)
531  {
532  int rs;
533 
534  if ((rs = ::send(sock, buffer, size, 0)) == -1)
535  {
536  fprintf(stderr, "send() failed: %s\n", strerror(errno));
537  return TTY_E_ERRORONWRITE;
538  }
539  else if (rs == 0)
540  {
541  fprintf(stderr, "send() unexpectedly returned 0\n");
542  return TTY_E_ERRORONWRITE;
543  }
544  /* update pointer and size to see if we've got more to send */
545  buffer += rs;
546  size -= rs;
547  }
548 
549  return TTY_E_NOERROR;
550 }
551 
552 inline
553 void telnetConn::event_handler( telnet_t *telnet,
554  telnet_event_t *ev,
555  void *user_data
556  )
557 {
558  telnetConn * cs = static_cast<telnetConn*>(user_data);
559  int sock = cs->m_sock;
560 
561  //Always reset the error at beginning.
562  cs->m_EHError = 0;
563 
564  switch (ev->type)
565  {
566  /* data received */
567  case TELNET_EV_DATA:
568  {
569  //First we remove the various control chars from the front.
570  if(ev->data.size == 0) break;
571 
572  char * buf = const_cast<char *>(ev->data.buffer);
573  buf[ev->data.size] = 0;
574  for(size_t i=0; i<ev->data.size; ++i)
575  {
576  if(ev->data.buffer[i] < 20)
577  {
578  ++buf;
579  continue;
580  }
581  break;
582  }
583 
584  //Now make it a string so we can make use of it.
585  std::string sbuf(buf);
586 
587  if(sbuf.size() == 0) break;
588 
589  if(cs->m_loggedin < TELNET_LOGGED_IN) //we aren't logged in yet
590  {
592  {
593  if(sbuf.find(cs->m_usernamePrompt) != std::string::npos)
594  {
596  }
597  break;
598  }
599 
601  {
602  if( sbuf.find(cs->m_passwordPrompt) != std::string::npos)
603  {
605  }
606  break;
607  }
608 
610  {
611  if( sbuf.find(cs->m_prompt) != std::string::npos)
612  {
614  }
615  break;
616  }
617  }
618 
619  //Always append
620  cs->m_strRead += sbuf;
621  break;
622  }
623  /* data must be sent */
624  case TELNET_EV_SEND:
625  {
626  send(sock, ev->data.buffer, ev->data.size);
627  break;
628  }
629  /* request to enable remote feature (or receipt) */
630  case TELNET_EV_WILL:
631  {
632  /* we'll agree to turn off our echo if server wants us to stop */
633  //if (ev->neg.telopt == TELNET_TELOPT_ECHO) do_echo = 0;
634  break;
635  }
636  /* notification of disabling remote feature (or receipt) */
637  case TELNET_EV_WONT:
638  {
639  break;
640  }
641  /* request to enable local feature (or receipt) */
642  case TELNET_EV_DO:
643  {
644  break;
645  }
646  /* demand to disable local feature (or receipt) */
647  case TELNET_EV_DONT:
648  {
649  break;
650  }
651  /* respond to TTYPE commands */
652  case TELNET_EV_TTYPE:
653  {
654  /* respond with our terminal type, if requested */
655  if (ev->ttype.cmd == TELNET_TTYPE_SEND)
656  {
657  telnet_ttype_is(telnet, getenv("TERM"));
658  }
659  break;
660  }
661  /* respond to particular subnegotiations */
662  case TELNET_EV_SUBNEGOTIATION:
663  {
664  break;
665  }
666  /* error */
667  case TELNET_EV_ERROR:
668  {
669  fprintf(stderr, "ERROR: %s\n", ev->error.msg);
671  break;
672  }
673  default:
674  {
675  /* ignore */
676  break;
677  }
678  }
679 }
680 
681 } //namespace tty
682 } //namespace MagAOX
683 
684 #endif //telnet_telnetConn_hpp
~telnetConn()
D&#39;tor, conducts connection cleanup.
Definition: telnetConn.hpp:208
int m_loggedin
Flag denoting the login state.
Definition: telnetConn.hpp:104
static const telnet_telopt_t telopts[]
libtelnet option table.
Definition: telnetConn.hpp:49
#define TELNET_E_BIND
Definition: ttyErrors.hpp:33
#define TELNET_BUFFSIZE
Definition: telnetConn.hpp:43
#define TTY_E_ERRORONREAD
Definition: ttyErrors.hpp:24
std::string m_strRead
The accumulated string read from the device.
Definition: telnetConn.hpp:113
int connect(const std::string &host, const std::string &port)
Connect to the device.
Definition: telnetConn.hpp:216
#define TELNET_WAITING_PASS
Definition: telnetConn.hpp:58
#define TTY_E_ERRORONREADPOLL
Definition: ttyErrors.hpp:23
int read(const std::string &eot, int timeoutRead, bool clear=true)
Read from a telnet connection, until end-of-transmission string is read.
Definition: telnetConn.hpp:394
std::string m_usernamePrompt
The device&#39;s username entry prompt, used for managing login.
Definition: telnetConn.hpp:87
#define TELNET_E_GETADDR
Definition: ttyErrors.hpp:31
A Telnet connection manager, wrapping libtelnet.
Definition: telnetConn.hpp:80
Error numbers for the tty utilities.
#define TTY_E_ERRORONWRITE
Definition: ttyErrors.hpp:20
#define TELNET_WAITING_USER
Definition: telnetConn.hpp:56
#define TELNET_E_TELNETINIT
Definition: ttyErrors.hpp:35
#define TELNET_E_EHERROR
Definition: ttyErrors.hpp:36
static int send(int sock, const char *buffer, size_t size)
Internal send for use by event_handler.
Definition: telnetConn.hpp:527
int login(const std::string &username, const std::string &password)
Manage the login process on this device.
Definition: telnetConn.hpp:282
#define TELNET_WAITING_PROMPT
Definition: telnetConn.hpp:60
#define TELNET_E_SOCKET
Definition: ttyErrors.hpp:32
int write(const std::string &buffWrite, int timeoutWrite)
Write to a telnet connection.
Definition: telnetConn.hpp:352
#define TTY_E_ERRORONWRITEPOLL
Definition: ttyErrors.hpp:19
int m_sock
The socket file descriptor.
Definition: telnetConn.hpp:82
int telnetCRLF(std::string &telnetStr, const std::string &inputStr)
Replace lone \r and \n with \r\n for telnet-ness.
Definition: ttyIOUtils.hpp:44
#define TTY_E_TIMEOUTONREADPOLL
Definition: ttyErrors.hpp:22
int noLogin()
Set flags as if we&#39;re logged in, used when device doesn&#39;t require it.
Definition: telnetConn.hpp:345
int writeRead(const std::string &strWrite, bool swallowEcho, int timeoutWrite, int timeoutRead)
Write to a telnet connection, then get the reply.
Definition: telnetConn.hpp:467
std::string m_prompt
The device&#39;s prompt, used for detecting end of transmission.
Definition: telnetConn.hpp:92
int m_EHError
Used to indicate an error occurred in the event handler callback.
Definition: telnetConn.hpp:107
#define TELNET_E_CONNECT
Definition: ttyErrors.hpp:34
#define TELNET_GOT_PASS
Definition: telnetConn.hpp:59
#define TELNET_LOGGED_IN
Definition: telnetConn.hpp:61
std::string m_passwordPrompt
The device&#39;s password entry prompt, used for managing login.
Definition: telnetConn.hpp:90
#define TELNET_GOT_USER
Definition: telnetConn.hpp:57
Utilities for i/o on a file descriptor pointing to a tty device.
#define TTY_E_TIMEOUTONWRITEPOLL
Definition: ttyErrors.hpp:18
telnet_t * m_telnet
libtelnet telnet_t structure
Definition: telnetConn.hpp:84
#define TTY_E_NOERROR
Definition: ttyErrors.hpp:13
bool isEndOfTrans(const std::string &strRead, const std::string &eot)
Check if the end of the buffer contains the end-of-transmission string.
Definition: ttyIOUtils.hpp:153
static void event_handler(telnet_t *telnet, telnet_event_t *ev, void *user_data)
Event handler callback for libtelnet processing.
Definition: telnetConn.hpp:553
#define TTY_E_TIMEOUTONWRITE
Definition: ttyErrors.hpp:21
#define TTY_E_TIMEOUTONREAD
Definition: ttyErrors.hpp:25