acl/app/net_tools/dns/nslookup.cpp

212 lines
4.6 KiB
C++
Raw Normal View History

#include "StdAfx.h"
2014-11-19 00:25:21 +08:00
#include "dns_store.h"
#include "global/util.h"
#include "rpc/rpc_manager.h"
#include "nslookup.h"
//////////////////////////////////////////////////////////////////////////
domain_info::domain_info(nslookup& ns, const char* domain)
: ns_(ns)
{
ACL_SAFE_STRNCPY(domain_, domain, sizeof(domain_));
}
domain_info::~domain_info()
{
std::vector<IP_INFO*>::iterator it = ip_list_.begin();
for (; it != ip_list_.end(); ++it)
acl_myfree(*it);
}
void domain_info::add_ip(const char* ip, int ttl)
{
IP_INFO* info = (IP_INFO*) acl_mycalloc(1, sizeof(IP_INFO));
ACL_SAFE_STRNCPY(info->ip, ip, sizeof(info->ip));
info->ttl = ttl;
ip_list_.push_back(info);
}
void domain_info::set_begin()
{
gettimeofday(&begin_, NULL);
}
void domain_info::set_end()
{
gettimeofday(&end_, NULL);
}
double domain_info::get_spent() const
{
return util::stamp_sub(&end_, &begin_);
}
//////////////////////////////////////////////////////////////////////////
nslookup::nslookup(const char* filepath, nslookup_callback* callback,
const char* dns_ip, int dns_port, int timeout)
: filepath_(filepath)
, callback_(callback)
, dns_ip_(dns_ip)
, dns_port_(dns_port)
, timeout_(timeout)
, nresult_(0)
, domain_list_(NULL)
{
}
nslookup::~nslookup()
{
if (domain_list_)
delete domain_list_;
}
//////////////////////////////////////////////////////////////////////////
// <20><><EFBFBD>߳<EFBFBD><DFB3><EFBFBD><EFBFBD><EFBFBD>
2014-11-19 00:25:21 +08:00
void nslookup::rpc_onover()
{
callback_->nslookup_report(domain_list_->size(),
domain_list_->size());
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ݿ⣬ͬʱ<CDAC><CAB1><EFBFBD>ص<EFBFBD><D8B5>ӿڴ<D3BF><DAB4><EFBFBD>
2014-11-19 00:25:21 +08:00
dns_store* ds = new dns_store(domain_list_, *callback_);
// <20><>Ϊ<EFBFBD>ñ<EFBFBD><C3B1><EFBFBD><EFBFBD>Ѿ<EFBFBD><D1BE><EFBFBD><EFBFBD>ӹܣ<D3B9><DCA3><EFBFBD><EFBFBD>Դ˴<D4B4><CBB4><EFBFBD>Ҫ<EFBFBD>ÿ<EFBFBD><C3BF><EFBFBD><EFBFBD><EFBFBD>ظ<EFBFBD><D8B8>ͷ<EFBFBD>
2014-11-19 00:25:21 +08:00
domain_list_ = NULL;
rpc_manager::get_instance().fork(ds);
delete this;
}
void nslookup::rpc_wakeup(void*)
{
callback_->nslookup_report(domain_list_->size(), nresult_);
}
//////////////////////////////////////////////////////////////////////////
// <20><><EFBFBD>߳<EFBFBD><DFB3><EFBFBD><EFBFBD><EFBFBD>
2014-11-19 00:25:21 +08:00
void nslookup::rpc_run()
{
if (load_file() == true)
lookup_all();
}
bool nslookup::load_file()
{
acl::ifstream in;
if (in.open_read(filepath_) == false)
{
logger_error("open file(%s) failed: %s",
filepath_.c_str(), acl::last_serror());
return false;
}
domain_list_ = new std::vector<domain_info*>;
acl::string line;
while (in.eof() == false)
{
if (in.gets(line) == false)
break;
domain_info* di = new domain_info(*this, line.c_str());
domain_list_->push_back(di);
}
if (domain_list_->empty())
{
logger_error("no ip in file %s", filepath_.c_str());
return false;
}
return true;
}
void nslookup::lookup_all()
{
if (domain_list_ == NULL || domain_list_->empty())
{
logger_error("domain empty");
return;
}
ACL_AIO *aio;
/* <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>첽ͨ<ECB2BD>ž<EFBFBD><C5BE><EFBFBD> */
2014-11-19 00:25:21 +08:00
aio = acl_aio_create(ACL_EVENT_SELECT);
// acl_aio_set_keep_read(aio, 0);
// <20><><EFBFBD><EFBFBD> DNS <20><>ѯ<EFBFBD><D1AF><EFBFBD><EFBFBD>
2014-11-19 00:25:21 +08:00
ACL_DNS* dns = acl_dns_create(aio, timeout_);
acl_dns_add_dns(dns, dns_ip_.c_str(), dns_port_, 24);
rpc_signal(NULL);
time_t last_signal = time(NULL), t;
// <20><><EFBFBD><EFBFBD>Ŀ<EFBFBD><C4BF> domain <20><>ַ
2014-11-19 00:25:21 +08:00
std::vector<domain_info*>::iterator it = domain_list_->begin();
for (; it != domain_list_->end(); ++it)
{
(*it)->set_begin();
acl_dns_lookup(dns, (*it)->get_domain(), dns_result, *it);
}
while (1) {
/* <20><EFBFBD>¼<EFBFBD>ѭ<EFBFBD><D1AD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
2014-11-19 00:25:21 +08:00
acl_aio_loop(aio);
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>в<EFBFBD>ѯ<EFBFBD><D1AF><EFBFBD>̶<EFBFBD><CCB6><EFBFBD><EFBFBD>ɣ<EFBFBD><C9A3><EFBFBD><EFBFBD>˳<EFBFBD><CBB3><EFBFBD>¼<EFBFBD>ѭ<EFBFBD><D1AD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
2014-11-19 00:25:21 +08:00
if (nresult_ >= domain_list_->size())
{
logger("DNS lookup over: %d, %d",
(int) domain_list_->size(), (int) nresult_);
break;
}
t = time(NULL);
if (t - last_signal >= 1)
{
last_signal = t;
rpc_signal(NULL);
}
}
/* <20><>ʾ<EFBFBD><CABE><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѯ<EFBFBD><D1AF><EFBFBD><EFBFBD> */
2014-11-19 00:25:21 +08:00
acl_dns_close(dns);
/* <20><><EFBFBD>ٷ<EFBFBD><D9B7><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
2014-11-19 00:25:21 +08:00
acl_aio_free(aio);
}
2022-05-19 23:57:35 +08:00
void nslookup::dns_result(ACL_DNS_DB *dns_db, void *ctx, int errnum,
const ACL_RFC1035_MESSAGE*)
2014-11-19 00:25:21 +08:00
{
domain_info* info = (domain_info*) ctx;
info->set_end();
if (dns_db == NULL) {
logger("ERROR: %s, domain: %s, spent: %0.2f",
acl_dns_serror(errnum), info->get_domain(),
info->get_spent());
info->add_ip("0.0.0.0", 0);
info->get_nslookup().nresult_++;
return;
}
ACL_ITER iter;
acl::string buf;
buf.format("OK, domain: %s, spent: %0.2f, ip_list: ",
info->get_domain(), info->get_spent());
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>в<EFBFBD>ѯ<EFBFBD><D1AF><EFBFBD><EFBFBD>
2014-11-19 00:25:21 +08:00
const ACL_HOST_INFO *hi;
acl_foreach(iter, dns_db) {
hi = (const ACL_HOST_INFO*) iter.data;
if (iter.i > 0)
buf << ", ";
buf.format_append("ip=%s, ttl=%d", hi->ip, hi->ttl);
info->add_ip(hi->ip, hi->ttl);
}
logger("%s", buf.c_str());
info->get_nslookup().nresult_++;
}