2013-10-07 12:03:26 -04:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
2015-02-14 15:02:41 -05:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
2013-06-09 09:20:55 -04:00
|
|
|
#include <errno.h>
|
|
|
|
#include <grp.h>
|
2014-11-13 13:54:28 -05:00
|
|
|
#include <unistd.h>
|
2014-11-13 12:29:30 -05:00
|
|
|
|
2015-03-12 19:25:32 -04:00
|
|
|
#include "fs.h"
|
2013-06-09 09:20:55 -04:00
|
|
|
#include "util.h"
|
|
|
|
|
2013-10-07 19:45:25 -04:00
|
|
|
static struct stat st;
|
2015-03-08 19:41:53 -04:00
|
|
|
static int hflag = 0;
|
|
|
|
static int Rflag = 0;
|
|
|
|
static gid_t gid = -1;
|
|
|
|
static int ret = 0;
|
2013-06-09 09:20:55 -04:00
|
|
|
|
2013-10-07 19:45:25 -04:00
|
|
|
static void
|
2015-03-12 19:25:32 -04:00
|
|
|
chgrp(const char *path, void *data, struct recursor *r)
|
2013-06-09 09:20:55 -04:00
|
|
|
{
|
2015-03-08 18:31:59 -04:00
|
|
|
char *chownf_name;
|
|
|
|
int (*chownf)(const char *, uid_t, gid_t);
|
|
|
|
|
2015-03-12 19:25:32 -04:00
|
|
|
if (r->follow == 'P' || (r->follow == 'H' && r->depth) || (hflag && !(r->depth))) {
|
2015-03-08 18:31:59 -04:00
|
|
|
chownf_name = "lchown";
|
|
|
|
chownf = lchown;
|
|
|
|
} else {
|
|
|
|
chownf_name = "chown";
|
|
|
|
chownf = chown;
|
|
|
|
}
|
|
|
|
|
2015-03-02 15:43:56 -05:00
|
|
|
if (chownf(path, st.st_uid, gid) < 0) {
|
|
|
|
weprintf("%s %s:", chownf_name, path);
|
2015-03-08 18:31:59 -04:00
|
|
|
ret = 1;
|
|
|
|
} else if (Rflag) {
|
2015-03-12 19:25:32 -04:00
|
|
|
recurse(path, NULL, r);
|
2015-03-08 18:31:59 -04:00
|
|
|
}
|
2013-06-09 09:20:55 -04:00
|
|
|
}
|
|
|
|
|
2015-02-12 15:56:06 -05:00
|
|
|
static void
|
|
|
|
usage(void)
|
|
|
|
{
|
2015-02-16 11:30:43 -05:00
|
|
|
eprintf("usage: chgrp [-h] [-R [-H | -L | -P]] group file ...\n");
|
2015-02-12 15:56:06 -05:00
|
|
|
}
|
|
|
|
|
2013-06-09 09:20:55 -04:00
|
|
|
int
|
2014-04-18 06:51:18 -04:00
|
|
|
main(int argc, char *argv[])
|
2013-06-09 09:20:55 -04:00
|
|
|
{
|
|
|
|
struct group *gr;
|
2015-03-12 19:25:32 -04:00
|
|
|
struct recursor r = { .fn = chgrp, .hist = NULL, .depth = 0, .follow = 'P', .flags = 0 };
|
2013-06-09 09:20:55 -04:00
|
|
|
|
|
|
|
ARGBEGIN {
|
2015-02-12 15:56:06 -05:00
|
|
|
case 'h':
|
2015-03-08 18:31:59 -04:00
|
|
|
hflag = 1;
|
2015-02-12 15:56:06 -05:00
|
|
|
break;
|
2013-06-09 09:20:55 -04:00
|
|
|
case 'R':
|
2015-02-16 11:30:43 -05:00
|
|
|
Rflag = 1;
|
2013-06-09 09:20:55 -04:00
|
|
|
break;
|
2015-02-09 14:53:24 -05:00
|
|
|
case 'H':
|
|
|
|
case 'L':
|
|
|
|
case 'P':
|
2015-03-12 19:25:32 -04:00
|
|
|
r.follow = ARGC();
|
2015-02-09 14:53:24 -05:00
|
|
|
break;
|
2013-06-09 09:20:55 -04:00
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
} ARGEND;
|
2013-10-07 19:45:25 -04:00
|
|
|
|
2015-03-02 15:43:56 -05:00
|
|
|
if (argc < 2)
|
2013-06-09 09:20:55 -04:00
|
|
|
usage();
|
|
|
|
|
2013-10-07 19:45:25 -04:00
|
|
|
errno = 0;
|
2015-03-08 18:31:59 -04:00
|
|
|
if (!(gr = getgrnam(argv[0]))) {
|
2014-12-21 07:12:38 -05:00
|
|
|
if (errno)
|
|
|
|
eprintf("getgrnam %s:", argv[0]);
|
|
|
|
else
|
|
|
|
eprintf("getgrnam %s: no such group\n", argv[0]);
|
|
|
|
}
|
2013-06-09 09:20:55 -04:00
|
|
|
gid = gr->gr_gid;
|
|
|
|
|
2015-03-12 19:25:32 -04:00
|
|
|
for (argc--, argv++; *argv; argc--, argv++) {
|
2014-11-19 14:59:37 -05:00
|
|
|
if (stat(*argv, &st) < 0) {
|
2014-11-17 11:27:12 -05:00
|
|
|
weprintf("stat %s:", *argv);
|
2015-03-08 18:31:59 -04:00
|
|
|
ret = 1;
|
2013-06-09 09:20:55 -04:00
|
|
|
continue;
|
|
|
|
}
|
2015-03-12 19:25:32 -04:00
|
|
|
chgrp(*argv, NULL, &r);
|
2013-06-09 09:20:55 -04:00
|
|
|
}
|
2015-03-08 18:31:59 -04:00
|
|
|
|
2015-03-12 19:25:32 -04:00
|
|
|
return ret || recurse_status;
|
2013-06-09 09:20:55 -04:00
|
|
|
}
|