sync: improved unit tests

refs: #4264

Change-Id: I981f3d8f653e4750ee6f9d77f2c89188f11d8ebb
diff --git a/src/common.hpp b/src/common.hpp
index 0d67efc..93b3e87 100644
--- a/src/common.hpp
+++ b/src/common.hpp
@@ -29,6 +29,7 @@
 
 #include <ndn-cxx/common.hpp>
 #include <ndn-cxx/util/time.hpp>
+#include <ndn-cxx/name.hpp>
 
 namespace nlsr {
 
@@ -55,6 +56,8 @@
    static constexpr bool value = true;
 };
 
+using IsLsaNew = std::function<bool(const ndn::Name&, const std::string&, const uint64_t&)>;
+
 } // namespace nlsr
 
 #endif // NLSR_COMMON_HPP
diff --git a/src/communication/sync-logic-handler.hpp b/src/communication/sync-logic-handler.hpp
index b46ac7c..afa1464 100644
--- a/src/communication/sync-logic-handler.hpp
+++ b/src/communication/sync-logic-handler.hpp
@@ -94,12 +94,13 @@
   void
   createSyncSocket(const ndn::Name& syncPrefix);
 
-private:
+PUBLIC_WITH_TESTS_ELSE_PRIVATE:
   /*! \brief Simple function to glue Name components together
    */
   void
   buildUpdatePrefix();
 
