/* chgrp.c - Change user and group ownership
*
* Copyright 2012 Georgi Chorbadzhiyski <georgi@unixsol.org>
*
* See http://opengroup.org/onlinepubs/9699919799/utilities/chown.html
* See http://opengroup.org/onlinepubs/9699919799/utilities/chgrp.html
USE_CHGRP(NEWTOY(chgrp, "<2hPLHRfv[-HLP]", TOYFLAG_BIN))
USE_CHOWN(OLDTOY(chown, chgrp, TOYFLAG_BIN))
config CHGRP
bool "chgrp"
default y
help
usage: chgrp/chown [-RHLP] [-fvh] group file...
Change group of one or more files.
-f suppress most error messages.
-h change symlinks instead of what they point to
-R recurse into subdirectories (implies -h).
-H with -R change target of symlink, follow command line symlinks
-L with -R change target of symlink, follow all symlinks
-P with -R change symlink, do not follow symlinks (default)
-v verbose output.
config CHOWN
bool "chown"
default y
help
see: chgrp
*/
#define FOR_chgrp
#define FORCE_FLAGS
#include "toys.h"
GLOBALS(
uid_t owner;
gid_t group;
char *owner_name, *group_name;
int symfollow;
)
static int do_chgrp(struct dirtree *node)
{
int fd, ret, flags = toys.optflags;
// Depth first search
if (!dirtree_notdotdot(node)) return 0;
if ((flags & FLAG_R) && !node->again && S_ISDIR(node->st.st_mode))
return DIRTREE_COMEAGAIN|(DIRTREE_SYMFOLLOW*!!(flags&FLAG_L));
fd = dirtree_parentfd(node);
ret = fchownat(fd, node->name, TT.owner, TT.group,
AT_SYMLINK_NOFOLLOW*(!(flags&(FLAG_L|FLAG_H)) && (flags&(FLAG_h|FLAG_R))));
if (ret || (flags & FLAG_v)) {
char *path = dirtree_path(node, 0);
if (flags & FLAG_v)
xprintf("%s %s%s%s %s\n", toys.which->name,
TT.owner_name ? TT.owner_name : "",
toys.which->name[2]=='o' && TT.group_name ? ":" : "",
TT.group_name ? TT.group_name : "", path);
if (ret == -1 && !(toys.optflags & FLAG_f))
perror_msg("'%s' to '%s:%s'", path, TT.owner_name, TT.group_name);
free(path);
}
toys.exitval |= ret;
return 0;
}
void chgrp_main(void)
{
int ischown = toys.which->name[2] == 'o';
char **s, *own;
TT.owner = TT.group = -1;
// Distinguish chown from chgrp
if (ischown) {
char *grp;
own = xstrdup(*toys.optargs);
if ((grp = strchr(own, ':')) || (grp = strchr(own, '.'))) {
*(grp++) = 0;
TT.group_name = grp;
}
if (*own) TT.owner = xgetpwnamid(TT.owner_name = own)->pw_uid;
} else TT.group_name = *toys.optargs;
if (TT.group_name && *TT.group_name)
TT.group = xgetgrnamid(TT.group_name)->gr_gid;
for (s=toys.optargs+1; *s; s++)
dirtree_handle_callback(dirtree_start(*s, toys.optflags&(FLAG_H|FLAG_L)),
do_chgrp);;
if (CFG_TOYBOX_FREE && ischown) free(own);
}
void chown_main()
{
chgrp_main();
}