blob: b213074a65c731211f2c2397632e9796fca39ee8 [file] [log] [blame]
Robert Griebl1fca5582002-06-04 20:45:46 +00001/* vi: set sw=4 ts=4: */
2/*
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +00003 * addgroup - add groups to /etc/group and /etc/gshadow
Robert Griebl1fca5582002-06-04 20:45:46 +00004 *
5 * Copyright (C) 1999 by Lineo, inc. and John Beppu
6 * Copyright (C) 1999,2000,2001 by John Beppu <beppu@codepoet.org>
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +00007 * Copyright (C) 2007 by Tito Ragusa <farmatito@tiscali.it>
Robert Griebl1fca5582002-06-04 20:45:46 +00008 *
Rob Landley9a2dd512006-04-04 19:19:53 +00009 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
Robert Griebl1fca5582002-06-04 20:45:46 +000010 *
11 */
12
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000013#include "libbb.h"
Robert Griebl1fca5582002-06-04 20:45:46 +000014
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000015static void xgroup_study(struct group *g)
Robert Griebl1fca5582002-06-04 20:45:46 +000016{
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000017 /* Make sure gr_name is unused */
18 if (getgrnam(g->gr_name)) {
19 goto error;
Robert Griebl1fca5582002-06-04 20:45:46 +000020 }
Robert Griebl1fca5582002-06-04 20:45:46 +000021
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000022 /* Check if the desired gid is free
23 * or find the first free one */
24 while (1) {
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000025 if (!getgrgid(g->gr_gid)) {
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000026 return; /* found free group: return */
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000027 }
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000028 if (option_mask32) {
29 /* -g N, cannot pick gid other than N: error */
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000030 g->gr_name = itoa(g->gr_gid);
31 goto error;
32 }
33 g->gr_gid++;
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000034 if (g->gr_gid <= 0) {
35 /* overflowed: error */
36 bb_error_msg_and_die("no gids left");
37 }
38 }
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000039
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000040 error:
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000041 /* exit */
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000042 bb_error_msg_and_die("group %s already exists", g->gr_name);
Robert Griebl1fca5582002-06-04 20:45:46 +000043}
44
45/* append a new user to the passwd file */
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000046static void new_group(char *group, gid_t gid)
Robert Griebl1fca5582002-06-04 20:45:46 +000047{
Rob Landley9a2dd512006-04-04 19:19:53 +000048 FILE *file;
Robert Griebl1fca5582002-06-04 20:45:46 +000049 struct group gr;
50
Robert Griebl1fca5582002-06-04 20:45:46 +000051 /* make sure gid and group haven't already been allocated */
52 gr.gr_gid = gid;
53 gr.gr_name = group;
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000054 xgroup_study(&gr);
Robert Griebl1fca5582002-06-04 20:45:46 +000055
56 /* add entry to group */
Rob Landleyd921b2e2006-08-03 15:41:12 +000057 file = xfopen(bb_path_group_file, "a");
Rob Landley9a2dd512006-04-04 19:19:53 +000058 /* group:passwd:gid:userlist */
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000059 fprintf(file, "%s:x:%d:\n", group, gr.gr_gid);
Denis Vlasenko5df955f2007-03-13 13:01:14 +000060 if (ENABLE_FEATURE_CLEAN_UP)
61 fclose(file);
Rob Landley9a2dd512006-04-04 19:19:53 +000062#if ENABLE_FEATURE_SHADOWPASSWDS
Denis Vlasenko5df955f2007-03-13 13:01:14 +000063 file = fopen_or_warn(bb_path_gshadow_file, "a");
64 if (file) {
65 fprintf(file, "%s:!::\n", group);
66 if (ENABLE_FEATURE_CLEAN_UP)
67 fclose(file);
68 }
Robert Griebl1fca5582002-06-04 20:45:46 +000069#endif
Robert Griebl1fca5582002-06-04 20:45:46 +000070}
71
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000072#if ENABLE_FEATURE_ADDUSER_TO_GROUP
73static void add_user_to_group(char **args,
74 const char *path,
75 FILE *(*fopen_func)(const char *fileName, const char *mode))
76{
77 char *line;
78 int len = strlen(args[1]);
79 llist_t *plist = NULL;
80 FILE *group_file;
81
82 group_file = fopen_func(path, "r");
83
84 if (!group_file) return;
85
86 while ((line = xmalloc_getline(group_file))) {
87 /* Find the group */
88 if (!strncmp(line, args[1], len)
Denis Vlasenkof1f1b692007-07-30 12:32:37 +000089 && line[len] == ':'
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +000090 ) {
91 /* Add the new user */
92 line = xasprintf("%s%s%s", line,
93 last_char_is(line, ':') ? "" : ",",
94 args[0]);
95 }
96 llist_add_to_end(&plist, line);
97 }
98
99 if (ENABLE_FEATURE_CLEAN_UP) {
100 fclose(group_file);
101 group_file = fopen_func(path, "w");
102 while ((line = llist_pop(&plist))) {
103 if (group_file)
104 fprintf(group_file, "%s\n", line);
105 free(line);
106 }
107 if (group_file)
108 fclose(group_file);
109 } else {
110 group_file = fopen_func(path, "w");
111 if (group_file)
112 while ((line = llist_pop(&plist)))
113 fprintf(group_file, "%s\n", line);
114 }
115}
116#endif
117
Robert Griebl1fca5582002-06-04 20:45:46 +0000118/*
119 * addgroup will take a login_name as its first parameter.
120 *
Denis Vlasenko5df955f2007-03-13 13:01:14 +0000121 * gid can be customized via command-line parameters.
Denis Vlasenkof1f1b692007-07-30 12:32:37 +0000122 * If called with two non-option arguments, addgroup
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +0000123 * will add an existing user to an existing group.
Denis Vlasenko5df955f2007-03-13 13:01:14 +0000124 */
Denis Vlasenko06af2162007-02-03 17:28:39 +0000125int addgroup_main(int argc, char **argv);
Robert Griebl1fca5582002-06-04 20:45:46 +0000126int addgroup_main(int argc, char **argv)
127{
Robert Griebl1fca5582002-06-04 20:45:46 +0000128 char *group;
129 gid_t gid = 0;
Denis Vlasenko9213a9e2006-09-17 16:28:10 +0000130
Rob Landley9a2dd512006-04-04 19:19:53 +0000131 /* need to be root */
Denis Vlasenko13858992006-10-08 12:49:22 +0000132 if (geteuid()) {
Rob Landley9a2dd512006-04-04 19:19:53 +0000133 bb_error_msg_and_die(bb_msg_perm_denied_are_you_root);
Eric Andersen09eb0002002-11-14 11:10:14 +0000134 }
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000135
Denis Vlasenkof1f1b692007-07-30 12:32:37 +0000136 /* Syntax:
137 * addgroup group
138 * addgroup -g num group
139 * addgroup user group
140 * Check for min, max and missing args */
141 opt_complementary = "-1:?2";
Denis Vlasenkofe7cd642007-08-18 15:32:12 +0000142 if (getopt32(argv, "g:", &group)) {
Denis Vlasenkof1f1b692007-07-30 12:32:37 +0000143 gid = xatoul_range(group, 0, ((unsigned long)(gid_t)ULONG_MAX) >> 1);
144 }
145 /* move past the commandline options */
146 argv += optind;
147 argc -= optind;
148
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +0000149#if ENABLE_FEATURE_ADDUSER_TO_GROUP
150 if (argc == 2) {
151 struct group *gr;
Denis Vlasenko4b924f32007-05-30 00:29:55 +0000152
Denis Vlasenkof1f1b692007-07-30 12:32:37 +0000153 if (option_mask32) {
154 /* -g was there, but "addgroup -g num user group"
155 * is a no-no */
156 bb_show_usage();
157 }
158
Bernhard Reutner-Fischer00c2c482007-04-05 13:16:39 +0000159 /* check if group and user exist */
160 xuname2uid(argv[0]); /* unknown user: exit */
161 xgroup2gid(argv[1]); /* unknown group: exit */
162 /* check if user is already in this group */
163 gr = getgrnam(argv[1]);
164 for (; *(gr->gr_mem) != NULL; (gr->gr_mem)++) {
165 if (!strcmp(argv[0], *(gr->gr_mem))) {
166 /* user is already in group: do nothing */
167 return EXIT_SUCCESS;
168 }
169 }
170 add_user_to_group(argv, bb_path_group_file, xfopen);
171#if ENABLE_FEATURE_SHADOWPASSWDS
172 add_user_to_group(argv, bb_path_gshadow_file, fopen_or_warn);
173#endif /* ENABLE_FEATURE_SHADOWPASSWDS */
174 } else
175#endif /* ENABLE_FEATURE_ADDUSER_TO_GROUP */
176 new_group(argv[0], gid);
177
178 /* Reached only on success */
179 return EXIT_SUCCESS;
Robert Griebl1fca5582002-06-04 20:45:46 +0000180}