varnish-cache/bin/varnishtest/vtc_server.c
0
/*-
1
 * Copyright (c) 2008-2010 Varnish Software AS
2
 * All rights reserved.
3
 *
4
 * Author: Poul-Henning Kamp <phk@phk.freebsd.dk>
5
 *
6
 * SPDX-License-Identifier: BSD-2-Clause
7
 *
8
 * Redistribution and use in source and binary forms, with or without
9
 * modification, are permitted provided that the following conditions
10
 * are met:
11
 * 1. Redistributions of source code must retain the above copyright
12
 *    notice, this list of conditions and the following disclaimer.
13
 * 2. Redistributions in binary form must reproduce the above copyright
14
 *    notice, this list of conditions and the following disclaimer in the
15
 *    documentation and/or other materials provided with the distribution.
16
 *
17
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20
 * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27
 * SUCH DAMAGE.
28
 */
29
30
#include "config.h"
31
32
#include <sys/socket.h>
33
#include <sys/types.h>
34
#include <sys/stat.h>
35
36
#include <stdio.h>
37
#include <stdlib.h>
38
#include <string.h>
39
#include <unistd.h>
40
41
#include "vsa.h"
42
#include "vtc.h"
43
44
#include "vtcp.h"
45
#include "vus.h"
46
47
struct server {
48
        unsigned                magic;
49
#define SERVER_MAGIC            0x55286619
50
        char                    *name;
51
        struct vtclog           *vl;
52
        VTAILQ_ENTRY(server)    list;
53
        struct vtc_sess         *vsp;
54
        char                    run;
55
56
        char                    *spec;
57
58
        int                     depth;
59
        int                     sock;
60
        int                     fd;
61
        unsigned                is_dispatch;
62
        char                    listen[256];
63
        char                    aaddr[VTCP_ADDRBUFSIZE];
64
        char                    aport[VTCP_PORTBUFSIZE];
65
66
        pthread_t               tp;
67
};
68
69
static pthread_mutex_t          server_mtx;
70
71
static VTAILQ_HEAD(, server)    servers =
72
    VTAILQ_HEAD_INITIALIZER(servers);
73
74
/**********************************************************************
75
 * Allocate and initialize a server
76
 */
77
78
static struct server *
79 5221
server_new(const char *name, struct vtclog *vl)
80
{
81
        struct server *s;
82
83 5221
        VTC_CHECK_NAME(vl, name, "Server", 's');
84 5221
        ALLOC_OBJ(s, SERVER_MAGIC);
85 5221
        AN(s);
86 5221
        REPLACE(s->name, name);
87 5221
        s->vl = vtc_logopen("%s", s->name);
88 5221
        AN(s->vl);
89 5221
        s->vsp = Sess_New(s->vl, name);
90 5221
        AN(s->vsp);
91
92 5221
        bprintf(s->listen, "%s", default_listen_addr);
93 5221
        s->depth = 10;
94 5221
        s->sock = -1;
95 5221
        s->fd = -1;
96 5221
        PTOK(pthread_mutex_lock(&server_mtx));
97 5221
        VTAILQ_INSERT_TAIL(&servers, s, list);
98 5221
        PTOK(pthread_mutex_unlock(&server_mtx));
99 5221
        return (s);
100
}
101
102
/**********************************************************************
103
 * Clean up a server
104
 */
105
106
static void
107 5221
server_delete(struct server *s)
108
{
109
110 5221
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
111 5221
        Sess_Destroy(&s->vsp);
112 5221
        macro_undef(s->vl, s->name, "addr");
113 5221
        macro_undef(s->vl, s->name, "port");
114 5221
        macro_undef(s->vl, s->name, "sock");
115 5221
        vtc_logclose(s->vl);
116 5221
        free(s->name);
117
        /* XXX: MEMLEAK (?) (VSS ??) */
118 5221
        FREE_OBJ(s);
119 5221
}
120
121
/**********************************************************************
122
 * Server listen
123
 */
