123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282 |
- // Copyright (C) 2012 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 <datasrc/master_loader_callbacks.h>
- #include <datasrc/memory/zone_data_loader.h>
- #include <datasrc/memory/zone_data_updater.h>
- #include <datasrc/memory/logger.h>
- #include <datasrc/memory/segment_object_holder.h>
- #include <datasrc/memory/util_internal.h>
- #include <datasrc/memory/rrset_collection.h>
- #include <dns/master_loader.h>
- #include <dns/rrcollator.h>
- #include <dns/rdataclass.h>
- #include <dns/rrset.h>
- #include <dns/zone_checker.h>
- #include <boost/foreach.hpp>
- #include <boost/bind.hpp>
- #include <boost/noncopyable.hpp>
- #include <map>
- using namespace isc::dns;
- using namespace isc::dns::rdata;
- namespace isc {
- namespace datasrc {
- namespace memory {
- using detail::SegmentObjectHolder;
- using detail::getCoveredType;
- namespace { // unnamed namespace
- // A functor type used for loading.
- typedef boost::function<void(isc::dns::ConstRRsetPtr)> LoadCallback;
- // A helper internal class for \c loadZoneData(). make it non-copyable
- // to avoid accidental copy.
- //
- // The current internal implementation no longer expects that both a
- // normal (non RRSIG) RRset and (when signed) its RRSIG are added at
- // once, but we do that here anyway to avoid merging RdataSets every
- // single time which can be inefficient.
- //
- // We hold all RRsets of the same owner name in node_rrsets_ and
- // node_rrsigsets_, and add the matching pairs of RRsets to the zone
- // when we see a new owner name. We do this to limit the size of
- // NodeRRsets below. However, RRsets can occur in any order.
- //
- // The caller is responsible for adding the RRsets of the last group
- // in the input sequence by explicitly calling flushNodeRRsets() at the
- // end. It's cleaner and more robust if we let the destructor of this class
- // do it, but since we cannot guarantee the adding operation is exception free,
- // we don't choose that option to maintain the common expectation for
- // destructors.
- class ZoneDataLoader : boost::noncopyable {
- public:
- ZoneDataLoader(util::MemorySegment& mem_sgmt,
- const isc::dns::RRClass rrclass,
- const isc::dns::Name& zone_name, ZoneData& zone_data) :
- updater_(mem_sgmt, rrclass, zone_name, zone_data)
- {}
- void addFromLoad(const isc::dns::ConstRRsetPtr& rrset);
- void flushNodeRRsets();
- private:
- typedef std::map<isc::dns::RRType, isc::dns::ConstRRsetPtr> NodeRRsets;
- typedef NodeRRsets::value_type NodeRRsetsVal;
- // A helper to identify the covered type of an RRSIG.
- const isc::dns::Name& getCurrentName() const;
- private:
- NodeRRsets node_rrsets_;
- NodeRRsets node_rrsigsets_;
- std::vector<isc::dns::ConstRRsetPtr> non_consecutive_rrsets_;
- ZoneDataUpdater updater_;
- };
- void
- ZoneDataLoader::addFromLoad(const ConstRRsetPtr& rrset) {
- // If we see a new name, flush the temporary holders, adding the
- // pairs of RRsets and RRSIGs of the previous name to the zone.
- if ((!node_rrsets_.empty() || !node_rrsigsets_.empty() ||
- !non_consecutive_rrsets_.empty()) &&
- (getCurrentName() != rrset->getName())) {
- flushNodeRRsets();
- }
- // Store this RRset until it can be added to the zone. If an rrtype
- // that's already been seen is found, queue it in a different vector
- // to be merged later.
- const bool is_rrsig = rrset->getType() == RRType::RRSIG();
- NodeRRsets& node_rrsets = is_rrsig ? node_rrsigsets_ : node_rrsets_;
- const RRType& rrtype = is_rrsig ? getCoveredType(rrset) : rrset->getType();
- if (!node_rrsets.insert(NodeRRsetsVal(rrtype, rrset)).second) {
- non_consecutive_rrsets_.insert(non_consecutive_rrsets_.begin(), rrset);
- }
- if (rrset->getRRsig()) {
- addFromLoad(rrset->getRRsig());
- }
- }
- void
- ZoneDataLoader::flushNodeRRsets() {
- BOOST_FOREACH(NodeRRsetsVal val, node_rrsets_) {
- // Identify the corresponding RRSIG for the RRset, if any. If
- // found add both the RRset and its RRSIG at once.
- ConstRRsetPtr sig_rrset;
- NodeRRsets::iterator sig_it = node_rrsigsets_.find(val.first);
- if (sig_it != node_rrsigsets_.end()) {
- sig_rrset = sig_it->second;
- node_rrsigsets_.erase(sig_it);
- }
- updater_.add(val.second, sig_rrset);
- }
- // Normally rrsigsets map should be empty at this point, but it's still
- // possible that an RRSIG that don't has covered RRset is added; they
- // still remain in the map. We add them to the zone separately.
- BOOST_FOREACH(NodeRRsetsVal val, node_rrsigsets_) {
- updater_.add(ConstRRsetPtr(), val.second);
- }
- // Add any non-consecutive rrsets too.
- BOOST_FOREACH(ConstRRsetPtr rrset, non_consecutive_rrsets_) {
- if (rrset->getType() == RRType::RRSIG()) {
- updater_.add(ConstRRsetPtr(), rrset);
- } else {
- updater_.add(rrset, ConstRRsetPtr());
- }
- }
- node_rrsets_.clear();
- node_rrsigsets_.clear();
- non_consecutive_rrsets_.clear();
- }
- const Name&
- ZoneDataLoader::getCurrentName() const {
- if (!node_rrsets_.empty()) {
- return (node_rrsets_.begin()->second->getName());
- }
- assert(!node_rrsigsets_.empty());
- return (node_rrsigsets_.begin()->second->getName());
- }
- void
- logWarning(const dns::Name* zone_name, const dns::RRClass* rrclass,
- const std::string& reason)
- {
- LOG_WARN(logger, DATASRC_MEMORY_CHECK_WARNING).arg(*zone_name).
- arg(*rrclass).arg(reason);
- }
- void
- logError(const dns::Name* zone_name, const dns::RRClass* rrclass,
- const std::string& reason)
- {
- LOG_ERROR(logger, DATASRC_MEMORY_CHECK_ERROR).arg(*zone_name).arg(*rrclass).
- arg(reason);
- }
- ZoneData*
- loadZoneDataInternal(util::MemorySegment& mem_sgmt,
- const isc::dns::RRClass& rrclass,
- const Name& zone_name,
- boost::function<void(LoadCallback)> rrset_installer)
- {
- SegmentObjectHolder<ZoneData, RRClass> holder(
- mem_sgmt, ZoneData::create(mem_sgmt, zone_name), rrclass);
- ZoneDataLoader loader(mem_sgmt, rrclass, zone_name, *holder.get());
- rrset_installer(boost::bind(&ZoneDataLoader::addFromLoad, &loader, _1));
- // Add any last RRsets that were left
- loader.flushNodeRRsets();
- const ZoneNode* origin_node = holder.get()->getOriginNode();
- const RdataSet* rdataset = origin_node->getData();
- // If the zone is NSEC3-signed, check if it has NSEC3PARAM
- if (holder.get()->isNSEC3Signed()) {
- if (RdataSet::find(rdataset, RRType::NSEC3PARAM()) == NULL) {
- LOG_WARN(logger, DATASRC_MEMORY_MEM_NO_NSEC3PARAM).
- arg(zone_name).arg(rrclass);
- }
- }
- RRsetCollection collection(*(holder.get()), rrclass);
- const dns::ZoneCheckerCallbacks
- callbacks(boost::bind(&logError, &zone_name, &rrclass, _1),
- boost::bind(&logWarning, &zone_name, &rrclass, _1));
- if (!dns::checkZone(zone_name, rrclass, collection, callbacks)) {
- isc_throw(ZoneValidationError,
- "Errors found when validating zone: "
- << zone_name << "/" << rrclass);
- }
- return (holder.release());
- }
- // A wrapper for dns::MasterLoader used by loadZoneData() below. Essentially
- // it converts the two callback types. Note the mostly redundant wrapper of
- // boost::bind. It converts function<void(ConstRRsetPtr)> to
- // function<void(RRsetPtr)> (MasterLoader expects the latter). SunStudio
- // doesn't seem to do this conversion if we just pass 'callback'.
- void
- masterLoaderWrapper(const char* const filename, const Name& origin,
- const RRClass& zone_class, LoadCallback callback)
- {
- bool load_ok = false; // (we don't use it)
- dns::RRCollator collator(boost::bind(callback, _1));
- try {
- dns::MasterLoader(filename, origin, zone_class,
- createMasterLoaderCallbacks(origin, zone_class,
- &load_ok),
- collator.getCallback()).load();
- collator.flush();
- } catch (const dns::MasterLoaderError& e) {
- isc_throw(ZoneLoaderException, e.what());
- }
- }
- // The installer called from the iterator version of loadZoneData().
- void
- generateRRsetFromIterator(ZoneIterator* iterator, LoadCallback callback) {
- ConstRRsetPtr rrset;
- while ((rrset = iterator->getNextRRset()) != NULL) {
- callback(rrset);
- }
- }
- } // end of unnamed namespace
- ZoneData*
- loadZoneData(util::MemorySegment& mem_sgmt,
- const isc::dns::RRClass& rrclass,
- const isc::dns::Name& zone_name,
- const std::string& zone_file)
- {
- LOG_DEBUG(logger, DBG_TRACE_BASIC, DATASRC_MEMORY_MEM_LOAD_FROM_FILE).
- arg(zone_name).arg(rrclass).arg(zone_file);
- return (loadZoneDataInternal(mem_sgmt, rrclass, zone_name,
- boost::bind(masterLoaderWrapper,
- zone_file.c_str(),
- zone_name, rrclass,
- _1)));
- }
- ZoneData*
- loadZoneData(util::MemorySegment& mem_sgmt,
- const isc::dns::RRClass& rrclass,
- const isc::dns::Name& zone_name,
- ZoneIterator& iterator)
- {
- LOG_DEBUG(logger, DBG_TRACE_BASIC, DATASRC_MEMORY_MEM_LOAD_FROM_DATASRC).
- arg(zone_name).arg(rrclass);
- return (loadZoneDataInternal(mem_sgmt, rrclass, zone_name,
- boost::bind(generateRRsetFromIterator,
- &iterator, _1)));
- }
- } // namespace memory
- } // namespace datasrc
- } // namespace isc
|