dd: clarify meaning of multiplication factors; put xM in order
[platform/upstream/coreutils.git] / src / groups.c
1 /* groups -- print the groups a user is in
2    Copyright (C) 1989-2008 Free Software Foundation, Inc.
3
4    This program is free software: you can redistribute it and/or modify
5    it under the terms of the GNU General Public License as published by
6    the Free Software Foundation, either version 3 of the License, or
7    (at your option) any later version.
8
9    This program is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12    GNU General Public License for more details.
13
14    You should have received a copy of the GNU General Public License
15    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
16
17 /* Written by James Youngman based on id.c and groups.sh,
18    which were written by Arnold Robbins and David MacKenzie. */
19
20 #include <config.h>
21 #include <stdio.h>
22 #include <getopt.h>
23 #include <sys/types.h>
24 #include <pwd.h>
25 #include <grp.h>
26 #include <getopt.h>
27
28 #include "system.h"
29 #include "error.h"
30 #include "group-list.h"
31
32 /* The official name of this program (e.g., no `g' prefix).  */
33 #define PROGRAM_NAME "groups"
34
35 #define AUTHORS \
36   proper_name ("David MacKenzie"), \
37   proper_name ("James Youngman")
38
39
40 static struct option const longopts[] =
41 {
42   {GETOPT_HELP_OPTION_DECL},
43   {GETOPT_VERSION_OPTION_DECL},
44   {NULL, 0, NULL, 0}
45 };
46
47 void
48 usage (int status)
49 {
50   if (status != EXIT_SUCCESS)
51     fprintf (stderr, _("Try `%s --help' for more information.\n"),
52              program_name);
53   else
54     {
55       printf (_("Usage: %s [OPTION]... [USERNAME]\n"), program_name);
56       fputs (_("\
57 Print information for USERNAME or, if no USERNAME is specified,\n\
58 the current process (which is different if the groups database has changed).\n"),
59              stdout);
60       fputs (HELP_OPTION_DESCRIPTION, stdout);
61       fputs (VERSION_OPTION_DESCRIPTION, stdout);
62       emit_bug_reporting_address ();
63     }
64   exit (status);
65 }
66
67 int
68 main (int argc, char **argv)
69 {
70   int optc;
71   bool ok = true;
72   gid_t rgid, egid;
73   uid_t ruid;
74
75   initialize_main (&argc, &argv);
76   set_program_name (argv[0]);
77   setlocale (LC_ALL, "");
78   bindtextdomain (PACKAGE, LOCALEDIR);
79   textdomain (PACKAGE);
80
81   atexit (close_stdout);
82
83   /* Processing the arguments this way makes groups.c behave differently to
84    * groups.sh if one of the arguments is "--".
85    */
86   while ((optc = getopt_long (argc, argv, "", longopts, NULL)) != -1)
87     {
88       switch (optc)
89         {
90         case_GETOPT_HELP_CHAR;
91         case_GETOPT_VERSION_CHAR (PROGRAM_NAME, AUTHORS);
92         default:
93           usage (EXIT_FAILURE);
94         }
95     }
96
97   if (optind == argc)
98     {
99       /* No arguments.  Divulge the details of the current process. */
100       ruid = getuid ();
101       egid = getegid ();
102       rgid = getgid ();
103
104       if (!print_group_list (NULL, ruid, rgid, egid, true))
105         ok = false;
106       putchar ('\n');
107     }
108   else
109     {
110       /* At least one argument.  Divulge the details of the specified users. */
111       while (optind < argc)
112         {
113           struct passwd *pwd = getpwnam (argv[optind]);
114           if (pwd == NULL)
115             error (EXIT_FAILURE, 0, _("%s: No such user"), argv[optind]);
116           ruid = pwd->pw_uid;
117           rgid = egid = pwd->pw_gid;
118
119           printf ("%s : ", argv[optind]);
120           if (!print_group_list (argv[optind++], ruid, rgid, egid, true))
121             ok = false;
122           putchar ('\n');
123         }
124     }
125
126   exit (ok ? EXIT_SUCCESS : EXIT_FAILURE);
127 }
128
129 /*
130  * Local variables:
131  *  indent-tabs-mode: nil
132  * End:
133  */