+private:
   /*! \brief Determine which kind of LSA was updated and fetch it.
    *
    * Checks that the received update is not from us, which can happen,
diff --git a/src/lsdb.cpp b/src/lsdb.cpp
index 4de69ef..789edf6 100644
--- a/src/lsdb.cpp
+++ b/src/lsdb.cpp
@@ -68,23 +68,8 @@
   , m_sync(m_nlsr.getNlsrFace(),
            [this] (const ndn::Name& routerName, const std::string& lsaType,
                    const uint64_t& sequenceNumber) {
-             ndn::Name lsaKey = routerName;
-             lsaKey.append(lsaType);
-
-             if (lsaType == NameLsa::TYPE_STRING) {
-                 return isNameLsaNew(lsaKey, sequenceNumber);
-             }
-             else if (lsaType == AdjLsa::TYPE_STRING) {
-                 return isAdjLsaNew(lsaKey, sequenceNumber);
-             }
-             else if (lsaType == CoordinateLsa::TYPE_STRING) {
-                 return isCoordinateLsaNew(lsaKey, sequenceNumber);
-             }
-             else {
-               return false;
-             }
-           },
-           m_nlsr.getConfParameter())
+             return isLsaNew(routerName, lsaType, sequenceNumber);
+           }, m_nlsr.getConfParameter())
   , m_lsaRefreshTime(0)
   , m_adjLsaBuildInterval(ADJ_LSA_BUILD_INTERVAL_DEFAULT)
   , m_sequencingManager()
@@ -1313,4 +1298,24 @@
   return false;
 }
 
+bool
+Lsdb::isLsaNew(const ndn::Name& routerName, const std::string& lsaType,
+               const uint64_t& sequenceNumber) {
+  ndn::Name lsaKey = routerName;
+  lsaKey.append(lsaType);
+
+  if (lsaType == NameLsa::TYPE_STRING) {
+    return isNameLsaNew(lsaKey, sequenceNumber);
+  }
+  else if (lsaType == AdjLsa::TYPE_STRING) {
+    return isAdjLsaNew(lsaKey, sequenceNumber);
+  }
+  else if (lsaType == CoordinateLsa::TYPE_STRING) {
+    return isCoordinateLsaNew(lsaKey, sequenceNumber);
+  }
+  else {
+    return false;
+  }
+}
+
 } // namespace nlsr
diff --git a/src/lsdb.hpp b/src/lsdb.hpp
index 042d727..4574242 100644
--- a/src/lsdb.hpp
+++ b/src/lsdb.hpp
@@ -51,8 +51,10 @@
   }
 
   bool
-  doesLsaExist(const ndn::Name& key, const std::string& lsType);
+  isLsaNew(const ndn::Name& routerName, const std::string& lsaType, const uint64_t& sequenceNumber);
 
+  bool
+  doesLsaExist(const ndn::Name& key, const std::string& lsType);
 
   /*! \brief Builds a name LSA for this router and then installs it
       into the LSDB.
diff --git a/tests/test-lsdb.cpp b/tests/test-lsdb.cpp
index ad2dfde..866c91b 100644
--- a/tests/test-lsdb.cpp
+++ b/tests/test-lsdb.cpp
@@ -331,7 +331,33 @@
   BOOST_CHECK_EQUAL(nameList, newPrefixes);
 }
 
-BOOST_AUTO_TEST_SUITE_END()
+BOOST_AUTO_TEST_CASE(TestIsLsaNew)
+{
+  const ndn::Name::Component CONFIG_NETWORK{"/ndn"};
+  const ndn::Name::Component CONFIG_SITE{"/memphis"};
+  ndn::Name originRouter{};
+  originRouter.append(CONFIG_NETWORK).append(CONFIG_SITE).append("/%C1.Router/other-router");
+
+  // Install Name LSA
+  NamePrefixList nameList;
+  NameLsa lsa(originRouter, 999, ndn::time::system_clock::TimePoint::max(), nameList);
+
+  lsdb.installNameLsa(lsa);
+
+  // Lower NameLSA sequence number
+  uint64_t lowerSeqNo = 998;
+  BOOST_CHECK(!lsdb.isLsaNew(originRouter, NameLsa::TYPE_STRING, lowerSeqNo));
+
+  // Same NameLSA sequence number
+  uint64_t sameSeqNo = 999;
+  BOOST_CHECK(!lsdb.isLsaNew(originRouter, NameLsa::TYPE_STRING, sameSeqNo));
+
+  // Higher NameLSA sequence number
+  uint64_t higherSeqNo = 1000;
+  BOOST_CHECK(lsdb.isLsaNew(originRouter, NameLsa::TYPE_STRING, higherSeqNo));
+}
+
+BOOST_AUTO_TEST_SUITE_END() // TestLsdb
 
 } // namespace test
 } // namespace nlsr
diff --git a/tests/test-sync-logic-handler.cpp b/tests/test-sync-logic-handler.cpp
index 5ebe679..bb6a6db 100644
--- a/tests/test-sync-logic-handler.cpp
+++ b/tests/test-sync-logic-handler.cpp
@@ -19,10 +19,11 @@
  * NLSR, e.g., in COPYING.md file.  If not, see <http://www.gnu.org/licenses/>.
  **/
 
-#include "test-common.hpp"
-
-#include "nlsr.hpp"
 #include "communication/sync-logic-handler.hpp"
+#include "test-common.hpp"
+#include "common.hpp"
+#include "nlsr.hpp"
+#include "logger.hpp"
 
 #include <ndn-cxx/util/dummy-client-face.hpp>
 
@@ -37,15 +38,25 @@
   SyncLogicFixture()
     : face(std::make_shared<ndn::util::DummyClientFace>())
     , nlsr(g_ioService, g_scheduler, std::ref(*face), g_keyChain)
-    , sync(nlsr.getLsdb().getSyncLogicHandler())
+    , testIsLsaNew([] (const ndn::Name& name, const std::string& lsaType,
+                       const uint64_t sequenceNumber) {
+                     return true;
+                   })
     , CONFIG_NETWORK("/ndn")
     , CONFIG_SITE("/site")
     , CONFIG_ROUTER_NAME("/%C1.Router/this-router")
