1 1.1 jmmv // Copyright 2010 Google Inc. 2 1.1 jmmv // All rights reserved. 3 1.1 jmmv // 4 1.1 jmmv // Redistribution and use in source and binary forms, with or without 5 1.1 jmmv // modification, are permitted provided that the following conditions are 6 1.1 jmmv // met: 7 1.1 jmmv // 8 1.1 jmmv // * Redistributions of source code must retain the above copyright 9 1.1 jmmv // notice, this list of conditions and the following disclaimer. 10 1.1 jmmv // * Redistributions in binary form must reproduce the above copyright 11 1.1 jmmv // notice, this list of conditions and the following disclaimer in the 12 1.1 jmmv // documentation and/or other materials provided with the distribution. 13 1.1 jmmv // * Neither the name of Google Inc. nor the names of its contributors 14 1.1 jmmv // may be used to endorse or promote products derived from this software 15 1.1 jmmv // without specific prior written permission. 16 1.1 jmmv // 17 1.1 jmmv // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 18 1.1 jmmv // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 19 1.1 jmmv // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 20 1.1 jmmv // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 21 1.1 jmmv // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 22 1.1 jmmv // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 23 1.1 jmmv // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 24 1.1 jmmv // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 25 1.1 jmmv // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 26 1.1 jmmv // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 27 1.1 jmmv // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 1.1 jmmv 29 1.1 jmmv #include "cli/main.hpp" 30 1.1 jmmv 31 1.1 jmmv #if defined(HAVE_CONFIG_H) 32 1.1 jmmv # include "config.h" 33 1.1 jmmv #endif 34 1.1 jmmv 35 1.1 jmmv extern "C" { 36 1.1 jmmv #include <signal.h> 37 1.1 jmmv #include <unistd.h> 38 1.1 jmmv } 39 1.1 jmmv 40 1.1 jmmv #include <cstdlib> 41 1.1 jmmv #include <iostream> 42 1.1 jmmv #include <string> 43 1.1 jmmv #include <utility> 44 1.1 jmmv 45 1.1 jmmv #include "cli/cmd_about.hpp" 46 1.1 jmmv #include "cli/cmd_config.hpp" 47 1.1 jmmv #include "cli/cmd_db_exec.hpp" 48 1.1 jmmv #include "cli/cmd_db_migrate.hpp" 49 1.1 jmmv #include "cli/cmd_debug.hpp" 50 1.1 jmmv #include "cli/cmd_help.hpp" 51 1.1 jmmv #include "cli/cmd_list.hpp" 52 1.1 jmmv #include "cli/cmd_report.hpp" 53 1.1 jmmv #include "cli/cmd_report_html.hpp" 54 1.1 jmmv #include "cli/cmd_test.hpp" 55 1.1 jmmv #include "cli/common.ipp" 56 1.1 jmmv #include "cli/config.hpp" 57 1.1 jmmv #include "store/exceptions.hpp" 58 1.1 jmmv #include "utils/cmdline/commands_map.ipp" 59 1.1 jmmv #include "utils/cmdline/exceptions.hpp" 60 1.1 jmmv #include "utils/cmdline/globals.hpp" 61 1.1 jmmv #include "utils/cmdline/options.hpp" 62 1.1 jmmv #include "utils/cmdline/parser.ipp" 63 1.1 jmmv #include "utils/cmdline/ui.hpp" 64 1.1 jmmv #include "utils/config/tree.ipp" 65 1.1 jmmv #include "utils/env.hpp" 66 1.1 jmmv #include "utils/format/macros.hpp" 67 1.1 jmmv #include "utils/fs/operations.hpp" 68 1.1 jmmv #include "utils/fs/path.hpp" 69 1.1 jmmv #include "utils/logging/macros.hpp" 70 1.1 jmmv #include "utils/logging/operations.hpp" 71 1.1 jmmv #include "utils/optional.ipp" 72 1.1 jmmv #include "utils/sanity.hpp" 73 1.1 jmmv #include "utils/signals/exceptions.hpp" 74 1.1 jmmv 75 1.1 jmmv namespace cmdline = utils::cmdline; 76 1.1 jmmv namespace config = utils::config; 77 1.1 jmmv namespace fs = utils::fs; 78 1.1 jmmv namespace logging = utils::logging; 79 1.1 jmmv namespace signals = utils::signals; 80 1.1 jmmv 81 1.1 jmmv using utils::none; 82 1.1 jmmv using utils::optional; 83 1.1 jmmv 84 1.1 jmmv 85 1.1 jmmv namespace { 86 1.1 jmmv 87 1.1 jmmv 88 1.1 jmmv /// Executes the given subcommand with proper usage_error reporting. 89 1.1 jmmv /// 90 1.1 jmmv /// \param ui Object to interact with the I/O of the program. 91 1.1 jmmv /// \param command The subcommand to execute. 92 1.1 jmmv /// \param args The part of the command line passed to the subcommand. The 93 1.1 jmmv /// first item of this collection must match the command name. 94 1.1 jmmv /// \param user_config The runtime configuration to pass to the subcommand. 95 1.1 jmmv /// 96 1.1 jmmv /// \return The exit code of the command. Typically 0 on success, some other 97 1.1 jmmv /// integer otherwise. 98 1.1 jmmv /// 99 1.1 jmmv /// \throw cmdline::usage_error If the user input to the subcommand is invalid. 100 1.1 jmmv /// This error does not encode the command name within it, so this function 101 1.1 jmmv /// extends the message in the error to specify which subcommand was 102 1.1 jmmv /// affected. 103 1.1 jmmv /// \throw std::exception This propagates any uncaught exception. Such 104 1.1 jmmv /// exceptions are bugs, but we let them propagate so that the runtime will 105 1.1 jmmv /// abort and dump core. 106 1.1 jmmv static int 107 1.1 jmmv run_subcommand(cmdline::ui* ui, cli::cli_command* command, 108 1.1 jmmv const cmdline::args_vector& args, 109 1.1 jmmv const config::tree& user_config) 110 1.1 jmmv { 111 1.1 jmmv try { 112 1.1 jmmv PRE(command->name() == args[0]); 113 1.1 jmmv return command->main(ui, args, user_config); 114 1.1 jmmv } catch (const cmdline::usage_error& e) { 115 1.1 jmmv throw std::pair< std::string, cmdline::usage_error >( 116 1.1 jmmv command->name(), e); 117 1.1 jmmv } 118 1.1 jmmv } 119 1.1 jmmv 120 1.1 jmmv 121 1.1 jmmv /// Exception-safe version of main. 122 1.1 jmmv /// 123 1.1 jmmv /// This function provides the real meat of the entry point of the program. It 124 1.1 jmmv /// is allowed to throw some known exceptions which are parsed by the caller. 125 1.1 jmmv /// Doing so keeps this function simpler and allow tests to actually validate 126 1.1 jmmv /// that the errors reported are accurate. 127 1.1 jmmv /// 128 1.1 jmmv /// \return The exit code of the program. Should be EXIT_SUCCESS on success and 129 1.1 jmmv /// EXIT_FAILURE on failure. The caller extends this to additional integers for 130 1.1 jmmv /// errors reported through exceptions. 131 1.1 jmmv /// 132 1.1 jmmv /// \param ui Object to interact with the I/O of the program. 133 1.1 jmmv /// \param argc The number of arguments passed on the command line. 134 1.1 jmmv /// \param argv NULL-terminated array containing the command line arguments. 135 1.1 jmmv /// \param mock_command An extra command provided for testing purposes; should 136 1.1 jmmv /// just be NULL other than for tests. 137 1.1 jmmv /// 138 1.1 jmmv /// \throw cmdline::usage_error If the user ran the program with invalid 139 1.1 jmmv /// arguments. 140 1.1 jmmv /// \throw std::exception This propagates any uncaught exception. Such 141 1.1 jmmv /// exceptions are bugs, but we let them propagate so that the runtime will 142 1.1 jmmv /// abort and dump core. 143 1.1 jmmv static int 144 1.1 jmmv safe_main(cmdline::ui* ui, int argc, const char* const argv[], 145 1.1 jmmv cli::cli_command_ptr mock_command) 146 1.1 jmmv { 147 1.1 jmmv cmdline::options_vector options; 148 1.1 jmmv options.push_back(&cli::config_option); 149 1.1 jmmv options.push_back(&cli::variable_option); 150 1.1 jmmv const cmdline::string_option loglevel_option( 151 1.1 jmmv "loglevel", "Level of the messages to log", "level", "info"); 152 1.1 jmmv options.push_back(&loglevel_option); 153 1.1 jmmv const cmdline::path_option logfile_option( 154 1.1 jmmv "logfile", "Path to the log file", "file", 155 1.1 jmmv cli::detail::default_log_name().c_str()); 156 1.1 jmmv options.push_back(&logfile_option); 157 1.1 jmmv 158 1.1 jmmv cmdline::commands_map< cli::cli_command > commands; 159 1.1 jmmv 160 1.1 jmmv commands.insert(new cli::cmd_about()); 161 1.1 jmmv commands.insert(new cli::cmd_config()); 162 1.1 jmmv commands.insert(new cli::cmd_db_exec()); 163 1.1 jmmv commands.insert(new cli::cmd_db_migrate()); 164 1.1 jmmv commands.insert(new cli::cmd_help(&options, &commands)); 165 1.1 jmmv 166 1.1 jmmv commands.insert(new cli::cmd_debug(), "Workspace"); 167 1.1 jmmv commands.insert(new cli::cmd_list(), "Workspace"); 168 1.1 jmmv commands.insert(new cli::cmd_test(), "Workspace"); 169 1.1 jmmv 170 1.1 jmmv commands.insert(new cli::cmd_report(), "Reporting"); 171 1.1 jmmv commands.insert(new cli::cmd_report_html(), "Reporting"); 172 1.1 jmmv 173 1.1 jmmv if (mock_command.get() != NULL) 174 1.2 lukem commands.insert(std::move(mock_command)); 175 1.1 jmmv 176 1.1 jmmv const cmdline::parsed_cmdline cmdline = cmdline::parse(argc, argv, options); 177 1.1 jmmv 178 1.1 jmmv const fs::path logfile(cmdline.get_option< cmdline::path_option >( 179 1.1 jmmv "logfile")); 180 1.1 jmmv fs::mkdir_p(logfile.branch_path(), 0755); 181 1.1 jmmv LD(F("Log file is %s") % logfile); 182 1.1 jmmv utils::install_crash_handlers(logfile.str()); 183 1.1 jmmv try { 184 1.1 jmmv logging::set_persistency(cmdline.get_option< cmdline::string_option >( 185 1.1 jmmv "loglevel"), logfile); 186 1.1 jmmv } catch (const std::range_error& e) { 187 1.1 jmmv throw cmdline::usage_error(e.what()); 188 1.1 jmmv } 189 1.1 jmmv 190 1.1 jmmv if (cmdline.arguments().empty()) 191 1.1 jmmv throw cmdline::usage_error("No command provided"); 192 1.1 jmmv const std::string cmdname = cmdline.arguments()[0]; 193 1.1 jmmv 194 1.1 jmmv const config::tree user_config = cli::load_config(cmdline, 195 1.1 jmmv cmdname != "help"); 196 1.1 jmmv 197 1.1 jmmv cli::cli_command* command = commands.find(cmdname); 198 1.1 jmmv if (command == NULL) 199 1.1 jmmv throw cmdline::usage_error(F("Unknown command '%s'") % cmdname); 200 1.1 jmmv return run_subcommand(ui, command, cmdline.arguments(), user_config); 201 1.1 jmmv } 202 1.1 jmmv 203 1.1 jmmv 204 1.1 jmmv } // anonymous namespace 205 1.1 jmmv 206 1.1 jmmv 207 1.1 jmmv /// Gets the name of the default log file. 208 1.1 jmmv /// 209 1.1 jmmv /// \return The path to the log file. 210 1.1 jmmv fs::path 211 1.1 jmmv cli::detail::default_log_name(void) 212 1.1 jmmv { 213 1.1 jmmv // Update doc/troubleshooting.texi if you change this algorithm. 214 1.1 jmmv const optional< std::string > home(utils::getenv("HOME")); 215 1.1 jmmv if (home) { 216 1.1 jmmv return logging::generate_log_name(fs::path(home.get()) / ".kyua" / 217 1.1 jmmv "logs", cmdline::progname()); 218 1.1 jmmv } else { 219 1.1 jmmv const optional< std::string > tmpdir(utils::getenv("TMPDIR")); 220 1.1 jmmv if (tmpdir) { 221 1.1 jmmv return logging::generate_log_name(fs::path(tmpdir.get()), 222 1.1 jmmv cmdline::progname()); 223 1.1 jmmv } else { 224 1.1 jmmv return logging::generate_log_name(fs::path("/tmp"), 225 1.1 jmmv cmdline::progname()); 226 1.1 jmmv } 227 1.1 jmmv } 228 1.1 jmmv } 229 1.1 jmmv 230 1.1 jmmv 231 1.1 jmmv /// Testable entry point, with catch-all exception handlers. 232 1.1 jmmv /// 233 1.1 jmmv /// This entry point does not perform any initialization of global state; it is 234 1.1 jmmv /// provided to allow unit-testing of the utility's entry point. 235 1.1 jmmv /// 236 1.1 jmmv /// \param ui Object to interact with the I/O of the program. 237 1.1 jmmv /// \param argc The number of arguments passed on the command line. 238 1.1 jmmv /// \param argv NULL-terminated array containing the command line arguments. 239 1.1 jmmv /// \param mock_command An extra command provided for testing purposes; should 240 1.1 jmmv /// just be NULL other than for tests. 241 1.1 jmmv /// 242 1.1 jmmv /// \return 0 on success, some other integer on error. 243 1.1 jmmv /// 244 1.1 jmmv /// \throw std::exception This propagates any uncaught exception. Such 245 1.1 jmmv /// exceptions are bugs, but we let them propagate so that the runtime will 246 1.1 jmmv /// abort and dump core. 247 1.1 jmmv int 248 1.1 jmmv cli::main(cmdline::ui* ui, const int argc, const char* const* const argv, 249 1.1 jmmv cli_command_ptr mock_command) 250 1.1 jmmv { 251 1.1 jmmv try { 252 1.2 lukem const int exit_code = safe_main(ui, argc, argv, std::move(mock_command)); 253 1.1 jmmv 254 1.1 jmmv // Codes above 1 are reserved to report conditions captured as 255 1.1 jmmv // exceptions below. 256 1.1 jmmv INV(exit_code == EXIT_SUCCESS || exit_code == EXIT_FAILURE); 257 1.1 jmmv 258 1.1 jmmv return exit_code; 259 1.1 jmmv } catch (const signals::interrupted_error& e) { 260 1.1 jmmv cmdline::print_error(ui, e.what()); 261 1.1 jmmv // Re-deliver the interruption signal to self so that we terminate with 262 1.1 jmmv // the right status. At this point we should NOT have any custom signal 263 1.1 jmmv // handlers in place. 264 1.1 jmmv ::kill(getpid(), e.signo()); 265 1.1 jmmv LD("Interrupt signal re-delivery did not terminate program"); 266 1.1 jmmv // If we reach this, something went wrong because we did not exit as 267 1.1 jmmv // intended. Return an internal error instead. (Would be nicer to 268 1.1 jmmv // abort in principle, but it wouldn't be a nice experience if it ever 269 1.1 jmmv // happened.) 270 1.1 jmmv return 2; 271 1.1 jmmv } catch (const std::pair< std::string, cmdline::usage_error >& e) { 272 1.1 jmmv const std::string message = F("Usage error for command %s: %s.") % 273 1.1 jmmv e.first % e.second.what(); 274 1.1 jmmv LE(message); 275 1.1 jmmv ui->err(message); 276 1.1 jmmv ui->err(F("Type '%s help %s' for usage information.") % 277 1.1 jmmv cmdline::progname() % e.first); 278 1.1 jmmv return 3; 279 1.1 jmmv } catch (const cmdline::usage_error& e) { 280 1.1 jmmv const std::string message = F("Usage error: %s.") % e.what(); 281 1.1 jmmv LE(message); 282 1.1 jmmv ui->err(message); 283 1.1 jmmv ui->err(F("Type '%s help' for usage information.") % 284 1.1 jmmv cmdline::progname()); 285 1.1 jmmv return 3; 286 1.1 jmmv } catch (const store::old_schema_error& e) { 287 1.1 jmmv const std::string message = F("The database has schema version %s, " 288 1.1 jmmv "which is too old; please use db-migrate " 289 1.1 jmmv "to upgrade it") % e.old_version(); 290 1.1 jmmv cmdline::print_error(ui, message); 291 1.1 jmmv return 2; 292 1.1 jmmv } catch (const std::runtime_error& e) { 293 1.1 jmmv cmdline::print_error(ui, e.what()); 294 1.1 jmmv return 2; 295 1.1 jmmv } 296 1.1 jmmv } 297 1.1 jmmv 298 1.1 jmmv 299 1.1 jmmv /// Delegate for ::main(). 300 1.1 jmmv /// 301 1.1 jmmv /// This function is supposed to be called directly from the top-level ::main() 302 1.1 jmmv /// function. It takes care of initializing internal libraries and then calls 303 1.1 jmmv /// main(ui, argc, argv). 304 1.1 jmmv /// 305 1.1 jmmv /// \pre This function can only be called once. 306 1.1 jmmv /// 307 1.1 jmmv /// \throw std::exception This propagates any uncaught exception. Such 308 1.1 jmmv /// exceptions are bugs, but we let them propagate so that the runtime will 309 1.1 jmmv /// abort and dump core. 310 1.1 jmmv int 311 1.1 jmmv cli::main(const int argc, const char* const* const argv) 312 1.1 jmmv { 313 1.1 jmmv logging::set_inmemory(); 314 1.1 jmmv 315 1.1 jmmv LI(F("%s %s") % PACKAGE % VERSION); 316 1.1 jmmv 317 1.1 jmmv std::string plain_args; 318 1.1 jmmv for (const char* const* arg = argv; *arg != NULL; arg++) 319 1.1 jmmv plain_args += F(" %s") % *arg; 320 1.1 jmmv LI(F("Command line:%s") % plain_args); 321 1.1 jmmv 322 1.1 jmmv cmdline::init(argv[0]); 323 1.1 jmmv cmdline::ui ui; 324 1.1 jmmv 325 1.1 jmmv const int exit_code = main(&ui, argc, argv); 326 1.1 jmmv LI(F("Clean exit with code %s") % exit_code); 327 1.1 jmmv return exit_code; 328 1.1 jmmv } 329