mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-24 02:12:09 -05:00
fc6d051dfd
The LexicalPath instance methods dirname(), basename(), title() and extension() will be changed to return StringView const& in a further commit. Due to this, users creating temporary LexicalPath objects just to call one of those getters will recieve a StringView const& pointing to a possible freed buffer. To avoid this, static methods for those APIs have been added, which will return a String by value to avoid those problems. All cases where temporary LexicalPath objects have been used as described above haven been changed to use the static APIs.
60 lines
2.1 KiB
C++
60 lines
2.1 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 <stdio.h>
|
|
#include <unistd.h>
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
if (pledge("stdio rpath wpath cpath fattr", nullptr) < 0) {
|
|
perror("pledge");
|
|
return 1;
|
|
}
|
|
|
|
bool link = false;
|
|
bool recursion_allowed = false;
|
|
bool verbose = false;
|
|
Vector<String> sources;
|
|
String destination;
|
|
|
|
Core::ArgsParser args_parser;
|
|
args_parser.add_option(link, "Link files instead of copying", "link", 'l');
|
|
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(argc, argv);
|
|
|
|
bool destination_is_existing_dir = Core::File::is_directory(destination);
|
|
|
|
for (auto& source : sources) {
|
|
auto destination_path = destination_is_existing_dir
|
|
? String::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);
|
|
|
|
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, result.error().error_code);
|
|
return 1;
|
|
}
|
|
|
|
if (verbose)
|
|
outln("'{}' -> '{}'", source, destination_path);
|
|
}
|
|
return 0;
|
|
}
|