constification
[exim.git] / src / src / lookups / dbmdb.c
CommitLineData
0756eb3c
PH
1/*************************************************
2* Exim - an Internet mail transport agent *
3*************************************************/
4
c4ceed07 5/* Copyright (c) University of Cambridge 1995 - 2012 */
0756eb3c
PH
6/* See the file NOTICE for conditions of use and distribution. */
7
8#include "../exim.h"
9#include "lf_functions.h"
0756eb3c
PH
10
11
12/*************************************************
13* Open entry point *
14*************************************************/
15
16/* See local README for interface description */
17
e6d225ae 18static void *
0756eb3c
PH
19dbmdb_open(uschar *filename, uschar **errmsg)
20{
4a6a987a 21EXIM_DB *yield = NULL;
0756eb3c
PH
22EXIM_DBOPEN(filename, O_RDONLY, 0, &yield);
23if (yield == NULL)
24 {
25 int save_errno = errno;
26 *errmsg = string_open_failed(errno, "%s as a %s file", filename, EXIM_DBTYPE);
27 errno = save_errno;
28 }
29return yield;
30}
31
32
33
34/*************************************************
35* Check entry point *
36*************************************************/
37
38/* This needs to know more about the underlying files than is good for it!
39We need to know what the real file names are in order to check the owners and
40modes. If USE_DB is set, we know it is Berkeley DB, which uses an unmodified
41file name. If USE_TDB or USE_GDBM is set, we know it is tdb or gdbm, which do
42the same. Otherwise, for safety, we have to check for x.db or x.dir and x.pag.
43*/
44
e6d225ae 45static BOOL
0756eb3c
PH
46dbmdb_check(void *handle, uschar *filename, int modemask, uid_t *owners,
47 gid_t *owngroups, uschar **errmsg)
48{
49int rc;
50handle = handle; /* Keep picky compilers happy */
51
52#if defined(USE_DB) || defined(USE_TDB) || defined(USE_GDBM)
53rc = lf_check_file(-1, filename, S_IFREG, modemask, owners, owngroups,
54 "dbm", errmsg);
55#else
56 {
57 uschar filebuffer[256];
f1e894f3 58 (void)sprintf(CS filebuffer, "%.250s.db", filename);
0756eb3c
PH
59 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
60 "dbm", errmsg);
61 if (rc < 0) /* stat() failed */
62 {
f1e894f3 63 (void)sprintf(CS filebuffer, "%.250s.dir", filename);
0756eb3c
PH
64 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
65 "dbm", errmsg);
66 if (rc == 0) /* x.dir was OK */
67 {
f1e894f3 68 (void)sprintf(CS filebuffer, "%.250s.pag", filename);
0756eb3c
PH
69 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
70 "dbm", errmsg);
71 }
72 }
73 }
74#endif
75
76return rc == 0;
77}
78
79
80
81/*************************************************
82* Find entry point *
83*************************************************/
84
85/* See local README for interface description. This function adds 1 to
86the keylength in order to include the terminating zero. */
87
e6d225ae 88static int
55414b25 89dbmdb_find(void *handle, uschar *filename, const uschar *keystring, int length,
0756eb3c
PH
90 uschar **result, uschar **errmsg, BOOL *do_cache)
91{
92EXIM_DB *d = (EXIM_DB *)handle;
93EXIM_DATUM key, data;
94
95filename = filename; /* Keep picky compilers happy */
96errmsg = errmsg;
97do_cache = do_cache;
98
99EXIM_DATUM_INIT(key); /* Some DBM libraries require datums to */
100EXIM_DATUM_INIT(data); /* be cleared before use. */
101EXIM_DATUM_DATA(key) = CS keystring;
102EXIM_DATUM_SIZE(key) = length + 1;
103
104if (EXIM_DBGET(d, key, data))
105 {
106 *result = string_copyn(US EXIM_DATUM_DATA(data), EXIM_DATUM_SIZE(data));
107 EXIM_DATUM_FREE(data); /* Some DBM libraries need a free() call */
108 return OK;
109 }
110return FAIL;
111}
112
113
114
115/*************************************************
116* Find entry point - no zero on key *
117*************************************************/
118
119/* See local README for interface description */
120
121int
55414b25 122static dbmnz_find(void *handle, uschar *filename, const uschar *keystring, int length,
0756eb3c
PH
123 uschar **result, uschar **errmsg, BOOL *do_cache)
124{
125return dbmdb_find(handle, filename, keystring, length-1, result, errmsg,
126 do_cache);
127}
128
129
130
4a6a987a
PP
131/*************************************************
132* Find entry point - zero-joined list key *
133*************************************************/
134
135/*
136 * The parameter passed as a key is a list in normal Exim list syntax.
137 * The elements of that list are joined together on NUL, with no trailing
138 * NUL, to form the key.
139 */
140
141static int
55414b25 142dbmjz_find(void *handle, uschar *filename, const uschar *keystring, int length,
4a6a987a
PP
143 uschar **result, uschar **errmsg, BOOL *do_cache)
144{
145uschar *key_item, *key_buffer, *key_p;
55414b25 146const uschar *key_elems = keystring;
4a6a987a
PP
147int buflen, bufleft, key_item_len, sep = 0;
148
149/* To a first approximation, the size of the lookup key needs to be about,
150or less than, the length of the delimited list passed in + 1. */
151
152buflen = length + 3;
153key_buffer = store_get(buflen);
154
155key_buffer[0] = '\0';
156
157key_p = key_buffer;
158bufleft = buflen;
159
160/* In all cases of an empty list item, we can set 1 and advance by 1 and then
161pick up the trailing NUL from the previous list item, EXCEPT when at the
162beginning of the output string, in which case we need to supply that NUL
163ourselves. */
164while ((key_item = string_nextinlist(&key_elems, &sep, key_p, bufleft)) != NULL)
165 {
166 key_item_len = Ustrlen(key_item) + 1;
167 if (key_item_len == 1)
168 {
169 key_p[0] = '\0';
170 if (key_p == key_buffer)
171 {
172 key_p[1] = '\0';
173 key_item_len += 1;
174 }
175 }
176
177 bufleft -= key_item_len;
178 if (bufleft <= 0)
179 {
180 /* The string_nextinlist() will stop at buffer size, but we should always
181 have at least 1 character extra, so some assumption has failed. */
182 *errmsg = string_copy(US"Ran out of buffer space for joining elements");
183 return DEFER;
184 }
185 key_p += key_item_len;
186 }
187
188if (key_p == key_buffer)
189 {
190 *errmsg = string_copy(US"empty list key");
191 return FAIL;
192 }
193
194/* We do not pass in the final NULL; if needed, the list should include an
195empty element to put one in. Boundary: key length 1, is a NULL */
196key_item_len = key_p - key_buffer - 1;
197
198DEBUG(D_lookup) debug_printf("NUL-joined key length: %d\n", key_item_len);
199
200/* beware that dbmdb_find() adds 1 to length to get back terminating NUL, so
201because we've calculated the real length, we need to subtract one more here */
202return dbmdb_find(handle, filename,
203 key_buffer, key_item_len - 1,
204 result, errmsg, do_cache);
205}
206
207
208
0756eb3c
PH
209/*************************************************
210* Close entry point *
211*************************************************/
212
213/* See local README for interface description */
214
215void
e6d225ae 216static dbmdb_close(void *handle)
0756eb3c
PH
217{
218EXIM_DBCLOSE((EXIM_DB *)handle);
219}
220
6545de78
PP
221
222
223/*************************************************
224* Version reporting entry point *
225*************************************************/
226
227/* See local README for interface description. */
228
229#include "../version.h"
230
231void
232dbm_version_report(FILE *f)
233{
234#ifdef DYNLOOKUP
235fprintf(f, "Library version: DBM: Exim version %s\n", EXIM_VERSION_STR);
236#endif
237}
238
239
e6d225ae
DW
240lookup_info dbm_lookup_info = {
241 US"dbm", /* lookup name */
242 lookup_absfile, /* uses absolute file name */
243 dbmdb_open, /* open function */
244 dbmdb_check, /* check function */
245 dbmdb_find, /* find function */
246 dbmdb_close, /* close function */
247 NULL, /* no tidy function */
6545de78
PP
248 NULL, /* no quoting function */
249 dbm_version_report /* version reporting */
e6d225ae
DW
250};
251
252lookup_info dbmz_lookup_info = {
253 US"dbmnz", /* lookup name */
254 lookup_absfile, /* uses absolute file name */
255 dbmdb_open, /* sic */ /* open function */
256 dbmdb_check, /* sic */ /* check function */
257 dbmnz_find, /* find function */
258 dbmdb_close, /* sic */ /* close function */
259 NULL, /* no tidy function */
6545de78
PP
260 NULL, /* no quoting function */
261 NULL /* no version reporting (redundant) */
e6d225ae
DW
262};
263
4a6a987a
PP
264lookup_info dbmjz_lookup_info = {
265 US"dbmjz", /* lookup name */
266 lookup_absfile, /* uses absolute file name */
267 dbmdb_open, /* sic */ /* open function */
268 dbmdb_check, /* sic */ /* check function */
269 dbmjz_find, /* find function */
270 dbmdb_close, /* sic */ /* close function */
271 NULL, /* no tidy function */
272 NULL, /* no quoting function */
273 NULL /* no version reporting (redundant) */
274};
275
e6d225ae
DW
276#ifdef DYNLOOKUP
277#define dbmdb_lookup_module_info _lookup_module_info
278#endif
279
4a6a987a
PP
280static lookup_info *_lookup_list[] = { &dbm_lookup_info, &dbmz_lookup_info, &dbmjz_lookup_info };
281lookup_module_info dbmdb_lookup_module_info = { LOOKUP_MODULE_INFO_MAGIC, _lookup_list, 3 };
e6d225ae 282
0756eb3c 283/* End of lookups/dbmdb.c */