diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 09:51:24 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-28 09:51:24 +0000 |
commit | f7548d6d28c313cf80e6f3ef89aed16a19815df1 (patch) | |
tree | a3f6f2a3f247293bee59ecd28e8cd8ceb6ca064a /src/plugins/quota/quota-dict.c | |
parent | Initial commit. (diff) | |
download | dovecot-upstream.tar.xz dovecot-upstream.zip |
Adding upstream version 1:2.3.19.1+dfsg1.upstream/1%2.3.19.1+dfsg1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/plugins/quota/quota-dict.c')
-rw-r--r-- | src/plugins/quota/quota-dict.c | 269 |
1 files changed, 269 insertions, 0 deletions
diff --git a/src/plugins/quota/quota-dict.c b/src/plugins/quota/quota-dict.c new file mode 100644 index 0000000..02e444a --- /dev/null +++ b/src/plugins/quota/quota-dict.c @@ -0,0 +1,269 @@ +/* Copyright (c) 2005-2018 Dovecot authors, see the included COPYING file */ + +#include "lib.h" +#include "ioloop.h" +#include "str.h" +#include "dict.h" +#include "mail-user.h" +#include "mail-namespace.h" +#include "quota-private.h" + + +#define DICT_QUOTA_CURRENT_PATH DICT_PATH_PRIVATE"quota/" +#define DICT_QUOTA_CURRENT_BYTES_PATH DICT_QUOTA_CURRENT_PATH"storage" +#define DICT_QUOTA_CURRENT_COUNT_PATH DICT_QUOTA_CURRENT_PATH"messages" + +struct dict_quota_root { + struct quota_root root; + struct dict *dict; + struct timeout *to_update; + bool disable_unset; +}; + +extern struct quota_backend quota_backend_dict; + +static struct quota_root *dict_quota_alloc(void) +{ + struct dict_quota_root *root; + + root = i_new(struct dict_quota_root, 1); + return &root->root; +} + +static void handle_nounset_param(struct quota_root *_root, const char *param_value ATTR_UNUSED) +{ + ((struct dict_quota_root *)_root)->disable_unset = TRUE; +} + +static int dict_quota_init(struct quota_root *_root, const char *args, + const char **error_r) +{ + struct dict_quota_root *root = (struct dict_quota_root *)_root; + struct dict_settings set; + const char *username, *p, *error; + + event_set_append_log_prefix(_root->backend.event, "quota-dict: "); + + const struct quota_param_parser dict_params[] = { + {.param_name = "no-unset", .param_handler = handle_nounset_param}, + quota_param_hidden, quota_param_ignoreunlimited, quota_param_noenforcing, quota_param_ns, + {.param_name = NULL} + }; + + p = args == NULL ? NULL : strchr(args, ':'); + if (p == NULL) { + *error_r = "URI missing from parameters"; + return -1; + } + + username = t_strdup_until(args, p); + args = p+1; + + if (quota_parse_parameters(_root, &args, error_r, dict_params, FALSE) < 0) + i_unreached(); + + if (*username == '\0') + username = _root->quota->user->username; + + e_debug(_root->backend.event, "user=%s, uri=%s, noenforcing=%d", + username, args, _root->no_enforcing ? 1 : 0); + + /* FIXME: we should use 64bit integer as datatype instead but before + it can actually be used don't bother */ + i_zero(&set); + set.base_dir = _root->quota->user->set->base_dir; + set.event_parent = _root->quota->user->event; + if (dict_init(args, &set, &root->dict, &error) < 0) { + *error_r = t_strdup_printf("dict_init(%s) failed: %s", args, error); + return -1; + } + return 0; +} + +static void dict_quota_deinit(struct quota_root *_root) +{ + struct dict_quota_root *root = (struct dict_quota_root *)_root; + + i_assert(root->to_update == NULL); + + if (root->dict != NULL) { + dict_wait(root->dict); + dict_deinit(&root->dict); + } + i_free(root); +} + +static const char *const * +dict_quota_root_get_resources(struct quota_root *root ATTR_UNUSED) +{ + static const char *resources[] = { + QUOTA_NAME_STORAGE_KILOBYTES, QUOTA_NAME_MESSAGES, NULL + }; + + return resources; +} + +static enum quota_get_result +dict_quota_count(struct dict_quota_root *root, + bool want_bytes, uint64_t *value_r, + const char **error_r) +{ + struct dict_transaction_context *dt; + struct event_reason *reason; + uint64_t bytes, count; + enum quota_get_result error_res; + const struct dict_op_settings *set; + + reason = event_reason_begin("quota:recalculate"); + int ret = quota_count(&root->root, &bytes, &count, &error_res, error_r); + event_reason_end(&reason); + if (ret < 0) + return error_res; + + set = mail_user_get_dict_op_settings(root->root.quota->user); + dt = dict_transaction_begin(root->dict, set); + /* these unsets are mainly necessary for pgsql, because its + trigger otherwise increases quota without deleting it. + but some people with other databases want to store the + quota usage among other data in the same row, which + shouldn't be deleted. */ + if (!root->disable_unset) { + dict_unset(dt, DICT_QUOTA_CURRENT_BYTES_PATH); + dict_unset(dt, DICT_QUOTA_CURRENT_COUNT_PATH); + } + dict_set(dt, DICT_QUOTA_CURRENT_BYTES_PATH, dec2str(bytes)); + dict_set(dt, DICT_QUOTA_CURRENT_COUNT_PATH, dec2str(count)); + + e_debug(root->root.backend.event, "Quota recalculated: " + "count=%"PRIu64" bytes=%"PRIu64, count, bytes); + + dict_transaction_commit_async_nocallback(&dt); + *value_r = want_bytes ? bytes : count; + return QUOTA_GET_RESULT_LIMITED; +} + +static enum quota_get_result +dict_quota_get_resource(struct quota_root *_root, + const char *name, uint64_t *value_r, + const char **error_r) +{ + struct dict_quota_root *root = (struct dict_quota_root *)_root; + bool want_bytes; + int ret; + const struct dict_op_settings *set; + + if (strcmp(name, QUOTA_NAME_STORAGE_BYTES) == 0) + want_bytes = TRUE; + else if (strcmp(name, QUOTA_NAME_MESSAGES) == 0) + want_bytes = FALSE; + else { + *error_r = QUOTA_UNKNOWN_RESOURCE_ERROR_STRING; + return QUOTA_GET_RESULT_UNKNOWN_RESOURCE; + } + + set = mail_user_get_dict_op_settings(root->root.quota->user); + const char *key, *value, *error; + key = want_bytes ? DICT_QUOTA_CURRENT_BYTES_PATH : + DICT_QUOTA_CURRENT_COUNT_PATH; + ret = dict_lookup(root->dict, set, unsafe_data_stack_pool, + key, &value, &error); + if (ret < 0) { + *error_r = t_strdup_printf( + "dict_lookup(%s) failed: %s", key, error); + *value_r = 0; + return QUOTA_GET_RESULT_INTERNAL_ERROR; + } + + intmax_t tmp; + /* recalculate quota if it's negative or if it wasn't found */ + if (ret == 0 || str_to_intmax(value, &tmp) < 0) + tmp = -1; + if (tmp >= 0) + *value_r = tmp; + else + return dict_quota_count(root, want_bytes, value_r, error_r); + return QUOTA_GET_RESULT_LIMITED; +} + +static void dict_quota_recalc_timeout(struct dict_quota_root *root) +{ + uint64_t value; + const char *error; + + timeout_remove(&root->to_update); + if (dict_quota_count(root, TRUE, &value, &error) + <= QUOTA_GET_RESULT_INTERNAL_ERROR) + e_error(root->root.backend.event, + "Recalculation failed: %s", error); +} + +static void dict_quota_update_callback(const struct dict_commit_result *result, + struct dict_quota_root *root) +{ + if (result->ret == 0) { + /* row doesn't exist, need to recalculate it */ + if (root->to_update == NULL) + root->to_update = timeout_add_short(0, dict_quota_recalc_timeout, root); + } else if (result->ret < 0) { + e_error(root->root.backend.event, + "Quota update failed: %s " + "- Quota is now desynced", result->error); + } +} + +static int +dict_quota_update(struct quota_root *_root, + struct quota_transaction_context *ctx, + const char **error_r) +{ + struct dict_quota_root *root = (struct dict_quota_root *) _root; + struct dict_transaction_context *dt; + uint64_t value; + const struct dict_op_settings *set; + + if (ctx->recalculate != QUOTA_RECALCULATE_DONT) { + if (dict_quota_count(root, TRUE, &value, error_r) + <= QUOTA_GET_RESULT_INTERNAL_ERROR) + return -1; + } else { + set = mail_user_get_dict_op_settings(root->root.quota->user); + dt = dict_transaction_begin(root->dict, set); + if (ctx->bytes_used != 0) { + dict_atomic_inc(dt, DICT_QUOTA_CURRENT_BYTES_PATH, + ctx->bytes_used); + } + if (ctx->count_used != 0) { + dict_atomic_inc(dt, DICT_QUOTA_CURRENT_COUNT_PATH, + ctx->count_used); + } + dict_transaction_no_slowness_warning(dt); + dict_transaction_commit_async(&dt, dict_quota_update_callback, + root); + } + return 0; +} + +static void dict_quota_flush(struct quota_root *_root) +{ + struct dict_quota_root *root = (struct dict_quota_root *)_root; + + dict_wait(root->dict); + if (root->to_update != NULL) { + dict_quota_recalc_timeout(root); + dict_wait(root->dict); + } +} + +struct quota_backend quota_backend_dict = { + .name = "dict", + + .v = { + .alloc = dict_quota_alloc, + .init = dict_quota_init, + .deinit = dict_quota_deinit, + .get_resources = dict_quota_root_get_resources, + .get_resource = dict_quota_get_resource, + .update = dict_quota_update, + .flush = dict_quota_flush, + } +}; |