/common.cpp

https://github.com/silviu/quill · C++ · 236 lines · 184 code · 26 blank · 26 comment · 40 complexity · 264b05eb86f45b43a0da8e8c0ab0c467 MD5 · raw file

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <map>
  4. #include <string.h>
  5. #include <string>
  6. #include <iostream>
  7. #include <sstream>
  8. #include <sys/socket.h>
  9. #include <sys/types.h>
  10. #include <sys/select.h>
  11. #include <netdb.h>
  12. #include <errno.h>
  13. #include "common.h"
  14. #include "base64.h"
  15. using namespace std;
  16. #define BUF_SIZE 10
  17. void prompt()
  18. {
  19. cout << prompt_line << flush;
  20. }
  21. void perros(const char* s)
  22. {
  23. cout << "\n" << flush;
  24. perror(s);
  25. prompt();
  26. }
  27. void insert_fd(fd_set &s, int &fdmax, int fd)
  28. {
  29. if (fdmax < fd)
  30. fdmax = fd;
  31. FD_SET(fd, &s);
  32. }
  33. void copy_fdset(fd_set &source, fd_set &dest)
  34. {
  35. memcpy(&dest, &source, sizeof(struct addrinfo));
  36. }
  37. /**
  38. * reads len characters from the socket into buf.
  39. * if the socket gets shutdown or a socket error occurs, return -1
  40. * else return len
  41. */
  42. int readall(int fd, char * buf, size_t len)
  43. {
  44. size_t remaining = len;
  45. while (remaining) {
  46. int rc = recv(fd, buf, remaining, 0);
  47. if (rc == -1) {
  48. perros("recv in readall");
  49. return -1;
  50. }
  51. if (rc == 0) {
  52. perros("recv: unexpected socket shutdown");
  53. return -1;
  54. }
  55. buf += rc;
  56. remaining -= rc;
  57. }
  58. return len;
  59. }
  60. /**
  61. * adds characters to 'dest' until needle is found (inclusive)
  62. */
  63. int read_to_char(int fd, string & dest, int needle)
  64. {
  65. char buf[BUF_SIZE+1];
  66. bool found = false;
  67. while(!found) {
  68. int rc = recv(fd, buf, BUF_SIZE, MSG_PEEK);
  69. if (rc == -1) {
  70. perros("recv peek");
  71. return -1;
  72. }
  73. buf[rc] = '\0';
  74. //cout << "READ " << rc << " chars :: " << buf << endl;
  75. char* pos = (char*) memchr(buf, needle, BUF_SIZE);
  76. int len = BUF_SIZE;
  77. if (pos != NULL) {
  78. len = pos - buf + 1;
  79. found = true;
  80. }
  81. rc = readall(fd, buf, len);
  82. if (rc == -1)
  83. return -1;
  84. dest.append(buf, rc);
  85. }
  86. return 0;
  87. }
  88. /**
  89. * reads a line from the socket
  90. * (terminator == '\n', included in line)
  91. */
  92. int readln(int fd, string & dest)
  93. {
  94. return read_to_char(fd, dest, '\n');
  95. }
  96. /** Writes len characters to the socket. */
  97. int writeall(int fd, const char * buf, size_t len)
  98. {
  99. size_t remaining = len;
  100. while (remaining) {
  101. int rc = send(fd, buf, remaining, 0);
  102. if (rc == -1) {
  103. perros("send in writeall");
  104. return -1;
  105. }
  106. buf += rc;
  107. remaining -= rc;
  108. }
  109. return len;
  110. }
  111. /** Same as above, but accepts a "string" in place
  112. * of a "char*" as an argument.
  113. */
  114. int writeall(int fd, const string & str)
  115. {
  116. return writeall(fd, str.c_str(), str.length());
  117. }
  118. /** Appends a "\n" to the string that is to be sent.
  119. * Calls "writeall" to send the created line to the client.
  120. */
  121. int writeln(int fd, const string & s_)
  122. {
  123. string s = s_;
  124. size_t len = s.length();
  125. string nl = "\n";
  126. if (len == 0)
  127. return writeall(fd, nl);
  128. if (s[s.length()-1] != '\n')
  129. s += "\n";
  130. return writeall(fd, s);
  131. }
  132. /** Creates a socket. Binds to a port.
  133. * Returns a socket fd.
  134. */
  135. static int connect_or_bind(int must_bind, const char* host, const char* port)
  136. {
  137. struct addrinfo hints;
  138. struct addrinfo *result, *rp;
  139. int sfd, s;
  140. memset(&hints, 0, sizeof(struct addrinfo));
  141. hints.ai_family = AF_UNSPEC; /* Allow IPv4 or IPv6 */
  142. hints.ai_socktype = SOCK_STREAM; /* Datagram socket */
  143. hints.ai_flags = AI_PASSIVE; /* For wildcard IP address */
  144. hints.ai_protocol = 0; /* Any protocol */
  145. hints.ai_canonname = NULL;
  146. hints.ai_addr = NULL;
  147. hints.ai_next = NULL;
  148. s = getaddrinfo(host, port, &hints, &result);
  149. if (s != 0) {
  150. fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(s));
  151. exit(EXIT_FAILURE);
  152. }
  153. for (rp = result; rp != NULL; rp = rp->ai_next) {
  154. sfd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol);
  155. if (sfd == -1)
  156. continue;
  157. /** Make the server override the 120 sec standard timeout
  158. * until it is allowed to reuse a certain port
  159. */
  160. int on = 1;
  161. int rc = setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
  162. if ( rc == -1)
  163. perros("setsockopt");
  164. if (must_bind) {
  165. if (bind(sfd, rp->ai_addr, rp->ai_addrlen) == 0)
  166. break;
  167. }
  168. else {
  169. if (connect(sfd, rp->ai_addr, rp->ai_addrlen) == 0)
  170. break;
  171. }
  172. close(sfd);
  173. }
  174. if (rp == NULL) { /* No address succeeded */
  175. fprintf(stderr, "Could not bind. HOST=[%s] -- PORT=[%s]\n", host, port);
  176. exit(EXIT_FAILURE);
  177. }
  178. freeaddrinfo(result); /* No longer needed */
  179. return sfd;
  180. }
  181. int bind_to(const char* host, const char* serv)
  182. {
  183. return connect_or_bind(1, host, serv);
  184. }
  185. int connect_to(const char* host, const char* serv)
  186. {
  187. return connect_or_bind(0, host, serv);
  188. }
  189. int make_base64(const string &ascii, string &b_64)
  190. {
  191. size_t b_64_len;
  192. const unsigned char* ascii_c_str = (const unsigned char*) ascii.c_str();
  193. char* b_64_c_str = (char*) base64_encode(ascii_c_str, ascii.length(), &b_64_len);
  194. if (b_64_c_str == NULL)
  195. return -1;
  196. b_64.append(b_64_c_str, b_64_len);
  197. free(b_64_c_str);
  198. return 0;
  199. }
  200. int decode_base64(const string &b_64, string &ascii)
  201. {
  202. size_t ascii_len;
  203. const unsigned char* b_64_c_str = (const unsigned char*) b_64.c_str();
  204. char* ascii_c_str= (char*) base64_decode(b_64_c_str, b_64.length(), &ascii_len);
  205. if (ascii_c_str == NULL)
  206. return -1;
  207. ascii.append(ascii_c_str, ascii_len);
  208. free(ascii_c_str);
  209. return 0;
  210. }