bcb4e51a409d94ae670de96afb8483a4f7855294Stephan Bosch/* Copyright (c) 2008-2018 Dovecot authors, see the included COPYING file */
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "lib.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "array.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "hash.h"
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen#include "str.h"
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen#include "strescape.h"
39ea5717264668e2c7f9f7986eb821d21785f47fTimo Sirainen#include "home-expand.h"
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen#include "mkdir-parents.h"
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen#include "eacces-error.h"
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen#include "file-lock.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "file-dotlock.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "nfs-workarounds.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "istream.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "ostream.h"
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen#include "dict-transaction-memory.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include "dict-private.h"
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen#include <stdio.h>
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include <unistd.h>
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include <fcntl.h>
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen#include <sys/stat.h>
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstruct file_dict {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct dict dict;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen pool_t hash_pool;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen enum file_lock_method lock_method;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen char *path;
678d0463849ba777106eb7875f27db07a5d8e3dfTimo Sirainen HASH_TABLE(char *, char *) hash;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen int fd;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen bool refreshed;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen};
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainenstruct file_dict_iterate_path {
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen const char *path;
2ac5f36aa7c2e7a07ba8815d43a6d7483f62e74cTimo Sirainen size_t len;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen};
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstruct file_dict_iterate_context {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct dict_iterate_context ctx;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen pool_t pool;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct hash_iterate_context *iter;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen struct file_dict_iterate_path *paths;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen enum dict_iterate_flags flags;
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen const char *error;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen};
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstatic struct dotlock_settings file_dict_dotlock_settings = {
f323e3f0de9841f399aba5919e3f25652a88fa65Timo Sirainen .timeout = 60*2,
f323e3f0de9841f399aba5919e3f25652a88fa65Timo Sirainen .stale_timeout = 60,
f323e3f0de9841f399aba5919e3f25652a88fa65Timo Sirainen .use_io_notify = TRUE
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen};
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
10399559650f552a23949772be79eb6a80198c5aTimo Sirainenstatic int
10399559650f552a23949772be79eb6a80198c5aTimo Sirainenfile_dict_init(struct dict *driver, const char *uri,
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen const struct dict_settings *set,
39ea5717264668e2c7f9f7986eb821d21785f47fTimo Sirainen struct dict **dict_r, const char **error_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict;
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen const char *p, *path;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen dict = i_new(struct file_dict, 1);
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen dict->lock_method = FILE_LOCK_METHOD_DOTLOCK;
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen p = strchr(uri, ':');
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen if (p == NULL) {
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen /* no parameters */
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen path = uri;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen } else {
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen path = t_strdup_until(uri, p++);
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen if (strcmp(p, "lock=fcntl") == 0)
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen dict->lock_method = FILE_LOCK_METHOD_FCNTL;
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen else if (strcmp(p, "lock=flock") == 0)
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen dict->lock_method = FILE_LOCK_METHOD_FLOCK;
8f2eb1ee9ec07661bd50275da99b5f351972a49aTimo Sirainen else {
eca38954bcf972618f6b85932a3690acbd2b673aTimo Sirainen *error_r = t_strdup_printf("Invalid parameter: %s", p+1);
8f2eb1ee9ec07661bd50275da99b5f351972a49aTimo Sirainen i_free(dict);
8f2eb1ee9ec07661bd50275da99b5f351972a49aTimo Sirainen return -1;
8f2eb1ee9ec07661bd50275da99b5f351972a49aTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen dict->path = set->home_dir == NULL ? i_strdup(path) :
a6a6ad107e509cf8952a28f740eb2023284497b9Timo Sirainen i_strdup(home_expand_tilde(path, set->home_dir));
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen dict->dict = *driver;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen dict->hash_pool = pool_alloconly_create("file dict", 1024);
678d0463849ba777106eb7875f27db07a5d8e3dfTimo Sirainen hash_table_create(&dict->hash, dict->hash_pool, 0, str_hash, strcmp);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen dict->fd = -1;
10399559650f552a23949772be79eb6a80198c5aTimo Sirainen *dict_r = &dict->dict;
dc599de6096c51e6c922e069bfbbcb7d68c50ffaStephan Bosch return 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstatic void file_dict_deinit(struct dict *_dict)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict = (struct file_dict *)_dict;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
a943ed0f901e312445fd393249b91932797bba79Josef 'Jeff' Sipek i_close_fd_path(&dict->fd, dict->path);
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_destroy(&dict->hash);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen pool_unref(&dict->hash_pool);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen i_free(dict->path);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen i_free(dict);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstatic bool file_dict_need_refresh(struct file_dict *dict)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct stat st1, st2;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen if (dict->dict.iter_count > 0) {
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen /* Change nothing while there are iterators or they can crash
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen because the hash table content recreated. */
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen return FALSE;
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen }
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (dict->fd == -1)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return TRUE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
62ff6002b1e37a42303c2c0107f324860232e204Timo Sirainen /* Disable NFS flushing for now since it can cause unnecessary
62ff6002b1e37a42303c2c0107f324860232e204Timo Sirainen problems and there's no easy way for us to know here if
62ff6002b1e37a42303c2c0107f324860232e204Timo Sirainen mail_nfs_storage=yes. In any case it's pretty much an unsupported
62ff6002b1e37a42303c2c0107f324860232e204Timo Sirainen setting nowadays. */
62ff6002b1e37a42303c2c0107f324860232e204Timo Sirainen /*nfs_flush_file_handle_cache(dict->path);*/
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (nfs_safe_stat(dict->path, &st1) < 0) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen i_error("stat(%s) failed: %m", dict->path);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return FALSE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (fstat(dict->fd, &st2) < 0) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (errno != ESTALE)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen i_error("fstat(%s) failed: %m", dict->path);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return TRUE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (st1.st_ino != st2.st_ino ||
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen !CMP_DEV_T(st1.st_dev, st2.st_dev)) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen /* file changed */
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return TRUE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return FALSE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainenstatic int file_dict_open_latest(struct file_dict *dict, const char **error_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen int open_type;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (!file_dict_need_refresh(dict))
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
a943ed0f901e312445fd393249b91932797bba79Josef 'Jeff' Sipek i_close_fd_path(&dict->fd, dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen open_type = dict->lock_method == FILE_LOCK_METHOD_DOTLOCK ?
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen O_RDONLY : O_RDWR;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen dict->fd = open(dict->path, open_type);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (dict->fd == -1) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (errno == ENOENT)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return 0;
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen if (errno == EACCES)
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = eacces_error_get("open", dict->path);
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen else
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = t_strdup_printf("open(%s) failed: %m", dict->path);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return -1;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen dict->refreshed = FALSE;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return 1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen}
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainenstatic int file_dict_refresh(struct file_dict *dict, const char **error_r)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen{
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen struct istream *input;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen char *key, *value;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen if (file_dict_open_latest(dict, error_r) < 0)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen if (dict->refreshed || dict->dict.iter_count > 0)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_clear(dict->hash, TRUE);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen p_clear(dict->hash_pool);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen if (dict->fd != -1) {
e93184a9055c2530366dfe617e07199603c399ddMartti Rannanjärvi input = i_stream_create_fd(dict->fd, (size_t)-1);
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen while ((key = i_stream_read_next_line(input)) != NULL) {
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen /* strdup() before the second read */
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen key = str_tabunescape(p_strdup(dict->hash_pool, key));
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen if ((value = i_stream_read_next_line(input)) == NULL)
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen break;
3350b29fce44f8bc8fa015dad57024a8de301d38Timo Sirainen
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen value = str_tabunescape(p_strdup(dict->hash_pool, value));
c224fff79d18480a65e9b4504b891b8ea176f5b1Timo Sirainen hash_table_update(dict->hash, key, value);
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen }
bdb0f594a5673a0c5a16b92dc49eb2a8a66bdaceTimo Sirainen i_stream_destroy(&input);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen dict->refreshed = TRUE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainenstatic int file_dict_lookup(struct dict *_dict, pool_t pool, const char *key,
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen const char **value_r, const char **error_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict = (struct file_dict *)_dict;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen if (file_dict_refresh(dict, error_r) < 0)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return -1;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen *value_r = p_strdup(pool, hash_table_lookup(dict->hash, key));
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return *value_r == NULL ? 0 : 1;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstatic struct dict_iterate_context *
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainenfile_dict_iterate_init(struct dict *_dict, const char *const *paths,
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen enum dict_iterate_flags flags)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict_iterate_context *ctx;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict = (struct file_dict *)_dict;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen unsigned int i, path_count;
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen const char *error;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen pool_t pool;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen pool = pool_alloconly_create("file dict iterate", 256);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen ctx = p_new(pool, struct file_dict_iterate_context, 1);
8ad53e0bb29f61350f608fc519210f2442c20775Timo Sirainen ctx->ctx.dict = _dict;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen ctx->pool = pool;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen for (path_count = 0; paths[path_count] != NULL; path_count++) ;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen ctx->paths = p_new(pool, struct file_dict_iterate_path, path_count + 1);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen for (i = 0; i < path_count; i++) {
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen ctx->paths[i].path = p_strdup(pool, paths[i]);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen ctx->paths[i].len = strlen(paths[i]);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen ctx->flags = flags;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen if (file_dict_refresh(dict, &error) < 0)
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen ctx->error = p_strdup(pool, error);
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen
37fed1bc1545f7eb1755b61d6a5ac4d083a693b3Timo Sirainen ctx->iter = hash_table_iterate_init(dict->hash);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return &ctx->ctx;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainenstatic const struct file_dict_iterate_path *
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainenfile_dict_iterate_find_path(struct file_dict_iterate_context *ctx,
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen const char *key)
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen{
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen unsigned int i;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen for (i = 0; ctx->paths[i].path != NULL; i++) {
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen if (strncmp(ctx->paths[i].path, key, ctx->paths[i].len) == 0)
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen return &ctx->paths[i];
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen }
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen return NULL;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen}
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen
8d25b6ad05b99e75613cb045a121efd51e6afbb6Timo Sirainenstatic bool file_dict_iterate(struct dict_iterate_context *_ctx,
8d25b6ad05b99e75613cb045a121efd51e6afbb6Timo Sirainen const char **key_r, const char **value_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict_iterate_context *ctx =
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen (struct file_dict_iterate_context *)_ctx;
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen const struct file_dict_iterate_path *path;
678d0463849ba777106eb7875f27db07a5d8e3dfTimo Sirainen char *key, *value;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
c73415e93ecf1c699ef054d2b179b10976fa23f3Timo Sirainen while (hash_table_iterate(ctx->iter,
c73415e93ecf1c699ef054d2b179b10976fa23f3Timo Sirainen ((struct file_dict *)_ctx->dict)->hash,
c73415e93ecf1c699ef054d2b179b10976fa23f3Timo Sirainen &key, &value)) {
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen path = file_dict_iterate_find_path(ctx, key);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen if (path == NULL)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen continue;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen if ((ctx->flags & DICT_ITERATE_FLAG_RECURSE) != 0) {
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen /* match everything */
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen } else if ((ctx->flags & DICT_ITERATE_FLAG_EXACT_KEY) != 0) {
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen if (key[path->len] != '\0')
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen continue;
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen } else {
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen if (strchr(key + path->len, '/') != NULL)
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen continue;
76b4207273534f71365bc7f900c23a5160692802Timo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen *key_r = key;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen *value_r = value;
8d25b6ad05b99e75613cb045a121efd51e6afbb6Timo Sirainen return TRUE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
8d25b6ad05b99e75613cb045a121efd51e6afbb6Timo Sirainen return FALSE;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainenstatic int file_dict_iterate_deinit(struct dict_iterate_context *_ctx,
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen const char **error_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict_iterate_context *ctx =
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen (struct file_dict_iterate_context *)_ctx;
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen int ret = ctx->error != NULL ? -1 : 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
055389c58fa3915e12fb4e72ec86782ce77c5c72Timo Sirainen *error_r = t_strdup(ctx->error);
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_iterate_deinit(&ctx->iter);
9c7f6dbf65ca01026e5f9c8c8b67c7e629c0b5e7Timo Sirainen pool_unref(&ctx->pool);
8d25b6ad05b99e75613cb045a121efd51e6afbb6Timo Sirainen return ret;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstatic struct dict_transaction_context *
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenfile_dict_transaction_init(struct dict *_dict)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen struct dict_transaction_memory_context *ctx;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen pool_t pool;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
dba8af1faaf9fd3957254bb6f2234b285f77096fTimo Sirainen pool = pool_alloconly_create("file dict transaction", 2048);
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen ctx = p_new(pool, struct dict_transaction_memory_context, 1);
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen dict_transaction_memory_init(ctx, _dict, pool);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return &ctx->ctx;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainenstatic void file_dict_apply_changes(struct dict_transaction_memory_context *ctx,
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen bool *atomic_inc_not_found_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict = (struct file_dict *)ctx->ctx.dict;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen const char *tmp;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen char *key, *value, *old_value;
678d0463849ba777106eb7875f27db07a5d8e3dfTimo Sirainen char *orig_key, *orig_value;
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen const struct dict_transaction_memory_change *change;
2ac5f36aa7c2e7a07ba8815d43a6d7483f62e74cTimo Sirainen size_t new_len;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen long long diff;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
7bd72e4deca3cbf757dd1ea298486d9f3bc24226Timo Sirainen array_foreach(&ctx->changes, change) {
a75d470c9223a75801418fcdda258885c36317e0Timo Sirainen if (hash_table_lookup_full(dict->hash, change->key,
a75d470c9223a75801418fcdda258885c36317e0Timo Sirainen &orig_key, &orig_value)) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen key = orig_key;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen old_value = orig_value;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen } else {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen key = NULL;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen old_value = NULL;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen value = NULL;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
7bd72e4deca3cbf757dd1ea298486d9f3bc24226Timo Sirainen switch (change->type) {
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen case DICT_CHANGE_TYPE_INC:
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen if (old_value == NULL) {
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen *atomic_inc_not_found_r = TRUE;
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen break;
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen }
e48f289d2e5b2546a2c5dcc90f7ab624cc58cca2Stephan Bosch if (str_to_llong(old_value, &diff) < 0)
e48f289d2e5b2546a2c5dcc90f7ab624cc58cca2Stephan Bosch i_unreached();
e48f289d2e5b2546a2c5dcc90f7ab624cc58cca2Stephan Bosch diff += change->value.diff;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen tmp = t_strdup_printf("%lld", diff);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen new_len = strlen(tmp);
3c63c219ae7854b4f1d44a671a65572aa242cbcfTimo Sirainen if (old_value == NULL || new_len > strlen(old_value))
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen value = p_strdup(dict->hash_pool, tmp);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen else {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen memcpy(old_value, tmp, new_len + 1);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen value = old_value;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen /* fall through */
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen case DICT_CHANGE_TYPE_SET:
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (key == NULL)
7bd72e4deca3cbf757dd1ea298486d9f3bc24226Timo Sirainen key = p_strdup(dict->hash_pool, change->key);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (value == NULL) {
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen value = p_strdup(dict->hash_pool,
7bd72e4deca3cbf757dd1ea298486d9f3bc24226Timo Sirainen change->value.str);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_update(dict->hash, key, value);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen break;
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen case DICT_CHANGE_TYPE_UNSET:
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen if (old_value != NULL)
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_remove(dict->hash, key);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen break;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainenstatic int
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainenfd_copy_stat_permissions(const struct stat *src_st,
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen int dest_fd, const char *dest_path)
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen{
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen struct stat dest_st;
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen if (fstat(dest_fd, &dest_st) < 0) {
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen i_error("fstat(%s) failed: %m", dest_path);
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen return -1;
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (src_st->st_gid != dest_st.st_gid &&
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen ((src_st->st_mode & 0070) >> 3 != (src_st->st_mode & 0007))) {
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen /* group has different permissions from world.
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen preserve the group. */
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (fchown(dest_fd, (uid_t)-1, src_st->st_gid) < 0) {
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen i_error("fchown(%s, -1, %s) failed: %m",
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen dest_path, dec2str(src_st->st_gid));
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen return -1;
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if ((src_st->st_mode & 07777) != (dest_st.st_mode & 07777)) {
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (fchmod(dest_fd, src_st->st_mode & 07777) < 0) {
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen i_error("fchmod(%s, %o) failed: %m",
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen dest_path, (int)(src_st->st_mode & 0777));
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen return -1;
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen return 0;
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen}
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainenstatic int fd_copy_permissions(int src_fd, const char *src_path,
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen int dest_fd, const char *dest_path)
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen{
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen struct stat src_st;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (fstat(src_fd, &src_st) < 0) {
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen i_error("fstat(%s) failed: %m", src_path);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen return -1;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen }
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen return fd_copy_stat_permissions(&src_st, dest_fd, dest_path);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen}
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainenstatic int
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainenfd_copy_parent_dir_permissions(const char *src_path, int dest_fd,
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen const char *dest_path)
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen{
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen struct stat src_st;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen const char *src_dir, *p;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen p = strrchr(src_path, '/');
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (p == NULL)
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen src_dir = ".";
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen else
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen src_dir = t_strdup_until(src_path, p);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen if (stat(src_dir, &src_st) < 0) {
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen i_error("stat(%s) failed: %m", src_dir);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen return -1;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen }
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen src_st.st_mode &= 0666;
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen return fd_copy_stat_permissions(&src_st, dest_fd, dest_path);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen}
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainenstatic int file_dict_mkdir(struct file_dict *dict, const char **error_r)
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen{
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen const char *path, *p, *root;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen struct stat st;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen mode_t mode = 0700;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen p = strrchr(dict->path, '/');
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen if (p == NULL)
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return 0;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen path = t_strdup_until(dict->path, p);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen if (stat_first_parent(path, &root, &st) < 0) {
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen if (errno == EACCES)
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = eacces_error_get("stat", root);
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen else
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = t_strdup_printf("stat(%s) failed: %m", root);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return -1;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen }
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen if ((st.st_mode & S_ISGID) != 0) {
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen /* preserve parent's permissions when it has setgid bit */
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen mode = st.st_mode;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen }
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen
c85f661daf164687fc5af22d74902f94c26597d0Timo Sirainen if (mkdir_parents(path, mode) < 0 && errno != EEXIST) {
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen if (errno == EACCES)
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = eacces_error_get("mkdir_parents", path);
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen else
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = t_strdup_printf("mkdir_parents(%s) failed: %m", path);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return -1;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen }
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return 0;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen}
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainenstatic int
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainenfile_dict_lock(struct file_dict *dict, struct file_lock **lock_r,
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen const char **error_r)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen{
c719f74d3fd41d9b9fea0edaea1e00ab90da72dcTimo Sirainen int ret;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen if (file_dict_open_latest(dict, error_r) < 0)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (dict->fd == -1) {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen /* quota file doesn't exist yet, we need to create it */
c719f74d3fd41d9b9fea0edaea1e00ab90da72dcTimo Sirainen dict->fd = open(dict->path, O_CREAT | O_RDWR, 0600);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen if (dict->fd == -1 && errno == ENOENT) {
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen if (file_dict_mkdir(dict, error_r) < 0)
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return -1;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen dict->fd = open(dict->path, O_CREAT | O_RDWR, 0600);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen }
c719f74d3fd41d9b9fea0edaea1e00ab90da72dcTimo Sirainen if (dict->fd == -1) {
3bc62efe513ebc7450cffe9a4e8f0b07424bf190Timo Sirainen if (errno == EACCES)
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = eacces_error_get("creat", dict->path);
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen else {
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = t_strdup_printf(
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen "creat(%s) failed: %m", dict->path);
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
c719f74d3fd41d9b9fea0edaea1e00ab90da72dcTimo Sirainen (void)fd_copy_parent_dir_permissions(dict->path, dict->fd,
c719f74d3fd41d9b9fea0edaea1e00ab90da72dcTimo Sirainen dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen do {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (file_wait_lock(dict->fd, dict->path, F_WRLCK,
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen dict->lock_method,
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen file_dict_dotlock_settings.timeout,
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen lock_r) <= 0) {
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen *error_r = t_strdup_printf(
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen "file_wait_lock(%s) failed: %m", dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen /* check again if we need to reopen the file because it was
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen just replaced */
b5052fbfdbc2678cc8f12899afe55c998f43b740Timo Sirainen } while ((ret = file_dict_open_latest(dict, error_r)) > 0);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return ret < 0 ? -1 : 0;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen}
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainenstatic int
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainenfile_dict_write_changes(struct dict_transaction_memory_context *ctx,
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen bool *atomic_inc_not_found_r, const char **error_r)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct file_dict *dict = (struct file_dict *)ctx->ctx.dict;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen struct dotlock *dotlock = NULL;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen struct file_lock *lock = NULL;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen const char *temp_path = NULL;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct hash_iterate_context *iter;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen struct ostream *output;
678d0463849ba777106eb7875f27db07a5d8e3dfTimo Sirainen char *key, *value;
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen string_t *str;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen int fd = -1;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen *atomic_inc_not_found_r = FALSE;
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen switch (dict->lock_method) {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen case FILE_LOCK_METHOD_FCNTL:
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen case FILE_LOCK_METHOD_FLOCK:
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen if (file_dict_lock(dict, &lock, error_r) < 0)
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen temp_path = t_strdup_printf("%s.tmp", dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen fd = creat(temp_path, 0600);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (fd == -1) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen *error_r = t_strdup_printf(
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen "dict-file: creat(%s) failed: %m", temp_path);
262eeae35ee285981b37294628ec8d054daa1b23Timo Sirainen file_unlock(&lock);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen break;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen case FILE_LOCK_METHOD_DOTLOCK:
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen fd = file_dotlock_open(&file_dict_dotlock_settings, dict->path, 0,
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen &dotlock);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen if (fd == -1 && errno == ENOENT) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen if (file_dict_mkdir(dict, error_r) < 0)
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen return -1;
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen fd = file_dotlock_open(&file_dict_dotlock_settings,
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen dict->path, 0, &dotlock);
97943a36e08923d625898f5ca8ffd38325a3986dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (fd == -1) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen *error_r = t_strdup_printf(
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen "dict-file: file_dotlock_open(%s) failed: %m",
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen temp_path = file_dotlock_get_lock_path(dotlock);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen break;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen /* refresh once more now that we're locked */
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen if (file_dict_refresh(dict, error_r) < 0) {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (dotlock != NULL)
b66d803de86bfb411165b3465b0d9ef64ecfe2a1Timo Sirainen file_dotlock_delete(&dotlock);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen else {
4307c886579381dbb1897ea1388ae6978c96f560Timo Sirainen i_close_fd(&fd);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen file_unlock(&lock);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return -1;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen if (dict->fd != -1) {
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen /* preserve the permissions */
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen (void)fd_copy_permissions(dict->fd, dict->path, fd, temp_path);
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen } else {
e5ff2112aea089f3de2badf9b1635677791d1384Timo Sirainen /* get initial permissions from parent directory */
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen (void)fd_copy_parent_dir_permissions(dict->path, fd, temp_path);
b141c303bf09b6ae43e1eb4aac1e1a6b796b9d35Timo Sirainen }
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen file_dict_apply_changes(ctx, atomic_inc_not_found_r);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e93184a9055c2530366dfe617e07199603c399ddMartti Rannanjärvi output = o_stream_create_fd(fd, 0);
313c43343f711f04a8d4d1f8dd21aaab5c16b2d2Timo Sirainen o_stream_cork(output);
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen iter = hash_table_iterate_init(dict->hash);
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str = t_str_new(256);
a75d470c9223a75801418fcdda258885c36317e0Timo Sirainen while (hash_table_iterate(iter, dict->hash, &key, &value)) {
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str_truncate(str, 0);
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str_append_tabescaped(str, key);
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str_append_c(str, '\n');
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str_append_tabescaped(str, value);
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen str_append_c(str, '\n');
1f366614aaafcc9496ff85b25988f19c3254ab7cTimo Sirainen o_stream_nsend(output, str_data(str), str_len(str));
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
3ccab0bac68040f179a7de45c516cec258e28fdbTimo Sirainen hash_table_iterate_deinit(&iter);
efd9d73a7fba2830431aa1186fd65372f6631399Timo Sirainen
ad9afb64630511d5e25bc5bc11c5304986156928Timo Sirainen if (o_stream_finish(output) <= 0) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen *error_r = t_strdup_printf("write(%s) failed: %s", temp_path,
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen o_stream_get_error(output));
efd9d73a7fba2830431aa1186fd65372f6631399Timo Sirainen o_stream_destroy(&output);
4307c886579381dbb1897ea1388ae6978c96f560Timo Sirainen i_close_fd(&fd);
efd9d73a7fba2830431aa1186fd65372f6631399Timo Sirainen return -1;
efd9d73a7fba2830431aa1186fd65372f6631399Timo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen o_stream_destroy(&output);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (dotlock != NULL) {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (file_dotlock_replace(&dotlock,
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen DOTLOCK_REPLACE_FLAG_DONT_CLOSE_FD) < 0) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen *error_r = t_strdup_printf("file_dotlock_replace() failed: %m");
4307c886579381dbb1897ea1388ae6978c96f560Timo Sirainen i_close_fd(&fd);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen } else {
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen if (rename(temp_path, dict->path) < 0) {
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen *error_r = t_strdup_printf("rename(%s, %s) failed: %m",
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen temp_path, dict->path);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen file_unlock(&lock);
4307c886579381dbb1897ea1388ae6978c96f560Timo Sirainen i_close_fd(&fd);
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen return -1;
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen }
f739c92a9237db03327dc82e3792e39c160a1e4dTimo Sirainen file_lock_free(&lock);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
7b032348d7bbb93ff96188289d3dfc1899b9abb3Josef 'Jeff' Sipek i_close_fd(&dict->fd);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen dict->fd = fd;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen return 0;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
c2a66e7950cb4d3fc4d68e4480ea8f39bdd7c871Timo Sirainenstatic void
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainenfile_dict_transaction_commit(struct dict_transaction_context *_ctx,
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen bool async ATTR_UNUSED,
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen dict_transaction_commit_callback_t *callback,
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen void *context)
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen{
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen struct dict_transaction_memory_context *ctx =
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen (struct dict_transaction_memory_context *)_ctx;
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen struct dict_commit_result result;
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen bool atomic_inc_not_found;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch i_zero(&result);
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen if (file_dict_write_changes(ctx, &atomic_inc_not_found, &result.error) < 0)
9a382894724292e2af60ef94fc471d761f45e5d5Timo Sirainen result.ret = DICT_COMMIT_RET_FAILED;
8bf42ce5ef783b96a2ded67524173e95e9b45adaTimo Sirainen else if (atomic_inc_not_found)
9a382894724292e2af60ef94fc471d761f45e5d5Timo Sirainen result.ret = DICT_COMMIT_RET_NOTFOUND;
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen else
9a382894724292e2af60ef94fc471d761f45e5d5Timo Sirainen result.ret = DICT_COMMIT_RET_OK;
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen pool_unref(&ctx->pool);
3954326e793bdef1e94e0ad781ed6cc7e48beebbTimo Sirainen
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen callback(&result, context);
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen}
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainenstruct dict dict_driver_file = {
7bafda1813454621e03615e83d55bccfa7cc56bdTimo Sirainen .name = "file",
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen {
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .init = file_dict_init,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .deinit = file_dict_deinit,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .lookup = file_dict_lookup,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .iterate_init = file_dict_iterate_init,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .iterate = file_dict_iterate,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .iterate_deinit = file_dict_iterate_deinit,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .transaction_init = file_dict_transaction_init,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .transaction_commit = file_dict_transaction_commit,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .transaction_rollback = dict_transaction_memory_rollback,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .set = dict_transaction_memory_set,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .unset = dict_transaction_memory_unset,
ade5567577dadb0b275c840208d3ad21a9f00a36Timo Sirainen .atomic_inc = dict_transaction_memory_atomic_inc,
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen }
e3fabe8d0faa9aab7cae2d0eee9653f581a3061dTimo Sirainen};