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 25738
server_new(const char *name, struct vtclog *vl)
80
{
81
        struct server *s;
82
83 25738
        VTC_CHECK_NAME(vl, name, "Server", 's');
84 25738
        ALLOC_OBJ(s, SERVER_MAGIC);
85 25738
        AN(s);
86 25738
        REPLACE(s->name, name);
87 25738
        s->vl = vtc_logopen("%s", s->name);
88 25738
        AN(s->vl);
89 25738
        s->vsp = Sess_New(s->vl, name);
90 25738
        AN(s->vsp);
91
92 25738
        bprintf(s->listen, "%s", default_listen_addr);
93 25738
        s->depth = 10;
94 25738
        s->sock = -1;
95 25738
        s->fd = -1;
96 25738
        PTOK(pthread_mutex_lock(&server_mtx));
97 25738
        VTAILQ_INSERT_TAIL(&servers, s, list);
98 25738
        PTOK(pthread_mutex_unlock(&server_mtx));
99 25738
        return (s);
100
}
101
102
/**********************************************************************
103
 * Clean up a server
104
 */
105
106
static void
107 25738
server_delete(struct server *s)
108
{
109
110 25738
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
111 25738
        Sess_Destroy(&s->vsp);
112 25738
        macro_undef(s->vl, s->name, "addr");
113 25738
        macro_undef(s->vl, s->name, "port");
114 25738
        macro_undef(s->vl, s->name, "sock");
115 25738
        vtc_logclose(s->vl);
116 25738
        free(s->name);
117
        /* XXX: MEMLEAK (?) (VSS ??) */
118 25738
        FREE_OBJ(s);
119 25738
}
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 1475
uds_listen(void *priv, const struct sockaddr_un *uds)
133
{
134
        int sock, e;
135 1475
        struct helper *hp = priv;
136
137 1475
        sock = VUS_bind(uds, hp->errp);
138 1475
        if (sock >= 0)   {
139 1475
                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 1475
        }
148 1475
        if (sock > 0) {
149 1475
                *hp->errp = NULL;
150 1475
                return (sock);
151
        }
152 0
        AN(*hp->errp);
153 0
        return (0);
154 1475
}
155
156
static void
157 1475
server_listen_uds(struct server *s, const char **errp)
158
{
159
        mode_t m;
160
        struct helper h;
161
162 1475
        h.depth = s->depth;
163 1475
        h.errp = errp;
164
165 1475
        errno = 0;
166 1475
        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 1475
        m = umask(0);
174 1475
        s->sock = VUS_resolver(s->listen, uds_listen, &h, errp);
175 1475
        (void)umask(m);
176 1475
        if (*errp != NULL)
177 0
                return;
178 1475
        assert(s->sock > 0);
179 1475
        macro_def(s->vl, s->name, "addr", "0.0.0.0");
180 1475
        macro_def(s->vl, s->name, "port", "0");
181 1475
        macro_def(s->vl, s->name, "sock", "%s", s->listen);
182 1475
}
183
184
static void
185 24825
server_listen_tcp(struct server *s, const char **errp)
186
{
187 24825
        char buf[vsa_suckaddr_len];
188
        const struct suckaddr *sua;
189
190 24825
        s->sock = VTCP_listen_on(s->listen, "0", s->depth, errp);
191 24825
        if (*errp != NULL)
192 0
                return;
193 24825
        assert(s->sock > 0);
194 24825
        sua = VSA_getsockname(s->sock, buf, sizeof buf);
195 24825
        AN(sua);
196 49650
        VTCP_name(sua, s->aaddr, sizeof s->aaddr,
197 24825
            s->aport, sizeof s->aport);
198
199
        /* Record the actual port, and reuse it on subsequent starts */
200 24825
        if (VSA_Get_Proto(sua) == AF_INET)
201 24725
                bprintf(s->listen, "%s:%s", s->aaddr, s->aport);
202
        else
203 100
                bprintf(s->listen, "[%s]:%s", s->aaddr, s->aport);
204
205 24825
        macro_def(s->vl, s->name, "addr", "%s", s->aaddr);
206 24825
        macro_def(s->vl, s->name, "port", "%s", s->aport);
207 24825
        macro_def(s->vl, s->name, "sock", "%s", s->listen);
208 24825
}
209
210
static void
211 26300
server_listen(struct server *s)
212
{
213
        const char *err;
214
215 26300
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
216
217 26300
        if (s->sock >= 0)
218 2125
                VTCP_close(&s->sock);
219 26300
        if (VUS_is(s->listen))
220 1475
                server_listen_uds(s, &err);
221
        else
222 24825
                server_listen_tcp(s, &err);
223 26300
        if (err != NULL)
224 0
                vtc_fatal(s->vl,
225
                    "Server listen address (%s) cannot be resolved: %s",
226 0
                    s->listen, err);
227 26300
}
228
229
/**********************************************************************
230
 * Server thread
231
 */