124
125
struct helper {
126
        int             depth;
127
        const char      **errp;
128
};
129
130
/* cf. VTCP_listen_on() */
131
static int v_matchproto_(vus_resolved_f)
132 295
uds_listen(void *priv, const struct sockaddr_un *uds)
133
{
134
        int sock, e;
135 295
        struct helper *hp = priv;
136
137 295
        sock = VUS_bind(uds, hp->errp);
138 295
        if (sock >= 0)   {
139 295
                if (listen(sock, hp->depth) != 0) {
140 0
                        e = errno;
141 0
                        closefd(&sock);
142 0
                        errno = e;
143 0
                        if (hp->errp != NULL)
144 0
                                *hp->errp = "listen(2)";
145 0
                        return (-1);
146
                }
147 295
        }
148 295
        if (sock > 0) {
149 295
                *hp->errp = NULL;
150 295
                return (sock);
151
        }
152 0
        AN(*hp->errp);
153 0
        return (0);
154 295
}
155
156
static void
157 295
server_listen_uds(struct server *s, const char **errp)
158
{
159
        mode_t m;
160
        struct helper h;
161
162 295
        h.depth = s->depth;
163 295
        h.errp = errp;
164
165 295
        errno = 0;
166 295
        if (unlink(s->listen) != 0 && errno != ENOENT)
167 0
                vtc_fatal(s->vl, "Could not unlink %s before bind: %s",
168 0
                    s->listen, strerror(errno));
169
        /*
170
         * Temporarily set the umask to 0 to avoid issues with
171
         * permissions.
172
         */
173 295
        m = umask(0);
174 295
        s->sock = VUS_resolver(s->listen, uds_listen, &h, errp);
175 295
        (void)umask(m);
176 295
        if (*errp != NULL)
177 0
                return;
178 295
        assert(s->sock > 0);
179 295
        macro_def(s->vl, s->name, "addr", "0.0.0.0");
180 295
        macro_def(s->vl, s->name, "port", "0");
181 295
        macro_def(s->vl, s->name, "sock", "%s", s->listen);
182 295
}
183
184
static void
185 5050
server_listen_tcp(struct server *s, const char **errp)
186
{
187 5050
        char buf[vsa_suckaddr_len];
188
        const struct suckaddr *sua;
189
190 5050
        s->sock = VTCP_listen_on(s->listen, "0", s->depth, errp);
191 5050
        if (*errp != NULL)
192 0
                return;
193 5050
        assert(s->sock > 0);
194 5050
        sua = VSA_getsockname(s->sock, buf, sizeof buf);
195 5050
        AN(sua);
196 10100
        VTCP_name(sua, s->aaddr, sizeof s->aaddr,
197 5050
            s->aport, sizeof s->aport);
198
199
        /* Record the actual port, and reuse it on subsequent starts */
200 5050
        if (VSA_Get_Proto(sua) == AF_INET)
201 5030
                bprintf(s->listen, "%s:%s", s->aaddr, s->aport);
202
        else
203 20
                bprintf(s->listen, "[%s]:%s", s->aaddr, s->aport);
204
205 5050
        macro_def(s->vl, s->name, "addr", "%s", s->aaddr);
206 5050
        macro_def(s->vl, s->name, "port", "%s", s->aport);
207 5050
        macro_def(s->vl, s->name, "sock", "%s", s->listen);
208 5050
}
209
210
static void
211 5345
server_listen(struct server *s)
212
{
213
        const char *err;
214
215 5345
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
216
217 5345
        if (s->sock >= 0)
218 435
                VTCP_close(&s->sock);
219 5345
        if (VUS_is(s->listen))
220 295
                server_listen_uds(s, &err);
221
        else
222 5050
                server_listen_tcp(s, &err);
223 5345
        if (err != NULL)
224 0
                vtc_fatal(s->vl,
225
                    "Server listen address (%s) cannot be resolved: %s",
226 0
                    s->listen, err);
227 5345
}
228
229
/**********************************************************************
230
 * Server thread
231
 */
