env.cpp 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibCore/ArgsParser.h>
  7. #include <LibCore/DirIterator.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <unistd.h>
  11. int main(int argc, char** argv)
  12. {
  13. if (pledge("stdio rpath exec", nullptr) < 0) {
  14. perror("pledge");
  15. return 1;
  16. }
  17. bool ignore_env = false;
  18. const char* split_string = nullptr;
  19. Vector<const char*> values;
  20. Core::ArgsParser args_parser;
  21. args_parser.set_stop_on_first_non_option(true);
  22. args_parser.add_option(ignore_env, "Start with an empty environment", "ignore-environment", 'i');
  23. args_parser.add_option(split_string, "Process and split S into separate arguments; used to pass multiple arguments on shebang lines", "split-string", 'S', "S");
  24. args_parser.add_positional_argument(values, "Environment and commands", "env/command", Core::ArgsParser::Required::No);
  25. args_parser.parse(argc, argv);
  26. if (ignore_env)
  27. clearenv();
  28. size_t argv_start;
  29. for (argv_start = 0; argv_start < values.size(); ++argv_start) {
  30. if (StringView { values[argv_start] }.contains('=')) {
  31. putenv(const_cast<char*>(values[argv_start]));
  32. } else {
  33. break;
  34. }
  35. }
  36. Vector<String> split_string_storage;
  37. Vector<const char*> new_argv;
  38. if (split_string) {
  39. for (auto view : StringView(split_string).split_view(' ')) {
  40. split_string_storage.append(view);
  41. }
  42. for (auto& str : split_string_storage) {
  43. new_argv.append(str.characters());
  44. }
  45. }
  46. for (size_t i = argv_start; i < values.size(); ++i) {
  47. new_argv.append(values[i]);
  48. }
  49. if (new_argv.size() == 0) {
  50. for (auto entry = environ; *entry != nullptr; ++entry)
  51. outln("{}", *entry);
  52. return 0;
  53. }
  54. new_argv.append(nullptr);
  55. const char* executable = new_argv[0];
  56. char* const* new_argv_ptr = const_cast<char* const*>(&new_argv[0]);
  57. execvp(executable, new_argv_ptr);
  58. perror("execvp");
  59. return 1;
  60. }