src/server/daemon/location.c

Sun, 23 Nov 2025 12:44:59 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Sun, 23 Nov 2025 12:44:59 +0100
changeset 635
b85d45fd3b01
child 636
40f069ddda37
permissions
-rw-r--r--

add location config parser

635
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
1 /*
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
3 *
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
4 * Copyright 2025 Olaf Wintermann. All rights reserved.
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
5 *
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
6 * Redistribution and use in source and binary forms, with or without
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
7 * modification, are permitted provided that the following conditions are met:
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
8 *
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
9 * 1. Redistributions of source code must retain the above copyright
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
10 * notice, this list of conditions and the following disclaimer.
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
11 *
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
12 * 2. Redistributions in binary form must reproduce the above copyright
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
13 * notice, this list of conditions and the following disclaimer in the
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
14 * documentation and/or other materials provided with the distribution.
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
15 *
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
26 * POSSIBILITY OF SUCH DAMAGE.
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
27 */
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
28
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
29 #include "location.h"
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
30 #include "config.h"
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
31 #include "../util/util.h"
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
32 #include <cx/linked_list.h>
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
33
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
34 #define DIR_CHECK_ARGC(n) if(argc != n) { \
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
35 log_ereport(LOG_FAILURE, "%s directive argc != %d", name.ptr, n); \
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
36 return 1; \
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
37 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
38
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
39 static void location_list_add(WSLocation *location, WSLocation *sub) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
40 void **begin = (void**)&location->children_begin;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
41 void **end = (void**)&location->children_end;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
42 cx_linked_list_add(begin, end, offsetof(WSLocation, prev), offsetof(WSLocation, next), sub);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
43 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
44
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
45 static int add_location_config(const CxAllocator *a, WSLocation *location, ConfigNode *dir) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
46 cxmutstr name = dir->name;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
47 int argc = serverconfig_directive_count_args(dir);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
48 if(!cx_strcasecmp(name, "DirectoryIndex")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
49 DIR_CHECK_ARGC(1);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
50 location->config.set_dirindex = TRUE;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
51 location->config.dirindex = util_getboolean_s(cx_strcast(dir->args->value), FALSE);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
52 } else if(!cx_strcasecmp(name, "DocumentRoot")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
53 DIR_CHECK_ARGC(1);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
54 location->config.docroot = cx_strdup_a(a, dir->args->value);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
55 } else if(!cx_strcasecmp(name, "Location")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
56 WSLocation *sub_location = cfg_location_get(a, dir);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
57 if(!sub_location) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
58 return 1;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
59 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
60 location_list_add(location, sub_location);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
61 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
62
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
63 return 0;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
64 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
65
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
66
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
67 WSLocation* cfg_location_get(const CxAllocator *a, ConfigNode *obj) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
68 WSLocationMatch match = WS_LOCATION_MATCH_EXACT;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
69 cxmutstr match_str;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
70
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
71 int regex_flags = REG_EXTENDED;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
72 int argc = serverconfig_directive_count_args(obj);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
73 if(argc == 2) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
74 // arg0: match type
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
75 cxmutstr type_str = obj->args->value;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
76 if(!cx_strcmp(type_str, "=")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
77 // noop
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
78 } else if(!cx_strcmp(type_str, "^~")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
79 match = WS_LOCATION_MATCH_PREFIX;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
80 } else if(!cx_strcmp(type_str, "~")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
81 match = WS_LOCATION_MATCH_CS_REGEX;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
82 } else if(!cx_strcmp(type_str, "~*")) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
83 match = WS_LOCATION_MATCH_CI_REGEX;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
84 regex_flags |= REG_ICASE;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
85 } else {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
86 log_ereport(LOG_FAILURE, "Location: unknown operator %.*s", (int)type_str.length, type_str.ptr);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
87 return NULL;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
88 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
89
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
90 // arg1: match_string
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
91 match_str = obj->args->next->value;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
92 } else if(argc == 1) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
93 // arg0: match_string
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
94 match_str = obj->args->value;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
95 } else if(argc > 2) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
96 log_ereport(LOG_FAILURE, "Location directive must have 0-2 args");
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
97 return NULL;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
98 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
99
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
100 WSLocation *location = cxZalloc(a, sizeof(WSLocation));
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
101 location->match_string = cx_strdup_a((a), match_str);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
102
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
103 if(match == WS_LOCATION_MATCH_CS_REGEX || match == WS_LOCATION_MATCH_CI_REGEX) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
104 if(regcomp(&location->regex, match_str.ptr, regex_flags)) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
105 log_ereport(LOG_FAILURE, "Location: cannot compile regex pattern %s", match_str.ptr);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
106 return NULL;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
107 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
108 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
109
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
110 ConfigNode *dir = obj->children_begin;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
111 while(dir) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
112 if(dir->type == CONFIG_NODE_OBJECT || dir->type == CONFIG_NODE_DIRECTIVE) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
113 if(add_location_config(a, location, dir)) {
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
114 log_ereport(LOG_FAILURE, "Location %s: abort", match_str.ptr);
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
115 return NULL;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
116 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
117 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
118 dir = dir->next;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
119 }
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
120
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
121
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
122
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
123 return location;
b85d45fd3b01 add location config parser
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
124 }

mercurial