mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 22:02:44 +00:00 
			
		
		
		
	 6e19ab2bbc
			
		
	
	
		6e19ab2bbc
		
	
	
	
	
		
			
			We have a new, improved string type coming up in AK (OOM aware, no null state), and while it's going to use UTF-8, the name UTF8String is a mouthful - so let's free up the String name by renaming the existing class. Making the old one have an annoying name will hopefully also help with quick adoption :^)
		
			
				
	
	
		
			98 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			98 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/LexicalPath.h>
 | |
| #include <LibCore/ArgsParser.h>
 | |
| #include <LibCore/File.h>
 | |
| #include <LibCore/System.h>
 | |
| #include <LibMain/Main.h>
 | |
| #include <stdio.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| ErrorOr<int> serenity_main(Main::Arguments arguments)
 | |
| {
 | |
|     TRY(Core::System::pledge("stdio rpath wpath cpath fattr chown"));
 | |
| 
 | |
|     bool link = false;
 | |
|     auto preserve = Core::File::PreserveMode::Nothing;
 | |
|     bool recursion_allowed = false;
 | |
|     bool verbose = false;
 | |
|     Vector<StringView> sources;
 | |
|     DeprecatedString destination;
 | |
| 
 | |
|     Core::ArgsParser args_parser;
 | |
|     args_parser.add_option(link, "Link files instead of copying", "link", 'l');
 | |
|     args_parser.add_option({
 | |
|         Core::ArgsParser::OptionArgumentMode::Optional,
 | |
|         "Preserve a selection of mode, ownership and timestamps. Defaults to all three if the option is present but no list is given.",
 | |
|         "preserve",
 | |
|         'p',
 | |
|         "attributes",
 | |
|         [&preserve](char const* s) {
 | |
|             if (!s) {
 | |
|                 preserve = Core::File::PreserveMode::Permissions | Core::File::PreserveMode::Ownership | Core::File::PreserveMode::Timestamps;
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             bool values_ok = true;
 | |
| 
 | |
|             StringView { s, strlen(s) }.for_each_split_view(',', SplitBehavior::Nothing, [&](StringView value) {
 | |
|                 if (value == "mode"sv) {
 | |
|                     preserve |= Core::File::PreserveMode::Permissions;
 | |
|                 } else if (value == "ownership"sv) {
 | |
|                     preserve |= Core::File::PreserveMode::Ownership;
 | |
|                 } else if (value == "timestamps"sv) {
 | |
|                     preserve |= Core::File::PreserveMode::Timestamps;
 | |
|                 } else {
 | |
|                     warnln("cp: Unknown or unimplemented --preserve attribute: '{}'", value);
 | |
|                     values_ok = false;
 | |
|                 }
 | |
|             });
 | |
| 
 | |
|             return values_ok;
 | |
|         },
 | |
|         Core::ArgsParser::OptionHideMode::None,
 | |
|     });
 | |
|     args_parser.add_option(recursion_allowed, "Copy directories recursively", "recursive", 'R');
 | |
|     args_parser.add_option(recursion_allowed, "Same as -R", nullptr, 'r');
 | |
|     args_parser.add_option(verbose, "Verbose", "verbose", 'v');
 | |
|     args_parser.add_positional_argument(sources, "Source file paths", "source");
 | |
|     args_parser.add_positional_argument(destination, "Destination file path", "destination");
 | |
|     args_parser.parse(arguments);
 | |
| 
 | |
|     if (has_flag(preserve, Core::File::PreserveMode::Permissions)) {
 | |
|         umask(0);
 | |
|     } else {
 | |
|         TRY(Core::System::pledge("stdio rpath wpath cpath fattr"));
 | |
|     }
 | |
| 
 | |
|     bool destination_is_existing_dir = Core::File::is_directory(destination);
 | |
| 
 | |
|     for (auto& source : sources) {
 | |
|         auto destination_path = destination_is_existing_dir
 | |
|             ? DeprecatedString::formatted("{}/{}", destination, LexicalPath::basename(source))
 | |
|             : destination;
 | |
| 
 | |
|         auto result = Core::File::copy_file_or_directory(
 | |
|             destination_path, source,
 | |
|             recursion_allowed ? Core::File::RecursionMode::Allowed : Core::File::RecursionMode::Disallowed,
 | |
|             link ? Core::File::LinkMode::Allowed : Core::File::LinkMode::Disallowed,
 | |
|             Core::File::AddDuplicateFileMarker::No,
 | |
|             preserve);
 | |
| 
 | |
|         if (result.is_error()) {
 | |
|             if (result.error().tried_recursing)
 | |
|                 warnln("cp: -R not specified; omitting directory '{}'", source);
 | |
|             else
 | |
|                 warnln("cp: unable to copy '{}' to '{}': {}", source, destination_path, strerror(result.error().code()));
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         if (verbose)
 | |
|             outln("'{}' -> '{}'", source, destination_path);
 | |
|     }
 | |
|     return 0;
 | |
| }
 |