blob: 283ac78fc12c1eeba87b52781dd1b49c4f427776 [file] [log] [blame]
Denis Vlasenkocb04ff52006-12-30 21:11:57 +00001/*
2Copyright (c) 2001-2006, Gerrit Pape
3All rights reserved.
4
5Redistribution and use in source and binary forms, with or without
6modification, are permitted provided that the following conditions are met:
7
8 1. Redistributions of source code must retain the above copyright notice,
9 this list of conditions and the following disclaimer.
10 2. Redistributions in binary form must reproduce the above copyright
11 notice, this list of conditions and the following disclaimer in the
12 documentation and/or other materials provided with the distribution.
13 3. The name of the author may not be used to endorse or promote products
14 derived from this software without specific prior written permission.
15
Denys Vlasenko95f79532017-08-02 14:26:33 +020016THIS SOFTWARE IS PROVIDED BY THE AUTHOR ''AS IS'' AND ANY EXPRESS OR IMPLIED
Denis Vlasenkocb04ff52006-12-30 21:11:57 +000017WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*/
27
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000028#include "libbb.h"
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000029
Denis Vlasenkocce38582007-02-26 22:47:42 +000030/* Always sets uid and gid */
Denys Vlasenko526d8582015-10-19 04:27:17 +020031int FAST_FUNC get_uidgid(struct bb_uidgid_t *u, const char *ug)
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000032{
33 struct passwd *pwd;
34 struct group *gr;
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000035 char *user, *group;
36 unsigned n;
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000037
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000038 user = (char*)ug;
39 group = strchr(ug, ':');
40 if (group) {
41 int sz = (++group) - ug;
42 user = alloca(sz);
43 /* copies sz-1 bytes, stores terminating '\0' */
44 safe_strncpy(user, ug, sz);
45 }
Denys Vlasenko526d8582015-10-19 04:27:17 +020046 n = bb_strtou(user, NULL, 10);
47 if (!errno) {
48 u->uid = n;
49 pwd = getpwuid(n);
50 /* If we have e.g. "500" string without user */
51 /* with uid 500 in /etc/passwd, we set gid == uid */
52 u->gid = pwd ? pwd->pw_gid : n;
53 goto skip;
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000054 }
Denys Vlasenko526d8582015-10-19 04:27:17 +020055 /* it is not numeric */
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000056 pwd = getpwnam(user);
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000057 if (!pwd)
58 return 0;
59 u->uid = pwd->pw_uid;
60 u->gid = pwd->pw_gid;
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000061
62 skip:
63 if (group) {
Denys Vlasenko526d8582015-10-19 04:27:17 +020064 n = bb_strtou(group, NULL, 10);
65 if (!errno) {
66 u->gid = n;
67 return 1;
Denis Vlasenko9a44c4f2006-12-28 05:44:47 +000068 }
69 gr = getgrnam(group);
Denys Vlasenko4566e172011-05-16 00:01:08 +020070 if (!gr)
71 return 0;
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000072 u->gid = gr->gr_gid;
73 }
74 return 1;
75}
Bernhard Reutner-Fischerd73cbd32008-07-21 14:41:33 +000076void FAST_FUNC xget_uidgid(struct bb_uidgid_t *u, const char *ug)
77{
Denys Vlasenko526d8582015-10-19 04:27:17 +020078 if (!get_uidgid(u, ug))
Denis Vlasenkob74a2db2008-07-21 21:34:51 +000079 bb_error_msg_and_die("unknown user/group %s", ug);
Bernhard Reutner-Fischerd73cbd32008-07-21 14:41:33 +000080}
Denis Vlasenkode59c0f2006-10-05 22:50:22 +000081
Denis Vlasenkocce38582007-02-26 22:47:42 +000082/* chown-like:
83 * "user" sets uid only,
84 * ":group" sets gid only
85 * "user:" sets uid and gid (to user's primary group id)
86 * "user:group" sets uid and gid
Denis Vlasenko8d89bed2008-09-07 23:22:08 +000087 * ('unset' uid or gid retains the value it has on entry)
Denis Vlasenkocce38582007-02-26 22:47:42 +000088 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000089void FAST_FUNC parse_chown_usergroup_or_die(struct bb_uidgid_t *u, char *user_group)
Denis Vlasenkocce38582007-02-26 22:47:42 +000090{
91 char *group;
92
Denys Vlasenko3d0805e2015-10-19 04:37:19 +020093 u->uid = u->gid = (gid_t)-1L;
94
Denis Vlasenkocce38582007-02-26 22:47:42 +000095 /* Check if there is a group name */
96 group = strchr(user_group, '.'); /* deprecated? */
97 if (!group)
98 group = strchr(user_group, ':');
99 else
100 *group = ':'; /* replace '.' with ':' */
101
102 /* Parse "user[:[group]]" */
103 if (!group) { /* "user" */
104 u->uid = get_ug_id(user_group, xuname2uid);
105 } else if (group == user_group) { /* ":group" */
106 u->gid = get_ug_id(group + 1, xgroup2gid);
107 } else {
108 if (!group[1]) /* "user:" */
109 *group = '\0';
Bernhard Reutner-Fischerd73cbd32008-07-21 14:41:33 +0000110 xget_uidgid(u, user_group);
Denis Vlasenkocce38582007-02-26 22:47:42 +0000111 }
112}
113
Denis Vlasenkode59c0f2006-10-05 22:50:22 +0000114#if 0
115#include <stdio.h>
116int main()
117{
118 unsigned u;
119 struct bb_uidgid_t ug;
Denys Vlasenko526d8582015-10-19 04:27:17 +0200120 u = get_uidgid(&ug, "apache");
Denis Vlasenkode59c0f2006-10-05 22:50:22 +0000121 printf("%u = %u:%u\n", u, ug.uid, ug.gid);
122 ug.uid = ug.gid = 1111;
Denys Vlasenko526d8582015-10-19 04:27:17 +0200123 u = get_uidgid(&ug, "apache");
Denis Vlasenkode59c0f2006-10-05 22:50:22 +0000124 printf("%u = %u:%u\n", u, ug.uid, ug.gid);
125 ug.uid = ug.gid = 1111;
Denys Vlasenko526d8582015-10-19 04:27:17 +0200126 u = get_uidgid(&ug, "apache:users");
Denis Vlasenkode59c0f2006-10-05 22:50:22 +0000127 printf("%u = %u:%u\n", u, ug.uid, ug.gid);
128 ug.uid = ug.gid = 1111;
Denys Vlasenko526d8582015-10-19 04:27:17 +0200129 u = get_uidgid(&ug, "apache:users");
Denis Vlasenkode59c0f2006-10-05 22:50:22 +0000130 printf("%u = %u:%u\n", u, ug.uid, ug.gid);
131 return 0;
132}
133#endif