blob: d14464573316389df7ce4eeeb5cc47da38f77bb8 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/**
* Copyright (c) 2014-2015, Regents of the University of California,
* Arizona Board of Regents,
* Colorado State University,
* University Pierre & Marie Curie, Sorbonne University,
* Washington University in St. Louis,
* Beijing Institute of Technology,
* The University of Memphis.
*
* This file is part of NFD (Named Data Networking Forwarding Daemon).
* See AUTHORS.md for complete list of NFD authors and contributors.
*
* NFD is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*
* NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with
* NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef NFD_CORE_RESOLVER_H
#define NFD_CORE_RESOLVER_H
#include "common.hpp"
#include "core/global-io.hpp"
#include "core/scheduler.hpp"
namespace nfd {
namespace resolver {
typedef function<bool (const boost::asio::ip::address& address)> AddressSelector;
struct AnyAddress {
bool
operator()(const boost::asio::ip::address& address)
{
return true;
}
};
struct Ipv4Address {
bool
operator()(const boost::asio::ip::address& address)
{
return address.is_v4();
}
};
struct Ipv6Address {
bool
operator()(const boost::asio::ip::address& address)
{
return address.is_v6();
}
};
} // namespace resolver
template<class Protocol>
class Resolver
{
public:
struct Error : public std::runtime_error
{
Error(const std::string& what) : std::runtime_error(what) {}
};
typedef function<void (const typename Protocol::endpoint& endpoint)> SuccessCallback;
typedef function<void (const std::string& reason)> ErrorCallback;
typedef boost::asio::ip::basic_resolver< Protocol > resolver;
/** \brief Asynchronously resolve host and port
*
* If an address selector predicate is specified, then each resolved IP address
* is checked against the predicate.
*
* Available address selector predicates:
*
* - resolver::AnyAddress()
* - resolver::Ipv4Address()
* - resolver::Ipv6Address()
*/
static void
asyncResolve(const std::string& host, const std::string& port,
const SuccessCallback& onSuccess,
const ErrorCallback& onError,
const nfd::resolver::AddressSelector& addressSelector = nfd::resolver::AnyAddress(),
const time::seconds& timeout = time::seconds(4))
{
shared_ptr<Resolver> resolver =
shared_ptr<Resolver>(new Resolver(onSuccess, onError,
addressSelector));
resolver->asyncResolve(host, port, timeout, resolver);
// resolver will be destroyed when async operation finishes or global IO service stops
}
/** \brief Synchronously resolve host and port
*
* If an address selector predicate is specified, then each resolved IP address
* is checked against the predicate.
*
* Available address selector predicates:
*
* - resolver::AnyAddress()
* - resolver::Ipv4Address()
* - resolver::Ipv6Address()
*/
static typename Protocol::endpoint
syncResolve(const std::string& host, const std::string& port,
const nfd::resolver::AddressSelector& addressSelector = nfd::resolver::AnyAddress())
{
Resolver resolver(SuccessCallback(), ErrorCallback(), addressSelector);
typename resolver::query query(host, port
#if not defined(__FreeBSD__)
, resolver::query::all_matching
#endif
);
typename resolver::iterator remoteEndpoint = resolver.m_resolver.resolve(query);
typename resolver::iterator end;
for (; remoteEndpoint != end; ++remoteEndpoint)
{
if (addressSelector(typename Protocol::endpoint(*remoteEndpoint).address()))
return *remoteEndpoint;
}
throw Error("No endpoint matching the specified address selector found");
}
private:
Resolver(const SuccessCallback& onSuccess,
const ErrorCallback& onError,
const nfd::resolver::AddressSelector& addressSelector)
: m_resolver(getGlobalIoService())
, m_addressSelector(addressSelector)
, m_onSuccess(onSuccess)
, m_onError(onError)
{
}
void
asyncResolve(const std::string& host, const std::string& port,
const time::seconds& timeout,
const shared_ptr<Resolver>& self)
{
typename resolver::query query(host, port
#if not defined(__FreeBSD__)
, resolver::query::all_matching
#endif
);
m_resolver.async_resolve(query,
bind(&Resolver::onResolveSuccess, this, _1, _2, self));
m_resolveTimeout = scheduler::schedule(timeout,
bind(&Resolver::onResolveError, this,
"Timeout", self));
}
void
onResolveSuccess(const boost::system::error_code& error,
typename resolver::iterator remoteEndpoint,
const shared_ptr<Resolver>& self)
{
scheduler::cancel(m_resolveTimeout);
if (error)
{
if (error == boost::system::errc::operation_canceled)
return;
return m_onError("Remote endpoint hostname or port cannot be resolved: " +
error.category().message(error.value()));
}
typename resolver::iterator end;
for (; remoteEndpoint != end; ++remoteEndpoint)
{
if (m_addressSelector(typename Protocol::endpoint(*remoteEndpoint).address()))
return m_onSuccess(*remoteEndpoint);
}
m_onError("No endpoint matching the specified address selector found");
}
void
onResolveError(const std::string& errorInfo,
const shared_ptr<Resolver>& self)
{
m_resolver.cancel();
m_onError(errorInfo);
}
private:
resolver m_resolver;
scheduler::EventId m_resolveTimeout;
nfd::resolver::AddressSelector m_addressSelector;
SuccessCallback m_onSuccess;
ErrorCallback m_onError;
};
typedef Resolver<boost::asio::ip::tcp> TcpResolver;
typedef Resolver<boost::asio::ip::udp> UdpResolver;
} // namespace nfd
#endif // NFD_CORE_RESOLVER_H