static_unittest.cc 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. // Copyright (C) 2010 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. // $Id$
  15. #include <stdint.h>
  16. #include <string>
  17. #include <vector>
  18. #include <config.h>
  19. #include <gtest/gtest.h>
  20. #include <dns/name.h>
  21. #include <dns/message.h>
  22. #include <dns/rdata.h>
  23. #include <dns/rrclass.h>
  24. #include <dns/rrtype.h>
  25. #include <dns/rdataclass.h>
  26. #include <dns/rrsetlist.h>
  27. #include <cc/data.h>
  28. #include <datasrc/query.h>
  29. #include <datasrc/data_source.h>
  30. #include <datasrc/static_datasrc.h>
  31. using namespace std;
  32. using namespace isc::dns;
  33. using namespace isc::dns::rdata;
  34. using namespace isc::datasrc;
  35. namespace {
  36. class StaticDataSourceTest : public ::testing::Test {
  37. protected:
  38. StaticDataSourceTest() : version_name("version.bind"),
  39. authors_name("authors.bind"),
  40. nomatch_name("example.com"),
  41. rrclass(RRClass::CH()), rrtype(RRType::TXT()),
  42. rrttl(RRTTL(0)), find_flags(0), matched_rdata(0)
  43. {
  44. // version.bind is answered with package name+version
  45. // (defined as PACKAGE_STRING in config.h)
  46. version_data.push_back(PACKAGE_STRING);
  47. // XXX: in addition, the order the following items matter.
  48. authors_data.push_back("Chen Zhengzhang");
  49. authors_data.push_back("Evan Hunt");
  50. authors_data.push_back("Han Feng");
  51. authors_data.push_back("Jelte Jansen");
  52. authors_data.push_back("Jeremy C. Reed");
  53. authors_data.push_back("Jin Jian");
  54. authors_data.push_back("JINMEI Tatuya");
  55. authors_data.push_back("Kazunori Fujiwara");
  56. authors_data.push_back("Michael Graff");
  57. authors_data.push_back("Michal Vaner");
  58. authors_data.push_back("Naoki Kambe");
  59. authors_data.push_back("Shane Kerr");
  60. authors_data.push_back("Shen Tingting");
  61. authors_data.push_back("Stephen Morris");
  62. authors_data.push_back("Zhang Likun");
  63. version_ns_data.push_back("version.bind.");
  64. authors_ns_data.push_back("authors.bind.");
  65. version_soa_data.push_back("version.bind. hostmaster.version.bind. "
  66. "0 28800 7200 604800 86400");
  67. authors_soa_data.push_back("authors.bind. hostmaster.authors.bind. "
  68. "0 28800 7200 604800 86400");
  69. }
  70. StaticDataSrc data_source;
  71. const Name version_name;
  72. const Name authors_name;
  73. const Name nomatch_name;
  74. const RRClass rrclass;
  75. RRType rrtype; // we allow this to be modified in the test
  76. RRTTL rrttl;
  77. RRsetList result_sets;
  78. uint32_t find_flags;
  79. unsigned matched_rdata;
  80. vector<RRType> types;
  81. vector<RRTTL> ttls;
  82. vector<const vector<string>* > answers;
  83. vector<string> version_data;
  84. vector<string> authors_data;
  85. vector<string> version_ns_data;
  86. vector<string> authors_ns_data;
  87. vector<string> version_soa_data;
  88. vector<string> authors_soa_data;
  89. };
  90. void
  91. checkRRset(ConstRRsetPtr rrset, const Name& expected_name,
  92. const RRClass& expected_class, const RRType& expected_type,
  93. const RRTTL& rrttl, const vector<string>& expected_data)
  94. {
  95. EXPECT_EQ(expected_name, rrset->getName());
  96. EXPECT_EQ(expected_class, rrset->getClass());
  97. EXPECT_EQ(expected_type, rrset->getType());
  98. EXPECT_EQ(rrttl, rrset->getTTL());
  99. RdataIteratorPtr rdata_iterator = rrset->getRdataIterator();
  100. vector<string>::const_iterator data_it = expected_data.begin();
  101. for (; data_it != expected_data.end(); ++data_it) {
  102. EXPECT_FALSE(rdata_iterator->isLast());
  103. if (rdata_iterator->isLast()) {
  104. // buggy case, should stop here
  105. break;
  106. }
  107. EXPECT_EQ(0, (rdata_iterator->getCurrent()).compare(
  108. *createRdata(expected_type,
  109. expected_class,
  110. *data_it)));
  111. rdata_iterator->next();
  112. }
  113. EXPECT_TRUE(rdata_iterator->isLast());
  114. }
  115. void
  116. checkFind(const DataSrc& data_source,
  117. const Name& qname, const Name* zone_name,
  118. const RRClass& qclass, const RRClass& expected_class,
  119. const RRType& qtype, const vector<RRTTL>& expected_ttls,
  120. const uint32_t expected_flags, const vector<RRType>& expected_types,
  121. const vector<const vector<string>* >& expected_answers)
  122. {
  123. RRsetList result_sets;
  124. uint32_t find_flags;
  125. unsigned int rrset_matched = 0;
  126. unsigned int rrset_count = 0;
  127. EXPECT_EQ(DataSrc::SUCCESS,
  128. data_source.findRRset(qname, qclass, qtype, result_sets,
  129. find_flags, zone_name));
  130. EXPECT_EQ(expected_flags, find_flags);
  131. if ((find_flags & (DataSrc::NO_SUCH_ZONE | DataSrc::NAME_NOT_FOUND |
  132. DataSrc::TYPE_NOT_FOUND)) != 0) {
  133. EXPECT_TRUE(result_sets.begin() == result_sets.end());
  134. return;
  135. }
  136. RRsetList::iterator it = result_sets.begin();
  137. for (; it != result_sets.end(); ++it) {
  138. vector<RRType>::const_iterator found_type =
  139. find(expected_types.begin(), expected_types.end(),
  140. (*it)->getType());
  141. // there should be a match
  142. EXPECT_TRUE(found_type != expected_types.end());
  143. if (found_type != expected_types.end()) {
  144. unsigned int index = distance(expected_types.begin(), found_type);
  145. checkRRset(*it, qname, expected_class, (*it)->getType(),
  146. expected_ttls[index], *expected_answers[index]);
  147. ++rrset_matched;
  148. }
  149. ++rrset_count;
  150. }
  151. EXPECT_EQ(expected_types.size(), rrset_count);
  152. EXPECT_EQ(expected_types.size(), rrset_matched);
  153. }
  154. void
  155. checkFind(const DataSrc& data_source,
  156. const Name& qname, const Name* zone_name,
  157. const RRClass& qclass, const RRClass& expected_class,
  158. const RRType& qtype, // == expected RRType
  159. const RRTTL& expected_ttl, const uint32_t expected_flags,
  160. const vector<string>& expected_answers)
  161. {
  162. vector<RRType> types;
  163. vector<RRTTL> ttls;
  164. vector<const vector<string>* > answers;
  165. types.push_back(qtype);
  166. ttls.push_back(expected_ttl);
  167. answers.push_back(&expected_answers);
  168. checkFind(data_source, qname, zone_name, qclass, expected_class, qtype,
  169. ttls, expected_flags, types, answers);
  170. }
  171. TEST_F(StaticDataSourceTest, init) {
  172. EXPECT_EQ(DataSrc::SUCCESS, data_source.init());
  173. }
  174. TEST_F(StaticDataSourceTest, close) {
  175. EXPECT_EQ(DataSrc::SUCCESS, data_source.init());
  176. }
  177. TEST_F(StaticDataSourceTest, findClosestEnclosureForVersion) {
  178. DataSrcMatch match(version_name, rrclass);
  179. data_source.findClosestEnclosure(match);
  180. EXPECT_EQ(version_name, *match.getEnclosingZone());
  181. EXPECT_EQ(&data_source, match.getDataSource());
  182. }
  183. // Class Any query should result in the same answer.
  184. TEST_F(StaticDataSourceTest, findClosestEnclosureForVersionClassAny) {
  185. DataSrcMatch match(version_name, RRClass::ANY());
  186. data_source.findClosestEnclosure(match);
  187. EXPECT_EQ(version_name, *match.getEnclosingZone());
  188. EXPECT_EQ(&data_source, match.getDataSource());
  189. }
  190. // If class doesn't match the lookup should fail.
  191. TEST_F(StaticDataSourceTest, findClosestEnclosureForVersionClassMismatch) {
  192. DataSrcMatch match(version_name, RRClass::IN());
  193. data_source.findClosestEnclosure(match);
  194. EXPECT_EQ(static_cast<void*>(NULL), match.getEnclosingZone());
  195. EXPECT_EQ(static_cast<void*>(NULL), match.getDataSource());
  196. }
  197. TEST_F(StaticDataSourceTest, findClosestEnclosureForVersionPartial) {
  198. DataSrcMatch match(Name("foo").concatenate(version_name), rrclass);
  199. data_source.findClosestEnclosure(match);
  200. EXPECT_EQ(version_name, *match.getEnclosingZone());
  201. EXPECT_EQ(&data_source, match.getDataSource());
  202. }
  203. TEST_F(StaticDataSourceTest, findClosestEnclosureForAuthors) {
  204. DataSrcMatch match(authors_name, rrclass);
  205. data_source.findClosestEnclosure(match);
  206. EXPECT_EQ(authors_name, *match.getEnclosingZone());
  207. EXPECT_EQ(&data_source, match.getDataSource());
  208. }
  209. TEST_F(StaticDataSourceTest, findClosestEnclosureForAuthorsPartial) {
  210. DataSrcMatch match(Name("foo").concatenate(authors_name), rrclass);
  211. data_source.findClosestEnclosure(match);
  212. EXPECT_EQ(authors_name, *match.getEnclosingZone());
  213. EXPECT_EQ(&data_source, match.getDataSource());
  214. }
  215. TEST_F(StaticDataSourceTest, findClosestEnclosureNoMatch) {
  216. DataSrcMatch match(nomatch_name, rrclass);
  217. data_source.findClosestEnclosure(match);
  218. EXPECT_EQ(static_cast<void*>(NULL), match.getEnclosingZone());
  219. EXPECT_EQ(static_cast<void*>(NULL), match.getDataSource());
  220. }
  221. TEST_F(StaticDataSourceTest, findRRsetVersionTXT) {
  222. checkFind(data_source, version_name, NULL, rrclass, rrclass,
  223. rrtype, rrttl, 0, version_data);
  224. checkFind(data_source, version_name, &version_name, rrclass, rrclass,
  225. rrtype, rrttl, 0, version_data);
  226. }
  227. TEST_F(StaticDataSourceTest, findRRsetVersionNS) {
  228. rrtype = RRType::NS();
  229. checkFind(data_source, version_name, NULL, rrclass, rrclass,
  230. rrtype, rrttl, 0, version_ns_data);
  231. checkFind(data_source, version_name, &version_name, rrclass, rrclass,
  232. rrtype, rrttl, 0, version_ns_data);
  233. }
  234. TEST_F(StaticDataSourceTest, findRRsetVersionSOA) {
  235. rrtype = RRType::SOA();
  236. checkFind(data_source, version_name, NULL, rrclass, rrclass,
  237. rrtype, rrttl, 0, version_soa_data);
  238. checkFind(data_source, version_name, &version_name, rrclass, rrclass,
  239. rrtype, rrttl, 0, version_soa_data);
  240. }
  241. TEST_F(StaticDataSourceTest, findRRsetVersionANY) {
  242. rrtype = RRType::ANY();
  243. types.push_back(RRType::TXT());
  244. types.push_back(RRType::NS());
  245. types.push_back(RRType::SOA());
  246. ttls.push_back(rrttl);
  247. ttls.push_back(rrttl);
  248. ttls.push_back(rrttl);
  249. answers.push_back(&version_data);
  250. answers.push_back(&version_ns_data);
  251. answers.push_back(&version_soa_data);
  252. checkFind(data_source, version_name, NULL, rrclass, rrclass, rrtype, ttls,
  253. 0, types, answers);
  254. checkFind(data_source, version_name, &version_name, rrclass, rrclass,
  255. rrtype, ttls, 0, types, answers);
  256. }
  257. TEST_F(StaticDataSourceTest, findRRsetAuthorsTXT) {
  258. checkFind(data_source, authors_name, NULL, rrclass, rrclass,
  259. rrtype, rrttl, 0, authors_data);
  260. checkFind(data_source, authors_name, &authors_name, rrclass, rrclass,
  261. rrtype, rrttl, 0, authors_data);
  262. }
  263. TEST_F(StaticDataSourceTest, findRRsetAuthorsNS) {
  264. rrtype = RRType::NS();
  265. checkFind(data_source, authors_name, NULL, rrclass, rrclass,
  266. rrtype, rrttl, 0, authors_ns_data);
  267. checkFind(data_source, authors_name, &authors_name, rrclass, rrclass,
  268. rrtype, rrttl, 0, authors_ns_data);
  269. }
  270. TEST_F(StaticDataSourceTest, findRRsetAuthorsSOA) {
  271. rrtype = RRType::SOA();
  272. checkFind(data_source, authors_name, NULL, rrclass, rrclass,
  273. rrtype, rrttl, 0, authors_soa_data);
  274. checkFind(data_source, authors_name, &authors_name, rrclass, rrclass,
  275. rrtype, rrttl, 0, authors_soa_data);
  276. }
  277. TEST_F(StaticDataSourceTest, findRRsetAuthorsANY) {
  278. rrtype = RRType::ANY();
  279. types.push_back(RRType::TXT());
  280. types.push_back(RRType::NS());
  281. types.push_back(RRType::SOA());
  282. ttls.push_back(rrttl);
  283. ttls.push_back(rrttl);
  284. ttls.push_back(rrttl);
  285. answers.push_back(&authors_data);
  286. answers.push_back(&authors_ns_data);
  287. answers.push_back(&authors_soa_data);
  288. checkFind(data_source, authors_name, NULL, rrclass, rrclass, rrtype, ttls,
  289. 0, types, answers);
  290. checkFind(data_source, authors_name, &authors_name, rrclass, rrclass,
  291. rrtype, ttls, 0, types, answers);
  292. }
  293. // Class ANY lookup should result in the same answer.
  294. TEST_F(StaticDataSourceTest, findRRsetVersionClassAny) {
  295. checkFind(data_source, version_name, NULL, RRClass::ANY(), rrclass,
  296. rrtype, rrttl, 0, version_data);
  297. checkFind(data_source, version_name, &version_name, RRClass::ANY(), rrclass,
  298. rrtype, rrttl, 0, version_data);
  299. }
  300. // If the class doesn't match, it should simply fail.
  301. TEST_F(StaticDataSourceTest, findRRsetVersionClassMismatch) {
  302. EXPECT_EQ(DataSrc::ERROR,
  303. data_source.findRRset(version_name, RRClass::IN(), rrtype,
  304. result_sets, find_flags, &version_name));
  305. }
  306. TEST_F(StaticDataSourceTest, findRRsetOutOfZone) {
  307. // If the qname doesn't match any of the static zones, the result should
  308. // be "no such zone", regardless of whether the zone is explicitly
  309. // specified. Other "expected" result parameters will be ignored.
  310. checkFind(data_source, nomatch_name, NULL, rrclass, rrclass,
  311. rrtype, rrttl, DataSrc::NO_SUCH_ZONE, authors_ns_data);
  312. checkFind(data_source, nomatch_name, &version_name, rrclass, rrclass,
  313. rrtype, rrttl, DataSrc::NO_SUCH_ZONE, authors_ns_data);
  314. checkFind(data_source, nomatch_name, &authors_name, rrclass, rrclass,
  315. rrtype, rrttl, DataSrc::NO_SUCH_ZONE, authors_ns_data);
  316. }
  317. // If a zone name is given but doesn't match any of the static zones,
  318. // the result should be "no such zone"
  319. TEST_F(StaticDataSourceTest, findRRsetZoneMismatch) {
  320. const Name& short_zonename(Name("bind"));
  321. checkFind(data_source, version_name, &short_zonename, rrclass, rrclass,
  322. rrtype, rrttl, DataSrc::NO_SUCH_ZONE, authors_ns_data);
  323. checkFind(data_source, authors_name, &short_zonename, rrclass, rrclass,
  324. rrtype, rrttl, DataSrc::NO_SUCH_ZONE, authors_ns_data);
  325. }
  326. // Zone matches, but name doesn't exist in the zone
  327. TEST_F(StaticDataSourceTest, findRRsetNoName) {
  328. checkFind(data_source, Name("foo").concatenate(version_name), NULL, rrclass,
  329. rrclass, rrtype, rrttl, DataSrc::NAME_NOT_FOUND, authors_ns_data);
  330. checkFind(data_source, Name("foo").concatenate(version_name), &version_name,
  331. rrclass, rrclass, rrtype, rrttl, DataSrc::NAME_NOT_FOUND,
  332. authors_ns_data);
  333. checkFind(data_source, Name("foo").concatenate(authors_name), NULL, rrclass,
  334. rrclass, rrtype, rrttl, DataSrc::NAME_NOT_FOUND, authors_ns_data);
  335. checkFind(data_source, Name("foo").concatenate(authors_name), &authors_name,
  336. rrclass, rrclass, rrtype, rrttl, DataSrc::NAME_NOT_FOUND,
  337. authors_ns_data);
  338. }
  339. // Zone matches and qname exists, but type doesn't exist for the name.
  340. TEST_F(StaticDataSourceTest, findRRsetNoType) {
  341. const RRType& nomatch_type = RRType::A();
  342. checkFind(data_source, version_name, NULL, rrclass,
  343. rrclass, nomatch_type, rrttl, DataSrc::TYPE_NOT_FOUND,
  344. authors_ns_data);
  345. checkFind(data_source, version_name, &version_name, rrclass,
  346. rrclass, nomatch_type, rrttl, DataSrc::TYPE_NOT_FOUND,
  347. authors_ns_data);
  348. checkFind(data_source, authors_name, NULL, rrclass,
  349. rrclass, nomatch_type, rrttl, DataSrc::TYPE_NOT_FOUND,
  350. authors_ns_data);
  351. checkFind(data_source, authors_name, &authors_name, rrclass,
  352. rrclass, nomatch_type, rrttl, DataSrc::TYPE_NOT_FOUND,
  353. authors_ns_data);
  354. }
  355. // Simple tests for "unsupported" tests.
  356. TEST_F(StaticDataSourceTest, notImplemented) {
  357. Name target_name(version_name);
  358. EXPECT_EQ(DataSrc::NOT_IMPLEMENTED,
  359. data_source.findPreviousName(version_name, target_name,
  360. &version_name));
  361. string target_hash;
  362. EXPECT_EQ(DataSrc::NOT_IMPLEMENTED,
  363. data_source.findCoveringNSEC3(version_name, target_hash,
  364. result_sets));
  365. }
  366. }