doveadm-connection.c revision 11ee3b40320a31669bd717fecbe1e332dad4fd84
/* Copyright (c) 2010-2016 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "ioloop.h"
#include "net.h"
#include "istream.h"
#include "ostream.h"
#include "array.h"
#include "str.h"
#include "strescape.h"
#include "llist.h"
#include "master-service.h"
#include "user-directory.h"
#include "mail-host.h"
#include "director.h"
#include "director-host.h"
#include "director-request.h"
#include "director-connection.h"
#include "doveadm-connection.h"
#include <unistd.h>
#define DOVEADM_PROTOCOL_VERSION_MAJOR 1
#define DOVEADM_HANDSHAKE "VERSION\tdirector-doveadm\t1\t0\n"
#define MAX_VALID_VHOST_COUNT 1000
#define DEFAULT_MAX_MOVING_USERS 100
struct director_reset_cmd {
struct doveadm_connection *_conn;
struct user_directory_iter *iter;
unsigned int host_idx, hosts_count;
unsigned int max_moving_users;
};
struct doveadm_connection {
int fd;
struct director_reset_cmd *reset_cmd;
bool handshaked:1;
};
static struct doveadm_connection *doveadm_connections;
{
(*hostp)->user_count);
(long)(*hostp)->last_updown_change);
}
}
{
struct mail_host_list *orig_hosts_list;
unsigned int i, j, orig_hosts_count, cur_hosts_count;
int ret;
/* the hosts are sorted by IP */
for (i = j = 0; i < orig_hosts_count && j < cur_hosts_count; ) {
if (ret == 0)
i++, j++;
else if (ret > 0)
j++;
else {
i++;
}
}
for (; i < orig_hosts_count; i++)
}
{
if (!dir->ring_handshaked)
else if (dir->ring_synced)
else {
}
}
static void
{
else if (director_connection_is_synced(conn))
else
}
static void
const struct director_host *host,
{
else {
/* we might have a connection that is being connected */
struct director_connection *const *connp;
break;
}
}
}
}
{
struct director_host *const *hostp;
const char *type;
type = "removed";
type = "self";
else if (left)
else if (right)
type = "right";
else
type = "";
(unsigned long)last_failed);
else
}
}
static int
const char *const *args)
{
struct director_host *host;
i_error("doveadm sent invalid DIRECTOR-ADD parameters");
return -1;
}
}
return 1;
}
static int
const char *const *args)
{
struct director_host *host;
i_error("doveadm sent invalid DIRECTOR-REMOVE parameters");
return -1;
}
else {
}
return 1;
}
static int
{
unsigned int vhost_count = UINT_MAX;
tag = "";
else
}
i_error("doveadm sent invalid %s parameters",
return -1;
}
return 1;
}
if (update) {
return 1;
}
return 1;
"host is already being updated - try again later\n");
return 1;
}
if (vhost_count != UINT_MAX)
/* NOTE: we don't support changing a tag for an existing host.
it needs to be removed first. otherwise it would be a bit ugly to
handle. */
return 1;
}
static int
{
}
static int
{
}
static int
const char *const *args)
{
i_error("doveadm sent invalid %s parameters: %s",
return -1;
}
return 1;
}
;
"host is already being updated - try again later\n");
return 1;
} else {
}
return 1;
}
static int
const char *const *args)
{
i_error("doveadm sent invalid HOST-REMOVE parameters");
return -1;
}
else {
}
return 1;
}
static void
{
unsigned int total_user_count = 0;
}
i_warning("Flushed all backend hosts with %u users. This is an unsafe "
"operation and may cause the same users to end up in multiple backends.",
}
static int
{
return 1;
}
i_error("doveadm sent invalid HOST-FLUSH parameters");
return -1;
}
else {
}
return 1;
}
{
}
static bool
{
return FALSE;
continue;
} T_END;
break;
}
}
static bool
{
unsigned int count;
return FALSE;
}
}
return TRUE;
}
static int
const char *const *args)
{
struct director_reset_cmd *cmd;
unsigned int i = 0, count;
unsigned int max_moving_users = DEFAULT_MAX_MOVING_USERS;
i_error("doveadm sent invalid HOST-RESET-USERS parameters");
return -1;
}
i_error("doveadm sent invalid HOST-RESET-USERS ip: %s",
args[0]);
return -1;
}
for (i = 0; i < count; i++) {
break;
}
if (i == count) {
return 1;
}
count = i+1;
}
if (!director_reset_cmd_run(cmd)) {
/* we still have work to do. don't handle any more doveadm
input until we're finished. */
return 0;
}
return 1;
}
static int
const char *const *args)
{
unsigned int username_hash;
username = "";
tag = "";
} else {
}
/* get user's current host */
else {
}
/* get host if it wasn't in user directory */
else
/* get host with default configuration */
username_hash, tag);
else
return 1;
}
static int
{
struct user_directory_iter *iter;
i_error("doveadm sent invalid USER-LIST parameters");
return -1;
}
} else {
}
"%u\t%u\t%s\n",
} T_END;
}
return 1;
}
static int
{
unsigned int username_hash;
i_error("doveadm sent invalid USER-MOVE parameters");
return -1;
}
return 1;
}
return 1;
}
return 1;
}
static int
{
i_error("doveadm sent invalid USER-KICK parameters");
return -1;
}
return 1;
}
static int
{
int ret = 1;
i_error("doveadm sent empty command line");
return -1;
}
args++;
else {
ret = -1;
}
return ret;
}
{
const char *line;
int ret = 1;
if (!conn->handshaked) {
return;
}
i_error("doveadm not compatible with this server "
"(mixed old and new binaries?)");
return;
}
}
T_BEGIN {
} T_END;
}
}
{
}
struct doveadm_connection *
{
struct doveadm_connection *conn;
return conn;
}
{
/* finish the move even if doveadm disconnected */
}
i_error("close(doveadm connection) failed: %m");
}
void doveadm_connections_deinit(void)
{
while (reset_cmds != NULL)
while (doveadm_connections != NULL) {
}
}
void doveadm_connections_continue_reset_cmds(void)
{
while (reset_cmds != NULL) {
if (!director_reset_cmd_run(reset_cmds))
break;
}
}