trash-plugin.c revision 32b9cd0a982cd598f91d826ca9536f0897bad3ce
/* Copyright (c) 2005-2018 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "array.h"
#include "unichar.h"
#include "istream.h"
#include "mail-namespace.h"
#include "mail-search-build.h"
#include "quota-private.h"
#include "quota-plugin.h"
#include "trash-plugin.h"
#include <unistd.h>
#include <fcntl.h>
#define INIT_TRASH_MAILBOX_COUNT 4
#define MAX_RETRY_COUNT 3
#define TRASH_USER_CONTEXT(obj) \
#define TRASH_USER_CONTEXT_REQUIRE(obj) \
struct trash_mailbox {
const char *name;
int priority; /* lower number = higher priority */
struct mail_namespace *ns;
/* temporarily set while cleaning: */
struct mailbox_transaction_context *trans;
struct mail_search_context *search_ctx;
};
struct trash_user {
const char *config_file;
/* ordered by priority, highest first */
};
const char *trash_plugin_version = DOVECOT_ABI_VERSION;
static enum quota_alloc_result (*trash_next_quota_test_alloc)(
struct quota_transaction_context *, uoff_t,
const char **error_r);
{
struct mail_search_args *search_args;
return 0;
}
return -1;
}
{
int ret;
else {
}
if (ret <= 0) {
*received_time_r = 0;
return ret;
}
}
return -1;
return 1;
}
unsigned int count_needed)
{
struct trash_mailbox *trashes;
unsigned int i, j, count, oldest_idx;
unsigned int expunged_count = 0;
int ret = 0;
for (i = 0; i < count; ) {
/* expunge oldest mails first in all trash boxes with
same priority */
oldest_idx = count;
for (j = i; j < count; j++) {
break;
&received);
if (ret < 0)
goto err;
if (ret > 0) {
oldest_idx = j;
}
}
}
if (oldest_idx < count) {
&size) < 0) {
/* maybe expunged already? */
continue;
}
size_expunged += size;
if (size_expunged >= size_needed &&
break;
} else {
/* find more mails from next priority's mailbox */
i = j;
}
}
err:
for (i = 0; i < count; i++) {
continue;
if (size_expunged >= size_needed &&
expunged_count >= count_needed) {
} else {
/* couldn't get enough space, don't expunge anything */
}
}
if (size_expunged < size_needed) {
i_debug("trash plugin: Failed to remove enough messages "
}
return 0;
}
if (expunged_count < count_needed) {
i_debug("trash plugin: Failed to remove enough messages "
"(needed %u messages, expunged only %u messages)",
}
return 0;
}
if (ctx->bytes_over > 0) {
/* user is over quota. drop the over-bytes first. */
ctx->bytes_over = 0;
}
if (ctx->count_over > 0) {
/* user is over quota. drop the over-count first. */
ctx->count_over = 0;
}
} else {
}
} else {
}
return 1;
}
static enum quota_alloc_result
{
int i;
uint64_t size_needed = 0;
unsigned int count_needed = 0;
for (i = 0; ; i++) {
enum quota_alloc_result ret;
if (ret != QUOTA_ALLOC_RESULT_OVER_QUOTA) {
if (ret == QUOTA_ALLOC_RESULT_OVER_QUOTA_LIMIT &&
i_debug("trash plugin: Mail is larger than "
"quota, won't even try to handle");
return ret;
}
if (i == MAX_RETRY_COUNT) {
/* trash_try_clean_mails() should have returned 0 if
it couldn't get enough space, but allow retrying
it a couple of times if there was some extra space
that was needed.. */
break;
}
/* not enough space. try deleting some from mailbox. */
return QUOTA_ALLOC_RESULT_OVER_QUOTA;
}
}
return QUOTA_ALLOC_RESULT_OVER_QUOTA;
}
struct trash_mailbox *trash)
{
struct mail_namespace *ns;
return FALSE;
return TRUE;
}
const struct trash_mailbox *t2)
{
}
{
struct trash_mailbox *trash;
if (fd == -1) {
return -1;
}
/* <priority> <mailbox name> */
continue;
i_error("trash: Invalid priority for mailbox '%s'",
ret = -1;
}
i_error("trash: Mailbox name not UTF-8: %s",
ret = -1;
}
i_error("trash: Namespace not found for mailbox '%s'",
ret = -1;
}
if (user->mail_debug) {
i_debug("trash plugin: Added '%s' with priority %d",
}
}
i_close_fd(&fd);
return ret;
}
static void
{
struct trash_user *tuser;
const char *env;
if (user->mail_debug)
i_debug("trash: No trash setting - plugin disabled");
i_error("trash plugin: quota plugin not initialized");
} else {
}
}
static void
{
}
}
static struct mail_storage_hooks trash_mail_storage_hooks = {
};
{
}
void trash_plugin_deinit(void)
{
}