+    , OTHER_ROUTER_NAME("/%C1.Router/other-router/")
   {
     nlsr.getConfParameter().setNetwork(CONFIG_NETWORK);
     nlsr.getConfParameter().setSiteName(CONFIG_SITE);
     nlsr.getConfParameter().setRouterName(CONFIG_ROUTER_NAME);
     nlsr.getConfParameter().buildRouterPrefix();
+
+    conf.setNetwork(CONFIG_NETWORK);
+    conf.setSiteName(CONFIG_SITE);
+    conf.setRouterName(CONFIG_ROUTER_NAME);
+    conf.buildRouterPrefix();
+    INIT_LOGGERS("/tmp", "TRACE");
   }
 
   void
@@ -67,193 +78,207 @@
 public:
   std::shared_ptr<ndn::util::DummyClientFace> face;
   Nlsr nlsr;
-  SyncLogicHandler& sync;
+  ConfParameter conf;
+  IsLsaNew testIsLsaNew;
 
   const std::string CONFIG_NETWORK;
   const std::string CONFIG_SITE;
   const std::string CONFIG_ROUTER_NAME;
+  const std::string OTHER_ROUTER_NAME;
   const std::vector<std::string> lsaTypes = {NameLsa::TYPE_STRING, AdjLsa::TYPE_STRING,
                                              CoordinateLsa::TYPE_STRING};
 };
 
 BOOST_FIXTURE_TEST_SUITE(TestSyncLogicHandler, SyncLogicFixture)
 
+/* Tests that when SyncLogicHandler receives an LSA of either Name or
+   Adjacency type that appears to be newer, it will emit to its signal
+   with those LSA details.
+ */
 BOOST_AUTO_TEST_CASE(UpdateForOtherLS)
 {
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
+
   std::vector<std::string> lsaTypes = {NameLsa::TYPE_STRING, AdjLsa::TYPE_STRING};
 
   uint64_t syncSeqNo = 1;
 
   for (const std::string& lsaType : lsaTypes) {
-    std::string updateName = nlsr.getConfParameter().getLsaPrefix().toUri() +
-                             CONFIG_SITE + "/%C1.Router/other-router/" + lsaType;
+    std::string updateName = conf.getLsaPrefix().toUri() + CONFIG_SITE
+      + OTHER_ROUTER_NAME + lsaType;
+
+    // Actual testing done here -- signal function callback
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [&, this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_CHECK_EQUAL(ndn::Name{updateName}, routerName);
+        BOOST_CHECK_EQUAL(sequenceNumber, syncSeqNo);
+      });
 
     receiveUpdate(updateName, syncSeqNo, sync);
-
-    const auto& it = std::find_if(face->sentInterests.begin(), face->sentInterests.end(),
-                                  [updateName] (const ndn::Interest& interest) {
-                                    return interest.getName().getPrefix(-1) == updateName + "/";
-                                  });
-    BOOST_REQUIRE(it != face->sentInterests.end());
   }
 }
 
