// Copyright (C) 2010 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 // For random(), temporary until better forwarding is done #include // for some IPC/network system calls #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace asio; using asio::ip::udp; using asio::ip::tcp; using namespace std; using namespace isc::dns; using isc::log::dlog; using namespace boost; namespace asiolink { class IOServiceImpl { private: IOServiceImpl(const IOService& source); IOServiceImpl& operator=(const IOService& source); public: /// \brief The constructor IOServiceImpl() : io_service_(), work_(io_service_) {}; /// \brief The destructor. ~IOServiceImpl() {}; //@} /// \brief Start the underlying event loop. /// /// This method does not return control to the caller until /// the \c stop() method is called via some handler. void run() { io_service_.run(); }; /// \brief Run the underlying event loop for a single event. /// /// This method return control to the caller as soon as the /// first handler has completed. (If no handlers are ready when /// it is run, it will block until one is.) void run_one() { io_service_.run_one();} ; /// \brief Stop the underlying event loop. /// /// This will return the control to the caller of the \c run() method. void stop() { io_service_.stop();} ; /// \brief Return the native \c io_service object used in this wrapper. /// /// This is a short term work around to support other BIND 10 modules /// that share the same \c io_service with the authoritative server. /// It will eventually be removed once the wrapper interface is /// generalized. asio::io_service& get_io_service() { return io_service_; }; private: asio::io_service io_service_; asio::io_service::work work_; }; IOService::IOService() { io_impl_ = new IOServiceImpl(); } IOService::~IOService() { delete io_impl_; } void IOService::run() { io_impl_->run(); } void IOService::run_one() { io_impl_->run_one(); } void IOService::stop() { io_impl_->stop(); } asio::io_service& IOService::get_io_service() { return (io_impl_->get_io_service()); } class DNSServiceImpl { public: DNSServiceImpl(IOService& io_service, const char& port, const ip::address* v4addr, const ip::address* v6addr, SimpleCallback* checkin, DNSLookup* lookup, DNSAnswer* answer); IOService& io_service_; typedef boost::shared_ptr UDPServerPtr; typedef boost::shared_ptr TCPServerPtr; typedef boost::shared_ptr DNSServerPtr; vector servers_; SimpleCallback *checkin_; DNSLookup *lookup_; DNSAnswer *answer_; void addServer(uint16_t port, const ip::address& address) { try { dlog(std::string("Initialize TCP server at ") + address.to_string() + ":" + boost::lexical_cast(port)); TCPServerPtr tcpServer(new TCPServer(io_service_.get_io_service(), address, port, checkin_, lookup_, answer_)); (*tcpServer)(); servers_.push_back(tcpServer); dlog(std::string("Initialize UDP server at ") + address.to_string() + ":" + boost::lexical_cast(port)); UDPServerPtr udpServer(new UDPServer(io_service_.get_io_service(), address, port, checkin_, lookup_, answer_)); (*udpServer)(); servers_.push_back(udpServer); } catch (const asio::system_error& err) { // We need to catch and convert any ASIO level exceptions. // This can happen for unavailable address, binding a privilege port // without the privilege, etc. isc_throw(IOError, "Failed to initialize network servers: " << err.what()); } } void addServer(const char& port, const ip::address& address) { uint16_t portnum; try { // XXX: SunStudio with stlport4 doesn't reject some invalid // representation such as "-1" by lexical_cast, so // we convert it into a signed integer of a larger size and perform // range check ourselves. const int32_t portnum32 = boost::lexical_cast(&port); if (portnum32 < 0 || portnum32 > 65535) { isc_throw(IOError, "Invalid port number '" << &port); } portnum = portnum32; } catch (const boost::bad_lexical_cast& ex) { isc_throw(IOError, "Invalid port number '" << &port << "': " << ex.what()); } addServer(portnum, address); } }; DNSServiceImpl::DNSServiceImpl(IOService& io_service, const char& port, const ip::address* const v4addr, const ip::address* const v6addr, SimpleCallback* checkin, DNSLookup* lookup, DNSAnswer* answer) : io_service_(io_service), checkin_(checkin), lookup_(lookup), answer_(answer) { if (v4addr) { addServer(port, *v4addr); } if (v6addr) { addServer(port, *v6addr); } } DNSService::DNSService(IOService& io_service, const char& port, const char& address, SimpleCallback* checkin, DNSLookup* lookup, DNSAnswer* answer) : impl_(new DNSServiceImpl(io_service, port, NULL, NULL, checkin, lookup, answer)), io_service_(io_service) { addServer(port, &address); } DNSService::DNSService(IOService& io_service, const char& port, const bool use_ipv4, const bool use_ipv6, SimpleCallback* checkin, DNSLookup* lookup, DNSAnswer* answer) : impl_(NULL), io_service_(io_service) { const ip::address v4addr_any = ip::address(ip::address_v4::any()); const ip::address* const v4addrp = use_ipv4 ? &v4addr_any : NULL; const ip::address v6addr_any = ip::address(ip::address_v6::any()); const ip::address* const v6addrp = use_ipv6 ? &v6addr_any : NULL; impl_ = new DNSServiceImpl(io_service, port, v4addrp, v6addrp, checkin, lookup, answer); } DNSService::DNSService(IOService& io_service, SimpleCallback* checkin, DNSLookup* lookup, DNSAnswer *answer) : impl_(new DNSServiceImpl(io_service, *"0", NULL, NULL, checkin, lookup, answer)), io_service_(io_service) { } DNSService::~DNSService() { delete impl_; } namespace { typedef std::vector > AddressVector; } RecursiveQuery::RecursiveQuery(DNSService& dns_service, const AddressVector& upstream, int timeout, unsigned retries) : dns_service_(dns_service), upstream_(new AddressVector(upstream)), timeout_(timeout), retries_(retries) {} namespace { ip::address convertAddr(const string& address) { error_code err; ip::address addr = ip::address::from_string(address, err); if (err) { isc_throw(IOError, "Invalid IP address '" << &address << "': " << err.message()); } return addr; } } void DNSService::addServer(const char& port, const string& address) { impl_->addServer(port, convertAddr(address)); } void DNSService::addServer(uint16_t port, const string& address) { impl_->addServer(port, convertAddr(address)); } void DNSService::clearServers() { // FIXME: This does not work, it does not close the socket. // How is it done? impl_->servers_.clear(); } namespace { /* * This is a query in progress. When a new query is made, this one holds * the context information about it, like how many times we are allowed * to retry on failure, what to do when we succeed, etc. * * Used by RecursiveQuery::sendQuery. */ class RunningQuery : public UDPQuery::Callback { private: // The io service to handle async calls asio::io_service& io_; // Info for (re)sending the query (the question and destination) Question question_; shared_ptr upstream_; // Buffer to store the result. OutputBufferPtr buffer_; // Server to notify when we succeed or fail shared_ptr server_; /* * TODO Do something more clever with timeouts. In the long term, some * computation of average RTT, increase with each retry, etc. */ // Timeout information int timeout_; unsigned retries_; // (re)send the query to the server. void send() { const int uc = upstream_->size(); if (uc > 0) { int serverIndex(random() % uc); dlog("Sending upstream query (" + question_.toText() + ") to " + upstream_->at(serverIndex).first); UDPQuery query(io_, question_, upstream_->at(serverIndex).first, upstream_->at(serverIndex).second, buffer_, this, timeout_); io_.post(query); } else { dlog("Error, no upstream servers to send to."); } } public: RunningQuery(asio::io_service& io, const Question &question, shared_ptr upstream, OutputBufferPtr buffer, DNSServer* server, int timeout, unsigned retries) : io_(io), question_(question), upstream_(upstream), buffer_(buffer), server_(server->clone()), timeout_(timeout), retries_(retries) { send(); } // This function is used as callback from DNSQuery. virtual void operator()(UDPQuery::Result result) { if (result == UDPQuery::TIME_OUT && retries_ --) { dlog("Resending query"); // We timed out, but we have some retries, so send again send(); } else { server_->resume(result == UDPQuery::SUCCESS); delete this; } } }; } void RecursiveQuery::sendQuery(const Question& question, OutputBufferPtr buffer, DNSServer* server) { // XXX: eventually we will need to be able to determine whether // the message should be sent via TCP or UDP, or sent initially via // UDP and then fall back to TCP on failure, but for the moment // we're only going to handle UDP. asio::io_service& io = dns_service_.get_io_service(); // It will delete itself when it is done new RunningQuery(io, question, upstream_, buffer, server, timeout_, retries_); } }