test-server.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497
  1. /*
  2. * libwebsockets-test-server - libwebsockets test implementation
  3. *
  4. * Copyright (C) 2010-2016 Andy Green <andy@warmcat.com>
  5. *
  6. * This file is made available under the Creative Commons CC0 1.0
  7. * Universal Public Domain Dedication.
  8. *
  9. * The person who associated a work with this deed has dedicated
  10. * the work to the public domain by waiving all of his or her rights
  11. * to the work worldwide under copyright law, including all related
  12. * and neighboring rights, to the extent allowed by law. You can copy,
  13. * modify, distribute and perform the work, even for commercial purposes,
  14. * all without asking permission.
  15. *
  16. * The test apps are intended to be adapted for use in your code, which
  17. * may be proprietary. So unlike the library itself, they are licensed
  18. * Public Domain.
  19. */
  20. #include "test-server.h"
  21. int close_testing;
  22. int max_poll_elements;
  23. int debug_level = 7;
  24. #ifdef EXTERNAL_POLL
  25. struct lws_pollfd *pollfds;
  26. int *fd_lookup;
  27. int count_pollfds;
  28. #endif
  29. volatile int force_exit = 0;
  30. struct lws_context *context;
  31. struct lws_plat_file_ops fops_plat;
  32. /* http server gets files from this path */
  33. #define LOCAL_RESOURCE_PATH INSTALL_DATADIR"/libwebsockets-test-server"
  34. char *resource_path = LOCAL_RESOURCE_PATH;
  35. #if defined(LWS_USE_POLARSSL)
  36. #else
  37. #if defined(LWS_USE_MBEDTLS)
  38. #else
  39. #if defined(LWS_OPENSSL_SUPPORT) && defined(LWS_HAVE_SSL_CTX_set1_param)
  40. char crl_path[1024] = "";
  41. #endif
  42. #endif
  43. #endif
  44. /* singlethreaded version --> no locks */
  45. void test_server_lock(int care)
  46. {
  47. }
  48. void test_server_unlock(int care)
  49. {
  50. }
  51. /*
  52. * This demo server shows how to use libwebsockets for one or more
  53. * websocket protocols in the same server
  54. *
  55. * It defines the following websocket protocols:
  56. *
  57. * dumb-increment-protocol: once the socket is opened, an incrementing
  58. * ascii string is sent down it every 50ms.
  59. * If you send "reset\n" on the websocket, then
  60. * the incrementing number is reset to 0.
  61. *
  62. * lws-mirror-protocol: copies any received packet to every connection also
  63. * using this protocol, including the sender
  64. */
  65. enum demo_protocols {
  66. /* always first */
  67. PROTOCOL_HTTP = 0,
  68. PROTOCOL_DUMB_INCREMENT,
  69. PROTOCOL_LWS_MIRROR,
  70. PROTOCOL_LWS_ECHOGEN,
  71. PROTOCOL_LWS_STATUS,
  72. /* always last */
  73. DEMO_PROTOCOL_COUNT
  74. };
  75. /* list of supported protocols and callbacks */
  76. static struct lws_protocols protocols[] = {
  77. /* first protocol must always be HTTP handler */
  78. {
  79. "http-only", /* name */
  80. callback_http, /* callback */
  81. sizeof (struct per_session_data__http), /* per_session_data_size */
  82. 0, /* max frame size / rx buffer */
  83. },
  84. {
  85. "dumb-increment-protocol",
  86. callback_dumb_increment,
  87. sizeof(struct per_session_data__dumb_increment),
  88. 10, /* rx buf size must be >= permessage-deflate rx size */
  89. },
  90. {
  91. "lws-mirror-protocol",
  92. callback_lws_mirror,
  93. sizeof(struct per_session_data__lws_mirror),
  94. 128, /* rx buf size must be >= permessage-deflate rx size */
  95. },
  96. {
  97. "lws-echogen",
  98. callback_lws_echogen,
  99. sizeof(struct per_session_data__echogen),
  100. 128, /* rx buf size must be >= permessage-deflate rx size */
  101. },
  102. {
  103. "lws-status",
  104. callback_lws_status,
  105. sizeof(struct per_session_data__lws_status),
  106. 128, /* rx buf size must be >= permessage-deflate rx size */
  107. },
  108. { NULL, NULL, 0, 0 } /* terminator */
  109. };
  110. /* this shows how to override the lws file operations. You don't need
  111. * to do any of this unless you have a reason (eg, want to serve
  112. * compressed files without decompressing the whole archive)
  113. */
  114. static lws_filefd_type
  115. test_server_fops_open(struct lws *wsi, const char *filename,
  116. unsigned long *filelen, int flags)
  117. {
  118. lws_filefd_type n;
  119. /* call through to original platform implementation */
  120. n = fops_plat.open(wsi, filename, filelen, flags);
  121. lwsl_info("%s: opening %s, ret %ld, len %lu\n", __func__, filename,
  122. (long)n, *filelen);
  123. return n;
  124. }
  125. void sighandler(int sig)
  126. {
  127. force_exit = 1;
  128. lws_cancel_service(context);
  129. }
  130. static const struct lws_extension exts[] = {
  131. {
  132. "permessage-deflate",
  133. lws_extension_callback_pm_deflate,
  134. "permessage-deflate"
  135. },
  136. {
  137. "deflate-frame",
  138. lws_extension_callback_pm_deflate,
  139. "deflate_frame"
  140. },
  141. { NULL, NULL, NULL /* terminator */ }
  142. };
  143. static struct option options[] = {
  144. { "help", no_argument, NULL, 'h' },
  145. { "debug", required_argument, NULL, 'd' },
  146. { "port", required_argument, NULL, 'p' },
  147. { "ssl", no_argument, NULL, 's' },
  148. { "allow-non-ssl", no_argument, NULL, 'a' },
  149. { "interface", required_argument, NULL, 'i' },
  150. { "closetest", no_argument, NULL, 'c' },
  151. { "ssl-cert", required_argument, NULL, 'C' },
  152. { "ssl-key", required_argument, NULL, 'K' },
  153. { "ssl-ca", required_argument, NULL, 'A' },
  154. #if defined(LWS_OPENSSL_SUPPORT)
  155. { "ssl-verify-client", no_argument, NULL, 'v' },
  156. #if defined(LWS_HAVE_SSL_CTX_set1_param)
  157. { "ssl-crl", required_argument, NULL, 'R' },
  158. #endif
  159. #endif
  160. { "libev", no_argument, NULL, 'e' },
  161. #ifndef LWS_NO_DAEMONIZE
  162. { "daemonize", no_argument, NULL, 'D' },
  163. #endif
  164. { "resource_path", required_argument, NULL, 'r' },
  165. { NULL, 0, 0, 0 }
  166. };
  167. int main(int argc, char **argv)
  168. {
  169. struct lws_context_creation_info info;
  170. char interface_name[128] = "";
  171. unsigned int ms, oldms = 0;
  172. const char *iface = NULL;
  173. char cert_path[1024] = "";
  174. char key_path[1024] = "";
  175. char ca_path[1024] = "";
  176. int uid = -1, gid = -1;
  177. int use_ssl = 0;
  178. int opts = 0;
  179. int n = 0;
  180. #ifndef _WIN32
  181. /* LOG_PERROR is not POSIX standard, and may not be portable */
  182. #ifdef __sun
  183. int syslog_options = LOG_PID;
  184. #else
  185. int syslog_options = LOG_PID | LOG_PERROR;
  186. #endif
  187. #endif
  188. #ifndef LWS_NO_DAEMONIZE
  189. int daemonize = 0;
  190. #endif
  191. /*
  192. * take care to zero down the info struct, he contains random garbaage
  193. * from the stack otherwise
  194. */
  195. memset(&info, 0, sizeof info);
  196. info.port = 7681;
  197. while (n >= 0) {
  198. n = getopt_long(argc, argv, "eci:hsap:d:Dr:C:K:A:R:vu:g:", options, NULL);
  199. if (n < 0)
  200. continue;
  201. switch (n) {
  202. case 'e':
  203. opts |= LWS_SERVER_OPTION_LIBEV;
  204. break;
  205. #ifndef LWS_NO_DAEMONIZE
  206. case 'D':
  207. daemonize = 1;
  208. #if !defined(_WIN32) && !defined(__sun)
  209. syslog_options &= ~LOG_PERROR;
  210. #endif
  211. break;
  212. #endif
  213. case 'u':
  214. uid = atoi(optarg);
  215. break;
  216. case 'g':
  217. gid = atoi(optarg);
  218. break;
  219. case 'd':
  220. debug_level = atoi(optarg);
  221. break;
  222. case 's':
  223. use_ssl = 1;
  224. break;
  225. case 'a':
  226. opts |= LWS_SERVER_OPTION_ALLOW_NON_SSL_ON_SSL_PORT;
  227. break;
  228. case 'p':
  229. info.port = atoi(optarg);
  230. break;
  231. case 'i':
  232. strncpy(interface_name, optarg, sizeof interface_name);
  233. interface_name[(sizeof interface_name) - 1] = '\0';
  234. iface = interface_name;
  235. break;
  236. case 'c':
  237. close_testing = 1;
  238. fprintf(stderr, " Close testing mode -- closes on "
  239. "client after 50 dumb increments"
  240. "and suppresses lws_mirror spam\n");
  241. break;
  242. case 'r':
  243. resource_path = optarg;
  244. printf("Setting resource path to \"%s\"\n", resource_path);
  245. break;
  246. case 'C':
  247. strncpy(cert_path, optarg, sizeof(cert_path) - 1);
  248. cert_path[sizeof(cert_path) - 1] = '\0';
  249. break;
  250. case 'K':
  251. strncpy(key_path, optarg, sizeof(key_path) - 1);
  252. key_path[sizeof(key_path) - 1] = '\0';
  253. break;
  254. case 'A':
  255. strncpy(ca_path, optarg, sizeof(ca_path) - 1);
  256. ca_path[sizeof(ca_path) - 1] = '\0';
  257. break;
  258. #if defined(LWS_OPENSSL_SUPPORT)
  259. case 'v':
  260. use_ssl = 1;
  261. opts |= LWS_SERVER_OPTION_REQUIRE_VALID_OPENSSL_CLIENT_CERT;
  262. break;
  263. #if defined(LWS_USE_POLARSSL)
  264. #else
  265. #if defined(LWS_USE_MBEDTLS)
  266. #else
  267. #if defined(LWS_HAVE_SSL_CTX_set1_param)
  268. case 'R':
  269. strncpy(crl_path, optarg, sizeof(crl_path) - 1);
  270. crl_path[sizeof(crl_path) - 1] = '\0';
  271. break;
  272. #endif
  273. #endif
  274. #endif
  275. #endif
  276. case 'h':
  277. fprintf(stderr, "Usage: test-server "
  278. "[--port=<p>] [--ssl] "
  279. "[-d <log bitfield>] "
  280. "[--resource_path <path>]\n");
  281. exit(1);
  282. }
  283. }
  284. #if !defined(LWS_NO_DAEMONIZE) && !defined(WIN32)
  285. /*
  286. * normally lock path would be /var/lock/lwsts or similar, to
  287. * simplify getting started without having to take care about
  288. * permissions or running as root, set to /tmp/.lwsts-lock
  289. */
  290. if (daemonize && lws_daemonize("/tmp/.lwsts-lock")) {
  291. fprintf(stderr, "Failed to daemonize\n");
  292. return 10;
  293. }
  294. #endif
  295. signal(SIGINT, sighandler);
  296. #ifndef _WIN32
  297. /* we will only try to log things according to our debug_level */
  298. setlogmask(LOG_UPTO (LOG_DEBUG));
  299. openlog("lwsts", syslog_options, LOG_DAEMON);
  300. #endif
  301. /* tell the library what debug level to emit and to send it to syslog */
  302. lws_set_log_level(debug_level, lwsl_emit_syslog);
  303. lwsl_notice("libwebsockets test server - license LGPL2.1+SLE\n");
  304. lwsl_notice("(C) Copyright 2010-2016 Andy Green <andy@warmcat.com>\n");
  305. printf("Using resource path \"%s\"\n", resource_path);
  306. #ifdef EXTERNAL_POLL
  307. max_poll_elements = getdtablesize();
  308. pollfds = malloc(max_poll_elements * sizeof (struct lws_pollfd));
  309. fd_lookup = malloc(max_poll_elements * sizeof (int));
  310. if (pollfds == NULL || fd_lookup == NULL) {
  311. lwsl_err("Out of memory pollfds=%d\n", max_poll_elements);
  312. return -1;
  313. }
  314. #endif
  315. info.iface = iface;
  316. info.protocols = protocols;
  317. info.ssl_cert_filepath = NULL;
  318. info.ssl_private_key_filepath = NULL;
  319. if (use_ssl) {
  320. if (strlen(resource_path) > sizeof(cert_path) - 32) {
  321. lwsl_err("resource path too long\n");
  322. return -1;
  323. }
  324. if (!cert_path[0])
  325. sprintf(cert_path, "%s/libwebsockets-test-server.pem",
  326. resource_path);
  327. if (strlen(resource_path) > sizeof(key_path) - 32) {
  328. lwsl_err("resource path too long\n");
  329. return -1;
  330. }
  331. if (!key_path[0])
  332. sprintf(key_path, "%s/libwebsockets-test-server.key.pem",
  333. resource_path);
  334. info.ssl_cert_filepath = cert_path;
  335. info.ssl_private_key_filepath = key_path;
  336. if (ca_path[0])
  337. info.ssl_ca_filepath = ca_path;
  338. }
  339. info.gid = gid;
  340. info.uid = uid;
  341. info.max_http_header_pool = 16;
  342. info.options = opts | LWS_SERVER_OPTION_VALIDATE_UTF8;
  343. info.extensions = exts;
  344. info.timeout_secs = 5;
  345. info.ssl_cipher_list = "ECDHE-ECDSA-AES256-GCM-SHA384:"
  346. "ECDHE-RSA-AES256-GCM-SHA384:"
  347. "DHE-RSA-AES256-GCM-SHA384:"
  348. "ECDHE-RSA-AES256-SHA384:"
  349. "HIGH:!aNULL:!eNULL:!EXPORT:"
  350. "!DES:!MD5:!PSK:!RC4:!HMAC_SHA1:"
  351. "!SHA1:!DHE-RSA-AES128-GCM-SHA256:"
  352. "!DHE-RSA-AES128-SHA256:"
  353. "!AES128-GCM-SHA256:"
  354. "!AES128-SHA256:"
  355. "!DHE-RSA-AES256-SHA256:"
  356. "!AES256-GCM-SHA384:"
  357. "!AES256-SHA256";
  358. if (use_ssl)
  359. /* redirect guys coming on http */
  360. info.options |= LWS_SERVER_OPTION_REDIRECT_HTTP_TO_HTTPS;
  361. context = lws_create_context(&info);
  362. if (context == NULL) {
  363. lwsl_err("libwebsocket init failed\n");
  364. return -1;
  365. }
  366. /* this shows how to override the lws file operations. You don't need
  367. * to do any of this unless you have a reason (eg, want to serve
  368. * compressed files without decompressing the whole archive)
  369. */
  370. /* stash original platform fops */
  371. fops_plat = *(lws_get_fops(context));
  372. /* override the active fops */
  373. lws_get_fops(context)->open = test_server_fops_open;
  374. n = 0;
  375. #ifdef EXTERNAL_POLL
  376. int ms_1sec = 0;
  377. #endif
  378. while (n >= 0 && !force_exit) {
  379. struct timeval tv;
  380. gettimeofday(&tv, NULL);
  381. /*
  382. * This provokes the LWS_CALLBACK_SERVER_WRITEABLE for every
  383. * live websocket connection using the DUMB_INCREMENT protocol,
  384. * as soon as it can take more packets (usually immediately)
  385. */
  386. ms = (tv.tv_sec * 1000) + (tv.tv_usec / 1000);
  387. if ((ms - oldms) > 50) {
  388. lws_callback_on_writable_all_protocol(context,
  389. &protocols[PROTOCOL_DUMB_INCREMENT]);
  390. oldms = ms;
  391. }
  392. #ifdef EXTERNAL_POLL
  393. /*
  394. * this represents an existing server's single poll action
  395. * which also includes libwebsocket sockets
  396. */
  397. n = poll(pollfds, count_pollfds, 50);
  398. if (n < 0)
  399. continue;
  400. if (n) {
  401. for (n = 0; n < count_pollfds; n++)
  402. if (pollfds[n].revents)
  403. /*
  404. * returns immediately if the fd does not
  405. * match anything under libwebsockets
  406. * control
  407. */
  408. if (lws_service_fd(context,
  409. &pollfds[n]) < 0)
  410. goto done;
  411. /* if needed, force-service wsis that may not have read all input */
  412. while (!lws_service_adjust_timeout(context, 1, 0)) {
  413. lwsl_notice("extpoll doing forced service!\n");
  414. lws_plat_service_tsi(context, -1, 0);
  415. }
  416. } else {
  417. /* no revents, but before polling again, make lws check for any timeouts */
  418. if (ms - ms_1sec > 1000) {
  419. lwsl_notice("1 per sec\n");
  420. lws_service_fd(context, NULL);
  421. ms_1sec = ms;
  422. }
  423. }
  424. #else
  425. /*
  426. * If libwebsockets sockets are all we care about,
  427. * you can use this api which takes care of the poll()
  428. * and looping through finding who needed service.
  429. *
  430. * If no socket needs service, it'll return anyway after
  431. * the number of ms in the second argument.
  432. */
  433. n = lws_service(context, 50);
  434. #endif
  435. }
  436. #ifdef EXTERNAL_POLL
  437. done:
  438. #endif
  439. lws_context_destroy(context);
  440. lwsl_notice("libwebsockets-test-server exited cleanly\n");
  441. #ifndef _WIN32
  442. closelog();
  443. #endif
  444. return 0;
  445. }