2008-04-27 00:40:09 +00:00
|
|
|
/*
|
2021-12-05 09:35:27 -06:00
|
|
|
* SPDX-FileCopyrightText: 1990 - 1994, Julianne Frances Haugh
|
|
|
|
* SPDX-FileCopyrightText: 1996 - 2000, Marek Michałkiewicz
|
|
|
|
* SPDX-FileCopyrightText: 2002 - 2005, Tomasz Kłoczko
|
|
|
|
* SPDX-FileCopyrightText: 2007 - 2008, Nicolas François
|
2008-04-27 00:40:09 +00:00
|
|
|
*
|
2021-12-05 09:35:27 -06:00
|
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
2008-04-27 00:40:09 +00:00
|
|
|
*/
|
|
|
|
|
2007-10-07 11:44:02 +00:00
|
|
|
#include <config.h>
|
2007-10-07 11:47:01 +00:00
|
|
|
#include <errno.h>
|
2007-10-07 11:44:02 +00:00
|
|
|
#include <grp.h>
|
2007-10-07 11:47:01 +00:00
|
|
|
#include <pwd.h>
|
|
|
|
#include <stdio.h>
|
2007-10-07 11:44:02 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include "defines.h"
|
2007-10-07 11:47:01 +00:00
|
|
|
#include "prototypes.h"
|
2007-10-07 11:44:02 +00:00
|
|
|
|
|
|
|
#ifndef SUAUTHFILE
|
|
|
|
#define SUAUTHFILE "/etc/suauth"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define NOACTION 0
|
|
|
|
#define NOPWORD 1
|
|
|
|
#define DENY -1
|
|
|
|
#define OWNPWORD 2
|
|
|
|
|
2007-10-07 11:45:14 +00:00
|
|
|
#ifdef SU_ACCESS
|
|
|
|
|
2021-09-13 17:23:17 +01:00
|
|
|
/* Really, I could do with a few const char's here defining all the
|
2007-10-07 11:44:02 +00:00
|
|
|
* strings output to the user or the syslog. -- chris
|
|
|
|
*/
|
2007-10-07 11:45:23 +00:00
|
|
|
static int applies (const char *, char *);
|
2007-10-07 11:44:02 +00:00
|
|
|
|
2008-01-06 13:30:18 +00:00
|
|
|
static int isgrp (const char *, const char *);
|
2007-10-07 11:44:02 +00:00
|
|
|
|
|
|
|
static int lines = 0;
|
|
|
|
|
|
|
|
|
2011-06-13 18:26:26 +00:00
|
|
|
int check_su_auth (const char *actual_id,
|
|
|
|
const char *wanted_id,
|
|
|
|
bool su_to_root)
|
2007-10-07 11:44:02 +00:00
|
|
|
{
|
2007-10-07 11:45:23 +00:00
|
|
|
int posn, endline;
|
|
|
|
const char field[] = ":";
|
|
|
|
FILE *authfile_fd;
|
|
|
|
char temp[1024];
|
|
|
|
char *to_users;
|
|
|
|
char *from_users;
|
|
|
|
char *action;
|
|
|
|
|
|
|
|
if (!(authfile_fd = fopen (SUAUTHFILE, "r"))) {
|
2008-09-13 18:03:50 +00:00
|
|
|
int err = errno;
|
2007-10-07 11:44:02 +00:00
|
|
|
/*
|
|
|
|
* If the file doesn't exist - default to the standard su
|
|
|
|
* behaviour (no access control). If open fails for some
|
|
|
|
* other reason - maybe someone is trying to fool us with
|
|
|
|
* file descriptors limit etc., so deny access. --marekm
|
|
|
|
*/
|
2008-09-13 18:03:50 +00:00
|
|
|
if (ENOENT == err) {
|
2007-10-07 11:44:02 +00:00
|
|
|
return NOACTION;
|
2008-09-13 18:03:50 +00:00
|
|
|
}
|
2007-10-07 11:45:23 +00:00
|
|
|
SYSLOG ((LOG_ERR,
|
2008-09-13 18:03:50 +00:00
|
|
|
"could not open/read config file '%s': %s\n",
|
|
|
|
SUAUTHFILE, strerror (err)));
|
2007-10-07 11:44:02 +00:00
|
|
|
return DENY;
|
|
|
|
}
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
while (fgets (temp, sizeof (temp), authfile_fd) != NULL) {
|
2007-10-07 11:44:02 +00:00
|
|
|
lines++;
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
if (temp[endline = strlen (temp) - 1] != '\n') {
|
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d: line too long or missing newline",
|
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
while (endline > 0 && (temp[endline - 1] == ' '
|
|
|
|
|| temp[endline - 1] == '\t'
|
|
|
|
|| temp[endline - 1] == '\n'))
|
2007-10-07 11:44:02 +00:00
|
|
|
endline--;
|
|
|
|
temp[endline] = '\0';
|
|
|
|
|
|
|
|
posn = 0;
|
|
|
|
while (temp[posn] == ' ' || temp[posn] == '\t')
|
|
|
|
posn++;
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
if (temp[posn] == '\n' || temp[posn] == '#'
|
|
|
|
|| temp[posn] == '\0') {
|
2007-10-07 11:44:02 +00:00
|
|
|
continue;
|
|
|
|
}
|
2007-10-07 11:45:23 +00:00
|
|
|
if (!(to_users = strtok (temp + posn, field))
|
|
|
|
|| !(from_users = strtok ((char *) NULL, field))
|
|
|
|
|| !(action = strtok ((char *) NULL, field))
|
|
|
|
|| strtok ((char *) NULL, field)) {
|
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d. Bad number of fields.\n",
|
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
if (!applies (wanted_id, to_users))
|
2007-10-07 11:44:02 +00:00
|
|
|
continue;
|
2007-10-07 11:45:23 +00:00
|
|
|
if (!applies (actual_id, from_users))
|
2007-10-07 11:44:02 +00:00
|
|
|
continue;
|
2007-10-07 11:45:23 +00:00
|
|
|
if (!strcmp (action, "DENY")) {
|
2011-06-13 18:26:26 +00:00
|
|
|
SYSLOG ((su_to_root ? LOG_WARN : LOG_NOTICE,
|
* src/chfn.c, src/chsh.c, src/groupdel.c, src/groupmems.c,
src/groupmod.c, src/grpck.c, src/login.c, src/logoutd.c,
src/newgrp.c, src/newusers.c, src/passwd.c, src/pwck.c,
src/suauth.c, src/useradd.c, src/userdel.c, src/usermod.c,
src/vipw.c: Complete the switch from the `' quotation style to ''.
Do it also in SYSLOG messages. Quote some parameters. All this
permits to merge some messages.
2008-08-06 15:51:52 +00:00
|
|
|
"DENIED su from '%s' to '%s' (%s)\n",
|
2007-10-07 11:45:23 +00:00
|
|
|
actual_id, wanted_id, SUAUTHFILE));
|
2008-01-24 20:42:12 +00:00
|
|
|
fputs (_("Access to su to that account DENIED.\n"),
|
|
|
|
stderr);
|
2007-10-07 11:45:23 +00:00
|
|
|
fclose (authfile_fd);
|
2007-10-07 11:44:02 +00:00
|
|
|
return DENY;
|
2007-10-07 11:45:23 +00:00
|
|
|
} else if (!strcmp (action, "NOPASS")) {
|
2011-06-13 18:26:26 +00:00
|
|
|
SYSLOG ((su_to_root ? LOG_NOTICE : LOG_INFO,
|
* src/chfn.c, src/chsh.c, src/groupdel.c, src/groupmems.c,
src/groupmod.c, src/grpck.c, src/login.c, src/logoutd.c,
src/newgrp.c, src/newusers.c, src/passwd.c, src/pwck.c,
src/suauth.c, src/useradd.c, src/userdel.c, src/usermod.c,
src/vipw.c: Complete the switch from the `' quotation style to ''.
Do it also in SYSLOG messages. Quote some parameters. All this
permits to merge some messages.
2008-08-06 15:51:52 +00:00
|
|
|
"NO password asked for su from '%s' to '%s' (%s)\n",
|
2007-10-07 11:45:23 +00:00
|
|
|
actual_id, wanted_id, SUAUTHFILE));
|
2008-01-24 20:42:12 +00:00
|
|
|
fputs (_("Password authentication bypassed.\n"),stderr);
|
2007-10-07 11:45:23 +00:00
|
|
|
fclose (authfile_fd);
|
2007-10-07 11:44:02 +00:00
|
|
|
return NOPWORD;
|
2007-10-07 11:45:23 +00:00
|
|
|
} else if (!strcmp (action, "OWNPASS")) {
|
2011-06-13 18:26:26 +00:00
|
|
|
SYSLOG ((su_to_root ? LOG_NOTICE : LOG_INFO,
|
* src/chfn.c, src/chsh.c, src/groupdel.c, src/groupmems.c,
src/groupmod.c, src/grpck.c, src/login.c, src/logoutd.c,
src/newgrp.c, src/newusers.c, src/passwd.c, src/pwck.c,
src/suauth.c, src/useradd.c, src/userdel.c, src/usermod.c,
src/vipw.c: Complete the switch from the `' quotation style to ''.
Do it also in SYSLOG messages. Quote some parameters. All this
permits to merge some messages.
2008-08-06 15:51:52 +00:00
|
|
|
"su from '%s' to '%s': asking for user's own password (%s)\n",
|
2007-10-07 11:45:23 +00:00
|
|
|
actual_id, wanted_id, SUAUTHFILE));
|
2008-01-24 20:42:12 +00:00
|
|
|
fputs (_("Please enter your OWN password as authentication.\n"),
|
|
|
|
stderr);
|
2007-10-07 11:45:23 +00:00
|
|
|
fclose (authfile_fd);
|
2007-10-07 11:44:02 +00:00
|
|
|
return OWNPWORD;
|
|
|
|
} else {
|
2007-10-07 11:45:23 +00:00
|
|
|
SYSLOG ((LOG_ERR,
|
2017-10-22 21:30:30 +00:00
|
|
|
"%s, line %d: unrecognized action!\n",
|
2007-10-07 11:45:23 +00:00
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
}
|
|
|
|
}
|
2007-10-07 11:45:23 +00:00
|
|
|
fclose (authfile_fd);
|
2007-10-07 11:44:02 +00:00
|
|
|
return NOACTION;
|
|
|
|
}
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
static int applies (const char *single, char *list)
|
2007-10-07 11:44:02 +00:00
|
|
|
{
|
2007-10-07 11:45:23 +00:00
|
|
|
const char split[] = ", ";
|
|
|
|
char *tok;
|
2007-10-07 11:44:02 +00:00
|
|
|
|
|
|
|
int state = 0;
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
for (tok = strtok (list, split); tok != NULL;
|
|
|
|
tok = strtok (NULL, split)) {
|
2007-10-07 11:44:02 +00:00
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
if (!strcmp (tok, "ALL")) {
|
2007-10-07 11:44:02 +00:00
|
|
|
if (state != 0) {
|
2007-10-07 11:45:23 +00:00
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d: ALL in bad place\n",
|
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
state = 1;
|
2007-10-07 11:45:23 +00:00
|
|
|
} else if (!strcmp (tok, "EXCEPT")) {
|
2007-10-07 11:44:02 +00:00
|
|
|
if (state != 1) {
|
2007-10-07 11:45:23 +00:00
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d: EXCEPT in bas place\n",
|
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
state = 2;
|
2007-10-07 11:45:23 +00:00
|
|
|
} else if (!strcmp (tok, "GROUP")) {
|
2007-10-07 11:44:02 +00:00
|
|
|
if ((state != 0) && (state != 2)) {
|
2007-10-07 11:45:23 +00:00
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d: GROUP in bad place\n",
|
|
|
|
SUAUTHFILE, lines));
|
2007-10-07 11:44:02 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
state = (state == 0) ? 3 : 4;
|
|
|
|
} else {
|
|
|
|
switch (state) {
|
2007-10-07 11:45:23 +00:00
|
|
|
case 0: /* No control words yet */
|
|
|
|
if (!strcmp (tok, single))
|
|
|
|
return 1;
|
|
|
|
break;
|
|
|
|
case 1: /* An all */
|
|
|
|
SYSLOG ((LOG_ERR,
|
|
|
|
"%s, line %d: expect another token after ALL\n",
|
|
|
|
SUAUTHFILE, lines));
|
|
|
|
return 0;
|
|
|
|
case 2: /* All except */
|
|
|
|
if (!strcmp (tok, single))
|
|
|
|
return 0;
|
|
|
|
break;
|
|
|
|
case 3: /* Group */
|
|
|
|
if (isgrp (single, tok))
|
|
|
|
return 1;
|
|
|
|
break;
|
|
|
|
case 4: /* All except group */
|
|
|
|
if (isgrp (single, tok))
|
2007-10-07 11:44:02 +00:00
|
|
|
return 0;
|
2007-10-07 11:45:23 +00:00
|
|
|
/* FALL THRU */
|
2007-10-07 11:44:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((state != 0) && (state != 3))
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-01-06 13:30:18 +00:00
|
|
|
static int isgrp (const char *name, const char *group)
|
2007-10-07 11:44:02 +00:00
|
|
|
{
|
2007-10-07 11:45:23 +00:00
|
|
|
struct group *grp;
|
2007-10-07 11:44:02 +00:00
|
|
|
|
* lib/prototypes.h, configure.in, libmisc/Makefile.am,
libmisc/xgetXXbyYY.c, libmisc/xgetpwnam.c, libmisc/xgetpwuid.c,
libmisc/xgetgrnam.c, libmisc/xgetgrgid.c, libmisc/xgetspnam.c:
Added functions xgetpwnam(), xgetpwuid(), xgetgrnam(),
xgetgrgid(), and xgetspnam(). They allocate memory for the
returned structure and are more robust to successive calls. They
are implemented with the libc's getxxyyy_r() functions if
available.
* libmisc/limits.c, libmisc/entry.c, libmisc/chowntty.c,
libmisc/addgrps.c, libmisc/myname.c, libmisc/rlogin.c,
libmisc/pwdcheck.c, src/newgrp.c, src/login_nopam.c,
src/userdel.c, src/lastlog.c, src/grpck.c, src/gpasswd.c,
src/newusers.c, src/chpasswd.c, src/chfn.c, src/groupmems.c,
src/usermod.c, src/expiry.c, src/groupdel.c, src/chgpasswd.c,
src/su.c, src/useradd.c, src/groupmod.c, src/passwd.c, src/pwck.c,
src/groupadd.c, src/chage.c, src/login.c, src/suauth.c,
src/faillog.c, src/groups.c, src/chsh.c, src/id.c: Review all the
usage of one of the getpwnam(), getpwuid(), getgrnam(),
getgrgid(), and getspnam() functions. It was noticed on
http://bugs.debian.org/341230 that chfn and chsh use a passwd
structure after calling a pam function, which result in using
information from the passwd structure requested by pam, not the
original one. It is much easier to use the new xget... functions
to avoid these issues. I've checked which call to the original
get... functions could be left (reducing the scope of the
structure if possible), and I've left comments to ease future
reviews (e.g. /* local, no need for xgetpwnam */).
Note: the getpwent/getgrent calls should probably be checked also.
* src/groupdel.c, src/expiry.c: Fix typos in comments.
* src/groupmod.c: Re-indent.
* libmisc/Makefile.am, lib/groupmem.c, lib/groupio.c, lib/pwmem.c,
lib/pwio.c, lib/shadowmem.c, lib/shadowio.c: Move the __<xx>_dup
functions (used by the xget... functions) from the <xx>io.c files
to the new <xx>mem.c files. This avoid linking some utils against
the SELinux library.
2007-11-18 23:15:26 +00:00
|
|
|
grp = getgrnam (group); /* local, no need for xgetgrnam */
|
2007-10-07 11:44:02 +00:00
|
|
|
|
|
|
|
if (!grp || !grp->gr_mem)
|
|
|
|
return 0;
|
|
|
|
|
2007-10-07 11:45:23 +00:00
|
|
|
return is_on_list (grp->gr_mem, name);
|
2007-10-07 11:44:02 +00:00
|
|
|
}
|
2007-10-07 11:45:23 +00:00
|
|
|
#endif /* SU_ACCESS */
|