annotate lib/userspec.c @ 2810:aff553209434

(same_name): Invoke xalloc_die instead of printing our own message. (parse_user_spec): Don't translate a message until just before returning, to avoid unnecessary translation.
author Jim Meyering <jim@meyering.net>
date Mon, 07 Aug 2000 16:56:29 +0000
parents ce1bca7b77d5
children 658f210cdd89
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
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
60 #include "xalloc.h"
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
61
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
62 #if ENABLE_NLS
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
63 # include <libintl.h>
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
64 # define _(Text) gettext (Text)
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
65 #else
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
66 # define _(Text) Text
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
67 #endif
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
68 #define N_(Text) Text
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
69
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
70 #ifndef _POSIX_VERSION
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
71 struct passwd *getpwnam ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
72 struct group *getgrnam ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
73 struct group *getgrgid ();
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
74 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
75
499
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
76 #ifndef HAVE_ENDGRENT
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
77 # define endgrent() ((void) 0)
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
78 #endif
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
79
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
80 #ifndef HAVE_ENDPWENT
62aa0b766a11 [HAVE_ENDGRENT]: Define away endgrent.
Jim Meyering <jim@meyering.net>
parents: 475
diff changeset
81 # define endpwent() ((void) 0)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
82 #endif
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
83
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
84 /* Perform the equivalent of the statement `dest = strdup (src);',
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
85 but obtaining storage via alloca instead of from the heap. */
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
86
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
87 #define V_STRDUP(dest, src) \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
88 do \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
89 { \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
90 int _len = strlen ((src)); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
91 (dest) = (char *) alloca (_len + 1); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
92 strcpy (dest, src); \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
93 } \
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
94 while (0)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
95
2271
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
96 /* ISDIGIT differs from isdigit, as follows:
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
97 - 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
98 - It's guaranteed to evaluate its argument exactly once.
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
99 - It's typically faster.
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
100 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
101 only '0' through '9' are digits. Prefer ISDIGIT to isdigit unless
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
102 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
103 host does not conform to Posix. */
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
104 #define ISDIGIT(c) ((unsigned) (c) - '0' <= 9)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
105
1347
e7dc6e5a3be5 Don't declare strdup if it's defined as a macro.
Jim Meyering <jim@meyering.net>
parents: 1278
diff changeset
106 #ifndef strdup
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
107 char *strdup ();
1347
e7dc6e5a3be5 Don't declare strdup if it's defined as a macro.
Jim Meyering <jim@meyering.net>
parents: 1278
diff changeset
108 #endif
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
109
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
110 /* Return nonzero if STR represents an unsigned decimal integer,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
111 otherwise return 0. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
112
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
113 static int
1557
38fd8f5d359d ansideclify
Jim Meyering <jim@meyering.net>
parents: 1347
diff changeset
114 is_number (const char *str)
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
115 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
116 for (; *str; str++)
2271
78563cb66dc8 (ISDIGIT): Define it.
Jim Meyering <jim@meyering.net>
parents: 1581
diff changeset
117 if (!ISDIGIT (*str))
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
118 return 0;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
119 return 1;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
120 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
121
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
122 /* Extract from NAME, which has the form "[user][:.][group]",
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
123 a USERNAME, UID U, GROUPNAME, and GID G.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
124 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
125 If the group is omitted but the ":" separator is given,
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
126 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
127 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
128 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
129 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
130 interpreting the `.' as a separator.
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
131
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
132 USERNAME and GROUPNAME will be in newly malloc'd memory.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
133 Either one might be NULL instead, indicating that it was not
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
134 given and the corresponding numeric ID was left unchanged.
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
135
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
136 Return NULL if successful, a static error message string if not. */
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
137
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
138 const char *
1580
12fde28eb4ee break long line
Jim Meyering <jim@meyering.net>
parents: 1557
diff changeset
139 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
140 char **username_arg, char **groupname_arg)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
141 {
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
142 static const char *E_invalid_user = N_("invalid user");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
143 static const char *E_invalid_group = N_("invalid group");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
144 static const char *E_bad_spec =
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
145 N_("cannot get the login group of a numeric UID");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
146 static const char *E_cannot_omit_both =
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
147 N_("cannot omit both user and group");
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
148
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
149 const char *error_msg;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
150 char *spec; /* A copy we can write on. */
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
151 struct passwd *pwd;
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
152 struct group *grp;
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
153 char *g, *u, *separator;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
154 char *groupname;
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
155 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
156 char *dot = NULL;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
157
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
158 error_msg = NULL;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
159 *username_arg = *groupname_arg = NULL;
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
160 groupname = NULL;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
161
231
a3b51f9ee779 Include alloca directives.
Jim Meyering <jim@meyering.net>
parents: 230
diff changeset
162 V_STRDUP (spec, spec_arg);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
163
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
164 /* 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
165 separator = strchr (spec, ':');
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
166
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
167 /* If there is no colon, then see if there's a `.'. */
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
168 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
169 {
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
170 dot = strchr (spec, '.');
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
171 /* 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
172 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
173 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
174 as a separator. */
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
175 /* 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
176 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
177 if (dot)
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
178 maybe_retry = 1;
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
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
181 retry:
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
182
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
183 /* Replace separator with a NUL. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
184 if (separator != NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
185 *separator = '\0';
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
186
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
187 /* Set U and G to non-zero length strings corresponding to user and
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
188 group specifiers or to NULL. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
189 u = (*spec == '\0' ? NULL : spec);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
190
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
191 g = (separator == NULL || *(separator + 1) == '\0'
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
192 ? NULL
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
193 : separator + 1);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
195 if (u == NULL && g == NULL)
2273
ad48a40e44bb <libintl.h>: Include.
Jim Meyering <jim@meyering.net>
parents: 2272
diff changeset
196 return _(E_cannot_omit_both);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
197
1260
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
198 #ifdef __DJGPP__
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
199 /* 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
200 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
201 if (u && !is_number (u))
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
202 setenv ("USER", u, 1);
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
203 if (g && !is_number (g))
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
204 setenv ("GROUP", g, 1);
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
205 #endif
7aff74949325 (parse_user_spec) [__DJGPP__]: Make function know
Jim Meyering <jim@meyering.net>
parents: 966
diff changeset
206
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
207 if (u != NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
208 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
209 pwd = getpwnam (u);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
210 if (pwd == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
211 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
212
475
7e3e0dd559c7 (isnumber): Rename to is_number.
Jim Meyering <jim@meyering.net>
parents: 432
diff changeset
213 if (!is_number (u))
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
214 error_msg = E_invalid_user;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
215 else
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
216 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
217 int use_login_group;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
218 use_login_group = (separator != NULL && g == NULL);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
219 if (use_login_group)
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
220 error_msg = E_bad_spec;
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
221 else
2272
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
222 {
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
223 /* FIXME: don't use atoi! */
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
224 *uid = atoi (u);
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
225 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
226 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
227 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
228 else
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
229 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
230 *uid = pwd->pw_uid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
231 if (g == NULL && separator != NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
232 {
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
233 /* A separator was given, but a group was not specified,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
234 so get the login group. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
235 *gid = pwd->pw_gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
236 grp = getgrgid (pwd->pw_gid);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
237 if (grp == NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
238 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
239 /* This is enough room to hold the unsigned decimal
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
240 representation of any 32-bit quantity and the trailing
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
241 zero byte. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
242 char uint_buf[21];
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
243 sprintf (uint_buf, "%u", (unsigned) (pwd->pw_gid));
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
244 V_STRDUP (groupname, uint_buf);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
245 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
246 else
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
247 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
248 V_STRDUP (groupname, grp->gr_name);
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
249 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
250 endgrent ();
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
251 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
252 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
253 endpwent ();
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
254 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
255
195
Jim Meyering <jim@meyering.net>
parents: 194
diff changeset
256 if (g != NULL && error_msg == NULL)
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
257 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
258 /* Explicit group. */
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
259 grp = getgrnam (g);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
260 if (grp == NULL)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
261 {
475
7e3e0dd559c7 (isnumber): Rename to is_number.
Jim Meyering <jim@meyering.net>
parents: 432
diff changeset
262 if (!is_number (g))
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
263 error_msg = E_invalid_group;
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
264 else
2272
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
265 {
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
266 /* FIXME: don't use atoi! */
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
267 *gid = atoi (g);
9d6da67904b5 add FIXME comments for use of atoi
Jim Meyering <jim@meyering.net>
parents: 2271
diff changeset
268 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
269 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
270 else
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
271 *gid = grp->gr_gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
272 endgrent (); /* Save a file descriptor. */
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
273
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
274 if (error_msg == NULL)
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
275 V_STRDUP (groupname, g);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
276 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
277
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
278 if (error_msg == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
279 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
280 if (u != NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
281 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
282 *username_arg = strdup (u);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
283 if (*username_arg == NULL)
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
284 error_msg = xalloc_msg_memory_exhausted;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
285 }
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
286
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
287 if (groupname != NULL && error_msg == NULL)
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
288 {
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
289 *groupname_arg = strdup (groupname);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
290 if (*groupname_arg == NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
291 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
292 if (*username_arg != NULL)
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
293 {
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
294 free (*username_arg);
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
295 *username_arg = NULL;
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
296 }
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
297 error_msg = xalloc_msg_memory_exhausted;
230
372133c396e2 Rewrite.
Jim Meyering <jim@meyering.net>
parents: 195
diff changeset
298 }
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
299 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
300 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
301
2284
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
302 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
303 {
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
304 maybe_retry = 0;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
305 separator = dot;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
306 error_msg = NULL;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
307 goto retry;
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
308 }
ce1bca7b77d5 (parse_user_spec): If there is no `:' but there is a `.',
Jim Meyering <jim@meyering.net>
parents: 2273
diff changeset
309
2810
aff553209434 (same_name): Invoke xalloc_die instead of printing our own message.
Jim Meyering <jim@meyering.net>
parents: 2284
diff changeset
310 return _(error_msg);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
311 }
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
312
232
0d9395790eb7 (main): Change #ifdef TESTING to TEST.
Jim Meyering <jim@meyering.net>
parents: 231
diff changeset
313 #ifdef TEST
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
314
966
2db1c866ccce Change some #ifdef to #if.
Jim Meyering <jim@meyering.net>
parents: 650
diff changeset
315 # define NULL_CHECK(s) ((s) == NULL ? "(null)" : (s))
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
316
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
317 int
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
318 main (int argc, char **argv)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
319 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
320 int i;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
321
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
322 for (i = 1; i < argc; i++)
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
323 {
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
324 const char *e;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
325 char *username, *groupname;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
326 uid_t uid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
327 gid_t gid;
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
328 char *tmp;
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
329
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
330 tmp = strdup (argv[i]);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
331 e = parse_user_spec (tmp, &uid, &gid, &username, &groupname);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
332 free (tmp);
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
333 printf ("%s: %u %u %s %s %s\n",
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
334 argv[i],
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
335 (unsigned int) uid,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
336 (unsigned int) gid,
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
337 NULL_CHECK (username),
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
338 NULL_CHECK (groupname),
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
339 NULL_CHECK (e));
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
340 }
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
341
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
342 exit (0);
5
41c9d08b09d7 Initial revision
Jim Meyering <jim@meyering.net>
parents:
diff changeset
343 }
194
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
344
Jim Meyering <jim@meyering.net>
parents: 168
diff changeset
345 #endif