logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

env.c (3970B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. #define _POSIX_C_SOURCE 200809L
  5. #define _XOPEN_SOURCE 800 // wordexp
  6. #include "../config.h" // HAS_*
  7. #include "../lib/getopt_nolong.h"
  8. #include <assert.h> // assert
  9. #include <errno.h> // errno
  10. #include <stdbool.h> // bool, true, false
  11. #include <stdio.h> // puts, fprintf
  12. #include <stdlib.h> // putenv
  13. #include <string.h> // strchr, strerror
  14. #include <unistd.h> // getopt, opt*
  15. #ifdef HAS_WORDEXP
  16. #include <wordexp.h>
  17. #endif
  18. #ifdef HAS_GETOPT_LONG
  19. #include <getopt.h>
  20. #endif
  21. const char *argv0 = "env";
  22. extern char **environ;
  23. char *envclear[1];
  24. static int
  25. do_export(void)
  26. {
  27. int i = 0;
  28. for(; environ[i] != NULL; i++)
  29. {
  30. if(puts(environ[i]) < 0)
  31. {
  32. fprintf(stderr, "env: error: Failed writing environ: %s\n", strerror(errno));
  33. return 1;
  34. }
  35. }
  36. return 0;
  37. }
  38. static void
  39. usage(void)
  40. {
  41. fprintf(stderr, "env [-i] [-u key] [key=value ...] [command [args]]\n");
  42. }
  43. int
  44. main(int argc, char *argv[])
  45. {
  46. bool flag_i = false;
  47. #ifdef HAS_WORDEXP
  48. bool flag_S = false;
  49. wordexp_t opt_S;
  50. #define OPTARGS ":iu:S:"
  51. #else
  52. #define OPTARGS ":iu:"
  53. #endif
  54. #ifdef HAS_GETOPT_LONG
  55. // Strictly for GNUisms compatibility so no long-only options
  56. // clang-format off
  57. static struct option opts[] = {
  58. {"ignore-environment", no_argument, 0, 'i'},
  59. {"unset", required_argument, 0, 'u'},
  60. #ifdef HAS_WORDEXP
  61. {"split-string", required_argument, 0, 'S'},
  62. #endif
  63. {0, 0, 0, 0},
  64. };
  65. // clang-format on
  66. // Need + as first character to get POSIX-style option parsing
  67. for(int c = -1; (c = getopt_long(argc, argv, "+" OPTARGS, opts, NULL)) != -1;)
  68. #else
  69. for(int c = -1; (c = getopt_nolong(argc, argv, OPTARGS)) != -1;)
  70. #endif
  71. {
  72. switch(c)
  73. {
  74. case 'i':
  75. flag_i = true;
  76. break;
  77. case 'u':
  78. unsetenv(optarg);
  79. break;
  80. #ifdef HAS_WORDEXP
  81. case 'S':
  82. flag_S = true;
  83. int ret = wordexp(optarg, &opt_S, WRDE_NOCMD);
  84. switch(ret)
  85. {
  86. case 0:
  87. break;
  88. case WRDE_BADCHAR:
  89. fputs("env: error: wordexpr returned WRDE_BADCHAR\n", stderr);
  90. return 1;
  91. case WRDE_BADVAL:
  92. fputs("env: error: Undefined shell variable (WRDE_BADVAL)\n", stderr);
  93. return 1;
  94. case WRDE_CMDSUB:
  95. fputs("env: error: Command substitution forbidden (WRDE_CMDSUB)\n", stderr);
  96. return 1;
  97. case WRDE_NOSPACE:
  98. fputs("env: error: Out of memory (WRDE_NOSPACE)\n", stderr);
  99. return 1;
  100. case WRDE_SYNTAX:
  101. fputs("env: error: Syntax Error (WRDE_SYNTAX)\n", stderr);
  102. return 1;
  103. default:
  104. fprintf(stderr, "env: error: Unknown error %d from wordexp\n", ret);
  105. return 1;
  106. }
  107. break;
  108. #endif
  109. case ':':
  110. fprintf(stderr, "env: error: Missing operand for option: '-%c'\n", optopt);
  111. usage();
  112. return 1;
  113. case '?':
  114. if(!got_long_opt) fprintf(stderr, "env: error: Unrecognised option: '-%c'\n", optopt);
  115. usage();
  116. return 1;
  117. default:
  118. assert(false);
  119. }
  120. }
  121. argc -= optind;
  122. argv += optind;
  123. if(flag_i)
  124. {
  125. environ = envclear;
  126. envclear[0] = NULL;
  127. }
  128. for(; argv[0]; argv++, argc--)
  129. {
  130. char *sep = strchr(argv[0], '=');
  131. if(sep == NULL)
  132. {
  133. break;
  134. }
  135. *sep = 0;
  136. sep++;
  137. if(setenv(argv[0], sep, 1))
  138. {
  139. fprintf(stderr,
  140. "env: error: Failed setting environment variable '%s' to '%s': %s\n",
  141. argv[0],
  142. sep,
  143. strerror(errno));
  144. return 1;
  145. }
  146. }
  147. #ifdef HAS_WORDEXP
  148. if(flag_S)
  149. {
  150. errno = 0;
  151. if(execvp(opt_S.we_wordv[0], opt_S.we_wordv) < 0)
  152. {
  153. fprintf(
  154. stderr, "env: error: Failed executing '%s': %s\n", opt_S.we_wordv[0], strerror(errno));
  155. return (errno == ENOENT) ? 127 : 126;
  156. }
  157. assert(false);
  158. }
  159. #endif
  160. if(argc < 1)
  161. {
  162. return do_export();
  163. }
  164. assert(argv[0]);
  165. errno = 0;
  166. if(execvp(argv[0], argv) < 0)
  167. {
  168. fprintf(stderr, "env: error: Failed executing '%s': %s\n", argv[0], strerror(errno));
  169. return (errno == ENOENT) ? 127 : 126;
  170. }
  171. assert(false);
  172. }