// Copyright (C) 2009 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. // $Id$ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace isc::data; using namespace isc::cc; using namespace isc::config; using namespace isc::dns; using namespace isc::xfr; namespace { bool verbose_mode = false; const string PROGRAM = "Auth"; const char* DNSPORT = "5300"; /* need global var for config/command handlers. * todo: turn this around, and put handlers in the authserver * class itself? */ AuthSrv *auth_server; asio_link::IOService* io_service; ElementPtr my_config_handler(ElementPtr new_config) { return auth_server->updateConfig(new_config); } ElementPtr my_command_handler(const string& command, const ElementPtr args) { ElementPtr answer = createAnswer(); if (command == "print_message") { cout << args << endl; /* let's add that message to our answer as well */ answer->get("result")->add(args); } else if (command == "shutdown") { io_service->stop(); } return answer; } void usage() { cerr << "Usage: b10-auth [-a address] [-p port] [-4|-6] [-nv]" << endl; exit(1); } } // end of anonymous namespace int main(int argc, char* argv[]) { int ch; const char* port = DNSPORT; const char* address = NULL; bool use_ipv4 = true, use_ipv6 = true, cache = true; while ((ch = getopt(argc, argv, "46a:np:v")) != -1) { switch (ch) { case '4': // Note that -4 means "ipv4 only", we need to set "use_ipv6" here, // not "use_ipv4". We could use something like "ipv4_only", but // we found the negatively named variable could confuse the code // logic. use_ipv6 = false; break; case '6': // The same note as -4 applies. use_ipv4 = false; break; case 'n': cache = false; break; case 'a': address = optarg; break; case 'p': port = optarg; break; case 'v': verbose_mode = true; break; case '?': default: usage(); } } if (argc - optind > 0) { usage(); } if (!use_ipv4 && !use_ipv6) { cerr << "[b10-auth] Error: -4 and -6 can't coexist" << endl; usage(); } if ((!use_ipv4 || !use_ipv6) && address != NULL) { cerr << "[b10-auth] Error: -4|-6 and -a can't coexist" << endl; usage(); } int ret = 0; // XXX: we should eventually pass io_service here. Session* cc_session = NULL; Session* xfrin_session = NULL; bool xfrin_session_established = false; // XXX (see Trac #287) ModuleCCSession* config_session = NULL; XfroutClient xfrout_client(UNIX_SOCKET_FILE); 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); auth_server->setVerbose(verbose_mode); cout << "[b10-auth] Server created." << endl; if (address != NULL) { // XXX: we can only specify at most one explicit address. // This also means the server cannot run in the dual address // family mode if explicit addresses need to be specified. // We don't bother to fix this problem, however. The -a option // is a short term workaround until we support dynamic listening // port allocation. io_service = new asio_link::IOService(auth_server, *port, *address); } else { io_service = new asio_link::IOService(auth_server, *port, use_ipv4, use_ipv6); } cout << "[b10-auth] IOService created." << endl; cc_session = new Session(io_service->get_io_service()); cout << "[b10-auth] Configuration session channel created." << endl; config_session = new ModuleCCSession(specfile, *cc_session, my_config_handler, my_command_handler); cout << "[b10-auth] Configuration channel established." << endl; xfrin_session = new Session(io_service->get_io_service()); cout << "[b10-auth] Xfrin session channel created." << endl; xfrin_session->establish(NULL); xfrin_session_established = true; cout << "[b10-auth] Xfrin session channel established." << endl; // XXX: with the current interface to asio_link we have to create // auth_server before io_service while Session needs io_service. // In a next step of refactoring we should make asio_link independent // from auth_server, and create io_service, auth_server, and // sessions in that order. auth_server->setXfrinSession(xfrin_session); auth_server->setConfigSession(config_session); auth_server->updateConfig(ElementPtr()); cout << "[b10-auth] Server started." << endl; io_service->run(); } catch (const std::exception& ex) { cerr << "[b10-auth] Initialization failed: " << ex.what() << endl; ret = 1; } if (xfrin_session_established) { xfrin_session->disconnect(); } delete xfrin_session; delete config_session; delete cc_session; delete io_service; delete auth_server; return (ret); }