putchunks: respond to RDR discovery Interests

Change-Id: Id1039246304a3069f158aecf21724568db434958
refs: #4556
diff --git a/tests/chunks/producer.t.cpp b/tests/chunks/producer.t.cpp
index eea82ca..e2de21e 100644
--- a/tests/chunks/producer.t.cpp
+++ b/tests/chunks/producer.t.cpp
@@ -1,6 +1,6 @@
 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
 /*
- * Copyright (c) 2016-2018,  Regents of the University of California,
+ * Copyright (c) 2016-2019,  Regents of the University of California,
  *                           Colorado State University,
  *                           University Pierre & Marie Curie, Sorbonne University.
  *
@@ -28,6 +28,7 @@
 #include "tests/test-common.hpp"
 #include "tests/identity-management-fixture.hpp"
 
+#include <ndn-cxx/metadata-object.hpp>
 #include <ndn-cxx/security/pib/identity.hpp>
 #include <ndn-cxx/security/pib/key.hpp>
 #include <ndn-cxx/util/dummy-client-face.hpp>
@@ -199,6 +200,34 @@
   BOOST_REQUIRE_EQUAL(face.sentData.size(), 1);
 }
 
+BOOST_AUTO_TEST_CASE(RequestMetadata)
+{
+  Producer producer(prefix.appendVersion(version), face, m_keyChain, testString, options);
+  io.poll();
+
+  // ask for metadata with a valid discovery interest
+  face.receive(MetadataObject::makeDiscoveryInterest(Name(prefix).getPrefix(-1)));
+  face.processEvents();
+
+  BOOST_REQUIRE_EQUAL(face.sentData.size(), 1);
+  auto lastData = face.sentData.back();
+
+  // check the name of metadata packet
+  BOOST_CHECK(MetadataObject::isValidName(lastData.getName()));
+
+  // make metadata object from metadata packet
+  MetadataObject mobject(lastData);
+  BOOST_CHECK_EQUAL(mobject.getVersionedName(), prefix);
+
+  // ask for metadata with an invalid discovery interest
+  face.receive(MetadataObject::makeDiscoveryInterest(Name(prefix).getPrefix(-1))
+               .setCanBePrefix(false));
+  face.processEvents();
+
+  // we expect Nack in response to a discovery interest without CanBePrefix
+  BOOST_CHECK_EQUAL(face.sentNacks.size(), 1);
+}
+
 BOOST_AUTO_TEST_SUITE_END() // TestProducer
 BOOST_AUTO_TEST_SUITE_END() // Chunks
 
diff --git a/tools/chunks/putchunks/producer.cpp b/tools/chunks/putchunks/producer.cpp
index 4483558..ff4bafb 100644
--- a/tools/chunks/putchunks/producer.cpp
+++ b/tools/chunks/putchunks/producer.cpp
@@ -1,6 +1,6 @@
 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
 /*
- * Copyright (c) 2016-2018, Regents of the University of California,
+ * Copyright (c) 2016-2019, Regents of the University of California,
  *                          Colorado State University,
  *                          University Pierre & Marie Curie, Sorbonne University.
  *
@@ -25,10 +25,13 @@
  * @author Andrea Tosatto
  * @author Davide Pesavento
  * @author Klaus Schneider
+ * @author Chavoosh Ghasemi
  */
 
 #include "producer.hpp"
 
