summaryrefslogtreecommitdiffstats
path: root/src/bin/dhcp4/main.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/bin/dhcp4/main.cc')
-rw-r--r--src/bin/dhcp4/main.cc322
1 files changed, 322 insertions, 0 deletions
diff --git a/src/bin/dhcp4/main.cc b/src/bin/dhcp4/main.cc
new file mode 100644
index 0000000..7979a04
--- /dev/null
+++ b/src/bin/dhcp4/main.cc
@@ -0,0 +1,322 @@
+// Copyright (C) 2011-2023 Internet Systems Consortium, Inc. ("ISC")
+//
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#include <config.h>
+#include <kea_version.h>
+
+#include <dhcp4/ctrl_dhcp4_srv.h>
+#include <dhcp4/dhcp4_log.h>
+#include <dhcp4/parser_context.h>
+#include <dhcp4/json_config_parser.h>
+#include <dhcpsrv/cfgmgr.h>
+#include <exceptions/exceptions.h>
+#include <log/logger_support.h>
+#include <log/logger_manager.h>
+#include <log/output_option.h>
+#include <process/cfgrpt/config_report.h>
+#include <process/daemon.h>
+
+#include <boost/lexical_cast.hpp>
+
+#include <iostream>
+
+using namespace isc::data;
+using namespace isc::dhcp;
+using namespace isc::process;
+using namespace std;
+
+/// This file contains entry point (main() function) for standard DHCPv4 server
+/// component of Kea software suite. It parses command-line arguments and
+/// instantiates ControlledDhcpv4Srv class that is responsible for establishing
+/// connection with msgq (receiving commands and configuration) and also
+/// creating Dhcpv4 server object as well.
+///
+/// For detailed explanation or relations between main(), ControlledDhcpv4Srv,
+/// Dhcpv4Srv and other classes, see \ref dhcpv4Session.
+
+namespace {
+
+const char* const DHCP4_NAME = "kea-dhcp4";
+
+/// @brief Prints Kea Usage and exits
+///
+/// Note: This function never returns. It terminates the process.
+void
+usage() {
+ cerr << "Kea DHCPv4 server, "
+ << "version " << VERSION
+ << " (" << PACKAGE_VERSION_TYPE << ")"
+ << endl;
+ cerr << endl;
+ cerr << "Usage: " << DHCP4_NAME
+ << " -[v|V|W] [-d] [-{c|t|T} cfgfile] [-p number] [-P number]" << endl;
+ cerr << " -v: print version number and exit" << endl;
+ cerr << " -V: print extended version and exit" << endl;
+ cerr << " -W: display the configuration report and exit" << endl;
+ cerr << " -d: debug mode with extra verbosity (former -v)" << endl;
+ cerr << " -c file: specify configuration file" << endl;
+ cerr << " -t file: check the configuration file syntax and exit" << endl;
+ cerr << " -T file: check the configuration file doing hooks load and extra "
+ << "checks and exit" << endl;
+ cerr << " -p number: specify non-standard server port number 1-65535 "
+ << "(useful for testing only)" << endl;
+ cerr << " -P number: specify non-standard client port number 1-65535 "
+ << "(useful for testing only)" << endl;
+ exit(EXIT_FAILURE);
+}
+} // namespace
+
+int
+main(int argc, char* argv[]) {
+ int ch;
+ // The default. Any other values are useful for testing only.
+ int server_port_number = DHCP4_SERVER_PORT;
+ // Not zero values are useful for testing only.
+ int client_port_number = 0;
+ bool verbose_mode = false; // Should server be verbose?
+ bool check_mode = false; // Check syntax
+ bool load_hooks = false; // Check hooks config
+
+ // The standard config file
+ std::string config_file("");
+
+ while ((ch = getopt(argc, argv, "dvVWc:p:P:t:T:")) != -1) {
+ switch (ch) {
+ case 'd':
+ verbose_mode = true;
+ break;
+
+ case 'v':
+ cout << Dhcpv4Srv::getVersion(false) << endl;
+ return (EXIT_SUCCESS);
+
+ case 'V':
+ cout << Dhcpv4Srv::getVersion(true) << endl;
+ return (EXIT_SUCCESS);
+
+ case 'W':
+ cout << isc::detail::getConfigReport() << endl;
+ return (EXIT_SUCCESS);
+
+ case 'T':
+ load_hooks = true;
+ check_mode = true;
+ config_file = optarg;
+ break;
+
+ case 't':
+ check_mode = true;
+ config_file = optarg;
+ break;
+
+ case 'c': // config file
+ config_file = optarg;
+ break;
+
+ case 'p': // server port number
+ try {
+ server_port_number = boost::lexical_cast<int>(optarg);
+ } catch (const boost::bad_lexical_cast &) {
+ cerr << "Failed to parse server port number: [" << optarg
+ << "], 1-65535 allowed." << endl;
+ usage();
+ }
+ if (server_port_number <= 0 || server_port_number > 65535) {
+ cerr << "Failed to parse server port number: [" << optarg
+ << "], 1-65535 allowed." << endl;
+ usage();
+ }
+ break;
+
+ case 'P': // client port number
+ try {
+ client_port_number = boost::lexical_cast<int>(optarg);
+ } catch (const boost::bad_lexical_cast &) {
+ cerr << "Failed to parse client port number: [" << optarg
+ << "], 1-65535 allowed." << endl;
+ usage();
+ }
+ if (client_port_number <= 0 || client_port_number > 65535) {
+ cerr << "Failed to parse client port number: [" << optarg
+ << "], 1-65535 allowed." << endl;
+ usage();
+ }
+ break;
+
+ default:
+ usage();
+ }
+ }
+
+ // Check for extraneous parameters.
+ if (argc > optind) {
+ usage();
+ }
+
+ // Configuration file is required.
+ if (config_file.empty()) {
+ cerr << "Configuration file not specified." << endl;
+ usage();
+ }
+
+ // This is the DHCPv4 server
+ CfgMgr::instance().setFamily(AF_INET);
+
+ if (check_mode) {
+ try {
+ // We need to initialize logging, in case any error messages are to be printed.
+ // This is just a test, so we don't care about lockfile.
+ setenv("KEA_LOCKFILE_DIR", "none", 0);
+ Daemon::setDefaultLoggerName(DHCP4_ROOT_LOGGER_NAME);
+ Daemon::loggerInit(DHCP4_ROOT_LOGGER_NAME, verbose_mode);
+
+ // Check the syntax first.
+ Parser4Context parser;
+ ConstElementPtr json;
+ json = parser.parseFile(config_file, Parser4Context::PARSER_DHCP4);
+ if (!json) {
+ cerr << "No configuration found" << endl;
+ return (EXIT_FAILURE);
+ }
+ if (verbose_mode) {
+ cerr << "Syntax check OK" << endl;
+ }
+
+ // Check the logic next.
+ ConstElementPtr dhcp4 = json->get("Dhcp4");
+ if (!dhcp4) {
+ cerr << "Missing mandatory Dhcp4 element" << endl;
+ return (EXIT_FAILURE);
+ }
+ ControlledDhcpv4Srv server(0);
+ ConstElementPtr answer;
+
+ server.setProcName(DHCP4_NAME);
+
+ // Now we pass the Dhcp4 configuration to the server, but
+ // tell it to check the configuration only (check_only = true)
+ answer = configureDhcp4Server(server, dhcp4, true, load_hooks);
+
+ int status_code = 0;
+ answer = isc::config::parseAnswer(status_code, answer);
+ if (status_code == 0) {
+ return (EXIT_SUCCESS);
+ } else {
+ cerr << "Error encountered: " << answer->stringValue() << endl;
+ return (EXIT_FAILURE);
+ }
+ } catch (const std::exception& ex) {
+ cerr << "Syntax check failed with: " << ex.what() << endl;
+ }
+ return (EXIT_FAILURE);
+ }
+
+ int ret = EXIT_SUCCESS;
+ try {
+ // It is important that we set a default logger name because this name
+ // will be used when the user doesn't provide the logging configuration
+ // in the Kea configuration file.
+ Daemon::setDefaultLoggerName(DHCP4_ROOT_LOGGER_NAME);
+
+ // Initialize logging. If verbose, we'll use maximum verbosity.
+ Daemon::loggerInit(DHCP4_ROOT_LOGGER_NAME, verbose_mode);
+ LOG_DEBUG(dhcp4_logger, DBG_DHCP4_START, DHCP4_START_INFO)
+ .arg(getpid())
+ .arg(server_port_number)
+ .arg(client_port_number)
+ .arg(verbose_mode ? "yes" : "no");
+
+ LOG_INFO(dhcp4_logger, DHCP4_STARTING)
+ .arg(VERSION)
+ .arg(PACKAGE_VERSION_TYPE);
+
+ if (string(PACKAGE_VERSION_TYPE) == "development") {
+ LOG_WARN(dhcp4_logger, DHCP4_DEVELOPMENT_VERSION);
+ }
+
+ // Create the server instance.
+ ControlledDhcpv4Srv server(server_port_number, client_port_number);
+
+ // Remember verbose-mode
+ server.setVerbose(verbose_mode);
+
+ // Create our PID file.
+ server.setProcName(DHCP4_NAME);
+ server.setConfigFile(config_file);
+ server.createPIDFile();
+
+ try {
+ // Initialize the server.
+ server.init(config_file);
+ } catch (const std::exception& ex) {
+
+ // Let's log out what went wrong.
+ try {
+ // Log with the current logger, but only if it's not
+ // configured with console output so as to not log twice.
+ if (!dhcp4_logger.hasAppender(isc::log::OutputOption::DEST_CONSOLE)) {
+ LOG_ERROR(dhcp4_logger, DHCP4_INIT_FAIL).arg(ex.what());
+ }
+
+ // Log on the console as well.
+ isc::log::LoggerManager log_manager;
+ log_manager.process();
+ LOG_ERROR(dhcp4_logger, DHCP4_INIT_FAIL).arg(ex.what());
+ } catch (...) {
+ // The exception thrown during the initialization could
+ // originate from logger subsystem. Therefore LOG_ERROR()
+ // may fail as well.
+ cerr << "Failed to initialize server: " << ex.what() << endl;
+ }
+
+ return (EXIT_FAILURE);
+ }
+
+ // Tell the admin we are ready to process packets
+ LOG_INFO(dhcp4_logger, DHCP4_STARTED).arg(VERSION);
+
+ // And run the main loop of the server.
+ ret = server.run();
+
+ LOG_INFO(dhcp4_logger, DHCP4_SHUTDOWN);
+
+ } catch (const isc::process::DaemonPIDExists& ex) {
+ // First, we print the error on stderr (that should always work)
+ cerr << DHCP4_NAME << " already running? " << ex.what()
+ << endl;
+
+ // Let's also try to log it using logging system, but we're not
+ // sure if it's usable (the exception may have been thrown from
+ // the logger subsystem)
+ try {
+ LOG_FATAL(dhcp4_logger, DHCP4_ALREADY_RUNNING)
+ .arg(DHCP4_NAME).arg(ex.what());
+ } catch (...) {
+ // Already logged so ignore
+ }
+ ret = EXIT_FAILURE;
+ } catch (const std::exception& ex) {
+ // First, we print the error on stderr (that should always work)
+ cerr << DHCP4_NAME << ": Fatal error during start up: " << ex.what()
+ << endl;
+
+ // Let's also try to log it using logging system, but we're not
+ // sure if it's usable (the exception may have been thrown from
+ // the logger subsystem)
+ try {
+ LOG_FATAL(dhcp4_logger, DHCP4_SERVER_FAILED).arg(ex.what());
+ } catch (...) {
+ // Already logged so ignore
+ }
+ ret = EXIT_FAILURE;
+ } catch (...) {
+ cerr << DHCP4_NAME << ": Fatal error during start up"
+ << endl;
+ ret = EXIT_FAILURE;
+ }
+
+ return (ret);
+}