userdb-vpopmail.c revision 6fabfb7bbfd88d0c1de66981e52850f26067623b
/* Copyright (C) 2002-2003 Timo Sirainen */
/* Thanks to Courier-IMAP for showing how the vpopmail API should be used */
#include "common.h"
#if defined(PASSDB_VPOPMAIL) || defined(USERDB_VPOPMAIL)
#include "str.h"
#include "var-expand.h"
#include "userdb.h"
#include "userdb-vpopmail.h"
struct vpopmail_userdb_module {
struct userdb_module module;
const char *quota_template_key;
const char *quota_template_value;
};
struct vqpasswd *vpopmail_lookup_vqp(struct auth_request *request,
char vpop_user[VPOPMAIL_LIMIT],
char vpop_domain[VPOPMAIL_LIMIT])
{
struct vqpasswd *vpw;
/* vpop_user must be zero-filled or parse_email() leaves an
extra character after the user name. we'll fill vpop_domain
as well just to be sure... */
memset(vpop_user, '\0', VPOPMAIL_LIMIT);
memset(vpop_domain, '\0', VPOPMAIL_LIMIT);
if (parse_email(request->user, vpop_user, vpop_domain,
VPOPMAIL_LIMIT-1) < 0) {
auth_request_log_info(request, "vpopmail",
"parse_email() failed");
return NULL;
}
auth_request_log_debug(request, "vpopmail",
"lookup user=%s domain=%s",
vpop_user, vpop_domain);
vpw = vauth_getpw(vpop_user, vpop_domain);
if (vpw == NULL) {
auth_request_log_info(request, "vpopmail", "unknown user");
return NULL;
}
return vpw;
}
#ifdef USERDB_VPOPMAIL
static const char *
userdb_vpopmail_get_quota(const char *template, const char *vpop_str)
{
const struct var_expand_table *tab;
string_t *quota;
if (template == NULL || *vpop_str == '\0' ||
strcmp(vpop_str, "NOQUOTA") == 0)
return "";
tab = var_expand_table_build('q', format_maildirquota(vpop_str), '\0');
quota = t_str_new(128);
var_expand(quota, template, tab);
return str_c(quota);
}
static void vpopmail_lookup(struct auth_request *auth_request,
userdb_callback_t *callback)
{
struct userdb_module *_module = auth_request->userdb->userdb;
struct vpopmail_userdb_module *module =
(struct vpopmail_userdb_module *)_module;
char vpop_user[VPOPMAIL_LIMIT], vpop_domain[VPOPMAIL_LIMIT];
struct vqpasswd *vpw;
const char *quota;
uid_t uid;
gid_t gid;
vpw = vpopmail_lookup_vqp(auth_request, vpop_user, vpop_domain);
if (vpw == NULL) {
callback(USERDB_RESULT_USER_UNKNOWN, auth_request);
return;
}
/* we have to get uid/gid separately, because the gid field in
struct vqpasswd isn't really gid at all but just some flags... */
if (vget_assign(vpop_domain, NULL, 0, &uid, &gid) == NULL) {
auth_request_log_info(auth_request, "vpopmail",
"vget_assign(%s) failed", vpop_domain);
callback(USERDB_RESULT_INTERNAL_FAILURE, auth_request);
return;
}
if (auth_request->successful) {
/* update the last login only when we're really */
vset_lastauth(vpop_user, vpop_domain,
t_strdup_noconst(auth_request->service));
}
if (vpw->pw_dir == NULL || vpw->pw_dir[0] == '\0') {
/* user's homedir doesn't exist yet, create it */
auth_request_log_info(auth_request, "vpopmail",
"pw_dir isn't set, creating");
if (make_user_dir(vpop_user, vpop_domain, uid, gid) == NULL) {
auth_request_log_error(auth_request, "vpopmail",
"make_user_dir(%s, %s) failed",
vpop_user, vpop_domain);
callback(USERDB_RESULT_INTERNAL_FAILURE, auth_request);
return;
}
/* get the user again so pw_dir is visible */
vpw = vauth_getpw(vpop_user, vpop_domain);
if (vpw == NULL) {
callback(USERDB_RESULT_INTERNAL_FAILURE, auth_request);
return;
}
}
auth_request_init_userdb_reply(auth_request);
auth_request_set_userdb_field(auth_request, NULL, auth_request->user);
auth_request_set_userdb_field(auth_request, "uid", dec2str(uid));
auth_request_set_userdb_field(auth_request, "gid", dec2str(gid));
auth_request_set_userdb_field(auth_request, "home", vpw->pw_dir);
quota = userdb_vpopmail_get_quota(module->quota_template_value,
vpw->pw_shell);
if (*quota != '\0') {
auth_request_set_userdb_field(auth_request,
module->quota_template_key,
quota);
}
callback(USERDB_RESULT_OK, auth_request);
}
static struct userdb_module *
vpopmail_preinit(struct auth_userdb *auth_userdb, const char *args)
{
struct vpopmail_userdb_module *module;
const char *const *tmp, *p;
pool_t pool = auth_userdb->auth->pool;
module = p_new(pool, struct vpopmail_userdb_module, 1);
for (tmp = t_strsplit(args, " "); *tmp != NULL; tmp++) {
if (strncmp(*tmp, "cache_key=", 10) == 0)
module->module.cache_key = p_strdup(pool, *tmp + 10);
else if (strncmp(*tmp, "quota_template=", 15) == 0) {
p = strchr(*tmp + 15, '=');
if (p == NULL) {
i_fatal("vpopmail userdb: "
"quota_template missing '='");
}
module->quota_template_key =
p_strdup_until(pool, *tmp + 15, p);
module->quota_template_value = p_strdup(pool, p + 1);
} else
i_fatal("vpopmail userdb: Unknown args: %s", *tmp);
}
return &module->module;
}
struct userdb_module_interface userdb_vpopmail = {
"vpopmail",
vpopmail_preinit,
NULL,
NULL,
vpopmail_lookup
};
#endif
#endif