+/* Tests that when SyncLogicHandler in HR mode receives an LSA of
+   either Coordinate or Name type that appears to be newer, it will
+   emit to its signal with those LSA details.
+ */
 BOOST_AUTO_TEST_CASE(UpdateForOtherHR)
 {
-  Nlsr nlsr_hr(g_ioService, g_scheduler, std::ref(*face), g_keyChain);
-  SyncLogicHandler& sync_hr(nlsr_hr.getLsdb().getSyncLogicHandler());
+  conf.setHyperbolicState(HYPERBOLIC_STATE_ON);
 
-  nlsr_hr.getConfParameter().setNetwork(CONFIG_NETWORK);
-  nlsr_hr.getConfParameter().setSiteName(CONFIG_SITE);
-  nlsr_hr.getConfParameter().setRouterName(CONFIG_ROUTER_NAME);
-  nlsr_hr.getConfParameter().buildRouterPrefix();
-
-  nlsr_hr.getConfParameter().setHyperbolicState(HYPERBOLIC_STATE_ON);
-
-  nlsr_hr.initialize();
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
 
   uint64_t syncSeqNo = 1;
-
   std::vector<std::string> lsaTypes = {NameLsa::TYPE_STRING, CoordinateLsa::TYPE_STRING};
 
   for (const std::string& lsaType : lsaTypes) {
-    std::string updateName = nlsr_hr.getConfParameter().getLsaPrefix().toUri() +
-                           CONFIG_SITE + "/%C1.Router/other-router/" + lsaType;
+    std::string updateName = conf.getLsaPrefix().toUri() + CONFIG_SITE
+      + OTHER_ROUTER_NAME + lsaType;
 
-    receiveUpdate(updateName, syncSeqNo, sync_hr);
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [& ,this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_CHECK_EQUAL(ndn::Name{updateName}, routerName);
+        BOOST_CHECK_EQUAL(sequenceNumber, syncSeqNo);
+      });
 
-    const auto& it = std::find_if(face->sentInterests.begin(), face->sentInterests.end(),
-                                  [updateName] (const ndn::Interest& interest) {
-                                    return interest.getName().getPrefix(-1) == updateName + "/";
-                                  });
-
-    BOOST_REQUIRE(it != face->sentInterests.end());
+    receiveUpdate(updateName, syncSeqNo, sync);
   }
 }
 
+/* Tests that when SyncLogicHandler in HR-dry mode receives an LSA of
+   any type that appears to be newer, it will emit to its signal with
+   those LSA details.
+ */
 BOOST_AUTO_TEST_CASE(UpdateForOtherHRDry)
 {
+  conf.setHyperbolicState(HYPERBOLIC_STATE_DRY_RUN);
 
-  Nlsr nlsr_hrdry(g_ioService, g_scheduler, std::ref(*face),g_keyChain);
-  SyncLogicHandler& sync_hrdry(nlsr_hrdry.getLsdb().getSyncLogicHandler());
-
-  nlsr_hrdry.getConfParameter().setNetwork(CONFIG_NETWORK);
-  nlsr_hrdry.getConfParameter().setSiteName(CONFIG_SITE);
-  nlsr_hrdry.getConfParameter().setRouterName(CONFIG_ROUTER_NAME);
-  nlsr_hrdry.getConfParameter().buildRouterPrefix();
-
-  nlsr_hrdry.getConfParameter().setHyperbolicState(HYPERBOLIC_STATE_DRY_RUN);
-
-  nlsr_hrdry.initialize();
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
 
   for (const std::string& lsaType : lsaTypes) {
-
-    std::string updateName = nlsr.getConfParameter().getLsaPrefix().toUri() +
-                 CONFIG_SITE + "/%C1.Router/other-router/" + lsaType;
-
-
     uint64_t syncSeqNo = 1;
 
-    receiveUpdate(updateName, syncSeqNo, sync_hrdry);
+    std::string updateName = conf.getLsaPrefix().toUri() + CONFIG_SITE
+      + OTHER_ROUTER_NAME + lsaType;
 
-    // In HR dry-state all LSA's should be published
-    const auto& it = std::find_if(face->sentInterests.begin(), face->sentInterests.end(),
-                     [updateName] (const ndn::Interest& interest) {
-                       return interest.getName().getPrefix(-1) == updateName + "/";
-                     });
-    BOOST_REQUIRE(it != face->sentInterests.end());
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [& ,this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_CHECK_EQUAL(ndn::Name{updateName}, routerName);
+        BOOST_CHECK_EQUAL(sequenceNumber, syncSeqNo);
+      });
+
+    receiveUpdate(updateName, syncSeqNo, sync);
   }
 }
 
