install.cpp 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859
  1. /*
  2. * Copyright (c) 2022, Tim Schumacher <timschumi@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/LexicalPath.h>
  7. #include <AK/Vector.h>
  8. #include <LibCore/ArgsParser.h>
  9. #include <LibCore/Directory.h>
  10. #include <LibCore/FilePermissionsMask.h>
  11. #include <LibCore/System.h>
  12. #include <LibFileSystem/FileSystem.h>
  13. #include <LibMain/Main.h>
  14. ErrorOr<int> serenity_main(Main::Arguments arguments)
  15. {
  16. TRY(Core::System::pledge("stdio rpath wpath cpath fattr"));
  17. bool create_leading_dest_components = false;
  18. StringView mode = "0755"sv;
  19. Vector<StringView> sources;
  20. StringView destination;
  21. Core::ArgsParser args_parser;
  22. args_parser.add_ignored(nullptr, 'c'); // "copy files" is the default, no contradicting options exist.
  23. args_parser.add_option(create_leading_dest_components, "Create leading components of the destination path", nullptr, 'D');
  24. args_parser.add_option(mode, "Permissions to set (instead of 0755)", "mode", 'm', "mode");
  25. args_parser.add_positional_argument(sources, "Source path", "source");
  26. args_parser.add_positional_argument(destination, "Destination path", "destination");
  27. args_parser.parse(arguments);
  28. auto permission_mask = TRY(Core::FilePermissionsMask::parse(mode));
  29. ByteString destination_dir = (sources.size() > 1 ? ByteString { destination } : LexicalPath::dirname(destination));
  30. if (create_leading_dest_components) {
  31. auto destination_dir_absolute = TRY(FileSystem::absolute_path(destination_dir));
  32. MUST(Core::Directory::create(destination_dir_absolute, Core::Directory::CreateDirectories::Yes));
  33. }
  34. for (auto const& source : sources) {
  35. ByteString final_destination;
  36. if (sources.size() > 1) {
  37. final_destination = LexicalPath(destination).append(LexicalPath::basename(source)).string();
  38. } else {
  39. final_destination = destination;
  40. }
  41. TRY(FileSystem::copy_file_or_directory(final_destination, source, FileSystem::RecursionMode::Allowed,
  42. FileSystem::LinkMode::Disallowed, FileSystem::AddDuplicateFileMarker::No,
  43. FileSystem::PreserveMode::Nothing));
  44. auto current_access = TRY(Core::System::stat(final_destination));
  45. TRY(Core::System::chmod(final_destination, permission_mask.apply(current_access.st_mode)));
  46. }
  47. return 0;
  48. }