+#include <ndn-cxx/metadata-object.hpp>
+
 namespace ndn {
 namespace chunks {
 
@@ -52,10 +55,19 @@
   if (m_options.wantShowVersion)
     std::cout << m_versionedPrefix[-1] << std::endl;
 
-  m_face.setInterestFilter(m_prefix,
-                           bind(&Producer::onInterest, this, _2),
-                           RegisterPrefixSuccessCallback(),
-                           bind(&Producer::onRegisterFailed, this, _1, _2));
+  // register m_prefix without interest handler
+  m_face.registerPrefix(m_prefix, nullptr, bind(&Producer::onRegisterFailed, this, _1, _2));
+
+  // match Interests whose name starts with m_versionedPrefix
+  face.setInterestFilter(m_versionedPrefix, bind(&Producer::processSegmentInterest, this, _2));
+
+  // match Interests whose name is exactly m_prefix
+  face.setInterestFilter(InterestFilter(m_prefix, ""),
+                         bind(&Producer::processSegmentInterest, this, _2));
+
+  // match discovery Interests
+  face.setInterestFilter(MetadataObject::makeDiscoveryInterest(m_prefix).getName(),
+                         bind(&Producer::processDiscoveryInterest, this, _2));
 
   if (!m_options.isQuiet)
     std::cerr << "Data published with name: " << m_versionedPrefix << std::endl;
@@ -68,7 +80,32 @@
 }
 
 void
-Producer::onInterest(const Interest& interest)
+Producer::processDiscoveryInterest(const Interest& interest)
+{
+  if (m_options.isVerbose)
+    std::cerr << "Discovery Interest: " << interest << std::endl;
+
+  if (!interest.getCanBePrefix()) {
+    if (m_options.isVerbose)
+      std::cerr << "Discovery Interest lacks CanBePrefix, sending Nack" << std::endl;
+    m_face.put(lp::Nack(interest));
+    return;
+  }
+
+  MetadataObject mobject;
+  mobject.setVersionedName(m_versionedPrefix);
+
+  // make a metadata packet based on the received discovery Interest name
+  Data mdata(mobject.makeData(interest.getName(), m_keyChain, m_options.signingInfo));
+
+  if (m_options.isVerbose)
+    std::cerr << "Sending metadata: " << mdata << std::endl;
+
+  m_face.put(mdata);
+}
+
+void
+Producer::processSegmentInterest(const Interest& interest)
 {
   BOOST_ASSERT(m_store.size() > 0);
 
@@ -78,9 +115,7 @@
   const Name& name = interest.getName();
   shared_ptr<Data> data;
 
-  // is this a discovery Interest or a sequence retrieval?
-  if (name.size() == m_versionedPrefix.size() + 1 && m_versionedPrefix.isPrefixOf(name) &&
-      name[-1].isSegment()) {
+  if (name.size() == m_versionedPrefix.size() + 1 && name[-1].isSegment()) {
     const auto segmentNo = static_cast<size_t>(interest.getName()[-1].toSegment());
     // specific segment retrieval
     if (segmentNo < m_store.size()) {
@@ -88,7 +123,7 @@
     }
   }
   else if (interest.matchesData(*m_store[0])) {
-    // Interest has version and is looking for the first segment or has no version
+    // unspecified version or segment number, return first segment
     data = m_store[0];
   }
 
@@ -98,6 +133,11 @@
 
     m_face.put(*data);
   }
+  else {
+    if (m_options.isVerbose)
+      std::cerr << "Interest cannot be satisfied, sending Nack" << std::endl;
+    m_face.put(lp::Nack(interest));
+  }
 }
 
 void
diff --git a/tools/chunks/putchunks/producer.hpp b/tools/chunks/putchunks/producer.hpp
index f498ec7..722e248 100644
--- a/tools/chunks/putchunks/producer.hpp
+++ b/tools/chunks/putchunks/producer.hpp
@@ -1,6 +1,6 @@
 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
 /*
- * Copyright (c) 2016-2017, Regents of the University of California,
+ * Copyright (c) 2016-2019, Regents of the University of California,
  *                          Colorado State University,
  *                          University Pierre & Marie Curie, Sorbonne University.
  *
@@ -85,8 +85,17 @@
   void
   populateStore(std::istream& is);
 
+  /**
+   * @brief Respond with a metadata packet containing the versioned content name
+   */
   void
-  onInterest(const Interest& interest);
+  processDiscoveryInterest(const Interest& interest);
+
+  /**
+   * @brief Respond with the requested segment of content
+   */
+  void
+  processSegmentInterest(const Interest& interest);
 
   void
   onRegisterFailed(const Name& prefix, const std::string& reason);