+/* Tests that when SyncLogicHandler receives an update for an LSA with
+   details matching this router's details, it will *not* emit to its
+   signal those LSA details.
+ */
 BOOST_AUTO_TEST_CASE(NoUpdateForSelf)
 {
+  const uint64_t sequenceNumber = 1;
+
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
+
   for (const std::string& lsaType : lsaTypes) {
-    std::string updateName = nlsr.getConfParameter().getLsaPrefix().toUri() +
-                             CONFIG_SITE + CONFIG_ROUTER_NAME + lsaType;
+    // To ensure that we get correctly-separated components, create
+    // and modify a Name to hand off.
+    ndn::Name updateName = ndn::Name{conf.getLsaPrefix()};
+    updateName.append(CONFIG_SITE).append(CONFIG_ROUTER_NAME).append(lsaType);
 
-    receiveUpdate(updateName, 1, sync);
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [& ,this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_FAIL("Updates for self should not be emitted!");
+      });
 
-    std::vector<ndn::Interest>& interests = face->sentInterests;
-    BOOST_CHECK_EQUAL(interests.size(), 0);
+    receiveUpdate(updateName.toUri(), sequenceNumber, sync);
   }
 }
 
+/* Tests that when SyncLogicHandler receives an update for an LSA with
+   details that do not match the expected format, it will *not* emit
+   to its signal those LSA details.
+ */
 BOOST_AUTO_TEST_CASE(MalformedUpdate)
 {
-  for (const std::string& lsaType : lsaTypes) {
-    std::string updateName = CONFIG_SITE + nlsr.getConfParameter().getLsaPrefix().toUri() +
-                             CONFIG_ROUTER_NAME + lsaType;
+  const uint64_t sequenceNumber = 1;
 
-    std::vector<ndn::Interest>& interests = face->sentInterests;
-    BOOST_CHECK_EQUAL(interests.size(), 0);
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
+
+  for (const std::string& lsaType : lsaTypes) {
+    ndn::Name updateName{CONFIG_SITE};
+    updateName.append(CONFIG_ROUTER_NAME).append(lsaType);
+
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [& ,this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_FAIL("Malformed updates should not be emitted!");
+      });
+
+    receiveUpdate(updateName.toUri(), sequenceNumber, sync);
   }
 }
 
-BOOST_AUTO_TEST_CASE(SequenceNumber)
+/* Tests that when SyncLogicHandler receives an update for an LSA with
+   details that do not appear to be new, it will *not* emit to its
+   signal those LSA details.
+ */
+BOOST_AUTO_TEST_CASE(LsaNotNew)
 {
-  std::string originRouter = CONFIG_NETWORK + CONFIG_SITE + "/%C1.Router/other-router/";
+  auto testLsaAlwaysFalse = [] (const ndn::Name& routerName, const std::string& lsaType,
+                           const uint64_t& sequenceNumber) {
+    return false;
+  };
 
-  Lsdb& lsdb = nlsr.getLsdb();
-
-  // Install Name LSA
-  NamePrefixList nameList;
-  NameLsa lsa(originRouter, 999, ndn::time::system_clock::TimePoint::max(), nameList);
-  lsdb.installNameLsa(lsa);
-
-  // Install Adj LSA
-  AdjacencyList adjList;
-  AdjLsa adjLsa(originRouter, 1000, ndn::time::system_clock::TimePoint::max(),
-                3 , adjList);
-  lsdb.installAdjLsa(adjLsa);
-
-  std::vector<double> angles = {0.0};
-
-  // Install Cor LSA
-  CoordinateLsa corLsa(originRouter, 1000, ndn::time::system_clock::TimePoint::max(),
-                       0, angles);
-  lsdb.installCoordinateLsa(corLsa);
+  const uint64_t sequenceNumber = 1;
+  SyncLogicHandler sync{std::ref(*face), testLsaAlwaysFalse, conf};
+  sync.createSyncSocket(conf.getChronosyncPrefix());
+    ndn::util::signal::ScopedConnection connection = sync.onNewLsa->connect(
+      [& ,this] (const ndn::Name& routerName, const uint64_t& sequenceNumber) {
+        BOOST_FAIL("An update for an LSA with non-new sequence number should not emit!");
+      });
 
   std::string updateName = nlsr.getConfParameter().getLsaPrefix().toUri() +
                            CONFIG_SITE + "/%C1.Router/other-router/" + NameLsa::TYPE_STRING;
 
-  // Lower NameLSA sequence number
-  uint64_t lowerSeqNo = 998;
-  receiveUpdate(updateName, lowerSeqNo, sync);
-
-  std::vector<ndn::Interest>& interests = face->sentInterests;
-  BOOST_REQUIRE_EQUAL(interests.size(), 0);
-
-  // Same NameLSA sequence number
-  uint64_t sameSeqNo = 999;
-  receiveUpdate(updateName, sameSeqNo, sync);
-
-  interests = face->sentInterests;
-  BOOST_REQUIRE_EQUAL(interests.size(), 0);
-
-  // Higher NameLSA sequence number
-  uint64_t higherSeqNo = 1000;
-  receiveUpdate(updateName, higherSeqNo, sync);
-
-  interests = face->sentInterests;
-  BOOST_REQUIRE_EQUAL(interests.size(), 1);
-
-  std::vector<ndn::Interest>::iterator it = interests.begin();
-  BOOST_CHECK_EQUAL(it->getName().getPrefix(-1), updateName + "/");
+  receiveUpdate(updateName, sequenceNumber, sync);
 }
 
