imapc-client.c revision b09f510e3f7e3bd4a5216a60fa1406d1e83d001b
/* Copyright (c) 2011-2012 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "array.h"
#include "str.h"
#include "ioloop.h"
#include "safe-mkstemp.h"
#include "iostream-ssl.h"
#include "imapc-msgmap.h"
#include "imapc-connection.h"
#include "imapc-client-private.h"
#include <unistd.h>
const struct imapc_capability_name imapc_capability_names[] = {
{ "SASL-IR", IMAPC_CAPABILITY_SASL_IR },
{ "LITERAL+", IMAPC_CAPABILITY_LITERALPLUS },
{ "QRESYNC", IMAPC_CAPABILITY_QRESYNC },
{ "IDLE", IMAPC_CAPABILITY_IDLE },
{ "UIDPLUS", IMAPC_CAPABILITY_UIDPLUS },
{ "AUTH=PLAIN", IMAPC_CAPABILITY_AUTH_PLAIN },
{ "STARTTLS", IMAPC_CAPABILITY_STARTTLS },
{ "X-GM-EXT-1", IMAPC_CAPABILITY_X_GM_EXT_1 },
{ "CONDSTORE", IMAPC_CAPABILITY_CONDSTORE },
{ "IMAP4REV1", IMAPC_CAPABILITY_IMAP4REV1 },
{ NULL, 0 }
};
static void
default_untagged_callback(const struct imapc_untagged_reply *reply ATTR_UNUSED,
void *context ATTR_UNUSED)
{
}
struct imapc_client *
imapc_client_init(const struct imapc_client_settings *set)
{
struct imapc_client *client;
struct ssl_iostream_settings ssl_set;
const char *source;
pool_t pool;
pool = pool_alloconly_create("imapc client", 1024);
client = p_new(pool, struct imapc_client, 1);
client->pool = pool;
client->refcount = 1;
client->set.debug = set->debug;
client->set.host = p_strdup(pool, set->host);
client->set.port = set->port;
client->set.master_user = p_strdup_empty(pool, set->master_user);
client->set.username = p_strdup(pool, set->username);
client->set.password = p_strdup(pool, set->password);
client->set.dns_client_socket_path =
p_strdup(pool, set->dns_client_socket_path);
client->set.temp_path_prefix =
p_strdup(pool, set->temp_path_prefix);
client->set.rawlog_dir = p_strdup(pool, set->rawlog_dir);
if (set->ssl_mode != IMAPC_CLIENT_SSL_MODE_NONE) {
client->set.ssl_mode = set->ssl_mode;
client->set.ssl_ca_dir = p_strdup(pool, set->ssl_ca_dir);
client->set.ssl_verify = set->ssl_verify;
memset(&ssl_set, 0, sizeof(ssl_set));
ssl_set.ca_dir = set->ssl_ca_dir;
ssl_set.verify_remote_cert = set->ssl_verify;
ssl_set.crypto_device = set->ssl_crypto_device;
source = t_strdup_printf("%s:%u", set->host, set->port);
if (ssl_iostream_context_init_client(source, &ssl_set,
&client->ssl_ctx) < 0) {
i_error("imapc(%s): Couldn't initialize SSL context",
source);
}
}
client->untagged_callback = default_untagged_callback;
p_array_init(&client->conns, pool, 8);
return client;
}
void imapc_client_ref(struct imapc_client *client)
{
i_assert(client->refcount > 0);
client->refcount++;
}
void imapc_client_unref(struct imapc_client **_client)
{
struct imapc_client *client = *_client;
i_assert(client->refcount > 0);
if (--client->refcount > 0)
return;
if (client->ssl_ctx != NULL)
ssl_iostream_context_deinit(&client->ssl_ctx);
pool_unref(&client->pool);
}
void imapc_client_deinit(struct imapc_client **_client)
{
struct imapc_client *client = *_client;
struct imapc_client_connection **connp;
array_foreach_modifiable(&client->conns, connp) {
i_assert(imapc_connection_get_mailbox((*connp)->conn) == NULL);
imapc_connection_deinit(&(*connp)->conn);
i_free(*connp);
}
array_clear(&client->conns);
imapc_client_unref(_client);
}
void imapc_client_register_untagged(struct imapc_client *client,
imapc_untagged_callback_t *callback,
void *context)
{
client->untagged_callback = callback;
client->untagged_context = context;
}
static void imapc_client_run_pre(struct imapc_client *client)
{
struct imapc_client_connection *const *connp;
struct ioloop *prev_ioloop = current_ioloop;
i_assert(client->ioloop == NULL);
client->ioloop = io_loop_create();
io_loop_set_running(client->ioloop);
array_foreach(&client->conns, connp) {
imapc_connection_ioloop_changed((*connp)->conn);
imapc_connection_connect((*connp)->conn, NULL, NULL);
}
if (io_loop_is_running(client->ioloop))
io_loop_run(client->ioloop);
current_ioloop = prev_ioloop;
}
static void imapc_client_run_post(struct imapc_client *client)
{
struct imapc_client_connection *const *connp;
struct ioloop *ioloop = client->ioloop;
client->ioloop = NULL;
array_foreach(&client->conns, connp)
imapc_connection_ioloop_changed((*connp)->conn);
current_ioloop = ioloop;
io_loop_destroy(&ioloop);
}
void imapc_client_run(struct imapc_client *client)
{
imapc_client_run_pre(client);
imapc_client_run_post(client);
}
void imapc_client_stop(struct imapc_client *client)
{
if (client->ioloop != NULL)
io_loop_stop(client->ioloop);
}
bool imapc_client_is_running(struct imapc_client *client)
{
return client->ioloop != NULL;
}
static struct imapc_client_connection *
imapc_client_add_connection(struct imapc_client *client)
{
struct imapc_client_connection *conn;
conn = i_new(struct imapc_client_connection, 1);
conn->conn = imapc_connection_init(client);
array_append(&client->conns, &conn, 1);
return conn;
}
static struct imapc_connection *
imapc_client_find_connection(struct imapc_client *client)
{
struct imapc_client_connection *const *connp;
/* FIXME: stupid algorithm */
if (array_count(&client->conns) == 0)
return imapc_client_add_connection(client)->conn;
connp = array_idx(&client->conns, 0);
return (*connp)->conn;
}
struct imapc_command *
imapc_client_cmd(struct imapc_client *client,
imapc_command_callback_t *callback, void *context)
{
struct imapc_connection *conn;
conn = imapc_client_find_connection(client);
return imapc_connection_cmd(conn, callback, context);
}
static struct imapc_client_connection *
imapc_client_get_unboxed_connection(struct imapc_client *client)
{
struct imapc_client_connection *const *conns;
unsigned int i, count;
conns = array_get(&client->conns, &count);
for (i = 0; i < count; i++) {
if (conns[i]->box == NULL)
return conns[i];
}
return imapc_client_add_connection(client);
}
void imapc_client_login(struct imapc_client *client,
imapc_command_callback_t *callback, void *context)
{
struct imapc_client_connection *conn;
i_assert(array_count(&client->conns) == 0);
conn = imapc_client_add_connection(client);
imapc_connection_connect(conn->conn, callback, context);
}
struct imapc_client_mailbox *
imapc_client_mailbox_open(struct imapc_client *client,
void *untagged_box_context)
{
struct imapc_client_mailbox *box;
struct imapc_client_connection *conn;
box = i_new(struct imapc_client_mailbox, 1);
box->client = client;
box->untagged_box_context = untagged_box_context;
conn = imapc_client_get_unboxed_connection(client);
conn->box = box;
box->conn = conn->conn;
box->msgmap = imapc_msgmap_init();
return box;
}
void imapc_client_mailbox_set_reopen_cb(struct imapc_client_mailbox *box,
void (*callback)(void *context),
void *context)
{
box->reopen_callback = callback;
box->reopen_context = context;
}
static void
imapc_client_reconnect_cb(const struct imapc_command_reply *reply,
void *context)
{
struct imapc_client_mailbox *box = context;
i_assert(box->reconnecting);
box->reconnecting = FALSE;
if (reply->state == IMAPC_COMMAND_STATE_OK) {
/* reopen the mailbox */
box->reopen_callback(box->reopen_context);
} else {
imapc_connection_abort_commands(box->conn, NULL, FALSE);
}
}
void imapc_client_mailbox_reconnect(struct imapc_client_mailbox *box)
{
bool reconnect = box->reopen_callback != NULL && box->reconnect_ok;
if (reconnect) {
i_assert(!box->reconnecting);
box->reconnecting = TRUE;
}
imapc_connection_disconnect(box->conn);
if (reconnect) {
imapc_connection_connect(box->conn,
imapc_client_reconnect_cb, box);
}
box->reconnect_ok = FALSE;
}
void imapc_client_mailbox_close(struct imapc_client_mailbox **_box)
{
struct imapc_client_mailbox *box = *_box;
struct imapc_client_connection *const *connp;
box->closing = TRUE;
/* cancel any pending commands */
imapc_connection_unselect(box);
if (box->reconnecting) {
/* need to abort the reconnection so it won't try to access
the box */
imapc_connection_disconnect(box->conn);
}
/* set this only after unselect, which may cancel some commands that
reference this box */
*_box = NULL;
array_foreach(&box->client->conns, connp) {
if ((*connp)->box == box) {
(*connp)->box = NULL;
break;
}
}
imapc_msgmap_deinit(&box->msgmap);
i_free(box);
}
struct imapc_command *
imapc_client_mailbox_cmd(struct imapc_client_mailbox *box,
imapc_command_callback_t *callback, void *context)
{
struct imapc_command *cmd;
i_assert(!box->closing);
cmd = imapc_connection_cmd(box->conn, callback, context);
imapc_command_set_mailbox(cmd, box);
return cmd;
}
struct imapc_msgmap *
imapc_client_mailbox_get_msgmap(struct imapc_client_mailbox *box)
{
return box->msgmap;
}
void imapc_client_mailbox_idle(struct imapc_client_mailbox *box)
{
if (imapc_client_mailbox_is_opened(box))
imapc_connection_idle(box->conn);
box->reconnect_ok = TRUE;
}
bool imapc_client_mailbox_is_opened(struct imapc_client_mailbox *box)
{
struct imapc_client_mailbox *selected_box;
if (box->closing ||
imapc_connection_get_state(box->conn) != IMAPC_CONNECTION_STATE_DONE)
return FALSE;
selected_box = imapc_connection_get_mailbox(box->conn);
if (selected_box != box) {
if (selected_box != NULL)
i_error("imapc: Selected mailbox changed unexpectedly");
return FALSE;
}
return TRUE;
}
enum imapc_capability
imapc_client_get_capabilities(struct imapc_client *client)
{
struct imapc_client_connection *const *connp;
struct imapc_connection *conn = NULL;
/* try to find a connection that is already logged in */
array_foreach(&client->conns, connp) {
conn = (*connp)->conn;
if (imapc_connection_get_state(conn) == IMAPC_CONNECTION_STATE_DONE)
return imapc_connection_get_capabilities(conn);
}
/* fallback to whatever exists (there always exists one) */
return imapc_connection_get_capabilities(conn);
}
int imapc_client_create_temp_fd(struct imapc_client *client,
const char **path_r)
{
string_t *path;
int fd;
if (client->set.temp_path_prefix == NULL) {
i_error("imapc: temp_path_prefix not set, "
"can't create temp file");
return -1;
}
path = t_str_new(128);
str_append(path, client->set.temp_path_prefix);
fd = safe_mkstemp(path, 0600, (uid_t)-1, (gid_t)-1);
if (fd == -1) {
i_error("safe_mkstemp(%s) failed: %m", str_c(path));
return -1;
}
/* we just want the fd, unlink it */
if (unlink(str_c(path)) < 0) {
/* shouldn't happen.. */
i_error("unlink(%s) failed: %m", str_c(path));
(void)close(fd);
return -1;
}
*path_r = str_c(path);
return fd;
}