diff options
author | Tim Schumacher <timschumi@gmx.de> | 2022-07-25 15:04:39 +0200 |
---|---|---|
committer | Linus Groh <mail@linusgroh.de> | 2022-07-25 23:52:46 +0100 |
commit | 25e6ad8d3e294b231658dbdb47ab156c05982fff (patch) | |
tree | 542bc4afcbca0515cd30163b06a9fe7583c3cea9 /Userland/Utilities/chown.cpp | |
parent | 7b7c9a21946fd9ffd5c0868b66e238b4b7f84c0c (diff) | |
download | serenity-25e6ad8d3e294b231658dbdb47ab156c05982fff.zip |
chown: Implement recursion
Diffstat (limited to 'Userland/Utilities/chown.cpp')
-rw-r--r-- | Userland/Utilities/chown.cpp | 31 |
1 files changed, 27 insertions, 4 deletions
diff --git a/Userland/Utilities/chown.cpp b/Userland/Utilities/chown.cpp index c0173d042e..a487a09e4c 100644 --- a/Userland/Utilities/chown.cpp +++ b/Userland/Utilities/chown.cpp @@ -7,6 +7,7 @@ #include <AK/String.h> #include <AK/Vector.h> #include <LibCore/ArgsParser.h> +#include <LibCore/DirIterator.h> #include <LibCore/System.h> #include <LibMain/Main.h> #include <grp.h> @@ -22,11 +23,15 @@ ErrorOr<int> serenity_main(Main::Arguments arguments) String spec; Vector<StringView> paths; - bool dont_follow_symlinks = false; + bool no_dereference = false; + bool recursive = false; + bool follow_symlinks = false; Core::ArgsParser args_parser; args_parser.set_general_help("Change the ownership of a file or directory."); - args_parser.add_option(dont_follow_symlinks, "Don't follow symlinks", "no-dereference", 'h'); + args_parser.add_option(no_dereference, "Don't follow symlinks", "no-dereference", 'h'); + args_parser.add_option(recursive, "Change file ownership recursively", "recursive", 'R'); + args_parser.add_option(follow_symlinks, "Follow symlinks while recursing into directories", nullptr, 'L'); args_parser.add_positional_argument(spec, "User and group IDs", "USER[:GROUP]"); args_parser.add_positional_argument(paths, "Paths to files", "PATH"); args_parser.parse(arguments); @@ -66,12 +71,30 @@ ErrorOr<int> serenity_main(Main::Arguments arguments) } } - for (auto path : paths) { - if (dont_follow_symlinks) { + Function<ErrorOr<void>(StringView)> update_path_owner = [&](StringView path) -> ErrorOr<void> { + auto stat = TRY(Core::System::lstat(path)); + + if (S_ISLNK(stat.st_mode) && !follow_symlinks && !paths.contains_slow(path)) + return {}; + + if (no_dereference) { TRY(Core::System::lchown(path, new_uid, new_gid)); } else { TRY(Core::System::chown(path, new_uid, new_gid)); } + + if (recursive && S_ISDIR(stat.st_mode)) { + Core::DirIterator it(path, Core::DirIterator::Flags::SkipParentAndBaseDir); + + while (it.has_next()) + TRY(update_path_owner(it.next_full_path())); + } + + return {}; + }; + + for (auto path : paths) { + TRY(update_path_owner(path)); } return 0; |