+/* Tests that SyncLogicHandler successfully concatenates configured
+   variables together to form the necessary prefixes to advertise
+   through ChronoSync.
+ */
 BOOST_AUTO_TEST_CASE(UpdatePrefix)
 {
+
+  SyncLogicHandler sync{std::ref(*face), testIsLsaNew, conf};
+
   ndn::Name expectedPrefix = nlsr.getConfParameter().getLsaPrefix();
   expectedPrefix.append(CONFIG_SITE);
   expectedPrefix.append(CONFIG_ROUTER_NAME);
 
-  nlsr.initialize();
+  sync.buildUpdatePrefix();
 
-  BOOST_CHECK_EQUAL(sync.m_nameLsaUserPrefix, ndn::Name(expectedPrefix).append(NameLsa::TYPE_STRING));
-  BOOST_CHECK_EQUAL(sync.m_adjLsaUserPrefix, ndn::Name(expectedPrefix).append(AdjLsa::TYPE_STRING));
-  BOOST_CHECK_EQUAL(sync.m_coorLsaUserPrefix, ndn::Name(expectedPrefix).append(CoordinateLsa::TYPE_STRING));
+  BOOST_CHECK_EQUAL(sync.m_nameLsaUserPrefix,
+                    ndn::Name(expectedPrefix).append(NameLsa::TYPE_STRING));
+  BOOST_CHECK_EQUAL(sync.m_adjLsaUserPrefix,
+                    ndn::Name(expectedPrefix).append(AdjLsa::TYPE_STRING));
+  BOOST_CHECK_EQUAL(sync.m_coorLsaUserPrefix,
+                    ndn::Name(expectedPrefix).append(CoordinateLsa::TYPE_STRING));
 }
 
+/* Tests that SyncLogicHandler's socket will be created when
+   Nlsr::initialize is called, preventing use of sync before the
+   socket is created.
+
+   NB: This test is as much an Nlsr class test as a SyncLogicHandler
+   class test, but it rides the line and ends up here.
+ */
 BOOST_AUTO_TEST_CASE(CreateSyncSocketOnInitialization) // Bug #2649
 {
   nlsr.initialize();
@@ -264,7 +289,8 @@
   BOOST_REQUIRE(lsa == nullptr);
 
   // Publish a routing update before an Adjacency LSA is built
-  BOOST_CHECK_NO_THROW(sync.publishRoutingUpdate(AdjLsa::TYPE_STRING, 0));
+  BOOST_CHECK_NO_THROW(nlsr.getLsdb().getSyncLogicHandler()
+                       .publishRoutingUpdate(AdjLsa::TYPE_STRING, 0));
 }
 
 BOOST_AUTO_TEST_SUITE_END()