308 lines
5.5 KiB
C
308 lines
5.5 KiB
C
/*
|
|
* id - POSIX.2 user identity
|
|
*
|
|
* (INCOMPLETE -- supplementary groups for other users not yet done)
|
|
*
|
|
* usage: id [-Ggu] [-nr] [user]
|
|
*
|
|
* The default output format looks something like:
|
|
* uid=xxx(chet) gid=xx groups=aa(aname), bb(bname), cc(cname)
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <stdio.h>
|
|
#include "bashtypes.h"
|
|
#include <pwd.h>
|
|
#include <grp.h>
|
|
#include "bashansi.h"
|
|
|
|
#ifdef HAVE_LIMITS_H
|
|
# include <limits.h>
|
|
#else
|
|
# include <sys/param.h>
|
|
#endif
|
|
|
|
#if !defined (HAVE_GETPW_DECLS)
|
|
extern struct passwd *getpwuid ();
|
|
#endif
|
|
extern struct group *getgrgid ();
|
|
|
|
#include "shell.h"
|
|
#include "builtins.h"
|
|
#include "stdc.h"
|
|
#include "common.h"
|
|
#include "bashgetopt.h"
|
|
|
|
#define ID_ALLGROUPS 0x001 /* -G */
|
|
#define ID_GIDONLY 0x002 /* -g */
|
|
#define ID_USENAME 0x004 /* -n */
|
|
#define ID_USEREAL 0x008 /* -r */
|
|
#define ID_USERONLY 0x010 /* -u */
|
|
|
|
#define ID_FLAGSET(s) ((id_flags & (s)) != 0)
|
|
|
|
static int id_flags;
|
|
|
|
static uid_t ruid, euid;
|
|
static gid_t rgid, egid;
|
|
|
|
static char *id_user;
|
|
|
|
static int inituser ();
|
|
|
|
static int id_pruser ();
|
|
static int id_prgrp ();
|
|
static int id_prgroups ();
|
|
static int id_prall ();
|
|
|
|
int
|
|
id_builtin (list)
|
|
WORD_LIST *list;
|
|
{
|
|
int opt;
|
|
char *user;
|
|
|
|
id_flags = 0;
|
|
reset_internal_getopt ();
|
|
while ((opt = internal_getopt (list, "Ggnru")) != -1)
|
|
{
|
|
switch (opt)
|
|
{
|
|
case 'G': id_flags |= ID_ALLGROUPS; break;
|
|
case 'g': id_flags |= ID_GIDONLY; break;
|
|
case 'n': id_flags |= ID_USENAME; break;
|
|
case 'r': id_flags |= ID_USEREAL; break;
|
|
case 'u': id_flags |= ID_USERONLY; break;
|
|
default:
|
|
builtin_usage ();
|
|
return (EX_USAGE);
|
|
}
|
|
}
|
|
list = loptend;
|
|
|
|
user = list ? list->word->word : (char *)NULL;
|
|
|
|
/* Check for some invalid option combinations */
|
|
opt = ID_FLAGSET (ID_ALLGROUPS) + ID_FLAGSET (ID_GIDONLY) + ID_FLAGSET (ID_USERONLY);
|
|
if (opt > 1 || (opt == 0 && ((id_flags & (ID_USEREAL|ID_USENAME)) != 0)))
|
|
{
|
|
builtin_usage ();
|
|
return (EX_USAGE);
|
|
}
|
|
|
|
if (list && list->next)
|
|
{
|
|
builtin_usage ();
|
|
return (EX_USAGE);
|
|
}
|
|
|
|
if (inituser (user) < 0)
|
|
return (EXECUTION_FAILURE);
|
|
|
|
opt = 0;
|
|
if (id_flags & ID_USERONLY)
|
|
opt += id_pruser ((id_flags & ID_USEREAL) ? ruid : euid);
|
|
else if (id_flags & ID_GIDONLY)
|
|
opt += id_prgrp ((id_flags & ID_USEREAL) ? rgid : egid);
|
|
else if (id_flags & ID_ALLGROUPS)
|
|
opt += id_prgroups (user);
|
|
else
|
|
opt += id_prall (user);
|
|
putchar ('\n');
|
|
fflush (stdout);
|
|
|
|
return (opt == 0 ? EXECUTION_SUCCESS : EXECUTION_FAILURE);
|
|
}
|
|
|
|
static int
|
|
inituser (uname)
|
|
char *uname;
|
|
{
|
|
struct passwd *pwd;
|
|
|
|
if (uname)
|
|
{
|
|
pwd = getpwnam (uname);
|
|
if (pwd == 0)
|
|
{
|
|
builtin_error ("%s: no such user", uname);
|
|
return -1;
|
|
}
|
|
ruid = euid = pwd->pw_uid;
|
|
rgid = egid = pwd->pw_gid;
|
|
}
|
|
else
|
|
{
|
|
ruid = current_user.uid;
|
|
euid = current_user.euid;
|
|
rgid = current_user.gid;
|
|
egid = current_user.egid;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Print the name or value of user ID UID. */
|
|
static int
|
|
id_pruser (uid)
|
|
int uid;
|
|
{
|
|
struct passwd *pwd = NULL;
|
|
int r;
|
|
|
|
r = 0;
|
|
if (id_flags & ID_USENAME)
|
|
{
|
|
pwd = getpwuid (uid);
|
|
if (pwd == NULL)
|
|
r = 1;
|
|
}
|
|
if (pwd)
|
|
printf ("%s", pwd->pw_name);
|
|
else
|
|
printf ("%u", (unsigned) uid);
|
|
|
|
return r;
|
|
}
|
|
|
|
/* Print the name or value of group ID GID. */
|
|
|
|
static int
|
|
id_prgrp (gid)
|
|
int gid;
|
|
{
|
|
struct group *grp = NULL;
|
|
int r;
|
|
|
|
r = 0;
|
|
if (id_flags & ID_USENAME)
|
|
{
|
|
grp = getgrgid (gid);
|
|
if (grp == NULL)
|
|
r = 1;
|
|
}
|
|
|
|
if (grp)
|
|
printf ("%s", grp->gr_name);
|
|
else
|
|
printf ("%u", (unsigned) gid);
|
|
|
|
return r;
|
|
}
|
|
|
|
static int
|
|
id_prgroups (uname)
|
|
char *uname;
|
|
{
|
|
int *glist, ng, i, r;
|
|
|
|
r = 0;
|
|
id_prgrp (rgid);
|
|
if (egid != rgid)
|
|
{
|
|
putchar (' ');
|
|
id_prgrp (egid);
|
|
}
|
|
|
|
if (uname)
|
|
{
|
|
builtin_error ("supplementary groups for other users not yet implemented");
|
|
glist = (int *)NULL;
|
|
ng = 0;
|
|
r = 1;
|
|
}
|
|
else
|
|
glist = get_group_array (&ng);
|
|
|
|
for (i = 0; i < ng; i++)
|
|
if (glist[i] != rgid && glist[i] != egid)
|
|
{
|
|
putchar (' ');
|
|
id_prgrp (glist[i]);
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
static int
|
|
id_prall (uname)
|
|
char *uname;
|
|
{
|
|
int r, i, ng, *glist;
|
|
struct passwd *pwd;
|
|
struct group *grp;
|
|
|
|
r = 0;
|
|
printf ("uid=%u", (unsigned) ruid);
|
|
pwd = getpwuid (ruid);
|
|
if (pwd == NULL)
|
|
r = 1;
|
|
else
|
|
printf ("(%s)", pwd->pw_name);
|
|
|
|
printf (" gid=%u", (unsigned) rgid);
|
|
grp = getgrgid (rgid);
|
|
if (grp == NULL)
|
|
r = 1;
|
|
else
|
|
printf ("(%s)", grp->gr_name);
|
|
|
|
if (euid != ruid)
|
|
{
|
|
printf (" euid=%u", (unsigned) euid);
|
|
pwd = getpwuid (euid);
|
|
if (pwd == NULL)
|
|
r = 1;
|
|
else
|
|
printf ("(%s)", pwd->pw_name);
|
|
}
|
|
|
|
if (egid != rgid)
|
|
{
|
|
printf (" egid=%u", (unsigned) egid);
|
|
grp = getgrgid (egid);
|
|
if (grp == NULL)
|
|
r = 1;
|
|
else
|
|
printf ("(%s)", grp->gr_name);
|
|
}
|
|
|
|
if (uname)
|
|
{
|
|
builtin_error ("supplementary groups for other users not yet implemented");
|
|
glist = (int *)NULL;
|
|
ng = 0;
|
|
r = 1;
|
|
}
|
|
else
|
|
glist = get_group_array (&ng);
|
|
|
|
if (ng > 0)
|
|
printf (" groups=");
|
|
for (i = 0; i < ng; i++)
|
|
{
|
|
if (i > 0)
|
|
printf (", ");
|
|
printf ("%u", (unsigned) glist[i]);
|
|
grp = getgrgid (glist[i]);
|
|
if (grp == NULL)
|
|
r = 1;
|
|
else
|
|
printf ("(%s)", grp->gr_name);
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
char *id_doc[] = {
|
|
"return information about user identity",
|
|
(char *)NULL
|
|
};
|
|
|
|
struct builtin id_struct = {
|
|
"id",
|
|
id_builtin,
|
|
BUILTIN_ENABLED,
|
|
id_doc,
|
|
"id [user]\n\tid -G [-n] [user]\n\tid -g [-nr] [user]\n\tid -u [-nr] [user]",
|
|
0
|
|
};
|