232
233
static int
234 33039
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 33039
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
245
246 33039
        addr = (void*)&addr_s;
247 33039
        l = sizeof addr_s;
248 33039
        fd = accept(s->sock, addr, &l);
249 33039
        if (fd < 0)
250 0
                vtc_fatal(vl, "Accept failed: %s", strerror(errno));
251 33039
        if (VUS_is(s->listen))
252 1475
                vtc_log(vl, 3, "accepted fd %d 0.0.0.0 0", fd);
253
        else {
254 31564
                VTCP_hisname(fd, abuf, sizeof abuf, pbuf, sizeof pbuf);
255 31564
                vtc_log(vl, 3, "accepted fd %d %s %s", fd, abuf, pbuf);
256
        }
257 33039
        return (fd);
258
}
259
260
static void
261 32488
server_disc(void *priv, struct vtclog *vl, int *fdp)
262
{
263
        int j;
264
        struct server *s;
265
266 32488
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
267 32488
        vtc_log(vl, 3, "shutting fd %d", *fdp);
268 32488
        j = shutdown(*fdp, SHUT_WR);
269 32488
        if (!vtc_stop && !VTCP_Check(j))
270 0
                vtc_fatal(vl, "Shutdown failed: %s", strerror(errno));
271 32490
        VTCP_close(fdp);
272 32490
}
273
274
static void
275 25975
server_start_thread(struct server *s)
276
{
277
278 25975
        s->run = 1;
279 25975
        s->tp = Sess_Start_Thread(
280 25975
            s,
281 25975
            s->vsp,
282
            server_conn,
283
            server_disc,
284 25975
            s->listen,
285 25975
            &s->sock,
286 25975
            s->spec
287
        );
288 25975
}
289
290
/**********************************************************************
291
 * Start the server thread
292
 */
293
294
static void
295 25975
server_start(struct server *s)
296
{
297 25975
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
298 25975
        vtc_log(s->vl, 2, "Starting server");
299 25975
        server_listen(s);
300 25975
        vtc_log(s->vl, 1, "Listen on %s", s->listen);
301 25975
        server_start_thread(s);
302 25975
}
303
304
/**********************************************************************
305
 */
306
307
static void *
308 1638
server_dispatch_wrk(void *priv)
309
{
310
        struct server *s;
311
        struct vtclog *vl;
312
        int j, fd;
313
314 1638
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
315 1638
        assert(s->sock < 0);
316
317 1638
        vl = vtc_logopen("%s", s->name);
318 1638
        pthread_cleanup_push(vtc_logclose, vl);
319
320 1638
        fd = s->fd;
321
322 1638
        vtc_log(vl, 3, "start with fd %d", fd);
323 1638
        fd = sess_process(vl, s->vsp, s->spec, fd, &s->sock, s->listen);
324 1638
        vtc_log(vl, 3, "shutting fd %d", fd);
325 1638
        j = shutdown(fd, SHUT_WR);
326 1638
        if (!VTCP_Check(j))
327 0
                vtc_fatal(vl, "Shutdown failed: %s", strerror(errno));
328 1638
        VTCP_close(&s->fd);
329 1638
        vtc_log(vl, 2, "Ending");
330 1638
        pthread_cleanup_pop(0);
331 1638
        vtc_logclose(vl);
332 1638
        return (NULL);
333
}
334
335
static void *
336 325
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 325
        CAST_OBJ_NOTNULL(s, priv, SERVER_MAGIC);
348 325
        assert(s->sock >= 0);
349
350 325
        vl = vtc_logopen("%s", s->name);
351 325
        pthread_cleanup_push(vtc_logclose, vl);
