123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235 |
- // Copyright (C) 2009-2011 Internet Systems Consortium, Inc. ("ISC")
- //
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
- // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
- // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
- // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- // PERFORMANCE OF THIS SOFTWARE.
- #include <config.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <sys/select.h>
- #include <netdb.h>
- #include <netinet/in.h>
- #include <stdlib.h>
- #include <errno.h>
- #include <cassert>
- #include <iostream>
- #include <exceptions/exceptions.h>
- #include <util/buffer.h>
- #include <util/io/socketsession.h>
- #include <dns/message.h>
- #include <dns/messagerenderer.h>
- #include <cc/session.h>
- #include <cc/data.h>
- #include <config/ccsession.h>
- #include <xfr/xfrout_client.h>
- #include <auth/spec_config.h>
- #include <auth/common.h>
- #include <auth/auth_config.h>
- #include <auth/command.h>
- #include <auth/auth_srv.h>
- #include <auth/auth_log.h>
- #include <asiodns/asiodns.h>
- #include <asiolink/asiolink.h>
- #include <log/logger_support.h>
- #include <server_common/keyring.h>
- #include <server_common/socket_request.h>
- using namespace std;
- using namespace isc::asiodns;
- using namespace isc::asiolink;
- using namespace isc::auth;
- using namespace isc::cc;
- using namespace isc::config;
- using namespace isc::data;
- using namespace isc::dns;
- using namespace isc::log;
- using namespace isc::util;
- using namespace isc::util::io;
- using namespace isc::xfr;
- namespace {
- /* need global var for config/command handlers.
- * todo: turn this around, and put handlers in the authserver
- * class itself? */
- AuthSrv *auth_server;
- ConstElementPtr
- my_config_handler(ConstElementPtr new_config) {
- return (auth_server->updateConfig(new_config));
- }
- ConstElementPtr
- my_command_handler(const string& command, ConstElementPtr args) {
- assert(auth_server != NULL);
- return (execAuthServerCommand(*auth_server, command, args));
- }
- void
- usage() {
- cerr << "Usage: b10-auth [-u user] [-nv]"
- << endl;
- cerr << "\t-n: do not cache answers in memory" << endl;
- cerr << "\t-v: verbose output" << endl;
- exit(1);
- }
- } // end of anonymous namespace
- int
- main(int argc, char* argv[]) {
- int ch;
- bool cache = true;
- bool verbose = false;
- while ((ch = getopt(argc, argv, ":nu:v")) != -1) {
- switch (ch) {
- case 'n':
- cache = false;
- break;
- case 'v':
- verbose = true;
- break;
- case '?':
- default:
- usage();
- }
- }
- if (argc - optind > 0) {
- usage();
- }
- // Initialize logging. If verbose, we'll use maximum verbosity.
- isc::log::initLogger(AUTH_NAME,
- (verbose ? isc::log::DEBUG : isc::log::INFO),
- isc::log::MAX_DEBUG_LEVEL, NULL);
- int ret = 0;
- // XXX: we should eventually pass io_service here.
- Session* cc_session = NULL;
- Session* xfrin_session = NULL;
- Session* statistics_session = NULL;
- bool xfrin_session_established = false; // XXX (see Trac #287)
- bool statistics_session_established = false; // XXX (see Trac #287)
- ModuleCCSession* config_session = NULL;
- XfroutClient xfrout_client(getXfroutSocketPath());
- SocketSessionForwarder ddns_forwarder(getDDNSSocketPath());
- try {
- string specfile;
- if (getenv("B10_FROM_BUILD")) {
- specfile = string(getenv("B10_FROM_BUILD")) +
- "/src/bin/auth/auth.spec";
- } else {
- specfile = string(AUTH_SPECFILE_LOCATION);
- }
- auth_server = new AuthSrv(cache, xfrout_client, ddns_forwarder);
- LOG_INFO(auth_logger, AUTH_SERVER_CREATED);
- SimpleCallback* checkin = auth_server->getCheckinProvider();
- IOService& io_service = auth_server->getIOService();
- DNSLookup* lookup = auth_server->getDNSLookupProvider();
- DNSAnswer* answer = auth_server->getDNSAnswerProvider();
- DNSService dns_service(io_service, checkin, lookup, answer);
- auth_server->setDNSService(dns_service);
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_DNS_SERVICES_CREATED);
- cc_session = new Session(io_service.get_io_service());
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_CONFIG_CHANNEL_CREATED);
- // Initialize the Socket Requestor
- isc::server_common::initSocketRequestor(*cc_session, AUTH_NAME);
- // We delay starting listening to new commands/config just before we
- // go into the main loop to avoid confusion due to mixture of
- // synchronous and asynchronous operations (this would happen in
- // initial communication with the boss that takes place in
- // updateConfig() for listen_on and in initializing TSIG keys below).
- // Until then all operations on the CC session will take place
- // synchronously.
- config_session = new ModuleCCSession(specfile, *cc_session,
- my_config_handler,
- my_command_handler, false);
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_CONFIG_CHANNEL_ESTABLISHED);
- xfrin_session = new Session(io_service.get_io_service());
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_XFRIN_CHANNEL_CREATED);
- xfrin_session->establish(NULL);
- xfrin_session_established = true;
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_XFRIN_CHANNEL_ESTABLISHED);
- statistics_session = new Session(io_service.get_io_service());
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_STATS_CHANNEL_CREATED);
- statistics_session->establish(NULL);
- statistics_session_established = true;
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_STATS_CHANNEL_ESTABLISHED);
- auth_server->setXfrinSession(xfrin_session);
- auth_server->setStatisticsSession(statistics_session);
- // Configure the server. configureAuthServer() is expected to install
- // all initial configurations, but as a short term workaround we
- // handle the traditional "database_file" setup by directly calling
- // updateConfig().
- // if server load configure failed, we won't exit, give user second
- // chance to correct the configure.
- auth_server->setConfigSession(config_session);
- try {
- configureAuthServer(*auth_server, config_session->getFullConfig());
- auth_server->updateConfig(ElementPtr());
- } catch (const AuthConfigError& ex) {
- LOG_ERROR(auth_logger, AUTH_CONFIG_LOAD_FAIL).arg(ex.what());
- }
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_LOAD_TSIG);
- isc::server_common::initKeyring(*config_session);
- auth_server->setTSIGKeyRing(&isc::server_common::keyring);
- // Now start asynchronous read.
- config_session->start();
- LOG_DEBUG(auth_logger, DBG_AUTH_START, AUTH_CONFIG_CHANNEL_STARTED);
- // Successfully initialized.
- LOG_INFO(auth_logger, AUTH_SERVER_STARTED);
- io_service.run();
- } catch (const std::exception& ex) {
- LOG_FATAL(auth_logger, AUTH_SERVER_FAILED).arg(ex.what());
- ret = 1;
- }
- if (statistics_session_established) {
- statistics_session->disconnect();
- }
- if (xfrin_session_established) {
- xfrin_session->disconnect();
- }
- delete statistics_session;
- delete xfrin_session;
- delete config_session;
- delete cc_session;
- delete auth_server;
- return (ret);
- }
|