/* This file is part of GNUnet Copyright (C) 2021 GNUnet e.V. GNUnet is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. GNUnet is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . SPDX-License-Identifier: AGPL3.0-or-later */ /** * @file testing/testing_api_cmd_netjail_stop.c * @brief Command to stop the netjail script. * @author t3sserakt */ #include "platform.h" #include "gnunet_util_lib.h" #include "gnunet_testing_ng_lib.h" #define NETJAIL_STOP_SCRIPT "./../testing/netjail_stop.sh" // Child Wait handle struct GNUNET_ChildWaitHandle *cwh; /** * Struct to hold information for callbacks. * */ struct NetJailState { /** * Context for our asynchronous completion. */ struct GNUNET_TESTING_AsyncContext ac; // Number of local nodes in each namespace. char *local_m; // The number of namespaces. char *global_n; /** * The process id of the start script. */ struct GNUNET_OS_Process *stop_proc; }; /** * The cleanup function of this cmd frees resources the cmd allocated. * */ static void netjail_stop_cleanup (void *cls) { struct NetJailState *ns = cls; if (NULL != cwh) { GNUNET_wait_child_cancel (cwh); cwh = NULL; } if (NULL != ns->stop_proc) { GNUNET_assert (0 == GNUNET_OS_process_kill (ns->stop_proc, SIGKILL)); GNUNET_assert (GNUNET_OK == GNUNET_OS_process_wait (ns->stop_proc)); GNUNET_OS_process_destroy (ns->stop_proc); ns->stop_proc = NULL; } } /** * Trait function of this cmd does nothing. * */ static enum GNUNET_GenericReturnValue netjail_stop_traits (void *cls, const void **ret, const char *trait, unsigned int index) { return GNUNET_NO; } /** * Callback which will be called if the setup script finished. * */ static void child_completed_callback (void *cls, enum GNUNET_OS_ProcessStatusType type, long unsigned int exit_code) { struct NetJailState *ns = cls; cwh = NULL; GNUNET_OS_process_destroy (ns->stop_proc); ns->stop_proc = NULL; if (0 == exit_code) { GNUNET_TESTING_async_finish (&ns->ac); } else { // FIXME: log exit code! GNUNET_TESTING_async_fail (&ns->ac); } } /** * The run method starts the script which deletes the network namespaces. * * @param cls closure. * @param cmd CMD being run. * @param is interpreter state. */ static void netjail_stop_run (void *cls, struct GNUNET_TESTING_Interpreter *is) { struct NetJailState *ns = cls; char *const script_argv[] = { NETJAIL_STOP_SCRIPT, ns->local_m, ns->global_n, NULL }; unsigned int helper_check = GNUNET_OS_check_helper_binary ( NETJAIL_STOP_SCRIPT, GNUNET_YES, NULL); if (GNUNET_NO == helper_check) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "No SUID for %s!\n", NETJAIL_STOP_SCRIPT); GNUNET_TESTING_interpreter_fail (is); } else if (GNUNET_NO == helper_check) { GNUNET_log (GNUNET_ERROR_TYPE_ERROR, "%s not found!\n", NETJAIL_STOP_SCRIPT); GNUNET_TESTING_interpreter_fail (is); } ns->stop_proc = GNUNET_OS_start_process_vap (GNUNET_OS_INHERIT_STD_ERR, NULL, NULL, NULL, NETJAIL_STOP_SCRIPT, script_argv); cwh = GNUNET_wait_child (ns->stop_proc, &child_completed_callback, ns); GNUNET_break (NULL != cwh); } struct GNUNET_TESTING_Command GNUNET_TESTING_cmd_netjail_stop (const char *label, char *local_m, char *global_n) { struct NetJailState *ns; ns = GNUNET_new (struct NetJailState); ns->local_m = local_m; ns->global_n = global_n; { struct GNUNET_TESTING_Command cmd = { .cls = ns, .label = label, .run = &netjail_stop_run, .ac = &ns->ac, .cleanup = &netjail_stop_cleanup, .traits = &netjail_stop_traits }; return cmd; } }