232
233
static int
234 6690
server_conn(void *priv, struct vtclog *vl)
235
{
236
        struct server *s;
237
        struct sockaddr_storage addr_s;
238
        struct sockaddr *addr;
239
        char abuf[VTCP_ADDRBUFSIZE];
240
        char pbuf[VTCP_PORTBUFSIZE];
241
        socklen_t l;
242
        int fd;
243
244 6690
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
245
246 6690
        addr = (void*)&addr_s;
247 6690
        l = sizeof addr_s;
248 6690
        fd = accept(s->sock, addr, &l);
249 6690
        if (fd < 0)
250 0
                vtc_fatal(vl, "Accept failed: %s", strerror(errno));
251 6690
        if (VUS_is(s->listen))
252 295
                vtc_log(vl, 3, "accepted fd %d 0.0.0.0 0", fd);
253
        else {
254 6395
                VTCP_hisname(fd, abuf, sizeof abuf, pbuf, sizeof pbuf);
255 6395
                vtc_log(vl, 3, "accepted fd %d %s %s", fd, abuf, pbuf);
256
        }
257 6690
        return (fd);
258
}
259
260
static void
261 6569
server_disc(void *priv, struct vtclog *vl, int *fdp)
262
{
263
        int j;
264
        struct server *s;
265
266 6569
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
267 6569
        vtc_log(vl, 3, "shutting fd %d", *fdp);
268 6569
        j = shutdown(*fdp, SHUT_WR);
269 6569
        if (!vtc_stop && !VTCP_Check(j))
270 0
                vtc_fatal(vl, "Shutdown failed: %s", strerror(errno));
271 6571
        VTCP_close(fdp);
272 6571
}
273
274
static void
275 5280
server_start_thread(struct server *s)
276
{
277
278 5280
        s->run = 1;
279 5280
        s->tp = Sess_Start_Thread(
280 5280
            s,
281 5280
            s->vsp,
282
            server_conn,
283
            server_disc,
284 5280
            s->listen,
285 5280
            &s->sock,
286 5280
            s->spec
287
        );
288 5280
}
289
290
/**********************************************************************
291
 * Start the server thread
292
 */
293
294
static void
295 5280
server_start(struct server *s)
296
{
297 5280
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
298 5280
        vtc_log(s->vl, 2, "Starting server");
299 5280
        server_listen(s);
300 5280
        vtc_log(s->vl, 1, "Listen on %s", s->listen);
301 5280
        server_start_thread(s);
302 5280
}
303
304
/**********************************************************************
305
 */
306
307
static void *
308 326
server_dispatch_wrk(void *priv)
309
{
310
        struct server *s;
311
        struct vtclog *vl;
312
        int j, fd;
313
314 326
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
315 326
        assert(s->sock < 0);
316
317 326
        vl = vtc_logopen("%s", s->name);
318 326
        pthread_cleanup_push(vtc_logclose, vl);
319
320 326
        fd = s->fd;
321
322 326
        vtc_log(vl, 3, "start with fd %d", fd);
323 326
        fd = sess_process(vl, s->vsp, s->spec, fd, &s->sock, s->listen);
324 326
        vtc_log(vl, 3, "shutting fd %d", fd);
325 326
        j = shutdown(fd, SHUT_WR);
326 326
        if (!VTCP_Check(j))
327 0
                vtc_fatal(vl, "Shutdown failed: %s", strerror(errno));
328 326
        VTCP_close(&s->fd);
329 326
        vtc_log(vl, 2, "Ending");
330 326
        pthread_cleanup_pop(0);
331 326
        vtc_logclose(vl);
332 326
        return (NULL);
333
}
334
335
static void *
336 65
server_dispatch_thread(void *priv)
337
{
338
        struct server *s, *s2;
339
        static int sn = 1;
340
        int fd;
341
        char snbuf[8];
342
        struct vtclog *vl;
343
        struct sockaddr_storage addr_s;
344
        struct sockaddr *addr;
345
        socklen_t l;
346
347 65
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
348 65
        assert(s->sock >= 0);
349
350 65
        vl = vtc_logopen("%s", s->name);
351 65
        pthread_cleanup_push(vtc_logclose, vl);
352
353 65
        vtc_log(vl, 2, "Dispatch started on %s", s->listen);
354
355 391
        while (!vtc_stop) {
356 326
                addr = (void*)&addr_s;
357 326
                l = sizeof addr_s;
358 326
                fd = accept(s->sock, addr, &l);
359 326
                if (fd < 0)
360 0
                        vtc_fatal(vl, "Accepted failed: %s", strerror(errno));
361 326
                bprintf(snbuf, "s%d", sn++);
362 326
                vtc_log(vl, 3, "dispatch fd %d -> %s", fd, snbuf);
363 326
                s2 = server_new(snbuf, vl);
364 326
                s2->is_dispatch = 1;
365 326
                s2->spec = s->spec;
366 326
                bstrcpy(s2->listen, s->listen);
367 326
                s2->fd = fd;
368 326
                s2->run = 1;
369 326
                PTOK(pthread_create(&s2->tp, NULL, server_dispatch_wrk, s2));
370
        }
371 65
        pthread_cleanup_pop(0);
372 65
        vtc_logclose(vl);
373 65
        NEEDLESS(return (NULL));
374
}
375
376
static void
377 65
server_dispatch(struct server *s)
378
{
379 65
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
380 65
        server_listen(s);
381 65
        vtc_log(s->vl, 2, "Starting dispatch server");
382 65
        s->run = 1;
383 65
        PTOK(pthread_create(&s->tp, NULL, server_dispatch_thread, s));
384 65
}
385
386
/**********************************************************************
387
 * Force stop the server thread
388
 */
