1 /*************************************************
2 * Exim - an Internet mail transport agent *
3 *************************************************/
5 /* Copyright (c) University of Cambridge 1995 - 2012 */
6 /* See the file NOTICE for conditions of use and distribution. */
9 #include "lf_functions.h"
12 /*************************************************
14 *************************************************/
16 /* See local README for interface description */
19 dbmdb_open(uschar *filename, uschar **errmsg)
21 EXIM_DB *yield = NULL;
22 EXIM_DBOPEN(filename, O_RDONLY, 0, &yield);
25 int save_errno = errno;
26 *errmsg = string_open_failed(errno, "%s as a %s file", filename, EXIM_DBTYPE);
34 /*************************************************
36 *************************************************/
38 /* This needs to know more about the underlying files than is good for it!
39 We need to know what the real file names are in order to check the owners and
40 modes. If USE_DB is set, we know it is Berkeley DB, which uses an unmodified
41 file name. If USE_TDB or USE_GDBM is set, we know it is tdb or gdbm, which do
42 the same. Otherwise, for safety, we have to check for x.db or x.dir and x.pag.
46 dbmdb_check(void *handle, uschar *filename, int modemask, uid_t *owners,
47 gid_t *owngroups, uschar **errmsg)
50 handle = handle; /* Keep picky compilers happy */
52 #if defined(USE_DB) || defined(USE_TDB) || defined(USE_GDBM)
53 rc = lf_check_file(-1, filename, S_IFREG, modemask, owners, owngroups,
57 uschar filebuffer[256];
58 (void)sprintf(CS filebuffer, "%.250s.db", filename);
59 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
61 if (rc < 0) /* stat() failed */
63 (void)sprintf(CS filebuffer, "%.250s.dir", filename);
64 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
66 if (rc == 0) /* x.dir was OK */
68 (void)sprintf(CS filebuffer, "%.250s.pag", filename);
69 rc = lf_check_file(-1, filebuffer, S_IFREG, modemask, owners, owngroups,
81 /*************************************************
83 *************************************************/
85 /* See local README for interface description. This function adds 1 to
86 the keylength in order to include the terminating zero. */
89 dbmdb_find(void *handle, uschar *filename, uschar *keystring, int length,
90 uschar **result, uschar **errmsg, BOOL *do_cache)
92 EXIM_DB *d = (EXIM_DB *)handle;
95 filename = filename; /* Keep picky compilers happy */
99 EXIM_DATUM_INIT(key); /* Some DBM libraries require datums to */
100 EXIM_DATUM_INIT(data); /* be cleared before use. */
101 EXIM_DATUM_DATA(key) = CS keystring;
102 EXIM_DATUM_SIZE(key) = length + 1;
104 if (EXIM_DBGET(d, key, data))
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 */
115 /*************************************************
116 * Find entry point - no zero on key *
117 *************************************************/
119 /* See local README for interface description */
122 static dbmnz_find(void *handle, uschar *filename, uschar *keystring, int length,
123 uschar **result, uschar **errmsg, BOOL *do_cache)
125 return dbmdb_find(handle, filename, keystring, length-1, result, errmsg,
131 /*************************************************
132 * Find entry point - zero-joined list key *
133 *************************************************/
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.
142 dbmjz_find(void *handle, uschar *filename, uschar *keystring, int length,
143 uschar **result, uschar **errmsg, BOOL *do_cache)
145 uschar *key_item, *key_buffer, *key_p;
146 uschar *key_elems = keystring;
147 int buflen, bufleft, key_item_len, sep = 0;
149 /* To a first approximation, the size of the lookup key needs to be about,
150 or less than, the length of the delimited list passed in + 1. */
153 key_buffer = store_get(buflen);
155 key_buffer[0] = '\0';
160 /* In all cases of an empty list item, we can set 1 and advance by 1 and then
161 pick up the trailing NUL from the previous list item, EXCEPT when at the
162 beginning of the output string, in which case we need to supply that NUL
164 while ((key_item = string_nextinlist(&key_elems, &sep, key_p, bufleft)) != NULL)
166 key_item_len = Ustrlen(key_item) + 1;
167 if (key_item_len == 1)
170 if (key_p == key_buffer)
177 bufleft -= key_item_len;
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");
185 key_p += key_item_len;
188 if (key_p == key_buffer)
190 *errmsg = string_copy(US"empty list key");
194 /* We do not pass in the final NULL; if needed, the list should include an
195 empty element to put one in. Boundary: key length 1, is a NULL */
196 key_item_len = key_p - key_buffer - 1;
198 DEBUG(D_lookup) debug_printf("NUL-joined key length: %d\n", key_item_len);
200 /* beware that dbmdb_find() adds 1 to length to get back terminating NUL, so
201 because we've calculated the real length, we need to subtract one more here */
202 return dbmdb_find(handle, filename,
203 key_buffer, key_item_len - 1,
204 result, errmsg, do_cache);
209 /*************************************************
210 * Close entry point *
211 *************************************************/
213 /* See local README for interface description */
216 static dbmdb_close(void *handle)
218 EXIM_DBCLOSE((EXIM_DB *)handle);
223 /*************************************************
224 * Version reporting entry point *
225 *************************************************/
227 /* See local README for interface description. */
229 #include "../version.h"
232 dbm_version_report(FILE *f)
235 fprintf(f, "Library version: DBM: Exim version %s\n", EXIM_VERSION_STR);
240 lookup_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 */
248 NULL, /* no quoting function */
249 dbm_version_report /* version reporting */
252 lookup_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 */
260 NULL, /* no quoting function */
261 NULL /* no version reporting (redundant) */
264 lookup_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) */
277 #define dbmdb_lookup_module_info _lookup_module_info
280 static lookup_info *_lookup_list[] = { &dbm_lookup_info, &dbmz_lookup_info, &dbmjz_lookup_info };
281 lookup_module_info dbmdb_lookup_module_info = { LOOKUP_MODULE_INFO_MAGIC, _lookup_list, 3 };
283 /* End of lookups/dbmdb.c */