352
353 325
        vtc_log(vl, 2, "Dispatch started on %s", s->listen);
354
355 1963
        while (!vtc_stop) {
356 1638
                addr = (void*)&addr_s;
357 1638
                l = sizeof addr_s;
358 1638
                fd = accept(s->sock, addr, &l);
359 1638
                if (fd < 0)
360 0
                        vtc_fatal(vl, "Accepted failed: %s", strerror(errno));
361 1638
                bprintf(snbuf, "s%d", sn++);
362 1638
                vtc_log(vl, 3, "dispatch fd %d -> %s", fd, snbuf);
363 1638
                s2 = server_new(snbuf, vl);
364 1638
                s2->is_dispatch = 1;
365 1638
                s2->spec = s->spec;
366 1638
                bstrcpy(s2->listen, s->listen);
367 1638
                s2->fd = fd;
368 1638
                s2->run = 1;
369 1638
                PTOK(pthread_create(&s2->tp, NULL, server_dispatch_wrk, s2));
370
        }
371 325
        pthread_cleanup_pop(0);
372 325
        vtc_logclose(vl);
373 325
        NEEDLESS(return (NULL));
374
}
375
376
static void
377 325
server_dispatch(struct server *s)
378
{
379 325
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
380 325
        server_listen(s);
381 325
        vtc_log(s->vl, 2, "Starting dispatch server");
382 325
        s->run = 1;
383 325
        PTOK(pthread_create(&s->tp, NULL, server_dispatch_thread, s));
384 325
}
385
386
/**********************************************************************
387
 * Force stop the server thread
388
 */
389
390
static void
391 75
server_break(struct server *s)
392
{
393
        void *res;
394
395 75
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
396 75
        vtc_log(s->vl, 2, "Breaking for server");
397 75
        (void)pthread_cancel(s->tp);
398 75
        PTOK(pthread_join(s->tp, &res));
399 75
        VTCP_close(&s->sock);
400 75
        s->tp = 0;
401 75
        s->run = 0;
402 75
}
403
404
/**********************************************************************
405
 * Wait for server thread to stop
406
 */
407
408
static void
409 27863
server_wait(struct server *s)
410
{
411
        void *res;
412
413 27863
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
414 27863
        vtc_log(s->vl, 2, "Waiting for server (%d/%d)", s->sock, s->fd);
415 27863
        PTOK(pthread_join(s->tp, &res));
416 27863
        if (res != NULL && !vtc_stop)
417 0
                vtc_fatal(s->vl, "Server returned \"%p\"",
418 0
                    (char *)res);
419 27863
        s->tp = 0;
420 27863
        s->run = 0;
421 27863
}
422
423
/**********************************************************************
424
 * Generate VCL backend decls for our servers
425
 */
426
427
void
428 21150
cmd_server_gen_vcl(struct vsb *vsb)
429
{
430
        struct server *s;
431
432 21150
        PTOK(pthread_mutex_lock(&server_mtx));
433 45250
        VTAILQ_FOREACH(s, &servers, list) {
434 24100
                if (s->is_dispatch)
435 400
                        continue;
436
437 23700
                if (VUS_is(s->listen))
438 850
                        VSB_printf(vsb,
439
                           "backend %s { .path = \"%s\"; }\n",
440 425
                           s->name, s->listen);
441
                else
442 46550
                        VSB_printf(vsb,
443
                           "backend %s { .host = \"%s\"; .port = \"%s\"; }\n",
444 23275
                           s->name, s->aaddr, s->aport);
445 23700
        }
446 21150
        PTOK(pthread_mutex_unlock(&server_mtx));
447 21150
}
448
449
450
/**********************************************************************
451
 * Generate VCL backend decls for our servers
452
 */
