1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker * lws-unit-tests-smtp-client
3*1c60b9acSAndroid Build Coastguard Worker *
4*1c60b9acSAndroid Build Coastguard Worker * Written in 2010-2019 by Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker *
6*1c60b9acSAndroid Build Coastguard Worker * This file is made available under the Creative Commons CC0 1.0
7*1c60b9acSAndroid Build Coastguard Worker * Universal Public Domain Dedication.
8*1c60b9acSAndroid Build Coastguard Worker *
9*1c60b9acSAndroid Build Coastguard Worker * This performs unit tests for the SMTP client abstract protocol
10*1c60b9acSAndroid Build Coastguard Worker */
11*1c60b9acSAndroid Build Coastguard Worker
12*1c60b9acSAndroid Build Coastguard Worker #include <libwebsockets.h>
13*1c60b9acSAndroid Build Coastguard Worker
14*1c60b9acSAndroid Build Coastguard Worker #include <signal.h>
15*1c60b9acSAndroid Build Coastguard Worker
16*1c60b9acSAndroid Build Coastguard Worker static int interrupted, results[10], count_tests, count_passes;
17*1c60b9acSAndroid Build Coastguard Worker
18*1c60b9acSAndroid Build Coastguard Worker static int
email_sent_or_failed(struct lws_smtp_email * email,void * buf,size_t len)19*1c60b9acSAndroid Build Coastguard Worker email_sent_or_failed(struct lws_smtp_email *email, void *buf, size_t len)
20*1c60b9acSAndroid Build Coastguard Worker {
21*1c60b9acSAndroid Build Coastguard Worker free(email);
22*1c60b9acSAndroid Build Coastguard Worker
23*1c60b9acSAndroid Build Coastguard Worker return 0;
24*1c60b9acSAndroid Build Coastguard Worker }
25*1c60b9acSAndroid Build Coastguard Worker
26*1c60b9acSAndroid Build Coastguard Worker /*
27*1c60b9acSAndroid Build Coastguard Worker * The test helper calls this on the instance it created to prepare it for
28*1c60b9acSAndroid Build Coastguard Worker * the test. In our case, we need to queue up a test email to send on the
29*1c60b9acSAndroid Build Coastguard Worker * smtp client abstract protocol.
30*1c60b9acSAndroid Build Coastguard Worker */
31*1c60b9acSAndroid Build Coastguard Worker
32*1c60b9acSAndroid Build Coastguard Worker static int
smtp_test_instance_init(lws_abs_t * instance)33*1c60b9acSAndroid Build Coastguard Worker smtp_test_instance_init(lws_abs_t *instance)
34*1c60b9acSAndroid Build Coastguard Worker {
35*1c60b9acSAndroid Build Coastguard Worker lws_smtp_email_t *email = (lws_smtp_email_t *)
36*1c60b9acSAndroid Build Coastguard Worker malloc(sizeof(*email) + 2048);
37*1c60b9acSAndroid Build Coastguard Worker
38*1c60b9acSAndroid Build Coastguard Worker if (!email)
39*1c60b9acSAndroid Build Coastguard Worker return 1;
40*1c60b9acSAndroid Build Coastguard Worker
41*1c60b9acSAndroid Build Coastguard Worker /* attach an email to it */
42*1c60b9acSAndroid Build Coastguard Worker
43*1c60b9acSAndroid Build Coastguard Worker memset(email, 0, sizeof(*email));
44*1c60b9acSAndroid Build Coastguard Worker email->data = NULL /* email specific user data */;
45*1c60b9acSAndroid Build Coastguard Worker email->email_from = "[email protected]";
46*1c60b9acSAndroid Build Coastguard Worker email->email_to = "[email protected]";
47*1c60b9acSAndroid Build Coastguard Worker email->payload = (void *)&email[1];
48*1c60b9acSAndroid Build Coastguard Worker
49*1c60b9acSAndroid Build Coastguard Worker lws_snprintf((char *)email->payload, 2048,
50*1c60b9acSAndroid Build Coastguard Worker "From: [email protected]\n"
51*1c60b9acSAndroid Build Coastguard Worker "To: %s\n"
52*1c60b9acSAndroid Build Coastguard Worker "Subject: Test email for lws smtp-client\n"
53*1c60b9acSAndroid Build Coastguard Worker "\n"
54*1c60b9acSAndroid Build Coastguard Worker "Hello this was an api test for lws smtp-client\n"
55*1c60b9acSAndroid Build Coastguard Worker "\r\n.\r\n", "[email protected]");
56*1c60b9acSAndroid Build Coastguard Worker email->done = email_sent_or_failed;
57*1c60b9acSAndroid Build Coastguard Worker
58*1c60b9acSAndroid Build Coastguard Worker if (lws_smtpc_add_email(instance, email)) {
59*1c60b9acSAndroid Build Coastguard Worker lwsl_err("%s: failed to add email\n", __func__);
60*1c60b9acSAndroid Build Coastguard Worker return 1;
61*1c60b9acSAndroid Build Coastguard Worker }
62*1c60b9acSAndroid Build Coastguard Worker
63*1c60b9acSAndroid Build Coastguard Worker return 0;
64*1c60b9acSAndroid Build Coastguard Worker }
65*1c60b9acSAndroid Build Coastguard Worker
66*1c60b9acSAndroid Build Coastguard Worker /*
67*1c60b9acSAndroid Build Coastguard Worker * from https://en.wikipedia.org/wiki/Simple_Mail_Transfer_Protocol
68*1c60b9acSAndroid Build Coastguard Worker *
69*1c60b9acSAndroid Build Coastguard Worker * test vector sent to protocol
70*1c60b9acSAndroid Build Coastguard Worker * test vector received from protocol
71*1c60b9acSAndroid Build Coastguard Worker */
72*1c60b9acSAndroid Build Coastguard Worker
73*1c60b9acSAndroid Build Coastguard Worker static lws_unit_test_packet_t test_send1[] = {
74*1c60b9acSAndroid Build Coastguard Worker {
75*1c60b9acSAndroid Build Coastguard Worker "220 smtp.example.com ESMTP Postfix",
76*1c60b9acSAndroid Build Coastguard Worker smtp_test_instance_init, 34, LWS_AUT_EXPECT_RX
77*1c60b9acSAndroid Build Coastguard Worker }, {
78*1c60b9acSAndroid Build Coastguard Worker "HELO lws-test-client\x0a",
79*1c60b9acSAndroid Build Coastguard Worker NULL, 21, LWS_AUT_EXPECT_TX
80*1c60b9acSAndroid Build Coastguard Worker }, {
81*1c60b9acSAndroid Build Coastguard Worker "250 smtp.example.com, I am glad to meet you",
82*1c60b9acSAndroid Build Coastguard Worker NULL, 43, LWS_AUT_EXPECT_RX
83*1c60b9acSAndroid Build Coastguard Worker }, {
84*1c60b9acSAndroid Build Coastguard Worker "MAIL FROM: <[email protected]>\x0a",
85*1c60b9acSAndroid Build Coastguard Worker NULL, 33, LWS_AUT_EXPECT_TX
86*1c60b9acSAndroid Build Coastguard Worker }, {
87*1c60b9acSAndroid Build Coastguard Worker "250 Ok",
88*1c60b9acSAndroid Build Coastguard Worker NULL, 6, LWS_AUT_EXPECT_RX
89*1c60b9acSAndroid Build Coastguard Worker }, {
90*1c60b9acSAndroid Build Coastguard Worker "RCPT TO: <[email protected]>\x0a",
91*1c60b9acSAndroid Build Coastguard Worker NULL, 28, LWS_AUT_EXPECT_TX
92*1c60b9acSAndroid Build Coastguard Worker }, {
93*1c60b9acSAndroid Build Coastguard Worker "250 Ok",
94*1c60b9acSAndroid Build Coastguard Worker NULL, 6, LWS_AUT_EXPECT_RX
95*1c60b9acSAndroid Build Coastguard Worker }, {
96*1c60b9acSAndroid Build Coastguard Worker "DATA\x0a",
97*1c60b9acSAndroid Build Coastguard Worker NULL, 5, LWS_AUT_EXPECT_TX
98*1c60b9acSAndroid Build Coastguard Worker }, {
99*1c60b9acSAndroid Build Coastguard Worker "354 End data with <CR><LF>.<CR><LF>\x0a",
100*1c60b9acSAndroid Build Coastguard Worker NULL, 35, LWS_AUT_EXPECT_RX
101*1c60b9acSAndroid Build Coastguard Worker }, {
102*1c60b9acSAndroid Build Coastguard Worker "From: [email protected]\n"
103*1c60b9acSAndroid Build Coastguard Worker "To: [email protected]\n"
104*1c60b9acSAndroid Build Coastguard Worker "Subject: Test email for lws smtp-client\n"
105*1c60b9acSAndroid Build Coastguard Worker "\n"
106*1c60b9acSAndroid Build Coastguard Worker "Hello this was an api test for lws smtp-client\n"
107*1c60b9acSAndroid Build Coastguard Worker "\r\n.\r\n",
108*1c60b9acSAndroid Build Coastguard Worker NULL, 27 + 21 + 39 + 1 + 47 + 5, LWS_AUT_EXPECT_TX
109*1c60b9acSAndroid Build Coastguard Worker }, {
110*1c60b9acSAndroid Build Coastguard Worker "250 Ok: queued as 12345\x0a",
111*1c60b9acSAndroid Build Coastguard Worker NULL, 23, LWS_AUT_EXPECT_RX
112*1c60b9acSAndroid Build Coastguard Worker }, {
113*1c60b9acSAndroid Build Coastguard Worker "quit\x0a",
114*1c60b9acSAndroid Build Coastguard Worker NULL, 5, LWS_AUT_EXPECT_TX
115*1c60b9acSAndroid Build Coastguard Worker }, {
116*1c60b9acSAndroid Build Coastguard Worker "221 Bye\x0a",
117*1c60b9acSAndroid Build Coastguard Worker NULL, 7, LWS_AUT_EXPECT_RX |
118*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_LOCAL_CLOSE |
119*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_DO_REMOTE_CLOSE |
120*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_TEST_END
121*1c60b9acSAndroid Build Coastguard Worker }, { /* sentinel */
122*1c60b9acSAndroid Build Coastguard Worker
123*1c60b9acSAndroid Build Coastguard Worker }
124*1c60b9acSAndroid Build Coastguard Worker };
125*1c60b9acSAndroid Build Coastguard Worker
126*1c60b9acSAndroid Build Coastguard Worker
127*1c60b9acSAndroid Build Coastguard Worker static lws_unit_test_packet_t test_send2[] = {
128*1c60b9acSAndroid Build Coastguard Worker {
129*1c60b9acSAndroid Build Coastguard Worker "220 smtp.example.com ESMTP Postfix",
130*1c60b9acSAndroid Build Coastguard Worker smtp_test_instance_init, 34, LWS_AUT_EXPECT_RX
131*1c60b9acSAndroid Build Coastguard Worker }, {
132*1c60b9acSAndroid Build Coastguard Worker "HELO lws-test-client\x0a",
133*1c60b9acSAndroid Build Coastguard Worker NULL, 21, LWS_AUT_EXPECT_TX
134*1c60b9acSAndroid Build Coastguard Worker }, {
135*1c60b9acSAndroid Build Coastguard Worker "250 smtp.example.com, I am glad to meet you",
136*1c60b9acSAndroid Build Coastguard Worker NULL, 43, LWS_AUT_EXPECT_RX
137*1c60b9acSAndroid Build Coastguard Worker }, {
138*1c60b9acSAndroid Build Coastguard Worker "MAIL FROM: <[email protected]>\x0a",
139*1c60b9acSAndroid Build Coastguard Worker NULL, 33, LWS_AUT_EXPECT_TX
140*1c60b9acSAndroid Build Coastguard Worker }, {
141*1c60b9acSAndroid Build Coastguard Worker "500 Service Unavailable",
142*1c60b9acSAndroid Build Coastguard Worker NULL, 23, LWS_AUT_EXPECT_RX |
143*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_DO_REMOTE_CLOSE |
144*1c60b9acSAndroid Build Coastguard Worker LWS_AUT_EXPECT_TEST_END
145*1c60b9acSAndroid Build Coastguard Worker }, { /* sentinel */
146*1c60b9acSAndroid Build Coastguard Worker
147*1c60b9acSAndroid Build Coastguard Worker }
148*1c60b9acSAndroid Build Coastguard Worker };
149*1c60b9acSAndroid Build Coastguard Worker
150*1c60b9acSAndroid Build Coastguard Worker static lws_unit_test_t tests[] = {
151*1c60b9acSAndroid Build Coastguard Worker { "sending", test_send1, 3 },
152*1c60b9acSAndroid Build Coastguard Worker { "rejected", test_send2, 3 },
153*1c60b9acSAndroid Build Coastguard Worker { } /* sentinel */
154*1c60b9acSAndroid Build Coastguard Worker };
155*1c60b9acSAndroid Build Coastguard Worker
156*1c60b9acSAndroid Build Coastguard Worker static void
sigint_handler(int sig)157*1c60b9acSAndroid Build Coastguard Worker sigint_handler(int sig)
158*1c60b9acSAndroid Build Coastguard Worker {
159*1c60b9acSAndroid Build Coastguard Worker interrupted = 1;
160*1c60b9acSAndroid Build Coastguard Worker }
161*1c60b9acSAndroid Build Coastguard Worker
162*1c60b9acSAndroid Build Coastguard Worker /*
163*1c60b9acSAndroid Build Coastguard Worker * set the HELO our SMTP client will use
164*1c60b9acSAndroid Build Coastguard Worker */
165*1c60b9acSAndroid Build Coastguard Worker
166*1c60b9acSAndroid Build Coastguard Worker static const lws_token_map_t smtp_ap_tokens[] = {
167*1c60b9acSAndroid Build Coastguard Worker {
168*1c60b9acSAndroid Build Coastguard Worker .u = { .value = "lws-test-client" },
169*1c60b9acSAndroid Build Coastguard Worker .name_index = LTMI_PSMTP_V_HELO,
170*1c60b9acSAndroid Build Coastguard Worker }, { /* sentinel */
171*1c60b9acSAndroid Build Coastguard Worker }
172*1c60b9acSAndroid Build Coastguard Worker };
173*1c60b9acSAndroid Build Coastguard Worker
174*1c60b9acSAndroid Build Coastguard Worker void
tests_completion_cb(const void * cb_user)175*1c60b9acSAndroid Build Coastguard Worker tests_completion_cb(const void *cb_user)
176*1c60b9acSAndroid Build Coastguard Worker {
177*1c60b9acSAndroid Build Coastguard Worker interrupted = 1;
178*1c60b9acSAndroid Build Coastguard Worker }
179*1c60b9acSAndroid Build Coastguard Worker
main(int argc,const char ** argv)180*1c60b9acSAndroid Build Coastguard Worker int main(int argc, const char **argv)
181*1c60b9acSAndroid Build Coastguard Worker {
182*1c60b9acSAndroid Build Coastguard Worker int n = 1, logs = LLL_USER | LLL_ERR | LLL_WARN | LLL_NOTICE;
183*1c60b9acSAndroid Build Coastguard Worker struct lws_context_creation_info info;
184*1c60b9acSAndroid Build Coastguard Worker lws_test_sequencer_args_t args;
185*1c60b9acSAndroid Build Coastguard Worker struct lws_context *context;
186*1c60b9acSAndroid Build Coastguard Worker lws_abs_t *abs = NULL;
187*1c60b9acSAndroid Build Coastguard Worker struct lws_vhost *vh;
188*1c60b9acSAndroid Build Coastguard Worker const char *p;
189*1c60b9acSAndroid Build Coastguard Worker
190*1c60b9acSAndroid Build Coastguard Worker /* the normal lws init */
191*1c60b9acSAndroid Build Coastguard Worker
192*1c60b9acSAndroid Build Coastguard Worker signal(SIGINT, sigint_handler);
193*1c60b9acSAndroid Build Coastguard Worker
194*1c60b9acSAndroid Build Coastguard Worker if ((p = lws_cmdline_option(argc, argv, "-d")))
195*1c60b9acSAndroid Build Coastguard Worker logs = atoi(p);
196*1c60b9acSAndroid Build Coastguard Worker
197*1c60b9acSAndroid Build Coastguard Worker lws_set_log_level(logs, NULL);
198*1c60b9acSAndroid Build Coastguard Worker lwsl_user("LWS API selftest: SMTP client unit tests\n");
199*1c60b9acSAndroid Build Coastguard Worker
200*1c60b9acSAndroid Build Coastguard Worker memset(&info, 0, sizeof info); /* otherwise uninitialized garbage */
201*1c60b9acSAndroid Build Coastguard Worker info.port = CONTEXT_PORT_NO_LISTEN;
202*1c60b9acSAndroid Build Coastguard Worker info.options = LWS_SERVER_OPTION_EXPLICIT_VHOSTS;
203*1c60b9acSAndroid Build Coastguard Worker
204*1c60b9acSAndroid Build Coastguard Worker context = lws_create_context(&info);
205*1c60b9acSAndroid Build Coastguard Worker if (!context) {
206*1c60b9acSAndroid Build Coastguard Worker lwsl_err("lws init failed\n");
207*1c60b9acSAndroid Build Coastguard Worker return 1;
208*1c60b9acSAndroid Build Coastguard Worker }
209*1c60b9acSAndroid Build Coastguard Worker
210*1c60b9acSAndroid Build Coastguard Worker vh = lws_create_vhost(context, &info);
211*1c60b9acSAndroid Build Coastguard Worker if (!vh) {
212*1c60b9acSAndroid Build Coastguard Worker lwsl_err("Failed to create first vhost\n");
213*1c60b9acSAndroid Build Coastguard Worker goto bail1;
214*1c60b9acSAndroid Build Coastguard Worker }
215*1c60b9acSAndroid Build Coastguard Worker
216*1c60b9acSAndroid Build Coastguard Worker /* create the abs used to create connections */
217*1c60b9acSAndroid Build Coastguard Worker
218*1c60b9acSAndroid Build Coastguard Worker abs = lws_abstract_alloc(vh, NULL, "smtp.unit_test",
219*1c60b9acSAndroid Build Coastguard Worker &smtp_ap_tokens[0], NULL);
220*1c60b9acSAndroid Build Coastguard Worker if (!abs)
221*1c60b9acSAndroid Build Coastguard Worker goto bail1;
222*1c60b9acSAndroid Build Coastguard Worker
223*1c60b9acSAndroid Build Coastguard Worker /* configure the test sequencer */
224*1c60b9acSAndroid Build Coastguard Worker
225*1c60b9acSAndroid Build Coastguard Worker args.abs = abs;
226*1c60b9acSAndroid Build Coastguard Worker args.tests = tests;
227*1c60b9acSAndroid Build Coastguard Worker args.results = results;
228*1c60b9acSAndroid Build Coastguard Worker args.results_max = LWS_ARRAY_SIZE(results);
229*1c60b9acSAndroid Build Coastguard Worker args.count_tests = &count_tests;
230*1c60b9acSAndroid Build Coastguard Worker args.count_passes = &count_passes;
231*1c60b9acSAndroid Build Coastguard Worker args.cb = tests_completion_cb;
232*1c60b9acSAndroid Build Coastguard Worker args.cb_user = NULL;
233*1c60b9acSAndroid Build Coastguard Worker
234*1c60b9acSAndroid Build Coastguard Worker if (lws_abs_unit_test_sequencer(&args)) {
235*1c60b9acSAndroid Build Coastguard Worker lwsl_err("%s: failed to create test sequencer\n", __func__);
236*1c60b9acSAndroid Build Coastguard Worker goto bail1;
237*1c60b9acSAndroid Build Coastguard Worker }
238*1c60b9acSAndroid Build Coastguard Worker
239*1c60b9acSAndroid Build Coastguard Worker /* the usual lws event loop */
240*1c60b9acSAndroid Build Coastguard Worker
241*1c60b9acSAndroid Build Coastguard Worker while (n >= 0 && !interrupted)
242*1c60b9acSAndroid Build Coastguard Worker n = lws_service(context, 0);
243*1c60b9acSAndroid Build Coastguard Worker
244*1c60b9acSAndroid Build Coastguard Worker /* describe the overall test results */
245*1c60b9acSAndroid Build Coastguard Worker
246*1c60b9acSAndroid Build Coastguard Worker lwsl_user("%s: %d tests %d fail\n", __func__, count_tests,
247*1c60b9acSAndroid Build Coastguard Worker count_tests - count_passes);
248*1c60b9acSAndroid Build Coastguard Worker for (n = 0; n < count_tests; n++)
249*1c60b9acSAndroid Build Coastguard Worker lwsl_user(" test %d: %s\n", n,
250*1c60b9acSAndroid Build Coastguard Worker lws_unit_test_result_name(results[n]));
251*1c60b9acSAndroid Build Coastguard Worker
252*1c60b9acSAndroid Build Coastguard Worker bail1:
253*1c60b9acSAndroid Build Coastguard Worker lwsl_user("Completed: %s\n",
254*1c60b9acSAndroid Build Coastguard Worker !count_tests || count_passes != count_tests ? "FAIL" : "PASS");
255*1c60b9acSAndroid Build Coastguard Worker
256*1c60b9acSAndroid Build Coastguard Worker lws_context_destroy(context);
257*1c60b9acSAndroid Build Coastguard Worker
258*1c60b9acSAndroid Build Coastguard Worker lws_abstract_free(&abs);
259*1c60b9acSAndroid Build Coastguard Worker
260*1c60b9acSAndroid Build Coastguard Worker return !count_tests || count_passes != count_tests;
261*1c60b9acSAndroid Build Coastguard Worker }
262