2019-07-28 10:31:56 +08:00
|
|
|
|
#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_;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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());
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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_);
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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;
|
2019-07-28 10:31:56 +08:00
|
|
|
|
/* <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);
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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;
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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) {
|
2019-07-28 10:31:56 +08:00
|
|
|
|
/* <20>첽<EFBFBD>¼<EFBFBD>ѭ<EFBFBD><D1AD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> */
|
2014-11-19 00:25:21 +08:00
|
|
|
|
acl_aio_loop(aio);
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
/* <20><>ʾ<EFBFBD><CABE><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ѯ<EFBFBD><D1AF><EFBFBD><EFBFBD> */
|
2014-11-19 00:25:21 +08:00
|
|
|
|
|
|
|
|
|
acl_dns_close(dns);
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
/* <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());
|
|
|
|
|
|
2019-07-28 10:31:56 +08:00
|
|
|
|
// <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_++;
|
|
|
|
|
}
|