// 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. #include #include #include #include #include #include #include #include #include #include #include using isc::UnitTestUtil; using namespace std; using namespace isc::dns; using namespace isc::util; using namespace isc::dns::rdata; namespace { class Rdata_NSEC_Test : public RdataTest { // there's nothing to specialize }; const char* const nsec_txt = "www2.isc.org. CNAME RRSIG NSEC"; TEST_F(Rdata_NSEC_Test, toText_NSEC) { const generic::NSEC rdata_nsec(nsec_txt); EXPECT_EQ(nsec_txt, rdata_nsec.toText()); } TEST_F(Rdata_NSEC_Test, badText_NSEC) { EXPECT_THROW(generic::NSEC rdata_nsec("www.isc.org. BIFF POW SPOON"), InvalidRdataText); EXPECT_THROW(generic::NSEC rdata_nsec("www.isc.org."), InvalidRdataText); } TEST_F(Rdata_NSEC_Test, createFromWire_NSEC) { const generic::NSEC rdata_nsec(nsec_txt); EXPECT_EQ(0, rdata_nsec.compare( *rdataFactoryFromFile(RRType::NSEC(), RRClass::IN(), "rdata_nsec_fromWire1"))); // Too short RDLENGTH EXPECT_THROW(rdataFactoryFromFile(RRType::NSEC(), RRClass::IN(), "rdata_nsec_fromWire2"), DNSMessageFORMERR); // Invalid bitmap cases are tested in Rdata_NSECBITMAP_Test. } TEST_F(Rdata_NSEC_Test, createFromLexer_NSEC) { const generic::NSEC rdata_nsec(nsec_txt); EXPECT_EQ(0, rdata_nsec.compare( *test::createRdataUsingLexer(RRType::NSEC(), RRClass::IN(), nsec_txt))); // test::createRdataUsingLexer() constructs relative to // "example.org." origin. EXPECT_EQ(0, generic::NSEC("www2.example.org. CNAME RRSIG NSEC").compare( *test::createRdataUsingLexer(RRType::NSEC(), RRClass::IN(), "www2 CNAME RRSIG NSEC"))); // Exceptions cause NULL to be returned. EXPECT_FALSE(test::createRdataUsingLexer(RRType::NSEC(), RRClass::IN(), "www.isc.org.")); } TEST_F(Rdata_NSEC_Test, toWireRenderer_NSEC) { renderer.skip(2); const generic::NSEC rdata_nsec(nsec_txt); rdata_nsec.toWire(renderer); vector data; UnitTestUtil::readWireData("rdata_nsec_fromWire1", data); EXPECT_PRED_FORMAT4(UnitTestUtil::matchWireData, static_cast(renderer.getData()) + 2, renderer.getLength() - 2, &data[2], data.size() - 2); } TEST_F(Rdata_NSEC_Test, toWireBuffer_NSEC) { const generic::NSEC rdata_nsec(nsec_txt); rdata_nsec.toWire(obuffer); } TEST_F(Rdata_NSEC_Test, assign) { generic::NSEC rdata_nsec(nsec_txt); generic::NSEC rdata_nsec2 = rdata_nsec; EXPECT_EQ(0, rdata_nsec.compare(rdata_nsec2)); } TEST_F(Rdata_NSEC_Test, getNextName) { // The implementation is quite trivial, so we simply check it's actually // defined and does work as intended in a simple case. EXPECT_EQ(Name("www2.isc.org"), generic::NSEC((nsec_txt)).getNextName()); } TEST_F(Rdata_NSEC_Test, compare) { // trivial case: self equivalence EXPECT_EQ(0, generic::NSEC("example. A"). compare(generic::NSEC("example. A"))); EXPECT_EQ(0, generic::NSEC("EXAMPLE. A"). // should be case insensitive compare(generic::NSEC("example. A"))); // comparison attempt between incompatible RR types should be rejected EXPECT_THROW(generic::NSEC(nsec_txt).compare(*rdata_nomatch), bad_cast); // test RDATAs, sorted in the ascendent order. We only compare the // next name here. Bitmap comparison is tested in the bitmap tests. // Note that names are compared as wire-format data, not based on the // domain name comparison. vector compare_set; compare_set.push_back(generic::NSEC("a.example. A")); compare_set.push_back(generic::NSEC("example. A")); vector::const_iterator it; const vector::const_iterator it_end = compare_set.end(); for (it = compare_set.begin(); it != it_end - 1; ++it) { SCOPED_TRACE("compare " + it->toText() + " to " + (it + 1)->toText()); EXPECT_GT(0, (*it).compare(*(it + 1))); EXPECT_LT(0, (*(it + 1)).compare(*it)); } } }