/* This file is part of GNUnet. (C) 2009 Christian Grothoff (and other contributing authors) GNUnet is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, 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 General Public License for more details. You should have received a copy of the GNU General Public License along with GNUnet; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /** * @file arm/gnunet-arm.c * @brief arm for writing a tool * @author Christian Grothoff */ #include "platform.h" #include "gnunet_arm_service.h" #include "gnunet_client_lib.h" #include "gnunet_constants.h" #include "gnunet_getopt_lib.h" #include "gnunet_program_lib.h" #include "gnunet_time_lib.h" /** * Timeout for stopping services. Long to give some services a real chance. */ #define STOP_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_MINUTES, 1) /** * Timeout for stopping ARM. Extra-long since ARM needs to stop everyone else. */ #define STOP_TIMEOUT_ARM GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_MINUTES, 2) /** * Timeout for starting services, very short because of the strange way start works * (by checking if running before starting, so really this time is always waited on * startup (annoying)). */ #define START_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 1) /** * Timeout for listing all running services. */ #define LIST_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 2) /** * Set if we are to shutdown all services (including ARM). */ static int end; /** * Set if we are to start default services (including ARM). */ static int start; /** * Set if we are to stop/start default services (including ARM). */ static int restart; /** * Set if we should delete configuration and temp directory on exit. */ static int delete; /** * Set if we should not print status messages. */ static int quiet; /** * Set if we should print a list of currently running services. */ static int list; /** * Set to the name of a service to start. */ static char *init; /** * Set to the name of a service to kill. */ static char *term; /** * Set to the name of the config file used. */ static const char *config_file; /** * Set to the directory where runtime files are stored. */ static char *dir; /** * Final status code. */ static int ret; /** * Connection with ARM. */ static struct GNUNET_ARM_Handle *h; /** * Our configuration. */ static const struct GNUNET_CONFIGURATION_Handle *cfg; /** * Processing stage that we are in. Simple counter. */ static unsigned int phase; /** * User defined timestamp for completing operations. */ static struct GNUNET_TIME_Relative timeout; /** * Main continuation-passing-style loop. Runs the various * jobs that we've been asked to do in order. * * @param cls closure, unused * @param tc context, unused */ static void cps_loop (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc); /** * Callback invoked with the status of the last operation. Reports to the * user and then runs the next phase in the FSM. * * @param cls pointer to "const char*" identifying service that was manipulated * @param result result of the operation */ static void confirm_cb (void *cls, enum GNUNET_ARM_ProcessStatus result) { const char *service = cls; switch (result) { case GNUNET_ARM_PROCESS_UNKNOWN: FPRINTF (stderr, _("Service `%s' is unknown to ARM.\n"), service); ret = 1; break; case GNUNET_ARM_PROCESS_DOWN: if (quiet != GNUNET_YES) FPRINTF (stdout, _("Service `%s' has been stopped.\n"), service); break; case GNUNET_ARM_PROCESS_ALREADY_RUNNING: FPRINTF (stderr, _("Service `%s' was already running.\n"), service); ret = 1; break; case GNUNET_ARM_PROCESS_STARTING: if (quiet != GNUNET_YES) FPRINTF (stdout, _("Service `%s' has been started.\n"), service); break; case GNUNET_ARM_PROCESS_ALREADY_STOPPING: FPRINTF (stderr, _("Service `%s' was already being stopped.\n"), service); ret = 1; break; case GNUNET_ARM_PROCESS_ALREADY_DOWN: FPRINTF (stderr, _("Service `%s' was already not running.\n"), service); ret = 1; break; case GNUNET_ARM_PROCESS_SHUTDOWN: FPRINTF (stderr, "%s", _("Request ignored as ARM is shutting down.\n")); ret = 1; break; case GNUNET_ARM_PROCESS_COMMUNICATION_ERROR: FPRINTF (stderr, "%s", _("Error communicating with ARM service.\n")); ret = 1; break; case GNUNET_ARM_PROCESS_COMMUNICATION_TIMEOUT: FPRINTF (stderr, "%s", _("Timeout communicating with ARM service.\n")); ret = 1; break; case GNUNET_ARM_PROCESS_FAILURE: FPRINTF (stderr, "%s", _("Operation failed.\n")); ret = 1; break; default: FPRINTF (stderr, "%s", _("Unknown response code from ARM.\n")); break; } GNUNET_SCHEDULER_add_continuation (&cps_loop, NULL, GNUNET_SCHEDULER_REASON_PREREQ_DONE); } /** * Callback invoked with the list of running services. * Reports to the user and then runs the next phase in the FSM. * * @param cls currently not used * @param result result of the operation * @param count number of running services * @param list copy of the list of running services */ static void list_cb (void *cls, int result, unsigned int count, const char *const*list) { unsigned int i; if ( (result != GNUNET_YES) || (NULL == list) ) { FPRINTF (stderr, "%s", _("Error communicating with ARM. ARM not running?\n")); return; } FPRINTF (stdout, "%s", _("Running services:\n")); for (i=0; i 0) timeout.rel_value = temp_timeout_ms; if (GNUNET_OK == GNUNET_PROGRAM_run (argc, argv, "gnunet-arm", gettext_noop ("Control services and the Automated Restart Manager (ARM)"), options, &run, NULL)) { return ret; } return 1; } /* end of gnunet-arm.c */