blob: 3eb2cce11821e1acc1342c12906619b7e0f082ca [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2017, Regents of the University of California.
*
* This file is part of NDN repo-ng (Next generation of NDN repository).
* See AUTHORS.md for complete list of repo-ng authors and contributors.
*
* repo-ng is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*
* repo-ng is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with
* repo-ng, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#include "index.hpp"
#include <ndn-cxx/util/sha256.hpp>
#include <ndn-cxx/security/signature-sha256-with-rsa.hpp>
namespace repo {
/** @brief determines if entry can satisfy interest
* @param hash SHA256 hash of PublisherPublicKeyLocator if exists in interest, otherwise ignored
*/
static bool
matchesSimpleSelectors(const Interest& interest, ndn::ConstBufferPtr& hash,
const Index::Entry& entry)
{
const Name& fullName = entry.getName();
if (!interest.getName().isPrefixOf(fullName))
return false;
size_t nSuffixComponents = fullName.size() - interest.getName().size();
if (interest.getMinSuffixComponents() >= 0 &&
nSuffixComponents < static_cast<size_t>(interest.getMinSuffixComponents()))
return false;
if (interest.getMaxSuffixComponents() >= 0 &&
nSuffixComponents > static_cast<size_t>(interest.getMaxSuffixComponents()))
return false;
if (!interest.getExclude().empty() &&
entry.getName().size() > interest.getName().size() &&
interest.getExclude().isExcluded(entry.getName()[interest.getName().size()]))
return false;
if (!interest.getPublisherPublicKeyLocator().empty())
{
if (*entry.getKeyLocatorHash() != *hash)
return false;
}
return true;
}
Index::Index(size_t nMaxPackets)
: m_maxPackets(nMaxPackets)
, m_size(0)
{
}
bool
Index::insert(const Data& data, int64_t id)
{
if (isFull())
BOOST_THROW_EXCEPTION(Error("The Index is Full. Cannot Insert Any Data!"));
Entry entry(data, id);
bool isInserted = m_indexContainer.insert(entry).second;
if (isInserted)
++m_size;
return isInserted;
}
bool
Index::insert(const Name& fullName, int64_t id,
const ndn::ConstBufferPtr& keyLocatorHash)
{
if (isFull())
BOOST_THROW_EXCEPTION(Error("The Index is Full. Cannot Insert Any Data!"));
Entry entry(fullName, keyLocatorHash, id);
bool isInserted = m_indexContainer.insert(entry).second;
if (isInserted)
++m_size;
return isInserted;
}
std::pair<int64_t,Name>
Index::find(const Interest& interest) const
{
Name name = interest.getName();
IndexContainer::const_iterator result = m_indexContainer.lower_bound(name);
if (result != m_indexContainer.end())
{
return selectChild(interest, result);
}
else
{
return std::make_pair(0, Name());
}
}
std::pair<int64_t,Name>
Index::find(const Name& name) const
{
IndexContainer::const_iterator result = m_indexContainer.lower_bound(name);
if (result != m_indexContainer.end())
{
return findFirstEntry(name, result);
}
else
{
return std::make_pair(0, Name());
}
}
bool
Index::hasData(const Data& data) const
{
Index::Entry entry(data, -1); // the id number is useless
IndexContainer::const_iterator result = m_indexContainer.find(entry);
return result != m_indexContainer.end();
}
std::pair<int64_t,Name>
Index::findFirstEntry(const Name& prefix,
IndexContainer::const_iterator startingPoint) const
{
BOOST_ASSERT(startingPoint != m_indexContainer.end());
if (prefix.isPrefixOf(startingPoint->getName()))
{
return std::make_pair(startingPoint->getId(), startingPoint->getName());
}
else
{
return std::make_pair(0, Name());
}
}
bool
Index::erase(const Name& fullName)
{
Entry entry(fullName);
IndexContainer::const_iterator findIterator = m_indexContainer.find(entry);
if (findIterator != m_indexContainer.end())
{
m_indexContainer.erase(findIterator);
m_size--;
return true;
}
else
return false;
}
const ndn::ConstBufferPtr
Index::computeKeyLocatorHash(const KeyLocator& keyLocator)
{
const Block& block = keyLocator.wireEncode();
ndn::ConstBufferPtr keyLocatorHash = ndn::util::Sha256::computeDigest(block.wire(), block.size());
return keyLocatorHash;
}
std::pair<int64_t,Name>
Index::selectChild(const Interest& interest,
IndexContainer::const_iterator startingPoint) const
{
BOOST_ASSERT(startingPoint != m_indexContainer.end());
bool isLeftmost = (interest.getChildSelector() <= 0);
ndn::ConstBufferPtr hash;
if (!interest.getPublisherPublicKeyLocator().empty())
{
KeyLocator keyLocator = interest.getPublisherPublicKeyLocator();
const Block& block = keyLocator.wireEncode();
hash = ndn::util::Sha256::computeDigest(block.wire(), block.size());
}
if (isLeftmost)
{
for (IndexContainer::const_iterator it = startingPoint;
it != m_indexContainer.end(); ++it)
{
if (!interest.getName().isPrefixOf(it->getName()))
return std::make_pair(0, Name());
if (matchesSimpleSelectors(interest, hash, (*it)))
return std::make_pair(it->getId(), it->getName());
}
}
else
{
IndexContainer::const_iterator boundary = m_indexContainer.lower_bound(interest.getName());
if (boundary == m_indexContainer.end() || !interest.getName().isPrefixOf(boundary->getName()))
return std::make_pair(0, Name());
Name successor = interest.getName().getSuccessor();
IndexContainer::const_iterator last = interest.getName().size() == 0 ?
m_indexContainer.end() : m_indexContainer.lower_bound(interest.getName().getSuccessor());
while (true)
{
IndexContainer::const_iterator prev = last;
--prev;
if (prev == boundary)
{
bool isMatch = matchesSimpleSelectors(interest, hash, (*prev));
if (isMatch)
{
return std::make_pair(prev->getId(), prev->getName());
}
else
return std::make_pair(0, Name());
}
IndexContainer::const_iterator first =
m_indexContainer.lower_bound(prev->getName().getPrefix(interest.getName().size() + 1));
IndexContainer::const_iterator match =
std::find_if(first, last, bind(&matchesSimpleSelectors, interest, hash, _1));
if (match != last)
{
return std::make_pair(match->getId(), match->getName());
}
last = first;
}
}
return std::make_pair(0, Name());
}
Index::Entry::Entry(const Data& data, int64_t id)
: m_name(data.getFullName())
, m_id(id)
{
const ndn::Signature& signature = data.getSignature();
if (signature.hasKeyLocator())
m_keyLocatorHash = computeKeyLocatorHash(signature.getKeyLocator());
}
Index::Entry::Entry(const Name& fullName, const KeyLocator& keyLocator, int64_t id)
: m_name(fullName)
, m_keyLocatorHash(computeKeyLocatorHash(keyLocator))
, m_id(id)
{
}
Index::Entry::Entry(const Name& fullName,
const ndn::ConstBufferPtr& keyLocatorHash, int64_t id)
: m_name(fullName)
, m_keyLocatorHash(keyLocatorHash)
, m_id(id)
{
}
Index::Entry::Entry(const Name& name)
: m_name(name)
{
}
} // namespace repo