389
390
static void
391 15
server_break(struct server *s)
392
{
393
        void *res;
394
395 15
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
396 15
        vtc_log(s->vl, 2, "Breaking for server");
397 15
        (void)pthread_cancel(s->tp);
398 15
        PTOK(pthread_join(s->tp, &res));
399 15
        VTCP_close(&s->sock);
400 15
        s->tp = 0;
401 15
        s->run = 0;
402 15
}
403
404
/**********************************************************************
405
 * Wait for server thread to stop
406
 */
407
408
static void
409 5656
server_wait(struct server *s)
410
{
411
        void *res;
412
413 5656
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
414 5656
        vtc_log(s->vl, 2, "Waiting for server (%d/%d)", s->sock, s->fd);
415 5656
        PTOK(pthread_join(s->tp, &res));
416 5656
        if (res != NULL && !vtc_stop)
417 0
                vtc_fatal(s->vl, "Server returned \"%p\"",
418 0
                    (char *)res);
419 5656
        s->tp = 0;
420 5656
        s->run = 0;
421 5656
}
422
423
/**********************************************************************
424
 * Generate VCL backend decls for our servers
425
 */
426
427
void
428 4270
cmd_server_gen_vcl(struct vsb *vsb)
429
{
430
        struct server *s;
431
432 4270
        PTOK(pthread_mutex_lock(&server_mtx));
433 9130
        VTAILQ_FOREACH(s, &servers, list) {
434 4860
                if (s->is_dispatch)
435 80
                        continue;
436
437 4780
                if (VUS_is(s->listen))
438 170
                        VSB_printf(vsb,
439
                           "backend %s { .path = \"%s\"; }\n",
440 85
                           s->name, s->listen);
441
                else
442 9390
                        VSB_printf(vsb,
443
                           "backend %s { .host = \"%s\"; .port = \"%s\"; }\n",
444 4695
                           s->name, s->aaddr, s->aport);
445 4780
        }
446 4270
        PTOK(pthread_mutex_unlock(&server_mtx));
447 4270
}
448
449
450
/**********************************************************************
451
 * Generate VCL backend decls for our servers
452
 */
453
454
void
455 5
cmd_server_gen_haproxy_conf(struct vsb *vsb)
456
{
457
        struct server *s;
458
459 5
        PTOK(pthread_mutex_lock(&server_mtx));
460 10
        VTAILQ_FOREACH(s, &servers, list) {
461 5
                if (! VUS_is(s->listen))
462 10
                        VSB_printf(vsb,
463
                           "\n    backend be%s\n"
464
                           "\tserver srv%s %s:%s\n",
465 5
                           s->name + 1, s->name + 1, s->aaddr, s->aport);
466
                else
467 0
                        INCOMPL();
468 5
        }
469 10
        VTAILQ_FOREACH(s, &servers, list) {
470 5
                if (! VUS_is(s->listen))
471 10
                        VSB_printf(vsb,
472
                           "\n    frontend http%s\n"
473
                           "\tuse_backend be%s\n"
474
                           "\tbind \"fd@${fe%s}\"\n",
475 5
                           s->name + 1, s->name + 1, s->name + 1);
476
                else
477 0
                        INCOMPL();
478 5
        }
479 5
        PTOK(pthread_mutex_unlock(&server_mtx));
480 5
}
481
482
483
/**********************************************************************
484
 * Server command dispatch
485
 */
