+++ /dev/null
-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