• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * lws-minimal-http-server-eventlib-custom
3  *
4  * Written in 2010-2021 by 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  * This demonstrates a minimal http server using lws, on top of a custom "event
10  * library" that uses an existing application POLL loop.
11  *
12  * To keep it simple, it serves stuff from the subdirectory  "./mount-origin" of
13  * the dir it was started in.  Change mount.origin to serve from elsewhere.
14  */
15 
16 #include <libwebsockets.h>
17 #include <string.h>
18 #include <signal.h>
19 
20 static int interrupted;
21 static struct lws_context *context;
22 
23 #define MAX_CUSTOM_POLLFDS	64
24 
25 /* this represents the existing application poll loop context we want lws
26  * to cooperate with */
27 
28 typedef struct custom_poll_ctx {
29 	struct lws_pollfd	pollfds[MAX_CUSTOM_POLLFDS];
30 	int			count_pollfds;
31 } custom_poll_ctx_t;
32 
33 /* for this example we just have the one, but it is passed into lws as a
34  * foreign loop pointer, and all callbacks have access to it via that, so it
35  * is not needed to be defined at file scope. */
36 static custom_poll_ctx_t a_cpcx;
37 
38 /*
39  * These are the custom event loop operators that just make the custom event
40  * loop able to work by itself.  These would already exist in some form in an
41  * existing application.
42  */
43 
44 static struct lws_pollfd *
custom_poll_find_fd(custom_poll_ctx_t * cpcx,lws_sockfd_type fd)45 custom_poll_find_fd(custom_poll_ctx_t *cpcx, lws_sockfd_type fd)
46 {
47 	int n;
48 
49 	for (n = 0; n < cpcx->count_pollfds; n++)
50 		if (cpcx->pollfds[n].fd == fd)
51 			return &cpcx->pollfds[n];
52 
53 	return NULL;
54 }
55 
56 static int
custom_poll_add_fd(custom_poll_ctx_t * cpcx,lws_sockfd_type fd,int events)57 custom_poll_add_fd(custom_poll_ctx_t *cpcx, lws_sockfd_type fd, int events)
58 {
59 	struct lws_pollfd *pfd;
60 
61 	lwsl_info("%s: ADD fd %d, ev %d\n", __func__, fd, events);
62 
63 	pfd = custom_poll_find_fd(cpcx, fd);
64 	if (pfd) {
65 		lwsl_err("%s: ADD fd %d already in ext table\n", __func__, fd);
66 		return 1;
67 	}
68 
69 	if (cpcx->count_pollfds == LWS_ARRAY_SIZE(cpcx->pollfds)) {
70 		lwsl_err("%s: no room left\n", __func__);
71 		return 1;
72 	}
73 
74 	pfd = &cpcx->pollfds[cpcx->count_pollfds++];
75 	pfd->fd = fd;
76 	pfd->events = (short)events;
77 	pfd->revents = 0;
78 
79 	return 0;
80 }
81 
82 static int
custom_poll_del_fd(custom_poll_ctx_t * cpcx,lws_sockfd_type fd)83 custom_poll_del_fd(custom_poll_ctx_t *cpcx, lws_sockfd_type fd)
84 {
85 	struct lws_pollfd *pfd;
86 
87 	lwsl_info("%s: DEL fd %d\n", __func__, fd);
88 
89 	pfd = custom_poll_find_fd(cpcx, fd);
90 	if (!pfd) {
91 		lwsl_err("%s: DEL fd %d missing in ext table\n", __func__, fd);
92 		return 1;
93 	}
94 
95 	if (cpcx->count_pollfds > 1)
96 		*pfd = cpcx->pollfds[cpcx->count_pollfds - 1];
97 
98 	cpcx->count_pollfds--;
99 
100 	return 0;
101 }
102 
103 static int
custom_poll_change_fd(custom_poll_ctx_t * cpcx,lws_sockfd_type fd,int events_add,int events_remove)104 custom_poll_change_fd(custom_poll_ctx_t *cpcx, lws_sockfd_type fd,
105 		     int events_add, int events_remove)
106 {
107 	struct lws_pollfd *pfd;
108 
109 	lwsl_info("%s: CHG fd %d, ev_add %d, ev_rem %d\n", __func__, fd,
110 			events_add, events_remove);
111 
112 	pfd = custom_poll_find_fd(cpcx, fd);
113 	if (!pfd)
114 		return 1;
115 
116 	pfd->events = (short)((pfd->events & (~events_remove)) | events_add);
117 
118 	return 0;
119 }
120 
121 int
custom_poll_run(custom_poll_ctx_t * cpcx)122 custom_poll_run(custom_poll_ctx_t *cpcx)
123 {
124 	int n;
125 
126 	while (!interrupted) {
127 
128 		/*
129 		 * Notice that the existing loop must consult with lws about
130 		 * the maximum wait timeout to use.  Lws will reduce the
131 		 * timeout to the earliest scheduled event time if any earlier
132 		 * than the provided timeout.
133 		 */
134 
135 		n = lws_service_adjust_timeout(context, 5000, 0);
136 
137 		lwsl_debug("%s: entering poll wait %dms\n", __func__, n);
138 
139 		n = poll(cpcx->pollfds, (nfds_t)cpcx->count_pollfds, n);
140 
141 		lwsl_debug("%s: exiting poll ret %d\n", __func__, n);
142 
143 		if (n <= 0)
144 			continue;
145 
146 		for (n = 0; n < cpcx->count_pollfds; n++) {
147 			lws_sockfd_type fd = cpcx->pollfds[n].fd;
148 			int m;
149 
150 			if (!cpcx->pollfds[n].revents)
151 				continue;
152 
153 			m = lws_service_fd(context, &cpcx->pollfds[n]);
154 
155 			/* if something closed, retry this slot since may have been
156 			 * swapped with end fd */
157 			if (m && cpcx->pollfds[n].fd != fd)
158 				n--;
159 
160 			if (m < 0)
161 				/* lws feels something bad happened, but
162 				 * the outer application may not care */
163 				continue;
164 			if (!m) {
165 				/* check if it is an fd owned by the
166 				 * application */
167 			}
168 		}
169 	}
170 
171 	return 0;
172 }
173 
174 
175 /*
176  * These is the custom "event library" interface layer between lws event lib
177  * support and the custom loop implementation above.  We only need to support
178  * a few key apis.
179  *
180  * We are user code, so all the internal lws objects are opaque.  But there are
181  * enough public helpers to get everything done.
182  */
183 
184 /* one of these is appended to each pt for our use */
185 struct pt_eventlibs_custom {
186 	custom_poll_ctx_t		*io_loop;
187 };
188 
189 /*
190  * During lws context creation, we get called with the foreign loop pointer
191  * that was passed in the creation info struct.  Stash it in our private part
192  * of the pt, so we can reference it in the other callbacks subsequently.
193  */
194 
195 static int
init_pt_custom(struct lws_context * cx,void * _loop,int tsi)196 init_pt_custom(struct lws_context *cx, void *_loop, int tsi)
197 {
198 	struct pt_eventlibs_custom *priv = (struct pt_eventlibs_custom *)
199 					     lws_evlib_tsi_to_evlib_pt(cx, tsi);
200 
201 	/* store the loop we are bound to in our private part of the pt */
202 
203 	priv->io_loop = (custom_poll_ctx_t *)_loop;
204 
205 	return 0;
206 }
207 
208 static int
sock_accept_custom(struct lws * wsi)209 sock_accept_custom(struct lws *wsi)
210 {
211 	struct pt_eventlibs_custom *priv = (struct pt_eventlibs_custom *)
212 						lws_evlib_wsi_to_evlib_pt(wsi);
213 
214 	return custom_poll_add_fd(priv->io_loop, lws_get_socket_fd(wsi), POLLIN);
215 }
216 
217 static void
io_custom(struct lws * wsi,unsigned int flags)218 io_custom(struct lws *wsi, unsigned int flags)
219 {
220 	struct pt_eventlibs_custom *priv = (struct pt_eventlibs_custom *)
221 						lws_evlib_wsi_to_evlib_pt(wsi);
222 	int e_add = 0, e_remove = 0;
223 
224 	if (flags & LWS_EV_START) {
225 		if (flags & LWS_EV_WRITE)
226 			e_add |= POLLOUT;
227 
228 		if (flags & LWS_EV_READ)
229 			e_add |= POLLIN;
230 	} else {
231 		if (flags & LWS_EV_WRITE)
232 			e_remove |= POLLOUT;
233 
234 		if (flags & LWS_EV_READ)
235 			e_remove |= POLLIN;
236 	}
237 
238 	custom_poll_change_fd(priv->io_loop, lws_get_socket_fd(wsi),
239 			      e_add, e_remove);
240 }
241 
242 static int
wsi_logical_close_custom(struct lws * wsi)243 wsi_logical_close_custom(struct lws *wsi)
244 {
245 	struct pt_eventlibs_custom *priv = (struct pt_eventlibs_custom *)
246 						lws_evlib_wsi_to_evlib_pt(wsi);
247 	return custom_poll_del_fd(priv->io_loop, lws_get_socket_fd(wsi));
248 }
249 
250 static const struct lws_event_loop_ops event_loop_ops_custom = {
251 	.name				= "custom",
252 
253 	.init_pt			= init_pt_custom,
254 	.init_vhost_listen_wsi		= sock_accept_custom,
255 	.sock_accept			= sock_accept_custom,
256 	.io				= io_custom,
257 	.wsi_logical_close		= wsi_logical_close_custom,
258 
259 	.evlib_size_pt			= sizeof(struct pt_eventlibs_custom)
260 };
261 
262 static const lws_plugin_evlib_t evlib_custom = {
263 	.hdr = {
264 		"custom event loop",
265 		"lws_evlib_plugin",
266 		LWS_BUILD_HASH,
267 		LWS_PLUGIN_API_MAGIC
268 	},
269 
270 	.ops	= &event_loop_ops_custom
271 };
272 
273 /*
274  * The rest is just the normal minimal example for lws, with a couple of extra
275  * lines wiring up the custom event library handlers above.
276  */
277 
278 static const struct lws_http_mount mount = {
279 	/* .mount_next */		NULL,		/* linked-list "next" */
280 	/* .mountpoint */		"/",		/* mountpoint URL */
281 	/* .origin */			"./mount-origin", /* serve from dir */
282 	/* .def */			"index.html",	/* default filename */
283 	/* .protocol */			NULL,
284 	/* .cgienv */			NULL,
285 	/* .extra_mimetypes */		NULL,
286 	/* .interpret */		NULL,
287 	/* .cgi_timeout */		0,
288 	/* .cache_max_age */		0,
289 	/* .auth_mask */		0,
290 	/* .cache_reusable */		0,
291 	/* .cache_revalidate */		0,
292 	/* .cache_intermediaries */	0,
293 	/* .origin_protocol */		LWSMPRO_FILE,	/* files in a dir */
294 	/* .mountpoint_len */		1,		/* char count */
295 	/* .basic_auth_login_file */	NULL,
296 };
297 
298 /*
299  * This demonstrates a client connection operating on the same loop
300  * It's optional...
301  */
302 
303 static int
callback_http(struct lws * wsi,enum lws_callback_reasons reason,void * user,void * in,size_t len)304 callback_http(struct lws *wsi, enum lws_callback_reasons reason,
305 	      void *user, void *in, size_t len)
306 {
307 	switch (reason) {
308 
309 	case LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP:
310 		lwsl_user("LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP: resp %u\n",
311 				lws_http_client_http_response(wsi));
312 		break;
313 
314 	/* because we are protocols[0] ... */
315 	case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
316 		lwsl_err("CLIENT_CONNECTION_ERROR: %s\n",
317 			 in ? (char *)in : "(null)");
318 		break;
319 
320 	/* chunks of chunked content, with header removed */
321 	case LWS_CALLBACK_RECEIVE_CLIENT_HTTP_READ:
322 		lwsl_user("RECEIVE_CLIENT_HTTP_READ: read %d\n", (int)len);
323 		lwsl_hexdump_info(in, len);
324 		return 0; /* don't passthru */
325 
326 	/* uninterpreted http content */
327 	case LWS_CALLBACK_RECEIVE_CLIENT_HTTP:
328 		{
329 			char buffer[1024 + LWS_PRE];
330 			char *px = buffer + LWS_PRE;
331 			int lenx = sizeof(buffer) - LWS_PRE;
332 
333 			if (lws_http_client_read(wsi, &px, &lenx) < 0)
334 				return -1;
335 		}
336 		return 0; /* don't passthru */
337 
338 	case LWS_CALLBACK_COMPLETED_CLIENT_HTTP:
339 		lwsl_user("LWS_CALLBACK_COMPLETED_CLIENT_HTTP %s\n",
340 			  lws_wsi_tag(wsi));
341 		break;
342 
343 	case LWS_CALLBACK_CLOSED_CLIENT_HTTP:
344 		lwsl_info("%s: closed: %s\n", __func__, lws_wsi_tag(wsi));
345 		break;
346 
347 	default:
348 		break;
349 	}
350 
351 	return lws_callback_http_dummy(wsi, reason, user, in, len);
352 }
353 
354 static const struct lws_protocols protocols[] = {
355 	{ "httptest", callback_http, 0, 0, 0, NULL, 0},
356 	LWS_PROTOCOL_LIST_TERM
357 };
358 
359 static int
do_client_conn(void)360 do_client_conn(void)
361 {
362 	struct lws_client_connect_info i;
363 
364 	memset(&i, 0, sizeof i); /* otherwise uninitialized garbage */
365 
366 	i.context		= context;
367 
368 	i.ssl_connection	= LCCSCF_USE_SSL;
369 	i.port			= 443;
370 	i.address		= "warmcat.com";
371 
372 	i.ssl_connection	|= LCCSCF_H2_QUIRK_OVERFLOWS_TXCR |
373 				   LCCSCF_H2_QUIRK_NGHTTP2_END_STREAM;
374 	i.path			= "/";
375 	i.host			= i.address;
376 	i.origin		= i.address;
377 	i.method		= "GET";
378 	i.protocol	= protocols[0].name;
379 #if defined(LWS_WITH_SYS_FAULT_INJECTION)
380 	i.fi_wsi_name		= "user";
381 #endif
382 
383 	if (!lws_client_connect_via_info(&i)) {
384 		lwsl_err("Client creation failed\n");
385 
386 		return 1;
387 	}
388 
389 	lwsl_notice("Client creation OK\n");
390 
391 	return 0;
392 }
393 
394 /*
395  * End of client part
396  *
397  * Initialization part -->
398  */
399 
sigint_handler(int sig)400 void sigint_handler(int sig)
401 {
402 	interrupted = 1;
403 }
404 
main(int argc,const char ** argv)405 int main(int argc, const char **argv)
406 {
407 	struct lws_context_creation_info info;
408 	const char *p;
409 	int logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
410 	void *foreign_loops[1];
411 
412 	signal(SIGINT, sigint_handler);
413 
414 	if ((p = lws_cmdline_option(argc, argv, "-d")))
415 		logs = atoi(p);
416 
417 	/*
418 	 * init the existing custom event loop here if anything to do, don't
419 	 * run it yet. In our example, no init required.
420 	 */
421 
422 	lws_set_log_level(logs, NULL);
423 	lwsl_user("LWS minimal http server | visit http://localhost:7681\n");
424 
425 	memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
426 	info.port = 7681;
427 	info.mounts = &mount;
428 	info.error_document_404 = "/404.html";
429 	info.options = LWS_SERVER_OPTION_DO_SSL_GLOBAL_INIT |
430 		LWS_SERVER_OPTION_HTTP_HEADERS_SECURITY_BEST_PRACTICES_ENFORCE;
431 
432 	info.event_lib_custom = &evlib_custom; /* bind lws to our custom event
433 						* lib implementation above */
434 	foreign_loops[0] = &a_cpcx; /* pass in the custom poll object as the
435 				     * foreign loop object we will bind to */
436 	info.foreign_loops = foreign_loops;
437 
438 	/* optional to demonstrate client connection */
439 	info.protocols = protocols;
440 
441 	context = lws_create_context(&info);
442 	if (!context) {
443 		lwsl_err("lws init failed\n");
444 		return 1;
445 	}
446 
447 	/* optional to demonstrate client connection */
448 	do_client_conn();
449 
450 	/*
451 	 * We're going to run the custom loop now, instead of the lws loop.
452 	 * We have told lws to cooperate with this loop to get stuff done.
453 	 *
454 	 * We only come back from this when interrupted gets set by SIGINT
455 	 */
456 
457 	custom_poll_run(&a_cpcx);
458 
459 	/* clean up lws part */
460 
461 	lws_context_destroy(context);
462 
463 	return 0;
464 }
465