annotate lib/userspec.c @ 2284:ce1bca7b77d5

(parse_user_spec): If there is no `:' but there is a `.', then first look up the entire `.'-containing string as a login name.
author Jim Meyering <jim@meyering.net>
date Sun, 27 Feb 2000 11:13:16 +0000
parents ad48a40e44bb
children aff553209434
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
1 /* userspec.c -- Parse a user and group string.
2271
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
2 Copyright (C) 1989-1992, 1997, 1998, 2000 Free Software Foundation, Inc.
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
3
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
4 This program is free software; you can redistribute it and/or modify
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
5 it under the terms of the GNU General Public License as published by
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
6 the Free Software Foundation; either version 2, or (at your option)
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
7 any later version.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
8
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
9 This program is distributed in the hope that it will be useful,
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
12 GNU General Public License for more details.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
13
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
14 You should have received a copy of the GNU General Public License
650
b4ef1c1a0171 update FSF address in copyright
Jim Meyering <jim@meyering.net>
parents: 499
diff changeset
15 along with this program; if not, write to the Free Software Foundation,
b4ef1c1a0171 update FSF address in copyright
Jim Meyering <jim@meyering.net>
parents: 499
diff changeset
16 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
17
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
18 /* Written by David MacKenzie <djm@gnu.ai.mit.edu>. */
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
19
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
20 #if HAVE_CONFIG_H
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
21 # include <config.h>
105
12f81400139d merge with 3.8.3b
Jim Meyering <jim@meyering.net>
parents: 52
diff changeset
22 #endif
12f81400139d merge with 3.8.3b
Jim Meyering <jim@meyering.net>
parents: 52
diff changeset
23
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
24 #ifdef __GNUC__
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
25 # define alloca __builtin_alloca
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
26 #else
1278
9ad625a30b7d Use #if, not #ifdef with HAVE_ macros
Jim Meyering <jim@meyering.net>
parents: 1260
diff changeset
27 # if HAVE_ALLOCA_H
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
28 # include <alloca.h>
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
29 # else
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
30 # ifdef _AIX
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
31 # pragma alloca
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
32 # else
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
33 char *alloca ();
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
34 # endif
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
35 # endif
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
36 #endif
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
37
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
38 #include <stdio.h>
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
39 #include <sys/types.h>
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
40 #include <pwd.h>
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
41 #include <grp.h>
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
42
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
43 #if HAVE_STRING_H
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
44 # include <string.h>
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
45 #else
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
46 # include <strings.h>
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
47 # ifndef strchr
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
48 # define strchr index
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
49 # endif
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
50 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
51
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
52 #if STDC_HEADERS
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
53 # include <stdlib.h>
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
54 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
55
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
56 #if HAVE_UNISTD_H
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
57 # include <unistd.h>
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
58 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
59
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
60 #if ENABLE_NLS
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
61 # include <libintl.h>
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
62 # define _(Text) gettext (Text)
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
63 #else
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
64 # define _(Text) Text
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
65 #endif
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
66 #define N_(Text) Text
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
67
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
68 #ifndef _POSIX_VERSION
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
69 struct passwd *getpwnam ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
70 struct group *getgrnam ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
71 struct group *getgrgid ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
72 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
73
499
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
74 #ifndef HAVE_ENDGRENT
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
75 # define endgrent() ((void) 0)
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
76 #endif
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
77
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
78 #ifndef HAVE_ENDPWENT
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
79 # define endpwent() ((void) 0)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
80 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
81
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
82 /* Perform the equivalent of the statement `dest = strdup (src);',
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
83 but obtaining storage via alloca instead of from the heap. */
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
84
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
85 #define V_STRDUP(dest, src) \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
86 do \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
87 { \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
88 int _len = strlen ((src)); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
89 (dest) = (char *) alloca (_len + 1); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
90 strcpy (dest, src); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
91 } \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
92 while (0)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
93
2271
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
94 /* ISDIGIT differs from isdigit, as follows:
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
95 - Its arg may be any int or unsigned int; it need not be an unsigned char.
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
96 - It's guaranteed to evaluate its argument exactly once.
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
97 - It's typically faster.
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
98 Posix 1003.2-1992 section 2.5.2.1 page 50 lines 1556-1558 says that
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
99 only '0' through '9' are digits. Prefer ISDIGIT to isdigit unless
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
100 it's important to use the locale's definition of `digit' even when the
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
101 host does not conform to Posix. */
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
102 #define ISDIGIT(c) ((unsigned) (c) - '0' <= 9)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
103
1347
e7dc6e5a3be5 Don't declare strdup if it's defined as a macro.
Jim Meyering <jim@meyering.net>
parents: 1278
diff changeset
104 #ifndef strdup
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
105 char *strdup ();
1347
e7dc6e5a3be5 Don't declare strdup if it's defined as a macro.
Jim Meyering <jim@meyering.net>
parents: 1278
diff changeset
106 #endif
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
107
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
108 /* Return nonzero if STR represents an unsigned decimal integer,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
109 otherwise return 0. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
110
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
111 static int
1557
38fd8f5d359d ansideclify
Jim Meyering <jim@meyering.net>
parents: 1347
diff changeset
112 is_number (const char *str)
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
113 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
114 for (; *str; str++)
2271
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
115 if (!ISDIGIT (*str))
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
116 return 0;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
117 return 1;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
118 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
119
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
120 /* Extract from NAME, which has the form "[user][:.][group]",
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
121 a USERNAME, UID U, GROUPNAME, and GID G.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
122 Either user or group, or both, must be present.
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
123 If the group is omitted but the ":" separator is given,
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
124 use the given user's login group.
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
125 If SPEC_ARG contains a `:', then use that as the separator, ignoring
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
126 any `.'s. If there is no `:', but there is a `.', then first look
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
127 up SPEC_ARG as a login name. If that look-up fails, then try again
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
128 interpreting the `.' as a separator.
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
129
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
130 USERNAME and GROUPNAME will be in newly malloc'd memory.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
131 Either one might be NULL instead, indicating that it was not
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
132 given and the corresponding numeric ID was left unchanged.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
133
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
134 Return NULL if successful, a static error message string if not. */
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
135
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
136 const char *
1580
12fde28eb4ee break long line
Jim Meyering <jim@meyering.net>
parents: 1557
diff changeset
137 parse_user_spec (const char *spec_arg, uid_t *uid, gid_t *gid,
12fde28eb4ee break long line
Jim Meyering <jim@meyering.net>
parents: 1557
diff changeset
138 char **username_arg, char **groupname_arg)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
139 {
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
140 static const char *E_no_memory = N_("virtual memory exhausted");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
141 static const char *E_invalid_user = N_("invalid user");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
142 static const char *E_invalid_group = N_("invalid group");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
143 static const char *E_bad_spec =
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
144 N_("cannot get the login group of a numeric UID");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
145 static const char *E_cannot_omit_both =
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
146 N_("cannot omit both user and group");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
147
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
148 const char *error_msg;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
149 char *spec; /* A copy we can write on. */
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
150 struct passwd *pwd;
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
151 struct group *grp;
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
152 char *g, *u, *separator;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
153 char *groupname;
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
154 int maybe_retry = 0;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
155 char *dot = NULL;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
156
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
157 error_msg = NULL;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
158 *username_arg = *groupname_arg = NULL;
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
159 groupname = NULL;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
160
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
161 V_STRDUP (spec, spec_arg);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
162
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
163 /* Find the POSIX `:' separator if there is one. */
432
a56993d69333 (parse_user_spec): Use strchr, not index.
Jim Meyering <jim@meyering.net>
parents: 316
diff changeset
164 separator = strchr (spec, ':');
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
165
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
166 /* If there is no colon, then see if there's a `.'. */
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
167 if (separator == NULL)
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
168 {
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
169 dot = strchr (spec, '.');
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
170 /* If there's no colon but there is a `.', then first look up the
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
171 whole spec, in case it's an OWNER name that includes a dot.
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
172 If that fails, then we'll try again, but interpreting the `.'
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
173 as a separator. */
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
174 /* FIXME: accepting `.' as the separator is contrary to POSIX.
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
175 someday we should drop support for this. */
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
176 if (dot)
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
177 maybe_retry = 1;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
178 }
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
179
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
180 retry:
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
181
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
182 /* Replace separator with a NUL. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
183 if (separator != NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
184 *separator = '\0';
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
185
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
186 /* Set U and G to non-zero length strings corresponding to user and
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
187 group specifiers or to NULL. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
188 u = (*spec == '\0' ? NULL : spec);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
189
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
190 g = (separator == NULL || *(separator + 1) == '\0'
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
191 ? NULL
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
192 : separator + 1);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
193
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
194 if (u == NULL && g == NULL)
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
195 return _(E_cannot_omit_both);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
196
1260
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
197 #ifdef __DJGPP__
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
198 /* Pretend that we are the user U whose group is G. This makes
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
199 pwd and grp functions ``know'' about the UID and GID of these. */
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
200 if (u && !is_number (u))
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
201 setenv ("USER", u, 1);
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
202 if (g && !is_number (g))
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
203 setenv ("GROUP", g, 1);
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
204 #endif
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
205
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
206 if (u != NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
207 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
208 pwd = getpwnam (u);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
209 if (pwd == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
210 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
211
475
7e3e0dd559c7 (isnumber): Rename to is_number.
Jim Meyering <jim@meyering.net>
parents: 432
diff changeset
212 if (!is_number (u))
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
213 error_msg = _(E_invalid_user);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
214 else
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
215 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
216 int use_login_group;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
217 use_login_group = (separator != NULL && g == NULL);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
218 if (use_login_group)
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
219 error_msg = _(E_bad_spec);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
220 else
2272
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
221 {
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
222 /* FIXME: don't use atoi! */
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
223 *uid = atoi (u);
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
224 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
225 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
226 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
227 else
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
228 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
229 *uid = pwd->pw_uid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
230 if (g == NULL && separator != NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
231 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
232 /* A separator was given, but a group was not specified,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
233 so get the login group. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
234 *gid = pwd->pw_gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
235 grp = getgrgid (pwd->pw_gid);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
236 if (grp == NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
237 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
238 /* This is enough room to hold the unsigned decimal
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
239 representation of any 32-bit quantity and the trailing
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
240 zero byte. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
241 char uint_buf[21];
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
242 sprintf (uint_buf, "%u", (unsigned) (pwd->pw_gid));
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
243 V_STRDUP (groupname, uint_buf);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
244 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
245 else
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
246 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
247 V_STRDUP (groupname, grp->gr_name);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
248 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
249 endgrent ();
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
250 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
251 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
252 endpwent ();
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
253 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
254
195
Jim Meyering <jim@meyering.net>
parents: 194
diff changeset
255 if (g != NULL && error_msg == NULL)
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
256 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
257 /* Explicit group. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
258 grp = getgrnam (g);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
259 if (grp == NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
260 {
475
7e3e0dd559c7 (isnumber): Rename to is_number.
Jim Meyering <jim@meyering.net>
parents: 432
diff changeset
261 if (!is_number (g))
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
262 error_msg = _(E_invalid_group);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
263 else
2272
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
264 {
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
265 /* FIXME: don't use atoi! */
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
266 *gid = atoi (g);
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
267 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
268 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
269 else
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
270 *gid = grp->gr_gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
271 endgrent (); /* Save a file descriptor. */
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
272
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
273 if (error_msg == NULL)
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
274 V_STRDUP (groupname, g);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
275 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
276
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
277 if (error_msg == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
278 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
279 if (u != NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
280 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
281 *username_arg = strdup (u);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
282 if (*username_arg == NULL)
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
283 error_msg = _(E_no_memory);
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
284 }
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
285
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
286 if (groupname != NULL && error_msg == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
287 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
288 *groupname_arg = strdup (groupname);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
289 if (*groupname_arg == NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
290 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
291 if (*username_arg != NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
292 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
293 free (*username_arg);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
294 *username_arg = NULL;
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
295 }
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
296 error_msg = _(E_no_memory);
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
297 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
298 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
299 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
300
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
301 if (error_msg && maybe_retry)
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
302 {
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
303 maybe_retry = 0;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
304 separator = dot;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
305 error_msg = NULL;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
306 goto retry;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
307 }
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
308
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
309 return error_msg;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
310 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
311
232
0d9395790eb7 (main): Change #ifdef TESTING to TEST.
Jim Meyering <jim@meyering.net>
parents: 231
diff changeset
312 #ifdef TEST
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
313
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
314 # define NULL_CHECK(s) ((s) == NULL ? "(null)" : (s))
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
315
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
316 int
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
317 main (int argc, char **argv)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
318 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
319 int i;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
320
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
321 for (i = 1; i < argc; i++)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
322 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
323 const char *e;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
324 char *username, *groupname;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
325 uid_t uid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
326 gid_t gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
327 char *tmp;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
328
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
329 tmp = strdup (argv[i]);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
330 e = parse_user_spec (tmp, &uid, &gid, &username, &groupname);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
331 free (tmp);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
332 printf ("%s: %u %u %s %s %s\n",
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
333 argv[i],
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
334 (unsigned int) uid,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
335 (unsigned int) gid,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
336 NULL_CHECK (username),
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
337 NULL_CHECK (groupname),
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
338 NULL_CHECK (e));
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
339 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
340
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
341 exit (0);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
342 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
343
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
344 #endif