lazy-expunge-plugin.c revision fed564495b7d7c4dfbb025f52ff3df0e640a5959
/* Copyright (C) 2006 Timo Sirainen */
#include "common.h"
#include "ioloop.h"
#include "array.h"
#include "str.h"
#include "seq-range-array.h"
#include "maildir-storage.h"
#include "client.h"
#include "namespace.h"
#include "lazy-expunge-plugin.h"
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <dirent.h>
#include <time.h>
#define LAZY_EXPUNGE_CONTEXT(obj) \
enum lazy_namespace {
};
struct lazy_expunge_mail_storage {
struct mail_storage_vfuncs super;
bool internal_namespace;
};
struct lazy_expunge_mailbox {
struct mailbox_vfuncs super;
};
struct lazy_expunge_transaction {
struct mailbox *expunge_box;
};
struct lazy_expunge_mail {
struct mail_vfuncs super;
};
static void (*lazy_expunge_next_hook_mail_storage_created)
(struct mail_storage *storage);
static unsigned int lazy_expunge_storage_module_id = 0;
static bool lazy_expunge_storage_module_id_set = FALSE;
static struct mailbox *
{
return box;
return NULL;
/* probably the mailbox just doesn't exist. try creating it. */
return NULL;
/* and try opening again */
return box;
}
{
struct lazy_expunge_transaction *lt =
struct mail_storage *deststorage;
"lazy_expunge: Couldn't open expunge mailbox");
return -1;
}
}
return 0;
}
static struct mailbox_transaction_context *
{
struct mailbox_transaction_context *t;
struct lazy_expunge_transaction *lt;
return t;
}
struct lazy_expunge_move_context {
unsigned int dir_len;
};
{
const char *p;
return 1;
}
struct lazy_expunge_transaction *lt)
{
struct mailbox_transaction_context *trans;
struct index_transaction_context *itrans;
struct lazy_expunge_move_context ctx;
unsigned int i, count;
const char *dir;
bool is_file;
int ret = 0;
ret = -1;
break;
}
&ctx) < 0) {
ret = -1;
break;
}
}
}
if (mailbox_transaction_commit(&trans, 0) < 0)
ret = -1;
return ret;
}
{
}
static int
enum mailbox_sync_flags flags)
{
int ret;
return ret;
}
static void
{
}
static struct mail *
struct mailbox_header_lookup_ctx *wanted_headers)
{
struct lazy_expunge_mail *lmail;
struct mail_private *mail;
return _mail;
}
static struct mailbox *
{
struct lazy_expunge_mail_storage *lstorage =
struct lazy_expunge_mailbox *qbox;
return box;
return box;
}
{
unsigned int src_dirlen, dest_dirlen;
int ret = 0;
return 0;
if (!EDESTDIREXISTS(errno)) {
}
/* rename all the files separately */
"opendir(%s) failed: %m", srcdir);
return -1;
}
"rename(%s, %s) failed: %m",
ret = -1;
}
}
"closedir(%s) failed: %m", srcdir);
ret = -1;
}
if (ret == 0) {
"rmdir(%s) failed: %m", srcdir);
ret = -1;
}
}
return ret;
}
static int
{
const char *dest_name = *_dest_name;
bool is_file;
&is_file);
return 0;
if (!EDESTDIREXISTS(errno)) {
return -1;
}
/* mailbox is being deleted multiple times per second.
update the filename. */
&is_file);
}
t_push();
}
}
t_pop();
*_dest_name = dest_name;
return 1;
}
static int
{
struct lazy_expunge_mail_storage *lstorage =
struct mail_storage *dest_storage;
enum mailbox_name_status status;
const char *destname;
char timestamp[256];
int ret;
if (lstorage->internal_namespace)
/* first do the normal sanity checks */
return -1;
}
&status) < 0)
return -1;
if (status == MAILBOX_NAME_INVALID) {
return -1;
}
/* destination mailbox name needs to contain a timestamp */
/* first move the actual mailbox */
return -1;
if (ret == 0) {
return -1;
}
/* next move the expunged messages mailbox, if it exists */
return 0;
}
{
struct lazy_expunge_mail_storage *lstorage;
/* only maildir supported for now */
return;
}
}
{
struct lazy_expunge_mail_storage *lstorage;
const char *const *p;
int i;
/* FIXME: this works only as long as there's only one client. */
t_push();
for (i = 0; i < LAZY_NAMESPACE_COUNT; i++, p++) {
const char *name = *p;
lazy_namespaces[i] =
if (lazy_namespaces[i] == NULL)
i_fatal("lazy_expunge: Namespace must be in maildir "
"format: %s", name);
}
operations. */
}
t_pop();
}
void lazy_expunge_plugin_init(void)
{
return;
}
void lazy_expunge_plugin_deinit(void)
{
}
}