Upates to include copyright 2000 to everything
[platform/upstream/busybox.git] / chmod_chown_chgrp.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Mini chown/chmod/chgrp implementation for busybox
4  *
5  *
6  * Copyright (C) 1999,2000 by Lineo, inc.
7  * Written by Erik Andersen <andersen@lineo.com>, <andersee@debian.org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17  * General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22  *
23  */
24
25 #include "internal.h"
26 #define BB_DECLARE_EXTERN
27 #define bb_need_invalid_option
28 #include "messages.c"
29
30 #include <stdio.h>
31 #include <grp.h>
32 #include <pwd.h>
33
34
35 static uid_t uid = -1;
36 static gid_t gid = -1;
37 static int whichApp;
38 static char *invocationName = NULL;
39 static char *theMode = NULL;
40
41
42 #define CHGRP_APP   1
43 #define CHOWN_APP   2
44 #define CHMOD_APP   3
45
46 static const char chgrp_usage[] = "chgrp [OPTION]... GROUP FILE...\n\n"
47         "Change the group membership of each FILE to GROUP.\n"
48
49         "\nOptions:\n\t-R\tchange files and directories recursively\n";
50 static const char chown_usage[] =
51         "chown [OPTION]...  OWNER[.[GROUP] FILE...\n\n"
52         "Change the owner and/or group of each FILE to OWNER and/or GROUP.\n"
53
54         "\nOptions:\n\t-R\tchange files and directories recursively\n";
55 static const char chmod_usage[] =
56         "chmod [-R] MODE[,MODE]... FILE...\n\n"
57         "Each MODE is one or more of the letters ugoa, one of the symbols +-= and\n"
58
59         "one or more of the letters rwxst.\n\n"
60         "\nOptions:\n\t-R\tchange files and directories recursively.\n";
61
62
63 static int fileAction(const char *fileName, struct stat *statbuf, void* junk)
64 {
65         switch (whichApp) {
66         case CHGRP_APP:
67         case CHOWN_APP:
68 #if (__GLIBC__ >= 2) && (__GLIBC_MINOR__ >= 1)
69                 if (lchown
70                         (fileName, (whichApp == CHOWN_APP) ? uid : statbuf->st_uid,
71                          (gid == -1) ? statbuf->st_gid : gid) == 0)
72 #else
73                 if (chown
74                         (fileName, (whichApp == CHOWN_APP) ? uid : statbuf->st_uid,
75                          (gid == -1) ? statbuf->st_gid : gid) == 0)
76 #endif
77                 {
78                         return (TRUE);
79                 }
80                 break;
81         case CHMOD_APP:
82                 /* Parse the specified modes */
83                 if (parse_mode(theMode, &(statbuf->st_mode)) == FALSE) {
84                         fatalError( "%s: unknown mode: %s\n", invocationName, theMode);
85                 }
86                 if (chmod(fileName, statbuf->st_mode) == 0)
87                         return (TRUE);
88                 break;
89         }
90         perror(fileName);
91         return (FALSE);
92 }
93
94 int chmod_chown_chgrp_main(int argc, char **argv)
95 {
96         int recursiveFlag = FALSE;
97         char *groupName;
98         char *p;
99         const char *appUsage;
100
101         whichApp = (strcmp(*argv, "chown") == 0)? 
102                         CHOWN_APP : (strcmp(*argv, "chmod") == 0)? 
103                                 CHMOD_APP : CHGRP_APP;
104
105         appUsage = (whichApp == CHOWN_APP)? 
106                         chown_usage : (whichApp == CHMOD_APP) ? chmod_usage : chgrp_usage;
107
108         if (argc < 2)
109                 usage(appUsage);
110         invocationName = *argv;
111         argc--;
112         argv++;
113
114         /* Parse options */
115         while (**argv == '-') {
116                 while (*++(*argv))
117                         switch (**argv) {
118                         case 'R':
119                                 recursiveFlag = TRUE;
120                                 break;
121                         default:
122                                 fprintf(stderr, invalid_option, invocationName, **argv);
123                                 usage(appUsage);
124                         }
125                 argc--;
126                 argv++;
127         }
128
129         if (whichApp == CHMOD_APP) {
130                 theMode = *argv;
131         } else {
132
133                 /* Find the selected group */
134                 if (whichApp == CHGRP_APP) {
135                         groupName = *argv;
136                         gid = strtoul(groupName, &p, 10);       /* maybe it's already numeric */
137                         if (groupName == p)
138                                 gid = my_getgrnam(groupName);
139                         if (gid == -1)
140                                 goto bad_group;
141                 } else {
142                         groupName = strchr(*argv, '.');
143                         if (groupName) {
144                                 *groupName++ = '\0';
145                                 gid = strtoul(groupName, &p, 10);
146                                 if (groupName == p)
147                                         gid = my_getgrnam(groupName);
148                                 if (gid == -1)
149                                         goto bad_group;
150                         } else
151                                 gid = -1;
152                 }
153
154
155                 /* Find the selected user (if appropriate)  */
156                 if (whichApp == CHOWN_APP) {
157                         uid = strtoul(*argv, &p, 10);   /* if numeric ... */
158                         if (*argv == p)
159                                 uid = my_getpwnam(*argv);
160                         if (uid == -1) {
161                                 fatalError( "%s: unknown user name: %s\n", 
162                                                 invocationName, *argv);
163                         }
164                 }
165         }
166
167         /* Ok, ready to do the deed now */
168         if (argc <= 1) {
169                 fatalError( "%s: too few arguments\n", invocationName);
170         }
171         while (argc-- > 1) {
172                 if (recursiveAction (*(++argv), recursiveFlag, TRUE, FALSE, 
173                                         fileAction, fileAction, NULL) == FALSE)
174                         exit(FALSE);
175         }
176         exit(TRUE);
177
178   bad_group:
179         fatalError( "%s: unknown group name: %s\n", invocationName, groupName);
180 }
181
182 /*
183 Local Variables:
184 c-file-style: "linux"
185 c-basic-offset: 4
186 tab-width: 4
187 End:
188 */