test_process_slow.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. /* Copyright (c) 2018, The Tor Project, Inc. */
  2. /* See LICENSE for licensing information */
  3. /**
  4. * \file test_process_slow.c
  5. * \brief Slow test cases for the Process API.
  6. */
  7. #define MAINLOOP_PRIVATE
  8. #include "orconfig.h"
  9. #include "core/or/or.h"
  10. #include "core/mainloop/mainloop.h"
  11. #include "lib/evloop/compat_libevent.h"
  12. #include "lib/process/process.h"
  13. #include "lib/process/waitpid.h"
  14. #include "test/test.h"
  15. #ifndef BUILDDIR
  16. #define BUILDDIR "."
  17. #endif
  18. #ifdef _WIN32
  19. #define TEST_PROCESS "test-process.exe"
  20. #else
  21. #define TEST_PROCESS BUILDDIR "/src/test/test-process"
  22. #endif /* defined(_WIN32) */
  23. /** Timer that ticks once a second and stop the event loop after 5 ticks. */
  24. static periodic_timer_t *main_loop_timeout_timer;
  25. /** How many times have our timer ticked? */
  26. static int timer_tick_count;
  27. struct process_data_t {
  28. smartlist_t *stdout_data;
  29. smartlist_t *stderr_data;
  30. smartlist_t *stdin_data;
  31. process_exit_code_t exit_code;
  32. };
  33. typedef struct process_data_t process_data_t;
  34. static process_data_t *
  35. process_data_new(void)
  36. {
  37. process_data_t *process_data = tor_malloc_zero(sizeof(process_data_t));
  38. process_data->stdout_data = smartlist_new();
  39. process_data->stderr_data = smartlist_new();
  40. process_data->stdin_data = smartlist_new();
  41. return process_data;
  42. }
  43. static void
  44. process_data_free(process_data_t *process_data)
  45. {
  46. if (process_data == NULL)
  47. return;
  48. SMARTLIST_FOREACH(process_data->stdout_data, char *, x, tor_free(x));
  49. SMARTLIST_FOREACH(process_data->stderr_data, char *, x, tor_free(x));
  50. SMARTLIST_FOREACH(process_data->stdin_data, char *, x, tor_free(x));
  51. smartlist_free(process_data->stdout_data);
  52. smartlist_free(process_data->stderr_data);
  53. smartlist_free(process_data->stdin_data);
  54. tor_free(process_data);
  55. }
  56. static void
  57. process_stdout_callback(process_t *process, const char *data, size_t size)
  58. {
  59. tt_ptr_op(process, OP_NE, NULL);
  60. tt_ptr_op(data, OP_NE, NULL);
  61. tt_int_op(strlen(data), OP_EQ, size);
  62. process_data_t *process_data = process_get_data(process);
  63. smartlist_add(process_data->stdout_data, tor_strdup(data));
  64. done:
  65. return;
  66. }
  67. static void
  68. process_stderr_callback(process_t *process, const char *data, size_t size)
  69. {
  70. tt_ptr_op(process, OP_NE, NULL);
  71. tt_ptr_op(data, OP_NE, NULL);
  72. tt_int_op(strlen(data), OP_EQ, size);
  73. process_data_t *process_data = process_get_data(process);
  74. smartlist_add(process_data->stderr_data, tor_strdup(data));
  75. done:
  76. return;
  77. }
  78. static bool
  79. process_exit_callback(process_t *process, process_exit_code_t exit_code)
  80. {
  81. tt_ptr_op(process, OP_NE, NULL);
  82. process_data_t *process_data = process_get_data(process);
  83. process_data->exit_code = exit_code;
  84. /* Our process died. Let's check the values it returned. */
  85. tor_shutdown_event_loop_and_exit(0);
  86. done:
  87. /* Do not free up our process_t. */
  88. return false;
  89. }
  90. #ifdef _WIN32
  91. static const char *
  92. get_win32_test_binary_path(void)
  93. {
  94. static char buffer[MAX_PATH];
  95. /* Get the absolute path of our binary: \path\to\test-slow.exe. */
  96. GetModuleFileNameA(GetModuleHandle(0), buffer, sizeof(buffer));
  97. /* Find our process name. */
  98. char *offset = strstr(buffer, "test-slow.exe");
  99. tt_ptr_op(offset, OP_NE, NULL);
  100. /* Change test-slow.exe to test-process.exe. */
  101. memcpy(offset, TEST_PROCESS, strlen(TEST_PROCESS));
  102. return buffer;
  103. done:
  104. return NULL;
  105. }
  106. #endif
  107. static void
  108. main_loop_timeout_cb(periodic_timer_t *timer, void *data)
  109. {
  110. /* Sanity check. */
  111. tt_ptr_op(timer, OP_EQ, main_loop_timeout_timer);
  112. tt_ptr_op(data, OP_EQ, NULL);
  113. /* Have we been called 10 times we exit. */
  114. timer_tick_count++;
  115. tt_int_op(timer_tick_count, OP_LT, 10);
  116. #ifndef _WIN32
  117. /* Call waitpid callbacks. */
  118. notify_pending_waitpid_callbacks();
  119. #endif
  120. return;
  121. done:
  122. /* Exit with an error. */
  123. tor_shutdown_event_loop_and_exit(-1);
  124. }
  125. static void
  126. run_main_loop(void)
  127. {
  128. int ret;
  129. /* Wake up after 1 seconds. */
  130. static const struct timeval interval = {1, 0};
  131. timer_tick_count = 0;
  132. main_loop_timeout_timer = periodic_timer_new(tor_libevent_get_base(),
  133. &interval,
  134. main_loop_timeout_cb,
  135. NULL);
  136. /* Run our main loop. */
  137. ret = run_main_loop_until_done();
  138. /* Clean up our main loop timeout timer. */
  139. tt_int_op(ret, OP_EQ, 0);
  140. done:
  141. periodic_timer_free(main_loop_timeout_timer);
  142. }
  143. static void
  144. test_callbacks(void *arg)
  145. {
  146. (void)arg;
  147. const char *filename = NULL;
  148. #ifdef _WIN32
  149. filename = get_win32_test_binary_path();
  150. #else
  151. filename = TEST_PROCESS;
  152. #endif
  153. /* Initialize Process subsystem. */
  154. process_init();
  155. /* Process callback data. */
  156. process_data_t *process_data = process_data_new();
  157. /* Setup our process. */
  158. process_t *process = process_new(filename);
  159. process_set_data(process, process_data);
  160. process_set_stdout_read_callback(process, process_stdout_callback);
  161. process_set_stderr_read_callback(process, process_stderr_callback);
  162. process_set_exit_callback(process, process_exit_callback);
  163. /* Set environment variable. */
  164. process_set_environment(process, "TOR_TEST_ENV", "Hello, from Tor!");
  165. /* Add some arguments. */
  166. process_append_argument(process, "This is the first one");
  167. process_append_argument(process, "Second one");
  168. process_append_argument(process, "Third: Foo bar baz");
  169. /* Run our process. */
  170. process_status_t status;
  171. status = process_exec(process);
  172. tt_int_op(status, OP_EQ, PROCESS_STATUS_RUNNING);
  173. /* Write some lines to stdin. */
  174. process_printf(process, "Hi process!\r\n");
  175. process_printf(process, "Can you read more than one line?\n");
  176. process_printf(process, "Can you read partial ...");
  177. process_printf(process, " lines?\r\n");
  178. /* Start our main loop. */
  179. run_main_loop();
  180. /* Check if our process is still running? */
  181. status = process_get_status(process);
  182. tt_int_op(status, OP_EQ, PROCESS_STATUS_NOT_RUNNING);
  183. /* We returned. Let's see what our event loop said. */
  184. tt_int_op(smartlist_len(process_data->stdout_data), OP_EQ, 12);
  185. tt_int_op(smartlist_len(process_data->stderr_data), OP_EQ, 3);
  186. tt_int_op(process_data->exit_code, OP_EQ, 0);
  187. /* Check stdout output. */
  188. char argv0_expected[256];
  189. tor_snprintf(argv0_expected, sizeof(argv0_expected),
  190. "argv[0] = '%s'", filename);
  191. tt_str_op(smartlist_get(process_data->stdout_data, 0), OP_EQ,
  192. argv0_expected);
  193. tt_str_op(smartlist_get(process_data->stdout_data, 1), OP_EQ,
  194. "argv[1] = 'This is the first one'");
  195. tt_str_op(smartlist_get(process_data->stdout_data, 2), OP_EQ,
  196. "argv[2] = 'Second one'");
  197. tt_str_op(smartlist_get(process_data->stdout_data, 3), OP_EQ,
  198. "argv[3] = 'Third: Foo bar baz'");
  199. tt_str_op(smartlist_get(process_data->stdout_data, 4), OP_EQ,
  200. "Environment variable TOR_TEST_ENV = 'Hello, from Tor!'");
  201. tt_str_op(smartlist_get(process_data->stdout_data, 5), OP_EQ,
  202. "Output on stdout");
  203. tt_str_op(smartlist_get(process_data->stdout_data, 6), OP_EQ,
  204. "This is a new line");
  205. tt_str_op(smartlist_get(process_data->stdout_data, 7), OP_EQ,
  206. "Partial line on stdout ...end of partial line on stdout");
  207. tt_str_op(smartlist_get(process_data->stdout_data, 8), OP_EQ,
  208. "Read line from stdin: 'Hi process!'");
  209. tt_str_op(smartlist_get(process_data->stdout_data, 9), OP_EQ,
  210. "Read line from stdin: 'Can you read more than one line?'");
  211. tt_str_op(smartlist_get(process_data->stdout_data, 10), OP_EQ,
  212. "Read line from stdin: 'Can you read partial ... lines?'");
  213. tt_str_op(smartlist_get(process_data->stdout_data, 11), OP_EQ,
  214. "We are done for here, thank you!");
  215. /* Check stderr output. */
  216. tt_str_op(smartlist_get(process_data->stderr_data, 0), OP_EQ,
  217. "Output on stderr");
  218. tt_str_op(smartlist_get(process_data->stderr_data, 1), OP_EQ,
  219. "This is a new line");
  220. tt_str_op(smartlist_get(process_data->stderr_data, 2), OP_EQ,
  221. "Partial line on stderr ...end of partial line on stderr");
  222. done:
  223. process_data_free(process_data);
  224. process_free(process);
  225. process_free_all();
  226. }
  227. static void
  228. test_callbacks_terminate(void *arg)
  229. {
  230. (void)arg;
  231. const char *filename = NULL;
  232. #ifdef _WIN32
  233. filename = get_win32_test_binary_path();
  234. #else
  235. filename = TEST_PROCESS;
  236. #endif
  237. /* Initialize Process subsystem. */
  238. process_init();
  239. /* Process callback data. */
  240. process_data_t *process_data = process_data_new();
  241. /* Setup our process. */
  242. process_t *process = process_new(filename);
  243. process_set_data(process, process_data);
  244. process_set_exit_callback(process, process_exit_callback);
  245. /* Run our process. */
  246. process_status_t status;
  247. status = process_exec(process);
  248. tt_int_op(status, OP_EQ, PROCESS_STATUS_RUNNING);
  249. /* Zap our process. */
  250. process_terminate(process);
  251. /* Start our main loop. */
  252. run_main_loop();
  253. /* Check if our process is still running? */
  254. status = process_get_status(process);
  255. tt_int_op(status, OP_EQ, PROCESS_STATUS_NOT_RUNNING);
  256. done:
  257. process_data_free(process_data);
  258. process_free(process);
  259. process_free_all();
  260. }
  261. struct testcase_t slow_process_tests[] = {
  262. { "callbacks", test_callbacks, 0, NULL, NULL },
  263. { "callbacks_terminate", test_callbacks_terminate, 0, NULL, NULL },
  264. END_OF_TESTCASES
  265. };