1 /*
2  * ws protocol handler plugin for "lws-minimal"
3  *
4  * Written in 2010-2019 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 version holds a single message at a time, which may be lost if a new
10  * message comes.  See the minimal-ws-server-ring sample for the same thing
11  * but using an lws_ring ringbuffer to hold up to 8 messages at a time.
12  */
13 
14 #if !defined (LWS_PLUGIN_STATIC)
15 #define LWS_DLL
16 #define LWS_INTERNAL
17 #include <libwebsockets.h>
18 #endif
19 
20 #include <string.h>
21 
22 /*
23  * This came from...
24  *
25  * cat /dev/urandom | hexdump -C -n 1024 | tr -s ' ' | cut -d' ' -f 2-17 | head -n-1 | sed "s/\ /, 0x/g" | sed "s/^/0x/g" | sed "s/\$/,/g"
26  *
27  * ...then the length tuned by hand to get the ciphertext sizes that we want to
28  * confirm are OK.
29  *
30  * We can only pass in a maximum of one compression buffer of input at a time,
31  * which is 1024 by default.
32  */
33 
34 unsigned char uncompressible[] = {
35 	0xfe, 0xcc, 0x47, 0xcb, 0x10, 0xf4, 0x3c, 0x85,
36 	0x8e, 0xd4, 0xe2, 0xf6, 0xd1, 0xd1, 0xdb, 0x64,
37 	0x94, 0x50, 0xf6, 0x14, 0x25, 0x03, 0x09, 0x3a,
38 	0xb1, 0x47, 0x86, 0xa8, 0x3c, 0x4f, 0x3b, 0x98,
39 	0x7b, 0x3e, 0x67, 0x3e, 0x22, 0xc5, 0x4c, 0x45,
40 	0xf4, 0xf7, 0xb5, 0x79, 0xc0, 0x26, 0x6e, 0x5c,
41 	0xf4, 0x10, 0x04, 0xa9, 0x3c, 0x4f, 0xed, 0xc5,
42 	0x3d, 0xd4, 0x9f, 0x9f, 0xa3, 0xdb, 0x29, 0xeb,
43 	0x1e, 0xe1, 0x52, 0xab, 0xb5, 0x75, 0x25, 0x86,
44 	0x86, 0x02, 0x2c, 0x9d, 0x9c, 0x86, 0x46, 0x92,
45 	0xe9, 0x04, 0xd8, 0x2c, 0x7d, 0x8a, 0x56, 0xe1,
46 	0xe1, 0xb6, 0x84, 0x4d, 0x17, 0x30, 0x01, 0x60,
47 	0xa6, 0xf4, 0xba, 0xc9, 0x5a, 0x29, 0xe3, 0x05,
48 	0xe1, 0xb4, 0x0b, 0x23, 0x74, 0x93, 0x25, 0x76,
49 	0xce, 0x15, 0xe4, 0x82, 0x9f, 0xbf, 0xe8, 0x6a,
50 	0x4a, 0xc5, 0xc2, 0x22, 0x91, 0x80, 0xb5, 0xd7,
51 	0xb3, 0xce, 0x70, 0x0e, 0xf7, 0xbb, 0x2f, 0xc5,
52 	0x83, 0x39, 0x86, 0xe5, 0x3e, 0xb7, 0x83, 0x87,
53 	0xc2, 0xeb, 0xc8, 0xed, 0x59, 0x26, 0xc1, 0xe6,
54 	0x80, 0x17, 0x3c, 0x29, 0x53, 0x4c, 0x1c, 0x3f,
55 	0x54, 0xbe, 0x34, 0x26, 0x72, 0xed, 0x38, 0x10,
56 	0xd1, 0x37, 0x07, 0x2d, 0x12, 0x31, 0x9b, 0xc5,
57 	0x92, 0x09, 0x13, 0x5d, 0x8e, 0xef, 0xdb, 0x52,
58 	0x7f, 0x7d, 0x6f, 0x62, 0x1e, 0x17, 0xd2, 0xf9,
59 	0x72, 0x74, 0xc7, 0xd6, 0x1f, 0x8b, 0x9c, 0x4c,
60 	0x26, 0xd2, 0x6f, 0x7c, 0x33, 0x06, 0xee, 0xc2,
61 	0xa3, 0x41, 0x43, 0x4f, 0x40, 0x2a, 0x9c, 0xb3,
62 	0x4a, 0xb1, 0x88, 0x4e, 0x6f, 0xf2, 0xb7, 0x38,
63 	0xde, 0x87, 0x0d, 0xdc, 0x15, 0x6a, 0x36, 0x6b,
64 	0xf3, 0x6c, 0x61, 0xf5, 0x24, 0x8e, 0xb6, 0xcc,
65 	0x8a, 0x3a, 0xa0, 0xb4, 0x9b, 0xae, 0x85, 0x87,
66 	0x75, 0xf5, 0xbd, 0x50, 0x1f, 0xb5, 0x0c, 0xdb,
67 	0x6c, 0x68, 0x59, 0xef, 0x37, 0x5a, 0x2a, 0x85,
68 	0xf0, 0xce, 0x4d, 0x58, 0xa1, 0xa5, 0xde, 0x73,
69 	0x9b, 0x1a, 0x3d, 0x8a, 0x00, 0xba, 0x2f, 0xe2,
70 	0xda, 0xad, 0x3c, 0x63, 0x8a, 0x33, 0x39, 0xc4,
71 	0x07, 0x29, 0x1d, 0xa7, 0x40, 0x3b, 0xa4, 0xa6,
72 	0xae, 0xee, 0x37, 0x08, 0x83, 0xd1, 0x72, 0x66,
73 	0x3d, 0x43, 0xe3, 0x7a, 0x48, 0xfc, 0xf8, 0xd4,
74 	0xe3, 0xab, 0xd0, 0xe9, 0xb1, 0xf4, 0x4d, 0x3c,
75 	0x6b, 0x58, 0xde, 0x3c, 0x91, 0x0d, 0x3e, 0xec,
76 	0x35, 0x6d, 0x53, 0xe6, 0xb6, 0x4b, 0xc0, 0x80,
77 	0x18, 0xab, 0x96, 0x7f, 0x05, 0xd7, 0xd4, 0x81,
78 	0x0f, 0x92, 0x2b, 0xaf, 0x72, 0x59, 0xc2, 0x14,
79 	0xca, 0x62, 0x82, 0xac, 0xe3, 0x17, 0x43, 0x61,
80 	0x4d, 0x1e, 0xfc, 0x72, 0xaf, 0xfc, 0x55, 0x2a,
81 	0x2b, 0xb6, 0x8e, 0x6e, 0xe6, 0x86, 0xeb, 0xcc,
82 	0x26, 0x6c, 0xdf, 0xac, 0x02, 0x58, 0xa1, 0x5d,
83 	0x1b, 0x07, 0xe2, 0x5d, 0x50, 0xb9, 0xbf, 0x2e,
84 	0x1f, 0x49, 0x39, 0xe6, 0x7f, 0x2f, 0x0e, 0x9d,
85 	0x09, 0x42, 0xc7, 0xa1, 0xcc, 0xeb, 0x5b, 0x06,
86 	0x1c, 0x11, 0x9f, 0xea, 0xc1, 0x96, 0x82, 0xa9,
87 	0x30, 0x6a, 0xda, 0x98, 0x87, 0x43, 0xfd, 0x25,
88 	0xe7, 0x27, 0x53, 0x9a, 0xb3, 0x2f, 0x19, 0xa9,
89 	0x1a, 0xf4, 0xd6, 0xf3, 0x9e, 0xba, 0x9a, 0x91,
90 	0x52, 0x8f, 0x20, 0x6b, 0x4c, 0x3a, 0x2a, 0x3d,
91 	0xa0, 0xff, 0x8d, 0x61, 0x04, 0xee, 0x26, 0x55,
92 	0xdd, 0xd7, 0x67, 0xe4, 0x84, 0x0d, 0xf1, 0x5d,
93 	0xc7, 0xeb, 0xb3, 0x8c, 0x67, 0xa2, 0xc8, 0x1f,
94 	0x53, 0x02, 0xc4, 0x8c, 0x89, 0xd5, 0x51, 0xc8,
95 	0x8b, 0xb7, 0xc8, 0x11, 0xbe, 0x0e, 0xc2, 0xb1,
96 	0x00, 0x35, 0x81, 0x96, 0xac, 0x90, 0x9c, 0xbc,
97 	0x09, 0x82, 0x75, 0xc3, 0xe7, 0x66, 0x4e, 0x68,
98 	0xdc, 0xa1, 0xf0, 0xd0, 0x2d, 0x49, 0x3b, 0x47,
99 	0xba, 0x19, 0xc8, 0x9b, 0x90, 0x12, 0xc0, 0xdf,
100 	0xda, 0x32, 0x0f, 0x79, 0x6d, 0x1a, 0x5f, 0x92,
101 	0x51, 0x70, 0xfc, 0xca, 0x08, 0xd4, 0x7f, 0x1a,
102 	0x56, 0x04, 0x99, 0x33, 0x89, 0x3d, 0x6f, 0x89,
103 	0x10, 0x25, 0x81, 0xe2, 0xbd, 0x06, 0xd6, 0xaa,
104 	0x02, 0x8e, 0x4c, 0xa3, 0x60, 0xfd, 0xaf, 0x9c,
105 	0x81, 0x75, 0xaf, 0x2f, 0xe1, 0x72, 0xe0, 0x6e,
106 	0x15, 0xdd, 0xbb, 0x92, 0xd1, 0xbe, 0x8e, 0x9b,
107 	0xfb, 0x82, 0xb9, 0x47, 0x6f, 0x02, 0x28, 0x2a,
108 	0x67, 0x50, 0xed, 0x24, 0x9b, 0x4d, 0x69, 0xd7,
109 	0xa9, 0x66, 0x3e, 0x14, 0x4b, 0x00, 0x2a, 0xe4,
110 	0x3d, 0x63, 0xb2, 0x10, 0xd4, 0x05, 0x9d, 0xe3,
111 	0xde, 0xce, 0xd8, 0x04, 0x41, 0x03, 0xb5, 0xda,
112 	0xb0, 0x6f, 0xca, 0x63, 0x64, 0x04, 0xff, 0x07,
113 	0x58, 0x5f, 0x96, 0xf7, 0x6c, 0xb7, 0x67, 0x05,
114 	0xd6, 0x85, 0xf2, 0x1e, 0xc1, 0xdc, 0x76, 0x12,
115 	0x50, 0x83, 0x78, 0xa2, 0x51, 0x94, 0xe1, 0x2e,
116 	0xb8, 0x97, 0x5b, 0x08, 0x81, 0xac, 0x59, 0x43,
117 	0xe9, 0x01, 0x09, 0xa2, 0xed, 0x10, 0x4f, 0xb1,
118 	0x5b, 0xb8, 0x67, 0xe8, 0x61, 0x8d, 0xc8, 0xd9,
119 	0xc3, 0x5f, 0x65, 0xd7, 0xaa, 0x30, 0x0e, 0xc9,
120 	0x43, 0x98, 0x1d, 0xf1, 0xa5, 0x28, 0xd5, 0xa1,
121 	0x6b, 0x8f, 0x89, 0x76, 0x97, 0xa1, 0x3e, 0x6f,
122 	0x39, 0xf4, 0xb9, 0x6b, 0xa7, 0xfe, 0x58, 0x24,
123 	0xcd, 0x75, 0xa8, 0xec, 0x9e, 0x1c, 0x8e, 0x02,
124 	0x2a, 0xce, 0xe9, 0x0a, 0x24, 0x31, 0x89, 0x5a,
125 	0xd5, 0xdd, 0x70, 0x8e, 0x5f, 0xee, 0xc1, 0x34,
126 	0xf8, 0xe2, 0x8a, 0xca, 0xf1, 0xf2, 0x71, 0x4c,
127 	0x31, 0x56, 0xeb, 0x03, 0xf9, 0x6c, 0x0d, 0xa9,
128 	0x65, 0x6e, 0x88, 0x4f, 0x8e, 0x80, 0x69, 0xd7,
129 	0xd4, 0x63, 0x45, 0x9c, 0xab, 0x8c, 0x3d, 0x08,
130 	0x8b, 0xd9, 0x97, 0xdc, 0x88, 0x59, 0x19, 0x2d,
131 	0xb2, 0x84, 0xf4, 0x78, 0x3e, 0xce, 0x80, 0xba,
132 	0xeb, 0x34, 0x5a, 0x9e, 0x8e, 0x98, 0xc4, 0x45,
133 	0x9d, 0x59, 0xb2, 0x7e, 0xc1, 0x7e, 0x5b, 0x89,
134 	0xd0, 0x02, 0xcb, 0xa4, 0xf1, 0xf2, 0xa7, 0x3a,
135 	0x05, 0xc3, 0x7d, 0x43, 0x64, 0x7f, 0xf0, 0xc1,
136 	0xf8, 0x71, 0x3b, 0x38, 0x39, 0xc7, 0x1b, 0xf4,
137 	0x2f, 0x5a, 0x5c, 0x43, 0x1b, 0xe3, 0x93, 0xe8,
138 	0x79, 0xe8, 0x35, 0x63, 0x34, 0x7e, 0x25, 0x41,
139 	0x6f, 0x08, 0xce, 0x6f, 0x95, 0x2a, 0xc2, 0xdc,
140 	0x65, 0xe2, 0xa5, 0xc0, 0xfd, 0xf1, 0x78, 0x32,
141 	0x23, 0x09, 0x75, 0x99, 0x12, 0x7a, 0x83, 0xfd,
142 	0xae, 0x1e, 0xb2, 0xe9, 0x12, 0x5c, 0x3d, 0x03,
143 	0x68, 0x12, 0x1e, 0xe3, 0x8f, 0xff, 0x47, 0xe3,
144 	0xb4, 0x7e, 0x9b, 0x7e, 0x60, 0x2e, 0xf4, 0x06,
145 	0xba, 0x10, 0x08, 0x6b, 0xf9, 0x25, 0x59, 0xf3,
146 	0x61, 0x13, 0x2b, 0xd1, 0x2f, 0x04, 0x5f, 0xd6,
147 	0xd3, 0x42, 0xf6, 0x21, 0x57, 0xf6, 0xd3, 0xb3,
148 	0xec, 0xec, 0x07, 0x33, 0xbf, 0x69, 0x04, 0xec,
149 	0x88, 0x8d, 0x06, 0x2b, 0xfa, 0xee, 0xb2, 0x7b,
150 	0x41, 0x2a, 0x49, 0x0f, 0x30, 0x52, 0x41, 0x29,
151 	0x70, 0xd0, 0xf6, 0xb6, 0xbf, 0x27, 0x1a, 0x56,
152 	0x9a, 0x4b, 0x2a, 0x67, 0xfb, 0xc8, 0x16, 0x46,
153 	0x59, 0xc7, 0xf5, 0x5f, 0x20, 0x10, 0x25, 0x6c,
154 	0x1e, 0x36, 0x20, 0x0c, 0x3e, 0x7e, 0x15, 0x6c,
155 	0xa2, 0xbd, 0x22, 0xc4, 0x3d, 0xc9, 0x74, 0x56,
156 	0xab, 0x31, 0x92, 0xb8, 0x9f, 0xa1, 0x05, 0x2e,
157 	0xc4, 0xdb, 0x32, 0x91, 0xcb, 0x0f, 0x4a, 0x73,
158 	0x7f, 0xe1, 0xe6, 0x65, 0x2e, 0x5e, 0xa6, 0xaf,
159 	0xae, 0xa9, 0x04, 0x14, 0x83, 0xef, 0x19, 0x70,
160 	0x5e, 0xcb, 0xf5, 0x87, 0xcc, 0x45, 0xf7, 0x60,
161 	0xd7, 0x9d, 0x1e, 0x2e, /* 1012 */
162 
163 	/* up to here, this generates a 1022-byte single packet of compressed
164 	 * data that is well-formed and produces 1012 bytes of plaintext.
165 	 *
166 	 * The compressed packet ends
167 	 *
168 	 *  03F0: 70 5E CB F5 87 CC 45 F7 60 D7 9D 1E 2E 00
169 	 */
170 
171 	0x54, /* 1013 */
172 
173 	/* up to here, this generates a 1023-byte single packet of compressed
174 	 * data that is well-formed and produces 1013 bytes of plaintext.
175 	 *
176 	 * The compressed packet ends
177 	 *
178 	 *  03F0: 70 5E CB F5 87 CC 45 F7 60 D7 9D 1E 2E 54 00
179 	 */
180 
181 	0x83, /* 1014 */
182 
183 	/* up to here, a 1023-byte + 3-byte (1 byte payload) packet
184 	 * of uncompressed length 1014 */
185 
186 	0x09, 0x99, 0xf9, 0x71, 0x9f, 0x15, 0x49, 0xda, 0xa8, 0x99, /* 1024 */
187 
188 	/* up to here, a 1023-byte (1020 payload) + 3-byte (1 payload) packet
189 	 * of uncompressed length 1019 */
190 
191 	0xf5, 0xe6, 0xa1, 0x71, 0x64, 0x9a, 0x95, 0xed,
192 
193 
194 };
195 
196 /* generates ciphertext:        1022  1023  1023 + 3 1023 + 3 */
197 static int corner_lengths[] = {
198 /* bytes plaintext,	ciphertext */
199 	1012,	/*	1019 */
200 	1013,	/*	1020 */
201 	1014,	/*	1021 */
202 	1019,	/*	1021 */
203 	1024,	/*	1021*/
204 };
205 
206 
207 /* one of these is created for each client connecting to us */
208 
209 struct per_session_data__minimal {
210 	int which;
211 	int last; /* 0 no test, else test number in corner_lengths[] + 1 */
212 };
213 
214 static int
callback_minimal(struct lws * wsi,enum lws_callback_reasons reason,void * user,void * in,size_t len)215 callback_minimal(struct lws *wsi, enum lws_callback_reasons reason,
216 			void *user, void *in, size_t len)
217 {
218 	struct per_session_data__minimal *pss =
219 			(struct per_session_data__minimal *)user;
220 	unsigned char buf[LWS_PRE + 2048];
221 	int m;
222 
223 	switch (reason) {
224 	case LWS_CALLBACK_ESTABLISHED:
225 		if (lws_hdr_copy(wsi, (char *)buf, sizeof(buf),
226 				 WSI_TOKEN_GET_URI) < 0)
227 			return -1;
228 
229 		pss->last = atoi((char *)buf + 1);
230 
231 		if (pss->last > (int)LWS_ARRAY_SIZE(corner_lengths))
232 			pss->last = 0;
233 		lws_callback_on_writable(wsi);
234 		break;
235 
236 	case LWS_CALLBACK_SERVER_WRITEABLE:
237 		if (!pss->last)
238 			break;
239 
240 		lwsl_err("%s: writable %d, %d\n", __func__, pss->last,
241 				corner_lengths[pss->last - 1]);
242 
243 		memcpy(buf + LWS_PRE, uncompressible,
244 		       corner_lengths[pss->last - 1]);
245 
246 		/* notice we allowed for LWS_PRE in the payload already */
247 		m = lws_write(wsi, buf + LWS_PRE, corner_lengths[pss->last - 1],
248 				LWS_WRITE_BINARY);
249 		if (m < corner_lengths[pss->last - 1]) {
250 			lwsl_err("ERROR %d writing to ws socket\n", m);
251 			return -1;
252 		}
253 
254 		pss->last = 0;
255 		break;
256 
257 	default:
258 		break;
259 	}
260 
261 	return 0;
262 }
263 
264 #define LWS_PLUGIN_PROTOCOL_MINIMAL \
265 	{ \
266 		"lws-minimal", \
267 		callback_minimal, \
268 		sizeof(struct per_session_data__minimal), \
269 		2048, \
270 		0, NULL, 0 \
271 	}
272 
273 #if !defined (LWS_PLUGIN_STATIC)
274 
275 /* boilerplate needed if we are built as a dynamic plugin */
276 
277 static const struct lws_protocols protocols[] = {
278 	LWS_PLUGIN_PROTOCOL_MINIMAL
279 };
280 
281 int
init_protocol_minimal(struct lws_context * context,struct lws_plugin_capability * c)282 init_protocol_minimal(struct lws_context *context,
283 		      struct lws_plugin_capability *c)
284 {
285 	if (c->api_magic != LWS_PLUGIN_API_MAGIC) {
286 		lwsl_err("Plugin API %d, library API %d", LWS_PLUGIN_API_MAGIC,
287 			 c->api_magic);
288 		return 1;
289 	}
290 
291 	c->protocols = protocols;
292 	c->count_protocols = LWS_ARRAY_SIZE(protocols);
293 	c->extensions = NULL;
294 	c->count_extensions = 0;
295 
296 	return 0;
297 }
298 
299 int
destroy_protocol_minimal(struct lws_context * context)300 destroy_protocol_minimal(struct lws_context *context)
301 {
302 	return 0;
303 }
304 #endif
305