Index: coreutils/Config.in =================================================================== RCS file: /var/cvs/busybox/coreutils/Config.in,v retrieving revision 1.24 diff -u -r1.24 Config.in --- a/coreutils/Config.in 15 Mar 2004 08:28:19 -0000 1.24 +++ b/coreutils/Config.in 1 May 2004 11:39:04 -0000 @@ -218,6 +218,14 @@ help id displays the current user and group ID names. +config CONFIG_FEATURE_ID_GROUPS_ALIAS + bool " Support 'groups' as alias to 'id -Gn'" + default y + depends on CONFIG_ID + help + Print the groups a user is in. This is an alias to 'id -Gn' on + most systems. + config CONFIG_INSTALL bool "install" default n Index: coreutils/id.c =================================================================== RCS file: /var/cvs/busybox/coreutils/id.c,v retrieving revision 1.24 diff -u -r1.24 id.c --- a/coreutils/id.c 15 Mar 2004 08:28:20 -0000 1.24 +++ b/coreutils/id.c 1 May 2004 11:39:05 -0000 @@ -3,6 +3,8 @@ * Mini id implementation for busybox * * Copyright (C) 2000 by Randolph Chung <tausq@debian.org> + * Copyright (C) 2004 by Tony J. White <tjw@tjw.org> + * Copyright (C) 2004 by Glenn McGrath <bug1@iinet.net.au> * * 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 @@ -20,7 +22,6 @@ * */ -/* BB_AUDIT SUSv3 _NOT_ compliant -- option -G is not currently supported. */ #include "busybox.h" #include <stdio.h> @@ -33,78 +34,153 @@ #include <flask_util.h> #endif -#define JUST_USER 1 -#define JUST_GROUP 2 -#define PRINT_REAL 4 -#define NAME_NOT_NUMBER 8 +#define ID_OPT_JUST_USER 1 +#define ID_OPT_JUST_GROUP 2 +#define ID_OPT_ALL_GROUPS 4 +#define ID_OPT_PRINT_REAL 8 +#define ID_OPT_NAME_NOT_NUMBER 16 + +static void print_groups(unsigned long flags, const char sep) +{ + gid_t gids[64]; + int gid_count; + int i; + + gid_count = getgroups(64, gids); + + for (i = 0; i < gid_count; i++) { + struct group *tmp_grp; + + if (i != 0) { + putchar(sep); + } + tmp_grp = getgrgid(gids[i]); + if (flags & ID_OPT_NAME_NOT_NUMBER) { + if (tmp_grp == NULL) { + continue; + } + printf("%s", tmp_grp->gr_name); + } else { + printf("%u", gids[i]); + if (!(flags & ID_OPT_ALL_GROUPS)) { + if (tmp_grp == NULL) { + continue; + } + printf("(%s)", tmp_grp->gr_name); + } + } + } +} extern int id_main(int argc, char **argv) { - char user[9], group[9]; - long pwnam, grnam; - int uid, gid; - int flags; + struct group *grp; + struct passwd *usr; + unsigned long flags; + uid_t uid; + uid_t gid; + uid_t euid; + uid_t egid; #ifdef CONFIG_SELINUX int is_flask_enabled_flag = is_flask_enabled(); #endif - flags = bb_getopt_ulflags(argc, argv, "ugrn"); + bb_opt_complementaly = "u~gG:g~uG:G~ug:~n"; + flags = bb_getopt_ulflags(argc, argv, "ugGrn"); - if (((flags & (JUST_USER | JUST_GROUP)) == (JUST_USER | JUST_GROUP)) - || (argc > optind + 1) - ) { + /* Check one and only one context option was given */ + if ((flags & 0x80000000UL) || + (flags & (ID_OPT_PRINT_REAL | ID_OPT_ALL_GROUPS)) || + ((flags & (ID_OPT_PRINT_REAL | ID_OPT_NAME_NOT_NUMBER)) == + (ID_OPT_PRINT_REAL | ID_OPT_NAME_NOT_NUMBER))) { bb_show_usage(); } +#ifdef CONFIG_FEATURE_ID_GROUPS_ALIAS + /* groups command is an alias for 'id -Gn' */ + if (bb_applet_name[0] == 'g') { + flags |= (ID_OPT_ALL_GROUPS + ID_OPT_NAME_NOT_NUMBER); + } +#endif + + uid = getuid(); + gid = getgid(); + euid = geteuid(); + egid = getegid(); + + if (flags & ID_OPT_PRINT_REAL) { + euid = uid; + egid = gid; + } + if (argv[optind] == NULL) { - if (flags & PRINT_REAL) { - uid = getuid(); - gid = getgid(); - } else { - uid = geteuid(); - gid = getegid(); - } - my_getpwuid(user, uid); + usr = getpwuid(euid); + grp = getgrgid(egid); } else { - safe_strncpy(user, argv[optind], sizeof(user)); - gid = my_getpwnamegid(user); + usr = getpwnam(argv[optind]); + grp = getgrnam(argv[optind]); } - my_getgrgid(group, gid); - pwnam=my_getpwnam(user); - grnam=my_getgrnam(group); + if (usr == NULL) { + bb_perror_msg_and_die("cannot find user name"); + } + if (grp == NULL) { + bb_perror_msg_and_die("cannot find group name"); + } - if (flags & (JUST_GROUP | JUST_USER)) { - char *s = group; - if (flags & JUST_USER) { - s = user; - grnam = pwnam; + if (flags & ID_OPT_JUST_USER) { + if (flags & ID_OPT_NAME_NOT_NUMBER) { + printf("%s", grp->gr_name); + } else { + printf("%u", euid); } - if (flags & NAME_NOT_NUMBER) { - puts(s); + } + else if (flags & ID_OPT_JUST_GROUP) { + if (flags & ID_OPT_NAME_NOT_NUMBER) { + printf("%s", grp->gr_name); } else { - printf("%ld\n", grnam); + printf("%u", egid); } + } + else if (flags & ID_OPT_ALL_GROUPS) { + print_groups(flags, ' '); } else { -#ifdef CONFIG_SELINUX - printf("uid=%ld(%s) gid=%ld(%s)", pwnam, user, grnam, group); - if(is_flask_enabled_flag) - { - security_id_t mysid = getsecsid(); - char context[80]; - int len = sizeof(context); - context[0] = '\0'; - if(security_sid_to_context(mysid, context, &len)) - strcpy(context, "unknown"); - printf(" context=%s\n", context); - } - else - printf("\n"); -#else - printf("uid=%ld(%s) gid=%ld(%s)\n", pwnam, user, grnam, group); -#endif + printf("uid=%u(%s) gid=%u(%s)", uid, usr->pw_name, gid, grp->gr_name); + if (uid != euid) { + struct passwd *eusr; + printf(" euid=%u", euid); + eusr = getpwuid(euid); + if (eusr != NULL) { + printf("(%s)", eusr->pw_name); + } + } + if (gid != egid) { + struct group *egrp; + printf(" egid=%u", egid); + egrp = getgrgid(egid); + if (egrp != NULL) { + printf("(%s)", egrp->gr_name); + } + } + printf(" groups="); + print_groups(flags, ','); + } +#ifdef CONFIG_SELINUX + if (is_flask_enabled_flag) + { + security_id_t mysid = getsecsid(); + char context[80]; + int len = sizeof(context); + + context[0] = '\0'; + if (security_sid_to_context(mysid, len, &len)) { + strcpy(context, "unknown"); + } + printf(" context=%s", context); } +#endif + putchar('\n'); bb_fflush_stdout_and_exit(0); } Index: include/applets.h =================================================================== RCS file: /var/cvs/busybox/include/applets.h,v retrieving revision 1.113 diff -u -r1.113 applets.h --- a/include/applets.h 6 Apr 2004 16:59:43 -0000 1.113 +++ b/include/applets.h 1 May 2004 11:39:06 -0000 @@ -232,6 +232,9 @@ #ifdef CONFIG_GREP APPLET(grep, grep_main, _BB_DIR_BIN, _BB_SUID_NEVER) #endif +#if defined(CONFIG_FEATURE_ID_GROUPS_ALIAS) + APPLET(groups, id_main, _BB_DIR_USR_BIN, _BB_SUID_NEVER) +#endif #ifdef CONFIG_GUNZIP APPLET(gunzip, gunzip_main, _BB_DIR_BIN, _BB_SUID_NEVER) #endif Index: include/usage.h =================================================================== RCS file: /var/cvs/busybox/include/usage.h,v retrieving revision 1.207 diff -u -r1.207 usage.h --- a/include/usage.h 14 Apr 2004 17:59:21 -0000 1.207 +++ b/include/usage.h 1 May 2004 11:39:10 -0000 @@ -800,6 +800,16 @@ "$ grep ^[rR]oo. /etc/passwd\n" \ "root:x:0:0:root:/root:/bin/bash\n" +#define groups_trivial_usage \ + " [USERNAME]" +#define groups_full_usage \ + "Print all group names that USERNAME is a member of." +#define groups_example_usage \ + "$ groups\n" \ + "andersen users\n" \ + "$ groups tjw\n" \ + "tjw users\n" + #define gunzip_trivial_usage \ "[OPTION]... FILE" #define gunzip_full_usage \ @@ -1035,7 +1045,7 @@ #endif #define id_trivial_usage \ - "[OPTIONS]... [USERNAME]" + "[-Ggu[nr]]] [USERNAME]" #define id_full_usage \ "Print information for USERNAME or the current user\n\n" \ "Options:\n" \ @@ -1043,10 +1053,11 @@ "\t-g\tprints only the group ID\n" \ "\t-u\tprints only the user ID\n" \ "\t-n\tprint a name instead of a number\n" \ - "\t-r\tprints the real user ID instead of the effective ID" + "\t-r\tprints the real user ID instead of the effective ID\n" \ + "\t-G\tprints all groups the user belongs to" #define id_example_usage \ "$ id\n" \ - "uid=1000(andersen) gid=1000(andersen)\n" + "uid=1000(andersen) gid=1000(andersen) groups=1000(andersen),100(users)\n" #ifdef CONFIG_FEATURE_IFCONFIG_SLIP #define USAGE_SIOCSKEEPALIVE(a) a