486
487
void
488 11010
cmd_server(CMD_ARGS)
489
{
490
        struct server *s;
491
492 11010
        (void)priv;
493
494 11010
        if (av == NULL) {
495
                /* Reset and free */
496 10206
                while (1) {
497 10206
                        PTOK(pthread_mutex_lock(&server_mtx));
498 10206
                        s = VTAILQ_FIRST(&servers);
499 10206
                        CHECK_OBJ_ORNULL(s, SERVER_MAGIC);
500 10206
                        if (s != NULL)
501 5221
                                VTAILQ_REMOVE(&servers, s, list);
502 10206
                        PTOK(pthread_mutex_unlock(&server_mtx));
503 10206
                        if (s == NULL)
504 4985
                                break;
505 5221
                        if (s->run) {
506 4811
                                (void)pthread_cancel(s->tp);
507 4811
                                server_wait(s);
508 4811
                        }
509 5221
                        if (s->sock >= 0)
510 4895
                                VTCP_close(&s->sock);
511 5221
                        server_delete(s);
512
                }
513 4985
                return;
514
        }
515
516 6025
        AZ(strcmp(av[0], "server"));
517 6025
        av++;
518
519 6025
        PTOK(pthread_mutex_lock(&server_mtx));
520 7565
        VTAILQ_FOREACH(s, &servers, list)
521 2670
                if (!strcmp(s->name, av[0]))
522 1130
                        break;
523 6025
        PTOK(pthread_mutex_unlock(&server_mtx));
524 6025
        if (s == NULL)
525 4895
                s = server_new(av[0], vl);
526 6025
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
527 6025
        av++;
528
529 17835
        for (; *av != NULL; av++) {
530 11810
                if (vtc_error)
531 0
                        break;
532 11810
                if (!strcmp(*av, "-wait")) {
533 525
                        if (!s->run)
534 0
                                vtc_fatal(s->vl, "Server not -started");
535 525
                        server_wait(s);
536 525
                        continue;
537
                }
538
539 11285
                if (!strcmp(*av, "-break")) {
540 15
                        server_break(s);
541 15
                        continue;
542
                }
543
544
                /*
545
                 * We do an implicit -wait if people muck about with a
546
                 * running server.
547
                 */
548 11270
                if (s->run)
549 320
                        server_wait(s);
550
551 11270
                AZ(s->run);
552
553 11270
                if (Sess_GetOpt(s->vsp, &av))
554 395
                        continue;
555
556 10875
                if (!strcmp(*av, "-listen")) {
557 295
                        if (s->sock >= 0)
558 0
                                VTCP_close(&s->sock);
559 295
                        bprintf(s->listen, "%s", av[1]);
560 295
                        av++;
561 295
                        continue;
562
                }
563 10580
                if (!strcmp(*av, "-start")) {
564 5280
                        server_start(s);
565 5280
                        continue;
566
                }
567 5300
                if (!strcmp(*av, "-dispatch")) {
568 65
                        if (strcmp(s->name, "s0"))
569 0
                                vtc_fatal(s->vl,
570
                                    "server -dispatch only works on s0");
571 65
                        server_dispatch(s);
572 65
                        continue;
573
                }
574 5235
                if (**av == '-')
575 0
                        vtc_fatal(s->vl, "Unknown server argument: %s", *av);
576 5235
                s->spec = *av;
577 5235
        }
578 11010
}
579
580
void
581 4985
init_server(void)
582
{
583 4985
        PTOK(pthread_mutex_init(&server_mtx, NULL));
584 4985
}