9b1fe78ed3b4e9d17e20ee8863d74563ac096302
[exim.git] / src / src / lookups / json.c
1 /*************************************************
2 *     Exim - an Internet mail transport agent    *
3 *************************************************/
4
5 /* Copyright (c) Jeremy Harris 2019-2020 */
6 /* See the file NOTICE for conditions of use and distribution. */
7
8 #include "../exim.h"
9 #include "lf_functions.h"
10 #include <jansson.h>
11
12
13
14 /* All use of allocations will be done against the POOL_SEARCH memory,
15 which is freed once by search_tidyup(). Make the free call a dummy.
16 This burns some 300kB in handling a 37kB JSON file, for the benefit of
17 a fast free.  The alternative of staying with malloc is nearly as bad,
18 eyeballing the activity there are 20% the number of free vs. alloc
19 calls (before the big bunch at the end).
20
21 Assume that the file is trusted, so no tainting */
22
23 static void *
24 json_malloc(size_t nbytes)
25 {
26 void * p = store_get((int)nbytes, FALSE);
27 /* debug_printf("%s %d: %p\n", __FUNCTION__, (int)nbytes, p); */
28 return p;
29 }
30 static void
31 json_free(void * p)
32 {
33 /* debug_printf("%s: %p\n", __FUNCTION__, p); */
34 }
35
36 /*************************************************
37 *              Open entry point                  *
38 *************************************************/
39
40 /* See local README for interface description */
41
42 static void *
43 json_open(const uschar * filename, uschar ** errmsg)
44 {
45 FILE * f;
46
47 json_set_alloc_funcs(json_malloc, json_free);
48
49 if (!(f = Ufopen(filename, "rb")))
50   *errmsg = string_open_failed("%s for json search", filename);
51 return f;
52 }
53
54
55
56 /*************************************************
57 *             Check entry point                  *
58 *************************************************/
59
60 static BOOL
61 json_check(void *handle, const uschar *filename, int modemask, uid_t *owners,
62   gid_t *owngroups, uschar **errmsg)
63 {
64 return lf_check_file(fileno((FILE *)handle), filename, S_IFREG, modemask,
65   owners, owngroups, "json", errmsg) == 0;
66 }
67
68
69
70 /*************************************************
71 *         Find entry point for lsearch           *
72 *************************************************/
73
74 /* See local README for interface description */
75
76 static int
77 json_find(void * handle, const uschar * filename, const uschar * keystring,
78   int length, uschar ** result, uschar ** errmsg, uint * do_cache,
79   const uschar * opts)
80 {
81 FILE * f = handle;
82 json_t * j, * j0;
83 json_error_t jerr;
84 uschar * key;
85 int sep = 0;
86
87 rewind(f);
88 if (!(j = json_loadf(f, 0, &jerr)))
89   {
90   *errmsg = string_sprintf("json error on open: %.*s\n",
91        JSON_ERROR_TEXT_LENGTH, jerr.text);
92   return FAIL;
93   }
94 j0 = j;
95
96 for (int k = 1;  (key = string_nextinlist(&keystring, &sep, NULL, 0)); k++)
97   {
98   BOOL numeric = TRUE;
99   for (uschar * s = key; *s; s++) if (!isdigit(*s)) { numeric = FALSE; break; }
100
101   if (!(j = numeric
102         ? json_array_get(j, (size_t) strtoul(CS key, NULL, 10))
103         : json_object_get(j, CCS key)
104      ) )
105     {
106     DEBUG(D_lookup) debug_printf_indent("%s, for key %d: '%s'\n",
107       numeric
108       ? US"bad index, or not json array"
109       : US"no such key, or not json object",
110       k, key);
111     json_decref(j0);
112     return FAIL;
113     }
114   }
115
116 switch (json_typeof(j))
117   {
118   case JSON_STRING:
119     *result = string_copyn(CUS json_string_value(j), json_string_length(j));
120     break;
121   case JSON_INTEGER:
122     *result = string_sprintf("%" JSON_INTEGER_FORMAT, json_integer_value(j));
123     break;
124   case JSON_REAL:
125     *result = string_sprintf("%f", json_real_value(j));
126     break;
127   case JSON_TRUE:       *result = US"true";     break;
128   case JSON_FALSE:      *result = US"false";    break;
129   case JSON_NULL:       *result = NULL;         break;
130   default:              *result = US json_dumps(j, 0); break;
131   }
132 json_decref(j0);
133 return OK;
134 }
135
136
137
138 /*************************************************
139 *              Close entry point                 *
140 *************************************************/
141
142 /* See local README for interface description */
143
144 static void
145 json_close(void *handle)
146 {
147 (void)fclose((FILE *)handle);
148 }
149
150
151
152 /*************************************************
153 *         Version reporting entry point          *
154 *************************************************/
155
156 /* See local README for interface description. */
157
158 #include "../version.h"
159
160 void
161 json_version_report(FILE *f)
162 {
163 fprintf(f, "Library version: json: Jansonn version %s\n", JANSSON_VERSION);
164 }
165
166
167 static lookup_info json_lookup_info = {
168   .name = US"json",                     /* lookup name */
169   .type = lookup_absfile,               /* uses absolute file name */
170   .open = json_open,                    /* open function */
171   .check = json_check,                  /* check function */
172   .find = json_find,                    /* find function */
173   .close = json_close,                  /* close function */
174   .tidy = NULL,                         /* no tidy function */
175   .quote = NULL,                        /* no quoting function */
176   .version_report = json_version_report         /* version reporting */
177 };
178
179
180 #ifdef DYNLOOKUP
181 #define json_lookup_module_info _lookup_module_info
182 #endif
183
184 static lookup_info *_lookup_list[] = { &json_lookup_info };
185 lookup_module_info json_lookup_module_info = { LOOKUP_MODULE_INFO_MAGIC, _lookup_list, 1 };
186
187 /* End of lookups/json.c */