1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker *
4*1c60b9acSAndroid Build Coastguard Worker * Copyright (C) 2010 - 2019 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker *
6*1c60b9acSAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker *
13*1c60b9acSAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker *
16*1c60b9acSAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker *
24*1c60b9acSAndroid Build Coastguard Worker * A helper for running multiple unit tests against abstract protocols.
25*1c60b9acSAndroid Build Coastguard Worker *
26*1c60b9acSAndroid Build Coastguard Worker * An lws_seq_t is used to base its actions in the event loop and manage
27*1c60b9acSAndroid Build Coastguard Worker * the sequencing of multiple tests. A new abstract connection is instantiated
28*1c60b9acSAndroid Build Coastguard Worker * for each test using te
29*1c60b9acSAndroid Build Coastguard Worker */
30*1c60b9acSAndroid Build Coastguard Worker
31*1c60b9acSAndroid Build Coastguard Worker #include <private-lib-core.h>
32*1c60b9acSAndroid Build Coastguard Worker
33*1c60b9acSAndroid Build Coastguard Worker struct lws_seq_test_sequencer {
34*1c60b9acSAndroid Build Coastguard Worker lws_abs_t original_abs;
35*1c60b9acSAndroid Build Coastguard Worker
36*1c60b9acSAndroid Build Coastguard Worker lws_test_sequencer_args_t args;
37*1c60b9acSAndroid Build Coastguard Worker
38*1c60b9acSAndroid Build Coastguard Worker struct lws_context *context;
39*1c60b9acSAndroid Build Coastguard Worker struct lws_vhost *vhost;
40*1c60b9acSAndroid Build Coastguard Worker struct lws_sequencer *unit_test_seq;
41*1c60b9acSAndroid Build Coastguard Worker
42*1c60b9acSAndroid Build Coastguard Worker /* holds the per-test token for the unit-test transport to consume */
43*1c60b9acSAndroid Build Coastguard Worker lws_token_map_t uttt[4];
44*1c60b9acSAndroid Build Coastguard Worker
45*1c60b9acSAndroid Build Coastguard Worker lws_abs_t *instance;
46*1c60b9acSAndroid Build Coastguard Worker
47*1c60b9acSAndroid Build Coastguard Worker int state;
48*1c60b9acSAndroid Build Coastguard Worker };
49*1c60b9acSAndroid Build Coastguard Worker
50*1c60b9acSAndroid Build Coastguard Worker /* sequencer messages specific to this sequencer */
51*1c60b9acSAndroid Build Coastguard Worker
52*1c60b9acSAndroid Build Coastguard Worker enum {
53*1c60b9acSAndroid Build Coastguard Worker SEQ_MSG_PASS = LWSSEQ_USER_BASE,
54*1c60b9acSAndroid Build Coastguard Worker SEQ_MSG_FAIL,
55*1c60b9acSAndroid Build Coastguard Worker SEQ_MSG_FAIL_TIMEOUT,
56*1c60b9acSAndroid Build Coastguard Worker };
57*1c60b9acSAndroid Build Coastguard Worker
58*1c60b9acSAndroid Build Coastguard Worker /*
59*1c60b9acSAndroid Build Coastguard Worker * We get called back when the unit test transport has decided if the test
60*1c60b9acSAndroid Build Coastguard Worker * passed or failed. We get the priv, and report to the sequencer message queue
61*1c60b9acSAndroid Build Coastguard Worker * what the result was.
62*1c60b9acSAndroid Build Coastguard Worker */
63*1c60b9acSAndroid Build Coastguard Worker
64*1c60b9acSAndroid Build Coastguard Worker static int
unit_test_result_cb(const void * cb_user,int disposition)65*1c60b9acSAndroid Build Coastguard Worker unit_test_result_cb(const void *cb_user, int disposition)
66*1c60b9acSAndroid Build Coastguard Worker {
67*1c60b9acSAndroid Build Coastguard Worker const struct lws_seq_test_sequencer *s =
68*1c60b9acSAndroid Build Coastguard Worker (const struct lws_seq_test_sequencer *)cb_user;
69*1c60b9acSAndroid Build Coastguard Worker int r;
70*1c60b9acSAndroid Build Coastguard Worker
71*1c60b9acSAndroid Build Coastguard Worker lwsl_debug("%s: disp %d\n", __func__, disposition);
72*1c60b9acSAndroid Build Coastguard Worker
73*1c60b9acSAndroid Build Coastguard Worker switch (disposition) {
74*1c60b9acSAndroid Build Coastguard Worker case LPE_FAILED_UNEXPECTED_PASS:
75*1c60b9acSAndroid Build Coastguard Worker case LPE_FAILED_UNEXPECTED_CLOSE:
76*1c60b9acSAndroid Build Coastguard Worker case LPE_FAILED:
77*1c60b9acSAndroid Build Coastguard Worker r = SEQ_MSG_FAIL;
78*1c60b9acSAndroid Build Coastguard Worker break;
79*1c60b9acSAndroid Build Coastguard Worker
80*1c60b9acSAndroid Build Coastguard Worker case LPE_FAILED_UNEXPECTED_TIMEOUT:
81*1c60b9acSAndroid Build Coastguard Worker r = SEQ_MSG_FAIL_TIMEOUT;
82*1c60b9acSAndroid Build Coastguard Worker break;
83*1c60b9acSAndroid Build Coastguard Worker
84*1c60b9acSAndroid Build Coastguard Worker case LPE_SUCCEEDED:
85*1c60b9acSAndroid Build Coastguard Worker r = SEQ_MSG_PASS;
86*1c60b9acSAndroid Build Coastguard Worker break;
87*1c60b9acSAndroid Build Coastguard Worker
88*1c60b9acSAndroid Build Coastguard Worker default:
89*1c60b9acSAndroid Build Coastguard Worker assert(0);
90*1c60b9acSAndroid Build Coastguard Worker return -1;
91*1c60b9acSAndroid Build Coastguard Worker }
92*1c60b9acSAndroid Build Coastguard Worker
93*1c60b9acSAndroid Build Coastguard Worker lws_seq_queue_event(s->unit_test_seq, r, NULL, NULL);
94*1c60b9acSAndroid Build Coastguard Worker
95*1c60b9acSAndroid Build Coastguard Worker ((struct lws_seq_test_sequencer *)s)->instance = NULL;
96*1c60b9acSAndroid Build Coastguard Worker
97*1c60b9acSAndroid Build Coastguard Worker return 0;
98*1c60b9acSAndroid Build Coastguard Worker }
99*1c60b9acSAndroid Build Coastguard Worker
100*1c60b9acSAndroid Build Coastguard Worker /*
101*1c60b9acSAndroid Build Coastguard Worker * We receive the unit test result callback's messages via the message queue.
102*1c60b9acSAndroid Build Coastguard Worker *
103*1c60b9acSAndroid Build Coastguard Worker * We log the results and always move on to the next test until there are no
104*1c60b9acSAndroid Build Coastguard Worker * more tests.
105*1c60b9acSAndroid Build Coastguard Worker */
106*1c60b9acSAndroid Build Coastguard Worker
107*1c60b9acSAndroid Build Coastguard Worker static lws_seq_cb_return_t
test_sequencer_cb(struct lws_sequencer * seq,void * user,int event,void * data,void * aux)108*1c60b9acSAndroid Build Coastguard Worker test_sequencer_cb(struct lws_sequencer *seq, void *user, int event, void *data,
109*1c60b9acSAndroid Build Coastguard Worker void *aux)
110*1c60b9acSAndroid Build Coastguard Worker {
111*1c60b9acSAndroid Build Coastguard Worker struct lws_seq_test_sequencer *s =
112*1c60b9acSAndroid Build Coastguard Worker (struct lws_seq_test_sequencer *)user;
113*1c60b9acSAndroid Build Coastguard Worker lws_unit_test_packet_t *exp = (lws_unit_test_packet_t *)
114*1c60b9acSAndroid Build Coastguard Worker s->args.tests[s->state].expect_array;
115*1c60b9acSAndroid Build Coastguard Worker lws_abs_t test_abs;
116*1c60b9acSAndroid Build Coastguard Worker
117*1c60b9acSAndroid Build Coastguard Worker switch ((int)event) {
118*1c60b9acSAndroid Build Coastguard Worker case LWSSEQ_CREATED: /* our sequencer just got started */
119*1c60b9acSAndroid Build Coastguard Worker lwsl_notice("%s: %s: created\n", __func__,
120*1c60b9acSAndroid Build Coastguard Worker lws_seq_name(seq));
121*1c60b9acSAndroid Build Coastguard Worker s->state = 0; /* first thing we'll do is the first url */
122*1c60b9acSAndroid Build Coastguard Worker goto step;
123*1c60b9acSAndroid Build Coastguard Worker
124*1c60b9acSAndroid Build Coastguard Worker case LWSSEQ_DESTROYED:
125*1c60b9acSAndroid Build Coastguard Worker /*
126*1c60b9acSAndroid Build Coastguard Worker * We are going down... if we have a child unit test sequencer
127*1c60b9acSAndroid Build Coastguard Worker * still around inform and destroy it
128*1c60b9acSAndroid Build Coastguard Worker */
129*1c60b9acSAndroid Build Coastguard Worker if (s->instance) {
130*1c60b9acSAndroid Build Coastguard Worker s->instance->at->close(s->instance);
131*1c60b9acSAndroid Build Coastguard Worker s->instance = NULL;
132*1c60b9acSAndroid Build Coastguard Worker }
133*1c60b9acSAndroid Build Coastguard Worker break;
134*1c60b9acSAndroid Build Coastguard Worker
135*1c60b9acSAndroid Build Coastguard Worker case SEQ_MSG_FAIL_TIMEOUT: /* current step timed out */
136*1c60b9acSAndroid Build Coastguard Worker if (exp->flags & LWS_AUT_EXPECT_SHOULD_TIMEOUT) {
137*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: test %d got expected timeout\n",
138*1c60b9acSAndroid Build Coastguard Worker __func__, s->state);
139*1c60b9acSAndroid Build Coastguard Worker
140*1c60b9acSAndroid Build Coastguard Worker goto pass;
141*1c60b9acSAndroid Build Coastguard Worker }
142*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: seq timed out at step %d\n", __func__, s->state);
143*1c60b9acSAndroid Build Coastguard Worker
144*1c60b9acSAndroid Build Coastguard Worker s->args.results[s->state] = LPE_FAILED_UNEXPECTED_TIMEOUT;
145*1c60b9acSAndroid Build Coastguard Worker goto done; /* always move on to the next test */
146*1c60b9acSAndroid Build Coastguard Worker
147*1c60b9acSAndroid Build Coastguard Worker case SEQ_MSG_FAIL:
148*1c60b9acSAndroid Build Coastguard Worker if (exp->flags & LWS_AUT_EXPECT_SHOULD_FAIL) {
149*1c60b9acSAndroid Build Coastguard Worker /*
150*1c60b9acSAndroid Build Coastguard Worker * in this case, we expected to fail like this, it's OK
151*1c60b9acSAndroid Build Coastguard Worker */
152*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: test %d failed as expected\n",
153*1c60b9acSAndroid Build Coastguard Worker __func__, s->state);
154*1c60b9acSAndroid Build Coastguard Worker
155*1c60b9acSAndroid Build Coastguard Worker goto pass; /* always move on to the next test */
156*1c60b9acSAndroid Build Coastguard Worker }
157*1c60b9acSAndroid Build Coastguard Worker
158*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: seq failed at step %d\n", __func__, s->state);
159*1c60b9acSAndroid Build Coastguard Worker
160*1c60b9acSAndroid Build Coastguard Worker s->args.results[s->state] = LPE_FAILED;
161*1c60b9acSAndroid Build Coastguard Worker goto done; /* always move on to the next test */
162*1c60b9acSAndroid Build Coastguard Worker
163*1c60b9acSAndroid Build Coastguard Worker case SEQ_MSG_PASS:
164*1c60b9acSAndroid Build Coastguard Worker if (exp->flags & (LWS_AUT_EXPECT_SHOULD_FAIL |
165*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_SHOULD_TIMEOUT)) {
166*1c60b9acSAndroid Build Coastguard Worker /*
167*1c60b9acSAndroid Build Coastguard Worker * In these specific cases, done would be a failure,
168*1c60b9acSAndroid Build Coastguard Worker * we expected to timeout or fail
169*1c60b9acSAndroid Build Coastguard Worker */
170*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: seq failed at step %d\n", __func__,
171*1c60b9acSAndroid Build Coastguard Worker s->state);
172*1c60b9acSAndroid Build Coastguard Worker
173*1c60b9acSAndroid Build Coastguard Worker s->args.results[s->state] = LPE_FAILED_UNEXPECTED_PASS;
174*1c60b9acSAndroid Build Coastguard Worker
175*1c60b9acSAndroid Build Coastguard Worker goto done; /* always move on to the next test */
176*1c60b9acSAndroid Build Coastguard Worker }
177*1c60b9acSAndroid Build Coastguard Worker lwsl_info("%s: seq done test %d\n", __func__, s->state);
178*1c60b9acSAndroid Build Coastguard Worker pass:
179*1c60b9acSAndroid Build Coastguard Worker (*s->args.count_passes)++;
180*1c60b9acSAndroid Build Coastguard Worker s->args.results[s->state] = LPE_SUCCEEDED;
181*1c60b9acSAndroid Build Coastguard Worker
182*1c60b9acSAndroid Build Coastguard Worker done:
183*1c60b9acSAndroid Build Coastguard Worker lws_seq_timeout_us(lws_seq_from_user(s), LWSSEQTO_NONE);
184*1c60b9acSAndroid Build Coastguard Worker s->state++;
185*1c60b9acSAndroid Build Coastguard Worker step:
186*1c60b9acSAndroid Build Coastguard Worker if (!s->args.tests[s->state].name) {
187*1c60b9acSAndroid Build Coastguard Worker /* the sequence has completed */
188*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: sequence completed OK\n", __func__);
189*1c60b9acSAndroid Build Coastguard Worker
190*1c60b9acSAndroid Build Coastguard Worker if (s->args.cb)
191*1c60b9acSAndroid Build Coastguard Worker s->args.cb(s->args.cb_user);
192*1c60b9acSAndroid Build Coastguard Worker
193*1c60b9acSAndroid Build Coastguard Worker return LWSSEQ_RET_DESTROY;
194*1c60b9acSAndroid Build Coastguard Worker }
195*1c60b9acSAndroid Build Coastguard Worker lwsl_info("%s: starting test %d\n", __func__, s->state);
196*1c60b9acSAndroid Build Coastguard Worker
197*1c60b9acSAndroid Build Coastguard Worker if (s->state >= s->args.results_max) {
198*1c60b9acSAndroid Build Coastguard Worker lwsl_err("%s: results array is too small\n", __func__);
199*1c60b9acSAndroid Build Coastguard Worker
200*1c60b9acSAndroid Build Coastguard Worker return LWSSEQ_RET_DESTROY;
201*1c60b9acSAndroid Build Coastguard Worker }
202*1c60b9acSAndroid Build Coastguard Worker test_abs = s->original_abs;
203*1c60b9acSAndroid Build Coastguard Worker s->uttt[0].name_index = LTMI_PEER_V_EXPECT_TEST;
204*1c60b9acSAndroid Build Coastguard Worker s->uttt[0].u.value = (void *)&s->args.tests[s->state];
205*1c60b9acSAndroid Build Coastguard Worker s->uttt[1].name_index = LTMI_PEER_V_EXPECT_RESULT_CB;
206*1c60b9acSAndroid Build Coastguard Worker s->uttt[1].u.value = (void *)unit_test_result_cb;
207*1c60b9acSAndroid Build Coastguard Worker s->uttt[2].name_index = LTMI_PEER_V_EXPECT_RESULT_CB_ARG;
208*1c60b9acSAndroid Build Coastguard Worker s->uttt[2].u.value = (void *)s;
209*1c60b9acSAndroid Build Coastguard Worker /* give the unit test transport the test tokens */
210*1c60b9acSAndroid Build Coastguard Worker test_abs.at_tokens = s->uttt;
211*1c60b9acSAndroid Build Coastguard Worker
212*1c60b9acSAndroid Build Coastguard Worker s->instance = lws_abs_bind_and_create_instance(&test_abs);
213*1c60b9acSAndroid Build Coastguard Worker if (!s->instance) {
214*1c60b9acSAndroid Build Coastguard Worker lwsl_notice("%s: failed to create step %d unit test\n",
215*1c60b9acSAndroid Build Coastguard Worker __func__, s->state);
216*1c60b9acSAndroid Build Coastguard Worker
217*1c60b9acSAndroid Build Coastguard Worker return LWSSEQ_RET_DESTROY;
218*1c60b9acSAndroid Build Coastguard Worker }
219*1c60b9acSAndroid Build Coastguard Worker (*s->args.count_tests)++;
220*1c60b9acSAndroid Build Coastguard Worker break;
221*1c60b9acSAndroid Build Coastguard Worker
222*1c60b9acSAndroid Build Coastguard Worker default:
223*1c60b9acSAndroid Build Coastguard Worker break;
224*1c60b9acSAndroid Build Coastguard Worker }
225*1c60b9acSAndroid Build Coastguard Worker
226*1c60b9acSAndroid Build Coastguard Worker return LWSSEQ_RET_CONTINUE;
227*1c60b9acSAndroid Build Coastguard Worker }
228*1c60b9acSAndroid Build Coastguard Worker
229*1c60b9acSAndroid Build Coastguard Worker
230*1c60b9acSAndroid Build Coastguard Worker /*
231*1c60b9acSAndroid Build Coastguard Worker * Creates an lws_sequencer to manage the test sequence
232*1c60b9acSAndroid Build Coastguard Worker */
233*1c60b9acSAndroid Build Coastguard Worker
234*1c60b9acSAndroid Build Coastguard Worker int
lws_abs_unit_test_sequencer(const lws_test_sequencer_args_t * args)235*1c60b9acSAndroid Build Coastguard Worker lws_abs_unit_test_sequencer(const lws_test_sequencer_args_t *args)
236*1c60b9acSAndroid Build Coastguard Worker {
237*1c60b9acSAndroid Build Coastguard Worker struct lws_seq_test_sequencer *s;
238*1c60b9acSAndroid Build Coastguard Worker struct lws_sequencer *seq;
239*1c60b9acSAndroid Build Coastguard Worker lws_seq_info_t i;
240*1c60b9acSAndroid Build Coastguard Worker
241*1c60b9acSAndroid Build Coastguard Worker memset(&i, 0, sizeof(i));
242*1c60b9acSAndroid Build Coastguard Worker i.context = args->abs->vh->context;
243*1c60b9acSAndroid Build Coastguard Worker i.user_size = sizeof(struct lws_seq_test_sequencer);
244*1c60b9acSAndroid Build Coastguard Worker i.puser = (void **)&s;
245*1c60b9acSAndroid Build Coastguard Worker i.cb = test_sequencer_cb;
246*1c60b9acSAndroid Build Coastguard Worker i.name = "test-seq";
247*1c60b9acSAndroid Build Coastguard Worker
248*1c60b9acSAndroid Build Coastguard Worker /*
249*1c60b9acSAndroid Build Coastguard Worker * Create a sequencer in the event loop to manage the tests
250*1c60b9acSAndroid Build Coastguard Worker */
251*1c60b9acSAndroid Build Coastguard Worker
252*1c60b9acSAndroid Build Coastguard Worker seq = lws_seq_create(&i);
253*1c60b9acSAndroid Build Coastguard Worker if (!seq) {
254*1c60b9acSAndroid Build Coastguard Worker lwsl_err("%s: unable to create sequencer\n", __func__);
255*1c60b9acSAndroid Build Coastguard Worker return 1;
256*1c60b9acSAndroid Build Coastguard Worker }
257*1c60b9acSAndroid Build Coastguard Worker
258*1c60b9acSAndroid Build Coastguard Worker /*
259*1c60b9acSAndroid Build Coastguard Worker * Take a copy of the original lws_abs_t we were passed so we can use
260*1c60b9acSAndroid Build Coastguard Worker * it as the basis of the lws_abs_t we create the individual tests with
261*1c60b9acSAndroid Build Coastguard Worker */
262*1c60b9acSAndroid Build Coastguard Worker s->original_abs = *args->abs;
263*1c60b9acSAndroid Build Coastguard Worker
264*1c60b9acSAndroid Build Coastguard Worker s->args = *args;
265*1c60b9acSAndroid Build Coastguard Worker
266*1c60b9acSAndroid Build Coastguard Worker s->context = args->abs->vh->context;
267*1c60b9acSAndroid Build Coastguard Worker s->vhost = args->abs->vh;
268*1c60b9acSAndroid Build Coastguard Worker s->unit_test_seq = seq;
269*1c60b9acSAndroid Build Coastguard Worker
270*1c60b9acSAndroid Build Coastguard Worker *s->args.count_tests = 0;
271*1c60b9acSAndroid Build Coastguard Worker *s->args.count_passes = 0;
272*1c60b9acSAndroid Build Coastguard Worker
273*1c60b9acSAndroid Build Coastguard Worker return 0;
274*1c60b9acSAndroid Build Coastguard Worker }
275