Commit b5c4df4d b5c4df4d6e798c83c1ac379cf95aa5f03a9d93b4 by Sergey Poznyakoff

Virtual-domain-specific auth functions.

1 parent 7db00198
/* GNU mailutils - a suite of utilities for electronic mail
Copyright (C) 2002 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <unistd.h>
#include <sys/types.h>
#include <pwd.h>
#ifdef HAVE_SHADOW_H
# include <shadow.h>
#endif
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
#ifdef HAVE_SECURITY_PAM_APPL_H
# include <security/pam_appl.h>
#endif
#ifdef HAVE_CRYPT_H
# include <crypt.h>
#endif
#include <mailutils/list.h>
#include <mailutils/iterator.h>
#include <mailutils/mailbox.h>
#include <mailutils/argp.h>
#include <mailutils/mu_auth.h>
#ifdef USE_VIRTUAL_DOMAINS
struct passwd *
getpwnam_virtual (char *u)
{
struct passwd *pw = NULL;
FILE *pfile;
size_t i = 0, len = strlen (u), delim = 0;
char *filename;
mu_virtual_domain = 0;
for (i = 0; i < len && delim == 0; i++)
if (u[i] == '!' || u[i] == ':' || u[i] == '@')
delim = i;
if (delim == 0)
return NULL;
filename = malloc (strlen (SITE_VIRTUAL_PWDDIR) +
strlen (&u[delim + 1]) + 2 /* slash and null byte */);
if (filename == NULL)
return NULL;
sprintf (filename, "%s/%s", SITE_VIRTUAL_PWDDIR, &u[delim + 1]);
pfile = fopen (filename, "r");
free (filename);
if (pfile)
while ((pw = fgetpwent (pfile)) != NULL)
{
if (strlen (pw->pw_name) == delim && !strncmp (u, pw->pw_name, delim))
{
mu_virtual_domain = 1;
break;
}
}
return pw;
}
struct passwd *
getpwnam_ip_virtual (const char *u)
{
struct sockaddr_in addr;
struct passwd *pw = NULL;
int len = sizeof (addr);
char *user = NULL;
if (getsockname (fileno (ifile), (struct sockaddr *)&addr, &len) == 0)
{
char *ip;
char *user;
struct hostent *info = gethostbyaddr ((char *)&addr.sin_addr,
4, AF_INET);
if (info)
{
user = malloc (strlen (info->h_name) + strlen (u) + 2);
if (user)
{
sprintf (user, "%s!%s", u, info->h_name);
pw = getpwnam_virtual (user);
free (user);
}
}
if (!pw)
{
ip = inet_ntoa (addr.sin_addr);
user = malloc (strlen (ip) + strlen (u) + 2);
if (user)
{
sprintf (user, "%s!%s", u, ip);
pw = getpwnam_virtual (user);
free (user);
}
}
}
return pw;
}
/* Virtual domains */
int
mu_auth_virt_domain_by_name (void *return_data, void *key,
void *unused_func_data, void *unused_call_data)
{
int rc;
struct passwd *pw;
char *mailbox_name;
if (!key)
{
errno = EINVAL;
return 1;
}
pw = getpwnam_virtual (key);
if (!pw)
{
pw = getpwnam_ip_virtual (key);
if (!pw)
return 1;
}
mailbox_name = calloc (strlen (pw->pw_dir) + strlen ("/INBOX"), 1);
sprintf (mailbox_name, "%s/INBOX", pw->pw_dir);
rc = mu_auth_data_alloc ((struct mu_auth_data **) return_data,
pw->pw_name,
pw->pw_passwd,
pw->pw_uid,
pw->pw_gid,
pw->pw_gecos,
pw->pw_dir,
pw->pw_shell,
mailbox_name,
0);
free (mailbox_name);
return rc;
}
#else
int
mu_auth_virt_domain_by_name (void *return_data, void *key,
void *unused_func_data, void *unused_call_data)
{
errno = ENOSYS;
return 1;
}
#endif
struct mu_auth_module mu_auth_virtual_module = {
"virtdomain",
NULL,
mu_auth_nosupport,
NULL,
mu_auth_virt_domain_by_name,
NULL,
mu_auth_nosupport,
NULL
};