453
454
void
455 25
cmd_server_gen_haproxy_conf(struct vsb *vsb)
456
{
457
        struct server *s;
458
459 25
        PTOK(pthread_mutex_lock(&server_mtx));
460 50
        VTAILQ_FOREACH(s, &servers, list) {
461 25
                if (! VUS_is(s->listen))
462 50
                        VSB_printf(vsb,
463
                           "\n    backend be%s\n"
464
                           "\tserver srv%s %s:%s\n",
465 25
                           s->name + 1, s->name + 1, s->aaddr, s->aport);
466
                else
467 0
                        INCOMPL();
468 25
        }
469 50
        VTAILQ_FOREACH(s, &servers, list) {
470 25
                if (! VUS_is(s->listen))
471 50
                        VSB_printf(vsb,
472
                           "\n    frontend http%s\n"
473
                           "\tuse_backend be%s\n"
474
                           "\tbind \"fd@${fe%s}\"\n",
475 25
                           s->name + 1, s->name + 1, s->name + 1);
476
                else
477 0
                        INCOMPL();
478 25
        }
479 25
        PTOK(pthread_mutex_unlock(&server_mtx));
480 25
}
481
482
483
/**********************************************************************
484
 * Server command dispatch
485
 */
486
487
void
488 54000
cmd_server(CMD_ARGS)
489
{
490
        struct server *s;
491
492 54000
        (void)priv;
493
494 54000
        if (av == NULL) {
495
                /* Reset and free */
496 50163
                while (1) {
497 50163
                        PTOK(pthread_mutex_lock(&server_mtx));
498 50163
                        s = VTAILQ_FIRST(&servers);
499 50163
                        CHECK_OBJ_ORNULL(s, SERVER_MAGIC);
500 50163
                        if (s != NULL)
501 25738
                                VTAILQ_REMOVE(&servers, s, list);
502 50163
                        PTOK(pthread_mutex_unlock(&server_mtx));
503 50163
                        if (s == NULL)
504 24425
                                break;
505 25738
                        if (s->run) {
506 23788
                                (void)pthread_cancel(s->tp);
507 23788
                                server_wait(s);
508 23788
                        }
509 25738
                        if (s->sock >= 0)
510 24100
                                VTCP_close(&s->sock);
511 25738
                        server_delete(s);
512
                }
513 24425
                return;
514
        }
515
516 29575
        AZ(strcmp(av[0], "server"));
517 29575
        av++;
518
519 29575
        PTOK(pthread_mutex_lock(&server_mtx));
520 37200
        VTAILQ_FOREACH(s, &servers, list)
521 13100
                if (!strcmp(s->name, av[0]))
522 5475
                        break;
523 29575
        PTOK(pthread_mutex_unlock(&server_mtx));
524 29575
        if (s == NULL)
525 24100
                s = server_new(av[0], vl);
526 29575
        CHECK_OBJ_NOTNULL(s, SERVER_MAGIC);
527 29575
        av++;
528
529 87650
        for (; *av != NULL; av++) {
530 58075
                if (vtc_error)
531 0
                        break;
532 58075
                if (!strcmp(*av, "-wait")) {
533 2500
                        if (!s->run)
534 0
                                vtc_fatal(s->vl, "Server not -started");
535 2500
                        server_wait(s);
536 2500
                        continue;
537
                }
538
539 55575
                if (!strcmp(*av, "-break")) {
540 75
                        server_break(s);
541 75
                        continue;
542
                }
543
544
                /*
545
                 * We do an implict -wait if people muck about with a
546
                 * running server.
547
                 */
548 55500
                if (s->run)
549 1575
                        server_wait(s);
550
551 55500
                AZ(s->run);
552
553 55500
                if (Sess_GetOpt(s->vsp, &av))
554 1975
                        continue;
555
556 53525
                if (!strcmp(*av, "-listen")) {
557 1475
                        if (s->sock >= 0)
558 0
                                VTCP_close(&s->sock);
559 1475
                        bprintf(s->listen, "%s", av[1]);
560 1475
                        av++;
561 1475
                        continue;
562
                }
563 52050
                if (!strcmp(*av, "-start")) {
564 25975
                        server_start(s);
565 25975
                        continue;
566
                }
567 26075
                if (!strcmp(*av, "-dispatch")) {
568 325
                        if (strcmp(s->name, "s0"))
569 0
                                vtc_fatal(s->vl,
570
                                    "server -dispatch only works on s0");
571 325
                        server_dispatch(s);
572 325
                        continue;
573
                }
574 25750
                if (**av == '-')
575 0
                        vtc_fatal(s->vl, "Unknown server argument: %s", *av);
576 25750
                s->spec = *av;
577 25750
        }
578 54000
}
579
580
void
581 24425
init_server(void)
582
{
583 24425
        PTOK(pthread_mutex_init(&server_mtx, NULL));
584 24425
}