diff --git a/.clang-format b/.clang-format new file mode 100644 index 0000000..f4c6247 --- /dev/null +++ b/.clang-format @@ -0,0 +1,2 @@ +BasedOnStyle: Mozilla +IndentWidth: 2 \ No newline at end of file diff --git a/README.md b/README.md index a265355..1a249f5 100644 --- a/README.md +++ b/README.md @@ -47,6 +47,8 @@ where `sync-prefix` is `/ndn/svs` for the example application. ## Contributing Contributions are welcome through GitHub. +Use `clang-format` to format the code before submitting a pull request. +The VS Code extension for `clang-format` is recommended to format the code automatically. ## License diff --git a/examples/chat-pubsub.cpp b/examples/chat-pubsub.cpp index 1bfad0d..15c235a 100644 --- a/examples/chat-pubsub.cpp +++ b/examples/chat-pubsub.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include @@ -40,7 +41,8 @@ class Program // Use HMAC signing for Sync Interests // Note: this is not generally recommended, but is used here for simplicity SecurityOptions secOpts(m_keyChain); - secOpts.interestSigner->signingInfo.setSigningHmacKey("dGhpcyBpcyBhIHNlY3JldCBtZXNzYWdl"); + secOpts.interestSigner->signingInfo.setSigningHmacKey( + "dGhpcyBpcyBhIHNlY3JldCBtZXNzYWdl"); // Sign data packets using SHA256 (for simplicity) secOpts.dataSigner->signingInfo.setSha256Signing(); @@ -51,22 +53,22 @@ class Program opts.maxPubAge = ndn::time::seconds(10); // Create the Pub/Sub instance - m_svsps = std::make_shared( - ndn::Name(m_options.prefix), - ndn::Name(m_options.m_id), - face, - std::bind(&Program::onMissingData, this, _1), - opts, - secOpts); + m_svsps = + std::make_shared(ndn::Name(m_options.prefix), + ndn::Name(m_options.m_id), + face, + std::bind(&Program::onMissingData, this, _1), + opts, + secOpts); std::cout << "SVS client starting: " << m_options.m_id << std::endl; // Subscribe to all data packets with prefix /chat (the "topic") - m_svsps->subscribe(ndn::Name("/chat"), [] (const auto& subData) - { - std::string content(reinterpret_cast(subData.data.data()), subData.data.size()); - std::cout << subData.producerPrefix << " [" << subData.seqNo << "] : " << - subData.name << " : "; + m_svsps->subscribe(ndn::Name("/chat"), [](const auto& subData) { + std::string content(reinterpret_cast(subData.data.data()), + subData.data.size()); + std::cout << subData.producerPrefix << " [" << subData.seqNo + << "] : " << subData.name << " : "; if (content.length() > 200) { std::cout << "[LONG] " << content.length() << " bytes" << " [" << std::hash{}(content) << "]"; @@ -77,8 +79,7 @@ class Program }); } - void - run() + void run() { // Begin processing face events in a separate thread. std::thread svsThread([this] { face.processEvents(); }); @@ -101,11 +102,10 @@ class Program protected: /** * Callback on receving a new State Vector from another node. - * This will be called regardless of whether the missing data contains any topics - * or producers that we are subscribed to. + * This will be called regardless of whether the missing data contains any + * topics or producers that we are subscribed to. */ - void - onMissingData(const std::vector&) + void onMissingData(const std::vector&) { // Ignore any other missing data for this example } @@ -113,8 +113,7 @@ class Program /** * Publish a string message to the group */ - void - publishMsg(const std::string& msg) + void publishMsg(const std::string& msg) { // Message to send std::string content = msg; @@ -129,16 +128,20 @@ class Program for (auto& c : content) c = std::rand() % 26 + 'a'; - std::cout << "> Sending random message with hash [" << std::hash{}(content) << "]" << std::endl; + std::cout << "> Sending random message with hash [" + << std::hash{}(content) << "]" << std::endl; } // Note that unlike SVSync, names can be arbitrary, // and need not be prefixed with the producer prefix. - ndn::Name name("chat"); // topic of publication - name.append(m_options.m_id); // who sent this - name.appendTimestamp(); // and when - - m_svsps->publish(name, ndn::make_span(reinterpret_cast(content.data()), content.size())); + ndn::Name name("chat"); // topic of publication + name.append(m_options.m_id); // who sent this + name.appendTimestamp(); // and when + + m_svsps->publish( + name, + ndn::make_span(reinterpret_cast(content.data()), + content.size())); } private: diff --git a/examples/chat.cpp b/examples/chat.cpp index 9904690..750875e 100644 --- a/examples/chat.cpp +++ b/examples/chat.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include @@ -37,21 +38,24 @@ class Program // Use HMAC signing for Sync Interests // Note: this is not generally recommended, but is used here for simplicity ndn::svs::SecurityOptions securityOptions(m_keyChain); - securityOptions.interestSigner->signingInfo.setSigningHmacKey("dGhpcyBpcyBhIHNlY3JldCBtZXNzYWdl"); + securityOptions.interestSigner->signingInfo.setSigningHmacKey( + "dGhpcyBpcyBhIHNlY3JldCBtZXNzYWdl"); // Create the SVSync instance m_svs = std::make_shared( - ndn::Name(m_options.prefix), // Sync prefix, common for all nodes in the group - ndn::Name(m_options.m_id), // Unique data prefix for this node - face, // Shared NDN face - std::bind(&Program::onMissingData, this, _1), // Callback on learning new sequence numbers from SVS - securityOptions); // Security configuration + ndn::Name( + m_options.prefix), // Sync prefix, common for all nodes in the group + ndn::Name(m_options.m_id), // Unique data prefix for this node + face, // Shared NDN face + std::bind(&Program::onMissingData, + this, + _1), // Callback on learning new sequence numbers from SVS + securityOptions); // Security configuration std::cout << "SVS client starting: " << m_options.m_id << std::endl; } - void - run() + void run() { // Begin processing face events in a separate thread. std::thread svsThread([this] { face.processEvents(); }); @@ -75,21 +79,18 @@ class Program /** * Callback on receving a new State Vector from another node */ - void - onMissingData(const std::vector& v) + void onMissingData(const std::vector& v) { // Iterate over the entire difference set - for (size_t i = 0; i < v.size(); i++) - { + for (size_t i = 0; i < v.size(); i++) { // Iterate over each new sequence number that we learned - for (ndn::svs::SeqNo s = v[i].low; s <= v[i].high; ++s) - { + for (ndn::svs::SeqNo s = v[i].low; s <= v[i].high; ++s) { // Request a single data packet using the SVSync API ndn::svs::NodeID nid = v[i].nodeId; - m_svs->fetchData(nid, s, [nid] (const auto& data) - { - std::string content(reinterpret_cast(data.getContent().value()), - data.getContent().value_size()); + m_svs->fetchData(nid, s, [nid](const auto& data) { + std::string content( + reinterpret_cast(data.getContent().value()), + data.getContent().value_size()); std::cout << data.getName() << " : " << content << std::endl; }); } @@ -99,10 +100,10 @@ class Program /** * Publish a string message to the SVSync group */ - void - publishMsg(std::string_view msg) + void publishMsg(std::string_view msg) { - // Encode the message into a Content TLV block, which is what the SVSync API expects + // Encode the message into a Content TLV block, which is what the SVSync API + // expects auto block = ndn::encoding::makeStringBlock(ndn::tlv::Content, msg); m_svs->publishData(block, ndn::time::seconds(1)); } diff --git a/ndn-svs/common.hpp b/ndn-svs/common.hpp index b85737d..8bd5850 100644 --- a/ndn-svs/common.hpp +++ b/ndn-svs/common.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_COMMON_HPP @@ -44,7 +45,8 @@ using SeqNo = uint64_t; using ndn::security::ValidationError; using DataValidatedCallback = std::function; -using DataValidationErrorCallback = std::function ; +using DataValidationErrorCallback = + std::function; } // namespace ndn::svs diff --git a/ndn-svs/core.cpp b/ndn-svs/core.cpp index 75c52c8..760c0f9 100644 --- a/ndn-svs/core.cpp +++ b/ndn-svs/core.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "core.hpp" @@ -27,8 +28,8 @@ #ifdef NDN_SVS_COMPRESSION #include #include -#include #include +#include #endif namespace ndn::svs { @@ -54,11 +55,11 @@ SVSyncCore::SVSyncCore(ndn::Face& face, , m_scheduler(m_face.getIoContext()) { // Register sync interest filter - m_syncRegisteredPrefix = - m_face.setInterestFilter(syncPrefix, - std::bind(&SVSyncCore::onSyncInterest, this, _2), - std::bind(&SVSyncCore::sendInitialInterest, this), - [] (auto&&...) { NDN_THROW(Error("Failed to register sync prefix")); }); + m_syncRegisteredPrefix = m_face.setInterestFilter( + syncPrefix, + std::bind(&SVSyncCore::onSyncInterest, this, _2), + std::bind(&SVSyncCore::sendInitialInterest, this), + [](auto&&...) { NDN_THROW(Error("Failed to register sync prefix")); }); } static inline int @@ -91,24 +92,26 @@ SVSyncCore::sendInitialInterest() void SVSyncCore::onSyncInterest(const Interest& interest) { - switch (m_securityOptions.interestSigner->signingInfo.getSignerType()) - { + switch (m_securityOptions.interestSigner->signingInfo.getSignerType()) { case security::SigningInfo::SIGNER_TYPE_NULL: onSyncInterestValidated(interest); return; case security::SigningInfo::SIGNER_TYPE_HMAC: - if (security::verifySignature(interest, m_keyChainMem.getTpm(), - m_securityOptions.interestSigner->signingInfo.getSignerName(), - DigestAlgorithm::SHA256)) + if (security::verifySignature( + interest, + m_keyChainMem.getTpm(), + m_securityOptions.interestSigner->signingInfo.getSignerName(), + DigestAlgorithm::SHA256)) onSyncInterestValidated(interest); return; default: if (m_securityOptions.validator) - m_securityOptions.validator->validate(interest, - std::bind(&SVSyncCore::onSyncInterestValidated, this, _1), - nullptr); + m_securityOptions.validator->validate( + interest, + std::bind(&SVSyncCore::onSyncInterestValidated, this, _1), + nullptr); else onSyncInterestValidated(interest); return; @@ -137,14 +140,17 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) params.parse(); #ifdef NDN_SVS_COMPRESSION - // Decompress if necessary. The spec requires that if an LZMA block is present, - // then no other blocks are present (everything is compressed together) + // Decompress if necessary. The spec requires that if an LZMA block is + // present, then no other blocks are present (everything is compressed + // together) if (params.find(tlv::LzmaBlock) != params.elements_end()) { auto lzmaBlock = params.get(tlv::LzmaBlock); boost::iostreams::filtering_istreambuf in; in.push(boost::iostreams::lzma_decompressor()); - in.push(boost::iostreams::array_source(reinterpret_cast(lzmaBlock.value()), lzmaBlock.value_size())); + in.push(boost::iostreams::array_source( + reinterpret_cast(lzmaBlock.value()), + lzmaBlock.value_size())); ndn::OBufferStream decompressed; boost::iostreams::copy(in, decompressed); @@ -161,24 +167,18 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) // Get state vector std::shared_ptr vvOther; - try - { + try { vvOther = std::make_shared(params.get(tlv::StateVector)); - } - catch (ndn::tlv::Error&) - { + } catch (ndn::tlv::Error&) { // TODO: log error return; } // Read extra mapping blocks - if (m_recvExtraBlock) - { + if (m_recvExtraBlock) { try { m_recvExtraBlock(params.get(tlv::MappingData), *vvOther); - } - catch (std::exception&) - { + } catch (std::exception&) { // TODO: log error but continue } } @@ -190,8 +190,7 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) auto missingData = std::get<2>(result); // Callback if missing data found - if (!missingData.empty()) - { + if (!missingData.empty()) { for (auto& e : missingData) e.incomingFace = incomingFace; m_onUpdate(missingData); @@ -203,12 +202,9 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) // If incoming state identical/newer to local vector, reset timer // If incoming state is older, send sync interest immediately - if (!myVectorNew) - { + if (!myVectorNew) { retxSyncInterest(false, 0); - } - else - { + } else { enterSuppressionState(*vvOther); // Check how much time is left on the timer, // reset to ~m_intrReplyDist if more than that. @@ -218,8 +214,7 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) // TODO: efficient curve depends on number of active nodes delay = suppressionCurve(m_maxSuppressionTime.count(), delay); - if (getCurrentTime() + delay * 1000 < m_nextSyncInterest) - { + if (getCurrentTime() + delay * 1000 < m_nextSyncInterest) { retxSyncInterest(false, delay); } } @@ -228,8 +223,7 @@ SVSyncCore::onSyncInterestValidated(const Interest& interest) void SVSyncCore::retxSyncInterest(bool send, unsigned int delay) { - if (send) - { + if (send) { std::lock_guard lock(m_recordedVvMutex); // Only send interest if in steady state or local vector has newer state @@ -283,7 +277,8 @@ SVSyncCore::sendSyncInterest() #ifdef NDN_SVS_COMPRESSION boost::iostreams::filtering_istreambuf in; in.push(boost::iostreams::lzma_compressor()); - in.push(boost::iostreams::array_source(reinterpret_cast(wire.data()), wire.size())); + in.push(boost::iostreams::array_source( + reinterpret_cast(wire.data()), wire.size())); ndn::OBufferStream compressed; boost::iostreams::copy(in, compressed); wire = ndn::Block(tlv::LzmaBlock, compressed.buf()); @@ -295,13 +290,13 @@ SVSyncCore::sendSyncInterest() interest.setApplicationParameters(wire); interest.setInterestLifetime(1_ms); - switch (m_securityOptions.interestSigner->signingInfo.getSignerType()) - { + switch (m_securityOptions.interestSigner->signingInfo.getSignerType()) { case security::SigningInfo::SIGNER_TYPE_NULL: break; case security::SigningInfo::SIGNER_TYPE_HMAC: - m_keyChainMem.sign(interest, m_securityOptions.interestSigner->signingInfo); + m_keyChainMem.sign(interest, + m_securityOptions.interestSigner->signingInfo); break; default: @@ -317,49 +312,45 @@ SVSyncCore::mergeStateVector(const VersionVector& vvOther) { std::lock_guard lock(m_vvMutex); - bool myVectorNew = false, - otherVectorNew = false; + bool myVectorNew = false, otherVectorNew = false; // New data found in vvOther std::vector missingData; // Check if other vector has newer state - for (const auto& entry : vvOther) - { + for (const auto& entry : vvOther) { NodeID nidOther = entry.first; SeqNo seqOther = entry.second; SeqNo seqCurrent = m_vv.get(nidOther); - if (seqCurrent < seqOther) - { + if (seqCurrent < seqOther) { otherVectorNew = true; SeqNo startSeq = m_vv.get(nidOther) + 1; - missingData.push_back({nidOther, startSeq, seqOther, 0}); + missingData.push_back({ nidOther, startSeq, seqOther, 0 }); m_vv.set(nidOther, seqOther); } } // Check if I have newer state - for (const auto& entry : m_vv) - { + for (const auto& entry : m_vv) { NodeID nid = entry.first; SeqNo seq = entry.second; SeqNo seqOther = vvOther.get(nid); // Ignore this node if it was last updated within network RTT - if (time::system_clock::now() - m_vv.getLastUpdate(nid) < m_maxSuppressionTime) + if (time::system_clock::now() - m_vv.getLastUpdate(nid) < + m_maxSuppressionTime) continue; - if (seqOther < seq) - { + if (seqOther < seq) { myVectorNew = true; break; } } - return {myVectorNew, otherVectorNew, missingData}; + return { myVectorNew, otherVectorNew, missingData }; } void @@ -396,8 +387,7 @@ SVSyncCore::getNodeIds() const { std::lock_guard lock(m_vvMutex); std::set sessionNames; - for (const auto& nid : m_vv) - { + for (const auto& nid : m_vv) { sessionNames.insert(nid.first); } return sessionNames; @@ -407,7 +397,8 @@ long SVSyncCore::getCurrentTime() const { return std::chrono::duration_cast( - std::chrono::steady_clock::now().time_since_epoch()).count(); + std::chrono::steady_clock::now().time_since_epoch()) + .count(); } bool @@ -420,14 +411,12 @@ SVSyncCore::recordVector(const VersionVector& vvOther) std::lock_guard lock1(m_vvMutex); - for (const auto& entry : vvOther) - { + for (const auto& entry : vvOther) { NodeID nidOther = entry.first; SeqNo seqOther = entry.second; SeqNo seqCurrent = m_recordedVv->get(nidOther); - if (seqCurrent < seqOther) - { + if (seqCurrent < seqOther) { m_recordedVv->set(nidOther, seqOther); } } diff --git a/ndn-svs/core.hpp b/ndn-svs/core.hpp index 43b88bd..4f9ad79 100644 --- a/ndn-svs/core.hpp +++ b/ndn-svs/core.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_CORE_HPP @@ -81,21 +82,17 @@ class SVSyncCore : noncopyable /** * @brief Reset the sync tree (and restart synchronization again) * - * @param isOnInterest a flag that tells whether the reset is called by reset interest. + * @param isOnInterest a flag that tells whether the reset is called by reset + * interest. */ - void - reset(bool isOnInterest = false); + void reset(bool isOnInterest = false); /** * @brief Get the node ID of the local session. * * @param prefix prefix of the node */ - const NodeID& - getNodeId() - { - return m_id; - } + const NodeID& getNodeId() { return m_id; } /** * @brief Get current seqNo of the local session. @@ -105,8 +102,7 @@ class SVSyncCore : noncopyable * * @param prefix prefix of the node */ - SeqNo - getSeqNo(const NodeID& nid = EMPTY_NODE_ID) const; + SeqNo getSeqNo(const NodeID& nid = EMPTY_NODE_ID) const; /** * @brief Update the seqNo of the local session @@ -116,15 +112,14 @@ class SVSyncCore : noncopyable * @param seq The new seqNo. * @param nid The NodeID of node to update. */ - void - updateSeqNo(const SeqNo& seq, const NodeID& nid = EMPTY_NODE_ID); + void updateSeqNo(const SeqNo& seq, const NodeID& nid = EMPTY_NODE_ID); /// @brief Get all the nodeIDs - std::set - getNodeIds() const; + std::set getNodeIds() const; using GetExtraBlockCallback = std::function; - using RecvExtraBlockCallback = std::function; + using RecvExtraBlockCallback = + std::function; /** * @brief Callback to get extra data block for sync interest. @@ -132,8 +127,7 @@ class SVSyncCore : noncopyable * The version vector will be locked during the duration of this callback, * so it must return FAST! */ - void - setGetExtraBlockCallback(const GetExtraBlockCallback& callback) + void setGetExtraBlockCallback(const GetExtraBlockCallback& callback) { m_getExtraBlock = callback; } @@ -142,47 +136,36 @@ class SVSyncCore : noncopyable * @brief Callback on receiving extra data in a sync interest. * Will be called BEFORE the interest is processed. */ - void - setRecvExtraBlockCallback(const RecvExtraBlockCallback& callback) + void setRecvExtraBlockCallback(const RecvExtraBlockCallback& callback) { m_recvExtraBlock = callback; } /// @brief Get current version vector - VersionVector& - getState() - { - return m_vv; - } + VersionVector& getState() { return m_vv; } /// @brief Get human-readable representation of version vector - std::string - getStateStr() const - { - return m_vv.toStr(); - } + std::string getStateStr() const { return m_vv.toStr(); } -NDN_SVS_PUBLIC_WITH_TESTS_ELSE_PRIVATE: - void - onSyncInterest(const Interest& interest); + NDN_SVS_PUBLIC_WITH_TESTS_ELSE_PRIVATE + : void + onSyncInterest(const Interest& interest); - void - onSyncInterestValidated(const Interest& interest); + void onSyncInterestValidated(const Interest& interest); /** * @brief Mark the instance as initialized and send the first interest */ - void - sendInitialInterest(); + void sendInitialInterest(); /** * @brief sendSyncInterest and schedule a new retxSyncInterest event. * * @param send Send a sync interest immediately - * @param delay Delay in milliseconds to schedule next interest (0 for default). + * @param delay Delay in milliseconds to schedule next interest (0 for + * default). */ - void - retxSyncInterest(bool send, unsigned int delay); + void retxSyncInterest(bool send, unsigned int delay); /** * @brief Add one sync interest to queue. @@ -190,8 +173,7 @@ class SVSyncCore : noncopyable * Called by retxSyncInterest(), or after increasing a sequence * number with updateSeqNo() */ - void - sendSyncInterest(); + void sendSyncInterest(); /** * @brief Merge state vector into the current @@ -203,8 +185,8 @@ class SVSyncCore : noncopyable * @returns a tuple of representing: * . */ - std::tuple> - mergeStateVector(const VersionVector& vvOther); + std::tuple> mergeStateVector( + const VersionVector& vvOther); /** * @brief Record vector by merging it into m_recordedVv @@ -212,8 +194,7 @@ class SVSyncCore : noncopyable * @param vvOther state vector to merge in * @returns if recorded successfully */ - bool - recordVector(const VersionVector& vvOther); + bool recordVector(const VersionVector& vvOther); /** * @brief Enter suppression state by setting @@ -223,19 +204,13 @@ class SVSyncCore : noncopyable * * @param vvOther first vector to record */ - void - enterSuppressionState(const VersionVector& vvOther); + void enterSuppressionState(const VersionVector& vvOther); /// @brief Reference to scheduler - ndn::Scheduler& - getScheduler() - { - return m_scheduler; - } + ndn::Scheduler& getScheduler() { return m_scheduler; } /// @brief Get the current time in microseconds with arbitrary reference - long - getCurrentTime() const; + long getCurrentTime() const; public: static inline const NodeID EMPTY_NODE_ID; diff --git a/ndn-svs/fetcher.cpp b/ndn-svs/fetcher.cpp index 19e37eb..2942c4e 100644 --- a/ndn-svs/fetcher.cpp +++ b/ndn-svs/fetcher.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "fetcher.hpp" @@ -19,26 +20,32 @@ namespace ndn::svs { -Fetcher::Fetcher(Face& face, - const SecurityOptions& securityOptions) +Fetcher::Fetcher(Face& face, const SecurityOptions& securityOptions) : m_face(face) , m_scheduler(face.getIoContext()) , m_securityOptions(securityOptions) -{} +{ +} void -Fetcher::expressInterest(const ndn::Interest& interest, - const ndn::DataCallback& afterSatisfied, - const ndn::NackCallback& afterNacked, - const ndn::TimeoutCallback& afterTimeout, - int nRetries, - const ndn::security::DataValidationFailureCallback& afterValidationFailed) +Fetcher::expressInterest( + const ndn::Interest& interest, + const ndn::DataCallback& afterSatisfied, + const ndn::NackCallback& afterNacked, + const ndn::TimeoutCallback& afterTimeout, + int nRetries, + const ndn::security::DataValidationFailureCallback& afterValidationFailed) { uint64_t id = ++m_interestIdCounter; m_interestQueue.push({ - id, interest, afterSatisfied, afterNacked, - afterTimeout, nRetries, - m_securityOptions.nRetriesOnValidationFail, afterValidationFailed, + id, + interest, + afterSatisfied, + afterNacked, + afterTimeout, + nRetries, + m_securityOptions.nRetriesOnValidationFail, + afterValidationFailed, }); processQueue(); } @@ -60,8 +67,7 @@ Fetcher::expressInterest(const QueuedInterest& qi) void Fetcher::processQueue() { - while (!m_interestQueue.empty() && m_pendingInterests.size() < m_windowSize) - { + while (!m_interestQueue.empty() && m_pendingInterests.size() < m_windowSize) { QueuedInterest i = m_interestQueue.front(); m_interestQueue.pop(); @@ -74,26 +80,27 @@ Fetcher::processQueue() } void -Fetcher::onData(const Interest& interest, const Data& data, +Fetcher::onData(const Interest& interest, + const Data& data, const QueuedInterest& qi) { m_pendingInterests.erase(qi.id); processQueue(); - if (m_securityOptions.validator == nullptr) - { + if (m_securityOptions.validator == nullptr) { // No validator provided qi.afterSatisfied(interest, data); - } - else - { - auto onDataValidated = [qi] (const Data& data) { + } else { + auto onDataValidated = [qi](const Data& data) { qi.afterSatisfied(qi.interest, data); }; - auto onValidationFailed = [this, qi] (const Data& data, const ValidationError& error) { + auto onValidationFailed = [this, qi](const Data& data, + const ValidationError& error) { if (qi.nRetriesOnValidationFail > 0) { - this->m_scheduler.schedule(ndn::time::milliseconds(this->m_securityOptions.millisBeforeRetryOnValidationFail), + this->m_scheduler.schedule( + ndn::time::milliseconds( + this->m_securityOptions.millisBeforeRetryOnValidationFail), [this, qi] { QueuedInterest qiNew(qi); qiNew.nRetriesOnValidationFail--; @@ -107,12 +114,14 @@ Fetcher::onData(const Interest& interest, const Data& data, } }; - m_securityOptions.validator->validate(data, onDataValidated, onValidationFailed); + m_securityOptions.validator->validate( + data, onDataValidated, onValidationFailed); } } void -Fetcher::onNack(const ndn::Interest& interest, const ndn::lp::Nack& nack, +Fetcher::onNack(const ndn::Interest& interest, + const ndn::lp::Nack& nack, const QueuedInterest& qi) { m_pendingInterests.erase(qi.id); @@ -121,13 +130,11 @@ Fetcher::onNack(const ndn::Interest& interest, const ndn::lp::Nack& nack, } void -Fetcher::onTimeout(const Interest& interest, - const QueuedInterest& qi) +Fetcher::onTimeout(const Interest& interest, const QueuedInterest& qi) { m_pendingInterests.erase(qi.id); - if (qi.nRetries == 0) - { + if (qi.nRetries == 0) { processQueue(); return qi.afterTimeout(interest); } diff --git a/ndn-svs/fetcher.hpp b/ndn-svs/fetcher.hpp index eb82dbc..b6a0af5 100644 --- a/ndn-svs/fetcher.hpp +++ b/ndn-svs/fetcher.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_FETCHER_HPP @@ -31,34 +32,30 @@ class Fetcher public: Fetcher(Face& face, const SecurityOptions& securityOptions); - void - expressInterest(const ndn::Interest& interest, - const ndn::DataCallback& afterSatisfied, - const ndn::NackCallback& afterNacked, - const ndn::TimeoutCallback& afterTimeout, - int nRetries = 0, - const ndn::security::DataValidationFailureCallback& afterValidationFailed = nullptr); + void expressInterest(const ndn::Interest& interest, + const ndn::DataCallback& afterSatisfied, + const ndn::NackCallback& afterNacked, + const ndn::TimeoutCallback& afterTimeout, + int nRetries = 0, + const ndn::security::DataValidationFailureCallback& + afterValidationFailed = nullptr); private: struct QueuedInterest; - void - expressInterest(const QueuedInterest& qi); + void expressInterest(const QueuedInterest& qi); - void - onData(const Interest& interest, const Data& data, - const QueuedInterest& qi); + void onData(const Interest& interest, + const Data& data, + const QueuedInterest& qi); - void - onNack(const ndn::Interest& interest, const ndn::lp::Nack& nack, - const QueuedInterest& qi); + void onNack(const ndn::Interest& interest, + const ndn::lp::Nack& nack, + const QueuedInterest& qi); - void - onTimeout(const Interest& interest, - const QueuedInterest& qi); + void onTimeout(const Interest& interest, const QueuedInterest& qi); - void - processQueue(); + void processQueue(); private: Face& m_face; diff --git a/ndn-svs/mapping-provider.cpp b/ndn-svs/mapping-provider.cpp index 6895967..86de051 100644 --- a/ndn-svs/mapping-provider.cpp +++ b/ndn-svs/mapping-provider.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "mapping-provider.hpp" @@ -23,21 +24,20 @@ MappingList::MappingList() = default; MappingList::MappingList(const NodeID& nid) : nodeId(nid) -{} +{ +} MappingList::MappingList(const Block& block) { block.parse(); for (auto it = block.elements_begin(); it != block.elements_end(); it++) { - if (it->type() == ndn::tlv::Name) - { + if (it->type() == ndn::tlv::Name) { nodeId = NodeID(*it); continue; } - if (it->type() == tlv::MappingEntry) - { + if (it->type() == tlv::MappingEntry) { it->parse(); // SeqNo and ApplicationName @@ -46,7 +46,8 @@ MappingList::MappingList(const Block& block) // Additional blocks std::vector blocks; - for (auto it2 = it->elements().begin() + 2; it2 != it->elements().end(); it2++) + for (auto it2 = it->elements().begin() + 2; it2 != it->elements().end(); + it2++) blocks.push_back(*it2); pairs.push_back({ seqNo, std::make_pair(name, blocks) }); @@ -61,8 +62,7 @@ MappingList::encode() const ndn::encoding::EncodingBuffer enc; size_t totalLength = 0; - for (const auto& [seq, mapping] : pairs) - { + for (const auto& [seq, mapping] : pairs) { size_t entryLength = 0; // Additional blocks @@ -73,7 +73,8 @@ MappingList::encode() const entryLength += ndn::encoding::prependBlock(enc, mapping.first.wireEncode()); // SeqNo - entryLength += ndn::encoding::prependNonNegativeIntegerBlock(enc, tlv::SeqNo, seq); + entryLength += + ndn::encoding::prependNonNegativeIntegerBlock(enc, tlv::SeqNo, seq); totalLength += enc.prependVarNumber(entryLength); totalLength += enc.prependVarNumber(tlv::MappingEntry); @@ -97,14 +98,16 @@ MappingProvider::MappingProvider(const Name& syncPrefix, , m_fetcher(face, securityOptions) , m_securityOptions(securityOptions) { - m_registeredPrefix = - m_face.setInterestFilter(Name(m_id).append(m_syncPrefix).append("MAPPING"), - std::bind(&MappingProvider::onMappingQuery, this, _2), - [] (auto&&...) {}); + m_registeredPrefix = m_face.setInterestFilter( + Name(m_id).append(m_syncPrefix).append("MAPPING"), + std::bind(&MappingProvider::onMappingQuery, this, _2), + [](auto&&...) {}); } void -MappingProvider::insertMapping(const NodeID& nodeId, const SeqNo& seqNo, const MappingEntryPair& entry) +MappingProvider::insertMapping(const NodeID& nodeId, + const SeqNo& seqNo, + const MappingEntryPair& entry) { m_map[Name(nodeId).appendNumber(seqNo)] = entry; } @@ -121,13 +124,11 @@ MappingProvider::onMappingQuery(const Interest& interest) MissingDataInfo query = parseMappingQueryDataName(interest.getName()); MappingList queryResponse(query.nodeId); - for (SeqNo i = query.low; i <= std::max(query.high, query.low); i++) - { + for (SeqNo i = query.low; i <= std::max(query.high, query.low); i++) { try { auto mapping = getMapping(query.nodeId, i); queryResponse.pairs.emplace_back(i, mapping); - } - catch (const std::exception&) { + } catch (const std::exception&) { // TODO: don't give up if not everything is found // Instead return whatever we have and let the client request // the remaining mappings again @@ -151,7 +152,7 @@ MappingProvider::fetchNameMapping(const MissingDataInfo& info, const MappingListCallback& onValidated, int nRetries) { - TimeoutCallback onTimeout = [] (auto&&...) {}; + TimeoutCallback onTimeout = [](auto&&...) {}; return fetchNameMapping(info, onValidated, onTimeout, nRetries); } @@ -167,8 +168,7 @@ MappingProvider::fetchNameMapping(const MissingDataInfo& info, interest.setMustBeFresh(false); interest.setInterestLifetime(2_s); - auto onDataValidated = [this, onValidated, info] (const Data& data) - { + auto onDataValidated = [this, onValidated, info](const Data& data) { Block block = data.getContent().blockFromValue(); MappingList list(block); @@ -176,8 +176,7 @@ MappingProvider::fetchNameMapping(const MissingDataInfo& info, for (const auto& [seq, mapping] : list.pairs) { try { getMapping(info.nodeId, seq); - } - catch (const std::exception&) { + } catch (const std::exception&) { insertMapping(info.nodeId, seq, mapping); } } @@ -188,17 +187,19 @@ MappingProvider::fetchNameMapping(const MissingDataInfo& info, m_fetcher.expressInterest(interest, std::bind(onDataValidated, _2), std::bind(onTimeout, _1), // Nack - onTimeout, nRetries, - [] (auto&&...) {}); + onTimeout, + nRetries, + [](auto&&...) {}); } Name MappingProvider::getMappingQueryDataName(const MissingDataInfo& info) { - return Name(info.nodeId).append(m_syncPrefix) - .append("MAPPING") - .appendNumber(info.low) - .appendNumber(info.high); + return Name(info.nodeId) + .append(m_syncPrefix) + .append("MAPPING") + .appendNumber(info.low) + .appendNumber(info.high); } MissingDataInfo diff --git a/ndn-svs/mapping-provider.hpp b/ndn-svs/mapping-provider.hpp index 73490d6..59714a8 100644 --- a/ndn-svs/mapping-provider.hpp +++ b/ndn-svs/mapping-provider.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_MAPPING_PROVIDER_HPP @@ -34,16 +35,13 @@ class MappingList public: MappingList(); - explicit - MappingList(const NodeID& nid); + explicit MappingList(const NodeID& nid); /// @brief Decode from Block - explicit - MappingList(const Block& block); + explicit MappingList(const Block& block); /// @brief Encode to Block - Block - encode() const; + Block encode() const; public: NodeID nodeId; @@ -61,24 +59,23 @@ class MappingProvider : noncopyable ndn::Face& face, const SecurityOptions& securityOptions); - virtual - ~MappingProvider() = default; + virtual ~MappingProvider() = default; using MappingListCallback = std::function; /** * @brief Insert a mapping entry into the store */ - void - insertMapping(const NodeID& nodeId, const SeqNo& seqNo, const MappingEntryPair& entry); + void insertMapping(const NodeID& nodeId, + const SeqNo& seqNo, + const MappingEntryPair& entry); /** * @brief Get a mapping and throw if not found * * @returns Corresponding application name */ - MappingEntryPair - getMapping(const NodeID& nodeId, const SeqNo& seqNo); + MappingEntryPair getMapping(const NodeID& nodeId, const SeqNo& seqNo); /** * @brief Retrieve the data mappings for encapsulated data packets @@ -86,10 +83,9 @@ class MappingProvider : noncopyable * @param info Query info * @param onValidated Callback when mapping is fetched and validated */ - void - fetchNameMapping(const MissingDataInfo& info, - const MappingListCallback& onValidated, - int nRetries = 0); + void fetchNameMapping(const MissingDataInfo& info, + const MappingListCallback& onValidated, + int nRetries = 0); /** * @brief Retrieve the data mappings for encapsulated data packets @@ -98,27 +94,23 @@ class MappingProvider : noncopyable * @param onValidated Callback when mapping is fetched and validated * @param onTimeout Callback when mapping is not retrieved */ - void - fetchNameMapping(const MissingDataInfo& info, - const MappingListCallback& onValidated, - const TimeoutCallback& onTimeout, - int nRetries = 0); + void fetchNameMapping(const MissingDataInfo& info, + const MappingListCallback& onValidated, + const TimeoutCallback& onTimeout, + int nRetries = 0); private: /** * @brief Return data name for mapping query */ - Name - getMappingQueryDataName(const MissingDataInfo& info); + Name getMappingQueryDataName(const MissingDataInfo& info); /** * @brief Return the query from mapping data name */ - MissingDataInfo - parseMappingQueryDataName(const Name& name); + MissingDataInfo parseMappingQueryDataName(const Name& name); - void - onMappingQuery(const Interest& interest); + void onMappingQuery(const Interest& interest); private: const Name m_syncPrefix; diff --git a/ndn-svs/security-options.cpp b/ndn-svs/security-options.cpp index 59c2852..abd2a8e 100644 --- a/ndn-svs/security-options.cpp +++ b/ndn-svs/security-options.cpp @@ -2,23 +2,26 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "security-options.hpp" namespace ndn::svs { -const SecurityOptions SecurityOptions::DEFAULT{SecurityOptions::DEFAULT_KEYCHAIN}; +const SecurityOptions SecurityOptions::DEFAULT{ + SecurityOptions::DEFAULT_KEYCHAIN +}; BaseSigner::~BaseSigner() = default; @@ -39,7 +42,8 @@ SecurityOptions::SecurityOptions(KeyChain& keyChain) , dataSigner(std::make_shared(keyChain)) , pubSigner(std::make_shared(keyChain)) { - interestSigner->signingInfo.setSignedInterestFormat(security::SignedInterestFormat::V03); + interestSigner->signingInfo.setSignedInterestFormat( + security::SignedInterestFormat::V03); } } // namespace ndn::svs diff --git a/ndn-svs/security-options.hpp b/ndn-svs/security-options.hpp index 53517fb..38992f5 100644 --- a/ndn-svs/security-options.hpp +++ b/ndn-svs/security-options.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_SECURITY_OPTIONS_HPP @@ -27,18 +28,17 @@ namespace ndn::svs { class BaseValidator : noncopyable { public: - virtual - ~BaseValidator() = default; + virtual ~BaseValidator() = default; /** * @brief Asynchronously validate @p data * * @note @p successCb and @p failureCb must not be nullptr */ - virtual void - validate(const Data& data, - const ndn::security::DataValidationSuccessCallback& successCb, - const ndn::security::DataValidationFailureCallback& failureCb) + virtual void validate( + const Data& data, + const ndn::security::DataValidationSuccessCallback& successCb, + const ndn::security::DataValidationFailureCallback& failureCb) { successCb(data); } @@ -48,10 +48,10 @@ class BaseValidator : noncopyable * * @note @p successCb and @p failureCb must not be nullptr */ - virtual void - validate(const Interest& interest, - const ndn::security::InterestValidationSuccessCallback& successCb, - const ndn::security::InterestValidationFailureCallback& failureCb) + virtual void validate( + const Interest& interest, + const ndn::security::InterestValidationSuccessCallback& successCb, + const ndn::security::InterestValidationFailureCallback& failureCb) { successCb(interest); } @@ -63,14 +63,11 @@ class BaseValidator : noncopyable class BaseSigner : noncopyable { public: - virtual - ~BaseSigner(); + virtual ~BaseSigner(); - virtual void - sign(Interest& interest) const {} + virtual void sign(Interest& interest) const {} - virtual void - sign(Data& data) const {} + virtual void sign(Data& data) const {} public: security::SigningInfo signingInfo; @@ -82,16 +79,14 @@ class BaseSigner : noncopyable class KeyChainSigner : public BaseSigner { public: - explicit - KeyChainSigner(KeyChain& keyChain) + explicit KeyChainSigner(KeyChain& keyChain) : m_keyChain(keyChain) - {} + { + } - void - sign(Interest& interest) const override; + void sign(Interest& interest) const override; - void - sign(Data& data) const override; + void sign(Data& data) const override; private: KeyChain& m_keyChain; @@ -103,8 +98,7 @@ class KeyChainSigner : public BaseSigner class SecurityOptions { public: - explicit - SecurityOptions(KeyChain& keyChain); + explicit SecurityOptions(KeyChain& keyChain); public: /** Signing options for sync interests */ diff --git a/ndn-svs/store-memory.hpp b/ndn-svs/store-memory.hpp index 1483c8e..a814923 100644 --- a/ndn-svs/store-memory.hpp +++ b/ndn-svs/store-memory.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_STORE_MEMORY_HPP @@ -26,17 +27,12 @@ namespace ndn::svs { class MemoryDataStore : public DataStore { public: - std::shared_ptr - find(const Interest& interest) override + std::shared_ptr find(const Interest& interest) override { return m_ims.find(interest); } - void - insert(const Data& data) override - { - return m_ims.insert(data); - } + void insert(const Data& data) override { return m_ims.insert(data); } private: InMemoryStoragePersistent m_ims; diff --git a/ndn-svs/store.hpp b/ndn-svs/store.hpp index 04eba07..3af3eb1 100644 --- a/ndn-svs/store.hpp +++ b/ndn-svs/store.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_STORE_HPP @@ -24,14 +25,11 @@ namespace ndn::svs { class DataStore : noncopyable { public: - virtual - ~DataStore() = default; + virtual ~DataStore() = default; - virtual std::shared_ptr - find(const Interest& interest) = 0; + virtual std::shared_ptr find(const Interest& interest) = 0; - virtual void - insert(const Data& data) = 0; + virtual void insert(const Data& data) = 0; }; } // namespace ndn::svs diff --git a/ndn-svs/svspubsub.cpp b/ndn-svs/svspubsub.cpp index 723c890..2b096cd 100644 --- a/ndn-svs/svspubsub.cpp +++ b/ndn-svs/svspubsub.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2021-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "svspubsub.hpp" @@ -34,18 +35,25 @@ SVSPubSub::SVSPubSub(const Name& syncPrefix, , m_onUpdate(std::move(updateCallback)) , m_opts(options) , m_securityOptions(securityOptions) - , m_svsync(syncPrefix, nodePrefix, face, + , m_svsync(syncPrefix, + nodePrefix, + face, std::bind(&SVSPubSub::updateCallbackInternal, this, _1), - securityOptions, options.dataStore) + securityOptions, + options.dataStore) , m_mappingProvider(syncPrefix, nodePrefix, face, securityOptions) { - m_svsync.getCore().setGetExtraBlockCallback(std::bind(&SVSPubSub::onGetExtraData, this, _1)); - m_svsync.getCore().setRecvExtraBlockCallback(std::bind(&SVSPubSub::onRecvExtraData, this, _1)); + m_svsync.getCore().setGetExtraBlockCallback( + std::bind(&SVSPubSub::onGetExtraData, this, _1)); + m_svsync.getCore().setRecvExtraBlockCallback( + std::bind(&SVSPubSub::onRecvExtraData, this, _1)); } SeqNo -SVSPubSub::publish(const Name& name, span value, - const Name& nodePrefix, time::milliseconds freshnessPeriod, +SVSPubSub::publish(const Name& name, + span value, + const Name& nodePrefix, + time::milliseconds freshnessPeriod, std::vector mappingBlocks) { // Segment the data if larger than MAX_DATA_SIZE @@ -56,32 +64,35 @@ SVSPubSub::publish(const Name& name, span value, NodeID nid = nodePrefix == EMPTY_NAME ? m_dataPrefix : nodePrefix; SeqNo seqNo = m_svsync.getCore().getSeqNo(nid) + 1; - for (size_t i = 0; i < nSegments; i++) - { + for (size_t i = 0; i < nSegments; i++) { // Create encapsulated segment auto segmentName = Name(name).appendVersion(0).appendSegment(i); auto segment = Data(segmentName); segment.setFreshnessPeriod(freshnessPeriod); const uint8_t* segVal = value.data() + i * MAX_DATA_SIZE; - const size_t segValSize = std::min(value.size() - i * MAX_DATA_SIZE, MAX_DATA_SIZE); + const size_t segValSize = + std::min(value.size() - i * MAX_DATA_SIZE, MAX_DATA_SIZE); segment.setContent(ndn::make_span(segVal, segValSize)); segment.setFinalBlock(finalBlock); m_securityOptions.dataSigner->sign(segment); // Insert outer segment - m_svsync.insertDataSegment(segment.wireEncode(), freshnessPeriod, - nid, seqNo, i, finalBlock, ndn::tlv::Data); + m_svsync.insertDataSegment(segment.wireEncode(), + freshnessPeriod, + nid, + seqNo, + i, + finalBlock, + ndn::tlv::Data); } // Insert mapping and manually update the sequence number insertMapping(nid, seqNo, name, mappingBlocks); m_svsync.getCore().updateSeqNo(seqNo, nid); return seqNo; - } - else - { + } else { ndn::Data data(name); data.setContent(value); data.setFreshnessPeriod(freshnessPeriod); @@ -91,17 +102,21 @@ SVSPubSub::publish(const Name& name, span value, } SeqNo -SVSPubSub::publishPacket(const Data& data, const Name& nodePrefix, +SVSPubSub::publishPacket(const Data& data, + const Name& nodePrefix, std::vector mappingBlocks) { NodeID nid = nodePrefix == EMPTY_NAME ? m_dataPrefix : nodePrefix; - SeqNo seqNo = m_svsync.publishData(data.wireEncode(), data.getFreshnessPeriod(), nid, ndn::tlv::Data); + SeqNo seqNo = m_svsync.publishData( + data.wireEncode(), data.getFreshnessPeriod(), nid, ndn::tlv::Data); insertMapping(nid, seqNo, data.getName(), mappingBlocks); return seqNo; } void -SVSPubSub::insertMapping(const NodeID& nid, SeqNo seqNo, const Name& name, +SVSPubSub::insertMapping(const NodeID& nid, + SeqNo seqNo, + const Name& name, std::vector additional) { // additional is a copy deliberately @@ -109,10 +124,11 @@ SVSPubSub::insertMapping(const NodeID& nid, SeqNo seqNo, const Name& name, // add timestamp block if (m_opts.useTimestamp) { - unsigned long now = - std::chrono::duration_cast - (std::chrono::system_clock::now().time_since_epoch()).count(); - auto timestamp = Name::Component::fromNumber(now, tlv::TimestampNameComponent); + unsigned long now = std::chrono::duration_cast( + std::chrono::system_clock::now().time_since_epoch()) + .count(); + auto timestamp = + Name::Component::fromNumber(now, tlv::TimestampNameComponent); additional.push_back(timestamp); } @@ -120,8 +136,8 @@ SVSPubSub::insertMapping(const NodeID& nid, SeqNo seqNo, const Name& name, MappingEntryPair entry = { name, additional }; // notify subscribers in next sync interest - if (m_notificationMappingList.nodeId == EMPTY_NAME || m_notificationMappingList.nodeId == nid) - { + if (m_notificationMappingList.nodeId == EMPTY_NAME || + m_notificationMappingList.nodeId == nid) { m_notificationMappingList.nodeId = nid; m_notificationMappingList.pairs.push_back({ seqNo, entry }); } @@ -131,7 +147,9 @@ SVSPubSub::insertMapping(const NodeID& nid, SeqNo seqNo, const Name& name, } uint32_t -SVSPubSub::subscribe(const Name& prefix, const SubscriptionCallback& callback, bool packets) +SVSPubSub::subscribe(const Name& prefix, + const SubscriptionCallback& callback, + bool packets) { uint32_t handle = ++m_subscriptionCount; Subscription sub = { handle, prefix, callback, packets, false }; @@ -140,8 +158,10 @@ SVSPubSub::subscribe(const Name& prefix, const SubscriptionCallback& callback, b } uint32_t -SVSPubSub::subscribeToProducer(const Name& nodePrefix, const SubscriptionCallback& callback, - bool prefetch, bool packets) +SVSPubSub::subscribeToProducer(const Name& nodePrefix, + const SubscriptionCallback& callback, + bool prefetch, + bool packets) { uint32_t handle = ++m_subscriptionCount; Subscription sub = { handle, nodePrefix, callback, packets, prefetch }; @@ -152,12 +172,9 @@ SVSPubSub::subscribeToProducer(const Name& nodePrefix, const SubscriptionCallbac void SVSPubSub::unsubscribe(uint32_t handle) { - auto unsub = [handle](std::vector& subs) - { - for (auto it = subs.begin(); it != subs.end(); ++it) - { - if (it->id == handle) - { + auto unsub = [handle](std::vector& subs) { + for (auto it = subs.begin(); it != subs.end(); ++it) { + if (it->id == handle) { subs.erase(it); return; } @@ -171,68 +188,61 @@ SVSPubSub::unsubscribe(uint32_t handle) void SVSPubSub::updateCallbackInternal(const std::vector& info) { - for (const auto& stream : info) - { + for (const auto& stream : info) { Name streamName(stream.nodeId); // Producer subscriptions - for (const auto& sub : m_producerSubscriptions) - { - if (sub.prefix.isPrefixOf(streamName)) - { + for (const auto& sub : m_producerSubscriptions) { + if (sub.prefix.isPrefixOf(streamName)) { // Add to fetching queue for (SeqNo i = stream.low; i <= stream.high; i++) m_fetchMap[std::pair(stream.nodeId, i)].push_back(sub); // Prefetch next available data if (sub.prefetch) - m_svsync.fetchData(stream.nodeId, stream.high + 1, [] (auto&&...) {}); // do nothing with prefetch + m_svsync.fetchData(stream.nodeId, stream.high + 1, [](auto&&...) { + }); // do nothing with prefetch } } // Fetch all mappings if we have prefix subscription(s) - if (!m_prefixSubscriptions.empty()) - { + if (!m_prefixSubscriptions.empty()) { MissingDataInfo remainingInfo = stream; // Attemt to find what we already know about mapping // This typically refers to the Sync Interest mapping optimization, // where the Sync Interest contains the notification mapping list - for (SeqNo i = remainingInfo.low; i <= remainingInfo.high; i++) - { - try - { + for (SeqNo i = remainingInfo.low; i <= remainingInfo.high; i++) { + try { // throws if mapping not found this->processMapping(stream.nodeId, i); remainingInfo.low++; - } - catch (const std::exception&) - { + } catch (const std::exception&) { break; } } // Find from network what we don't yet know - while (remainingInfo.high >= remainingInfo.low) - { + while (remainingInfo.high >= remainingInfo.low) { // Fetch a max of 10 entries per request // This is to ensure the mapping response does not overflow // TODO: implement a better solution to this issue MissingDataInfo truncatedRemainingInfo = remainingInfo; - if (truncatedRemainingInfo.high - truncatedRemainingInfo.low > 10) - { + if (truncatedRemainingInfo.high - truncatedRemainingInfo.low > 10) { truncatedRemainingInfo.high = truncatedRemainingInfo.low + 10; } - m_mappingProvider.fetchNameMapping(truncatedRemainingInfo, - [this, remainingInfo, streamName] (const MappingList& list) { + m_mappingProvider.fetchNameMapping( + truncatedRemainingInfo, + [this, remainingInfo, streamName](const MappingList& list) { bool queued = false; for (const auto& [seq, mapping] : list.pairs) queued |= this->processMapping(streamName, seq); if (queued) this->fetchAll(); - }, -1); + }, + -1); remainingInfo.low += 11; } @@ -250,18 +260,16 @@ SVSPubSub::processMapping(const NodeID& nodeId, SeqNo seqNo) auto mapping = m_mappingProvider.getMapping(nodeId, seqNo); // check if timestamp is too old - if (m_opts.maxPubAge > 0_ms) - { + if (m_opts.maxPubAge > 0_ms) { // look for the additional timestamp block // if no timestamp block is present, we just skip this step - for (const auto& block : mapping.second) - { + for (const auto& block : mapping.second) { if (block.type() != tlv::TimestampNameComponent) continue; - unsigned long now = - std::chrono::duration_cast - (std::chrono::system_clock::now().time_since_epoch()).count(); + unsigned long now = std::chrono::duration_cast( + std::chrono::system_clock::now().time_since_epoch()) + .count(); unsigned long pubTime = Name::Component(block).toNumber(); unsigned long maxAge = time::microseconds(m_opts.maxPubAge).count(); @@ -273,10 +281,8 @@ SVSPubSub::processMapping(const NodeID& nodeId, SeqNo seqNo) // check if known mapping matches subscription bool queued = false; - for (const auto& sub : m_prefixSubscriptions) - { - if (sub.prefix.isPrefixOf(mapping.first)) - { + for (const auto& sub : m_prefixSubscriptions) { + if (sub.prefix.isPrefixOf(mapping.first)) { m_fetchMap[std::pair(nodeId, seqNo)].push_back(sub); queued = true; } @@ -288,8 +294,7 @@ SVSPubSub::processMapping(const NodeID& nodeId, SeqNo seqNo) void SVSPubSub::fetchAll() { - for (const auto& pair : m_fetchMap) - { + for (const auto& pair : m_fetchMap) { // Check if already fetching this publication auto key = pair.first; if (m_fetchingMap.find(key) != m_fetchingMap.end()) @@ -298,12 +303,14 @@ SVSPubSub::fetchAll() // Fetch first data packet const auto& [nodeId, seqNo] = key; - m_svsync.fetchData(nodeId, seqNo, std::bind(&SVSPubSub::onSyncData, this, _1, key), 12); + m_svsync.fetchData( + nodeId, seqNo, std::bind(&SVSPubSub::onSyncData, this, _1, key), 12); } } void -SVSPubSub::onSyncData(const Data& firstData, const std::pair& publication) +SVSPubSub::onSyncData(const Data& firstData, + const std::pair& publication) { // Make sure the data is encapsulated if (firstData.getContentType() != ndn::tlv::Data) { @@ -317,121 +324,124 @@ SVSPubSub::onSyncData(const Data& firstData, const std::pair& publi // Return data to packet subscriptions SubscriptionData subData = { - innerData.getName(), - innerContent.value_bytes(), - publication.first, - publication.second, + innerData.getName(), innerContent.value_bytes(), + publication.first, publication.second, innerData, }; // Function to return data to subscriptions - auto returnData = [this, firstData, subData, publication] () - { + auto returnData = [this, firstData, subData, publication]() { bool hasFinalBlock = subData.packet.value().getFinalBlock().has_value(); bool hasBlobSubcriptions = false; - for (const auto& sub : this->m_fetchMap[publication]) - { + for (const auto& sub : this->m_fetchMap[publication]) { if (sub.isPacketSubscription || !hasFinalBlock) sub.callback(subData); hasBlobSubcriptions |= !sub.isPacketSubscription; } - // If there are blob subscriptions and a final block, we need to fetch remaining segments - if (hasBlobSubcriptions && hasFinalBlock && firstData.getName().size() > 2) - { + // If there are blob subscriptions and a final block, we need to fetch + // remaining segments + if (hasBlobSubcriptions && hasFinalBlock && + firstData.getName().size() > 2) { // Fetch remaining segments auto pubName = firstData.getName().getPrefix(-2); Interest interest(pubName); // strip off version and segment number ndn::SegmentFetcher::Options opts; - auto fetcher = ndn::SegmentFetcher::start(m_face, interest, m_nullValidator, opts); - - fetcher->onComplete.connectSingleShot([this, publication] (const ndn::ConstBufferPtr& data) { - try { - // Binary BLOB to return to app - auto finalBuffer = std::make_shared>(std::vector(data->size())); - auto bufSize = std::make_shared(0); - bool hasValidator = !!m_securityOptions.encapsulatedDataValidator; - - // Read all TLVs as Data packets till the end of data buffer - ndn::Block block(6, data); - block.parse(); - - // Number of elements validated / failed to validate - auto numValidated = std::make_shared(0); - auto numFailed = std::make_shared(0); - auto numElem = block.elements_size(); - - if (numElem == 0) - return this->cleanUpFetch(publication); - - // Get name of inner data - auto innerName = Data(block.elements()[0]).getName().getPrefix(-2); - - // Function to send final buffer to subscriptions if possible - auto sendFinalBuffer = [this, innerName, publication, finalBuffer, bufSize, numFailed, numValidated, numElem] - { - if (*numValidated + *numFailed != numElem) - return; - - if (*numFailed > 0) // abort + auto fetcher = + ndn::SegmentFetcher::start(m_face, interest, m_nullValidator, opts); + + fetcher->onComplete.connectSingleShot( + [this, publication](const ndn::ConstBufferPtr& data) { + try { + // Binary BLOB to return to app + auto finalBuffer = std::make_shared>( + std::vector(data->size())); + auto bufSize = std::make_shared(0); + bool hasValidator = !!m_securityOptions.encapsulatedDataValidator; + + // Read all TLVs as Data packets till the end of data buffer + ndn::Block block(6, data); + block.parse(); + + // Number of elements validated / failed to validate + auto numValidated = std::make_shared(0); + auto numFailed = std::make_shared(0); + auto numElem = block.elements_size(); + + if (numElem == 0) return this->cleanUpFetch(publication); - // Resize buffer to actual size - finalBuffer->resize(*bufSize); - - // Return data to packet subscriptions - SubscriptionData subData = { - innerName, - *finalBuffer, - publication.first, - publication.second, - std::nullopt, + // Get name of inner data + auto innerName = Data(block.elements()[0]).getName().getPrefix(-2); + + // Function to send final buffer to subscriptions if possible + auto sendFinalBuffer = [this, + innerName, + publication, + finalBuffer, + bufSize, + numFailed, + numValidated, + numElem] { + if (*numValidated + *numFailed != numElem) + return; + + if (*numFailed > 0) // abort + return this->cleanUpFetch(publication); + + // Resize buffer to actual size + finalBuffer->resize(*bufSize); + + // Return data to packet subscriptions + SubscriptionData subData = { + innerName, *finalBuffer, publication.first, + publication.second, std::nullopt, + }; + + for (const auto& sub : this->m_fetchMap[publication]) + if (!sub.isPacketSubscription) + sub.callback(subData); + + this->cleanUpFetch(publication); }; - for (const auto& sub : this->m_fetchMap[publication]) - if (!sub.isPacketSubscription) - sub.callback(subData); - - this->cleanUpFetch(publication); - }; - - for (size_t i = 0; i < numElem; i++) - { - Data innerData(block.elements()[i]); - - // Copy actual binary data to final buffer - auto size = innerData.getContent().value_size(); - std::memcpy(finalBuffer->data() + *bufSize, innerData.getContent().value(), size); - *bufSize += size; - - // Validate inner data - if (hasValidator) { - this->m_securityOptions.encapsulatedDataValidator->validate(innerData, - [sendFinalBuffer, numValidated] (auto&&...) { - *numValidated += 1; - sendFinalBuffer(); - }, - [sendFinalBuffer, numFailed] (auto&&...) { - *numFailed += 1; - sendFinalBuffer(); - }); - } else { - *numValidated += 1; + for (size_t i = 0; i < numElem; i++) { + Data innerData(block.elements()[i]); + + // Copy actual binary data to final buffer + auto size = innerData.getContent().value_size(); + std::memcpy(finalBuffer->data() + *bufSize, + innerData.getContent().value(), + size); + *bufSize += size; + + // Validate inner data + if (hasValidator) { + this->m_securityOptions.encapsulatedDataValidator->validate( + innerData, + [sendFinalBuffer, numValidated](auto&&...) { + *numValidated += 1; + sendFinalBuffer(); + }, + [sendFinalBuffer, numFailed](auto&&...) { + *numFailed += 1; + sendFinalBuffer(); + }); + } else { + *numValidated += 1; + } } - } - sendFinalBuffer(); - } - catch (const std::exception&) { - cleanUpFetch(publication); - } - }); - fetcher->onError.connectSingleShot(std::bind(&SVSPubSub::cleanUpFetch, this, publication)); - } - else - { + sendFinalBuffer(); + } catch (const std::exception&) { + cleanUpFetch(publication); + } + }); + fetcher->onError.connectSingleShot( + std::bind(&SVSPubSub::cleanUpFetch, this, publication)); + } else { cleanUpFetch(publication); } }; @@ -439,11 +449,8 @@ SVSPubSub::onSyncData(const Data& firstData, const std::pair& publi // Validate encapsulated packet if (m_securityOptions.encapsulatedDataValidator) { m_securityOptions.encapsulatedDataValidator->validate( - innerData, - [&] (auto&&...) { returnData(); }, - [] (auto&&...) {}); - } - else { + innerData, [&](auto&&...) { returnData(); }, [](auto&&...) {}); + } else { returnData(); } } @@ -466,15 +473,13 @@ SVSPubSub::onGetExtraData(const VersionVector&) void SVSPubSub::onRecvExtraData(const Block& block) { - try - { + try { MappingList list(block); - for (const auto& p : list.pairs) - { + for (const auto& p : list.pairs) { m_mappingProvider.insertMapping(list.nodeId, p.first, p.second); } + } catch (const std::exception&) { } - catch (const std::exception&) {} } } // namespace ndn::svs diff --git a/ndn-svs/svspubsub.hpp b/ndn-svs/svspubsub.hpp index 941d4da..7b0fb59 100644 --- a/ndn-svs/svspubsub.hpp +++ b/ndn-svs/svspubsub.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2021-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_SVSPUBSUB_HPP @@ -19,8 +20,8 @@ #include "core.hpp" #include "mapping-provider.hpp" -#include "store.hpp" #include "security-options.hpp" +#include "store.hpp" #include "svsync.hpp" #include @@ -77,8 +78,7 @@ class SVSPubSub : noncopyable const SVSPubSubOptions& options = {}, const SecurityOptions& securityOptions = SecurityOptions::DEFAULT); - virtual - ~SVSPubSub() = default; + virtual ~SVSPubSub() = default; struct SubscriptionData { @@ -108,13 +108,14 @@ class SVSPubSub : noncopyable * @param value data buffer * @param nodePrefix Name to publish the data under * @param freshnessPeriod freshness period for the data - * @param mappingBlocks Additional blocks to be published with the mapping (use sparingly) + * @param mappingBlocks Additional blocks to be published with the mapping + * (use sparingly) */ - SeqNo - publish(const Name& name, span value, - const Name& nodePrefix = EMPTY_NAME, - time::milliseconds freshnessPeriod = FRESH_FOREVER, - std::vector mappingBlocks = {}); + SeqNo publish(const Name& name, + span value, + const Name& nodePrefix = EMPTY_NAME, + time::milliseconds freshnessPeriod = FRESH_FOREVER, + std::vector mappingBlocks = {}); /** * @brief Subscribe to a application name prefix. @@ -125,8 +126,9 @@ class SVSPubSub : noncopyable * * @returns Handle to the subscription */ - uint32_t - subscribe(const Name& prefix, const SubscriptionCallback& callback, bool packets = false); + uint32_t subscribe(const Name& prefix, + const SubscriptionCallback& callback, + bool packets = false); /** * @brief Subscribe to a data producer @@ -138,17 +140,17 @@ class SVSPubSub : noncopyable * * @returns Handle to the subscription */ - uint32_t - subscribeToProducer(const Name& nodePrefix, const SubscriptionCallback& callback, - bool prefetch = false, bool packets = false); + uint32_t subscribeToProducer(const Name& nodePrefix, + const SubscriptionCallback& callback, + bool prefetch = false, + bool packets = false); /** * @brief Unsubscribe from a stream using a handle * * @param handle Handle received during subscription */ - void - unsubscribe(uint32_t handle); + void unsubscribe(uint32_t handle); /** * @brief Publish a encapsulated Data packet in the session and trigger @@ -159,19 +161,15 @@ class SVSPubSub : noncopyable * * @param data Data packet to publish * @param nodePrefix Name to publish the data under - * @param mappingBlocks Additional blocks to be published with the mapping (use sparingly) + * @param mappingBlocks Additional blocks to be published with the mapping + * (use sparingly) */ - SeqNo - publishPacket(const Data& data, - const Name& nodePrefix = EMPTY_NAME, - std::vector mappingBlocks = {}); + SeqNo publishPacket(const Data& data, + const Name& nodePrefix = EMPTY_NAME, + std::vector mappingBlocks = {}); /** @brief Get the underlying sync */ - SVSync& - getSVSync() - { - return m_svsync; - } + SVSync& getSVSync() { return m_svsync; } private: struct Subscription @@ -183,41 +181,37 @@ class SVSPubSub : noncopyable bool prefetch; }; - void - onSyncData(const Data& syncData, const std::pair& publication); + void onSyncData(const Data& syncData, + const std::pair& publication); - void - updateCallbackInternal(const std::vector& info); + void updateCallbackInternal(const std::vector& info); - Block - onGetExtraData(const VersionVector& vv); + Block onGetExtraData(const VersionVector& vv); - void - onRecvExtraData(const Block& block); + void onRecvExtraData(const Block& block); /// @brief Insert a mapping entry into the store - void - insertMapping(const NodeID& nid, SeqNo seqNo, const Name& name, - std::vector additional); + void insertMapping(const NodeID& nid, + SeqNo seqNo, + const Name& name, + std::vector additional); /** * @brief Get and process mapping from store. * @returns true if new publications were queued for fetch * @throws std::exception error if mapping is not found */ - bool - processMapping(const NodeID& nodeId, SeqNo seqNo); + bool processMapping(const NodeID& nodeId, SeqNo seqNo); - void - fetchAll(); + void fetchAll(); - void - cleanUpFetch(const std::pair& publication); + void cleanUpFetch(const std::pair& publication); public: static inline const Name EMPTY_NAME; static constexpr size_t MAX_DATA_SIZE = 8000; - static constexpr time::milliseconds FRESH_FOREVER = time::years(10000); // well ... + static constexpr time::milliseconds FRESH_FOREVER = + time::years(10000); // well ... private: Face& m_face; diff --git a/ndn-svs/svsync-base.cpp b/ndn-svs/svsync-base.cpp index c5dda36..f1b3fa7 100644 --- a/ndn-svs/svsync-base.cpp +++ b/ndn-svs/svsync-base.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "svsync-base.hpp" @@ -46,19 +47,26 @@ SVSyncBase::SVSyncBase(const Name& syncPrefix, m_registeredDataPrefix = m_face.setInterestFilter(m_dataPrefix, std::bind(&SVSyncBase::onDataInterest, this, _2), - [] (auto&&...) {}); + [](auto&&...) {}); } SeqNo -SVSyncBase::publishData(const uint8_t* buf, size_t len, const ndn::time::milliseconds& freshness, +SVSyncBase::publishData(const uint8_t* buf, + size_t len, + const ndn::time::milliseconds& freshness, const NodeID& nid) { - return publishData(ndn::encoding::makeBinaryBlock(ndn::tlv::Content, {buf, len}), freshness, nid); + return publishData( + ndn::encoding::makeBinaryBlock(ndn::tlv::Content, { buf, len }), + freshness, + nid); } SeqNo -SVSyncBase::publishData(const Block& content, const ndn::time::milliseconds& freshness, - const NodeID& id, uint32_t contentType) +SVSyncBase::publishData(const Block& content, + const ndn::time::milliseconds& freshness, + const NodeID& id, + uint32_t contentType) { NodeID pubId = id != EMPTY_NODE_ID ? id : m_id; SeqNo newSeq = m_core.getSeqNo(pubId) + 1; @@ -79,9 +87,13 @@ SVSyncBase::publishData(const Block& content, const ndn::time::milliseconds& fre } void -SVSyncBase::insertDataSegment(const Block& content, const ndn::time::milliseconds& freshness, - const NodeID& nid, const SeqNo seq, const size_t segNo, - const Name::Component& finalBlock, uint32_t contentType) +SVSyncBase::insertDataSegment(const Block& content, + const ndn::time::milliseconds& freshness, + const NodeID& nid, + const SeqNo seq, + const size_t segNo, + const Name::Component& finalBlock, + uint32_t contentType) { Name dataName = getDataName(nid, seq).appendVersion(0).appendSegment(segNo); auto data = std::make_shared(dataName); @@ -102,17 +114,20 @@ SVSyncBase::onDataInterest(const Interest& interest) } void -SVSyncBase::fetchData(const NodeID& nid, const SeqNo& seqNo, - const DataValidatedCallback& onValidated, int nRetries) +SVSyncBase::fetchData(const NodeID& nid, + const SeqNo& seqNo, + const DataValidatedCallback& onValidated, + int nRetries) { DataValidationErrorCallback onValidationFailed = std::bind(&SVSyncBase::onDataValidationFailed, this, _1, _2); - TimeoutCallback onTimeout = [] (auto&&...) {}; + TimeoutCallback onTimeout = [](auto&&...) {}; fetchData(nid, seqNo, onValidated, onValidationFailed, onTimeout, nRetries); } void -SVSyncBase::fetchData(const NodeID& nid, const SeqNo& seqNo, +SVSyncBase::fetchData(const NodeID& nid, + const SeqNo& seqNo, const DataValidatedCallback& onValidated, const DataValidationErrorCallback& onValidationFailed, const TimeoutCallback& onTimeout, @@ -123,10 +138,13 @@ SVSyncBase::fetchData(const NodeID& nid, const SeqNo& seqNo, interest.setCanBePrefix(true); interest.setInterestLifetime(2_s); - m_fetcher.expressInterest(interest, - std::bind(&SVSyncBase::onDataValidated, this, _2, onValidated), - std::bind(onTimeout, _1), // Nack - onTimeout, nRetries, onValidationFailed); + m_fetcher.expressInterest( + interest, + std::bind(&SVSyncBase::onDataValidated, this, _2, onValidated), + std::bind(onTimeout, _1), // Nack + onTimeout, + nRetries, + onValidationFailed); } void diff --git a/ndn-svs/svsync-base.hpp b/ndn-svs/svsync-base.hpp index 5e22a25..7863051 100644 --- a/ndn-svs/svsync-base.hpp +++ b/ndn-svs/svsync-base.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_SVSYNC_BASE_HPP @@ -53,11 +54,11 @@ class SVSyncBase : noncopyable const SecurityOptions& securityOptions = SecurityOptions::DEFAULT, std::shared_ptr dataStore = DEFAULT_DATASTORE); - virtual - ~SVSyncBase() = default; + virtual ~SVSyncBase() = default; /** - * @brief Publish a data packet in the session and trigger synchronization updates + * @brief Publish a data packet in the session and trigger synchronization + * updates * * This method will create a data packet with the supplied content. * The packet name is the local session + seqNo. @@ -70,12 +71,14 @@ class SVSyncBase : noncopyable * * @returns Sequence number of the published data packet */ - SeqNo - publishData(const uint8_t* buf, size_t len, const ndn::time::milliseconds& freshness, - const NodeID& id = EMPTY_NODE_ID); + SeqNo publishData(const uint8_t* buf, + size_t len, + const ndn::time::milliseconds& freshness, + const NodeID& id = EMPTY_NODE_ID); /** - * @brief Publish a data packet in the session and trigger synchronization updates + * @brief Publish a data packet in the session and trigger synchronization + * updates * * This method will create a data packet with the supplied content. * The packet name is the local session + seqNo. @@ -87,10 +90,10 @@ class SVSyncBase : noncopyable * * @returns Sequence number of the published data packet */ - SeqNo - publishData(const Block& content, const ndn::time::milliseconds& freshness, - const NodeID& nid = EMPTY_NODE_ID, - uint32_t contentType = ndn::tlv::Content); + SeqNo publishData(const Block& content, + const ndn::time::milliseconds& freshness, + const NodeID& nid = EMPTY_NODE_ID, + uint32_t contentType = ndn::tlv::Content); /** * Insert segment into the store without changing the sequence number. @@ -103,55 +106,52 @@ class SVSyncBase : noncopyable * @param finalBlock FinalBlockId of the data packet * @param contentType Content type of the data packet */ - void - insertDataSegment(const Block& content, const ndn::time::milliseconds& freshness, - const NodeID& nid, const SeqNo seq, const size_t segNo, - const Name::Component& finalBlock, - uint32_t contentType = ndn::tlv::Content); + void insertDataSegment(const Block& content, + const ndn::time::milliseconds& freshness, + const NodeID& nid, + const SeqNo seq, + const size_t segNo, + const Name::Component& finalBlock, + uint32_t contentType = ndn::tlv::Content); /** * @brief Retrive a data packet with a particular seqNo from a session * * @param nid The name of the target node * @param seq The seqNo of the data packet. - * @param onValidated The callback when the retrieved packet has been validated. + * @param onValidated The callback when the retrieved packet has been + * validated. * @param nRetries The number of retries. */ - void - fetchData(const NodeID& nid, const SeqNo& seq, - const DataValidatedCallback& onValidated, - int nRetries = 0); + void fetchData(const NodeID& nid, + const SeqNo& seq, + const DataValidatedCallback& onValidated, + int nRetries = 0); /** * @brief Retrive a data packet with a particular seqNo from a session * * @param nid The name of the target node * @param seq The seqNo of the data packet. - * @param onValidated The callback when the retrieved packet has been validated. - * @param onValidationFailed The callback when the retrieved packet failed validation. + * @param onValidated The callback when the retrieved packet has been + * validated. + * @param onValidationFailed The callback when the retrieved packet failed + * validation. * @param onTimeout The callback when data is not retrieved. * @param nRetries The number of retries. */ - void - fetchData(const NodeID& nid, const SeqNo& seq, - const DataValidatedCallback& onValidated, - const DataValidationErrorCallback& onValidationFailed, - const TimeoutCallback& onTimeout, - int nRetries = 0); + void fetchData(const NodeID& nid, + const SeqNo& seq, + const DataValidatedCallback& onValidated, + const DataValidationErrorCallback& onValidationFailed, + const TimeoutCallback& onTimeout, + int nRetries = 0); /** @brief Get the underlying data store */ - DataStore& - getDataStore() - { - return *m_dataStore; - } + DataStore& getDataStore() { return *m_dataStore; } /** @brief Get the underlying SVS core */ - SVSyncCore& - getCore() - { - return m_core; - } + SVSyncCore& getCore() { return m_core; } protected: /** @@ -162,35 +162,26 @@ class SVSyncBase : noncopyable * data prefix for proper functionality, or the application must * independently produce data under the prefix. */ - virtual Name - getDataName(const NodeID& nid, const SeqNo& seqNo) = 0; + virtual Name getDataName(const NodeID& nid, const SeqNo& seqNo) = 0; public: static inline const NodeID EMPTY_NODE_ID; static inline const std::shared_ptr DEFAULT_DATASTORE; private: - void - onDataInterest(const Interest& interest); + void onDataInterest(const Interest& interest); - void - onDataValidated(const Data& data, - const DataValidatedCallback& dataCallback); + void onDataValidated(const Data& data, + const DataValidatedCallback& dataCallback); - void - onDataValidationFailed(const Data& data, - const ValidationError& error); + void onDataValidationFailed(const Data& data, const ValidationError& error); /** * Determines whether a particular data packet is to be cached * Can be used to cache data packets from other nodes when * using multicast data interests. */ - virtual bool - shouldCache(const Data& data) const - { - return false; - } + virtual bool shouldCache(const Data& data) const { return false; } protected: const Name m_syncPrefix; diff --git a/ndn-svs/svsync-shared.hpp b/ndn-svs/svsync-shared.hpp index 89a3e70..daca263 100644 --- a/ndn-svs/svsync-shared.hpp +++ b/ndn-svs/svsync-shared.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_SVSYNC_SHARED_HPP @@ -32,36 +33,33 @@ namespace ndn::svs { class SVSyncShared : public SVSyncBase { public: - SVSyncShared(const Name& grpPrefix, - const NodeID& id, - ndn::Face& face, - const UpdateCallback& updateCallback, - const SecurityOptions& securityOptions = SecurityOptions::DEFAULT, - std::shared_ptr dataStore = DEFAULT_DATASTORE) + SVSyncShared( + const Name& grpPrefix, + const NodeID& id, + ndn::Face& face, + const UpdateCallback& updateCallback, + const SecurityOptions& securityOptions = SecurityOptions::DEFAULT, + std::shared_ptr dataStore = DEFAULT_DATASTORE) : SVSyncBase(Name(grpPrefix).append("s"), Name(grpPrefix).append("d"), - id, face, updateCallback, securityOptions, std::move(dataStore)) - {} + id, + face, + updateCallback, + securityOptions, + std::move(dataStore)) + { + } - Name - getDataName(const NodeID& nid, const SeqNo& seqNo) override + Name getDataName(const NodeID& nid, const SeqNo& seqNo) override { return Name(m_dataPrefix).append(nid).appendNumber(seqNo); } /** @brief Set whether data of other nodes is also cached and served */ - void - setCacheAll(bool val) - { - m_cacheAll = val; - } + void setCacheAll(bool val) { m_cacheAll = val; } private: - bool - shouldCache(const Data&) const override - { - return m_cacheAll; - } + bool shouldCache(const Data&) const override { return m_cacheAll; } private: bool m_cacheAll = false; diff --git a/ndn-svs/svsync.hpp b/ndn-svs/svsync.hpp index 8e88de5..907048e 100644 --- a/ndn-svs/svsync.hpp +++ b/ndn-svs/svsync.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_SVSYNC_HPP @@ -37,12 +38,17 @@ class SVSync : public SVSyncBase const UpdateCallback& updateCallback, const SecurityOptions& securityOptions = SecurityOptions::DEFAULT, std::shared_ptr dataStore = DEFAULT_DATASTORE) - : SVSyncBase(syncPrefix, Name(nodePrefix).append(syncPrefix), nodePrefix, - face, updateCallback, securityOptions, std::move(dataStore)) - {} + : SVSyncBase(syncPrefix, + Name(nodePrefix).append(syncPrefix), + nodePrefix, + face, + updateCallback, + securityOptions, + std::move(dataStore)) + { + } - Name - getDataName(const NodeID& nid, const SeqNo& seqNo) override + Name getDataName(const NodeID& nid, const SeqNo& seqNo) override { return Name(nid).append(m_syncPrefix).appendNumber(seqNo); } diff --git a/ndn-svs/tlv.hpp b/ndn-svs/tlv.hpp index 574c6d5..03303ec 100644 --- a/ndn-svs/tlv.hpp +++ b/ndn-svs/tlv.hpp @@ -5,13 +5,14 @@ namespace ndn::svs::tlv { -enum : uint32_t { - StateVector = 201, - StateVectorEntry = 202, - SeqNo = 204, - MappingData = 205, - MappingEntry = 206, - LzmaBlock = 211, +enum : uint32_t +{ + StateVector = 201, + StateVectorEntry = 202, + SeqNo = 204, + MappingData = 205, + MappingEntry = 206, + LzmaBlock = 211, }; } // namespace ndn::svs::tlv diff --git a/ndn-svs/version-vector.cpp b/ndn-svs/version-vector.cpp index 96d3f7d..4cbfb87 100644 --- a/ndn-svs/version-vector.cpp +++ b/ndn-svs/version-vector.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "version-vector.hpp" @@ -44,10 +45,10 @@ VersionVector::encode() const ndn::encoding::EncodingBuffer enc; size_t totalLength = 0; - for (auto it = m_map.rbegin(); it != m_map.rend(); it++) - { + for (auto it = m_map.rbegin(); it != m_map.rend(); it++) { // SeqNo - size_t entryLength = ndn::encoding::prependNonNegativeIntegerBlock(enc, tlv::SeqNo, it->second); + size_t entryLength = ndn::encoding::prependNonNegativeIntegerBlock( + enc, tlv::SeqNo, it->second); // NodeID (Name) entryLength += ndn::encoding::prependBlock(enc, it->first.wireEncode()); @@ -66,8 +67,7 @@ std::string VersionVector::toStr() const { std::ostringstream stream; - for (const auto& elem : m_map) - { + for (const auto& elem : m_map) { stream << elem.first << ":" << elem.second << " "; } return stream.str(); diff --git a/ndn-svs/version-vector.hpp b/ndn-svs/version-vector.hpp index 9a18ae2..46b8b39 100644 --- a/ndn-svs/version-vector.hpp +++ b/ndn-svs/version-vector.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2022 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_VERSION_VECTOR_HPP @@ -37,56 +38,39 @@ class VersionVector VersionVector() = default; /** Decode a version vector from ndn::Block */ - explicit - VersionVector(const ndn::Block& encoded); + explicit VersionVector(const ndn::Block& encoded); /** Encode the version vector to a string */ - ndn::Block - encode() const; + ndn::Block encode() const; /** Get a human-readable representation */ - std::string - toStr() const; + std::string toStr() const; - SeqNo - set(const NodeID& nid, SeqNo seqNo) + SeqNo set(const NodeID& nid, SeqNo seqNo) { m_map[nid] = seqNo; m_lastUpdate[nid] = time::system_clock::now(); return seqNo; } - SeqNo - get(const NodeID& nid) const + SeqNo get(const NodeID& nid) const { auto elem = m_map.find(nid); return elem == m_map.end() ? 0 : elem->second; } - time::system_clock::time_point - getLastUpdate(const NodeID& nid) const + time::system_clock::time_point getLastUpdate(const NodeID& nid) const { auto elem = m_lastUpdate.find(nid); - return elem == m_lastUpdate.end() ? time::system_clock::time_point::min() : elem->second; + return elem == m_lastUpdate.end() ? time::system_clock::time_point::min() + : elem->second; } - const_iterator - begin() const noexcept - { - return m_map.begin(); - } + const_iterator begin() const noexcept { return m_map.begin(); } - const_iterator - end() const noexcept - { - return m_map.end(); - } + const_iterator end() const noexcept { return m_map.end(); } - bool - has(const NodeID& nid) const - { - return m_map.find(nid) != end(); - } + bool has(const NodeID& nid) const { return m_map.find(nid) != end(); } private: std::map m_map; diff --git a/tests/boost-test.hpp b/tests/boost-test.hpp index 7449f38..bcf5262 100644 --- a/tests/boost-test.hpp +++ b/tests/boost-test.hpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2021 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #ifndef NDN_SVS_TESTS_BOOST_TEST_HPP diff --git a/tests/main.cpp b/tests/main.cpp index ead1da6..cc6b127 100644 --- a/tests/main.cpp +++ b/tests/main.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2021 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #define BOOST_TEST_MODULE ndnsvs diff --git a/tests/unit-tests/core.t.cpp b/tests/unit-tests/core.t.cpp index ec576db..7f9e49e 100644 --- a/tests/unit-tests/core.t.cpp +++ b/tests/unit-tests/core.t.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "core.hpp" @@ -27,7 +28,7 @@ class CoreFixture protected: CoreFixture() : m_syncPrefix("/ndn/test") - , m_core(m_face, m_syncPrefix, [] (auto&&...) {}) + , m_core(m_face, m_syncPrefix, [](auto&&...) {}) { } diff --git a/tests/unit-tests/version-vector.t.cpp b/tests/unit-tests/version-vector.t.cpp index f4fd111..3a99996 100644 --- a/tests/unit-tests/version-vector.t.cpp +++ b/tests/unit-tests/version-vector.t.cpp @@ -2,16 +2,17 @@ /* * Copyright (c) 2012-2023 University of California, Los Angeles * - * This file is part of ndn-svs, synchronization library for distributed realtime - * applications for NDN. + * This file is part of ndn-svs, synchronization library for distributed + * realtime applications for NDN. * - * ndn-svs library is free software: you can redistribute it and/or modify it under the - * terms of the GNU Lesser General Public License as published by the Free Software - * Foundation, in version 2.1 of the License. + * ndn-svs library is free software: you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published by the + * Free Software Foundation, in version 2.1 of the License. * - * ndn-svs library 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 Lesser General Public License for more details. + * ndn-svs library 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 Lesser General Public License + * for more details. */ #include "version-vector.hpp" @@ -54,8 +55,7 @@ BOOST_AUTO_TEST_CASE(Set) BOOST_AUTO_TEST_CASE(Iterate) { std::unordered_map umap; - for (auto elem : v) - { + for (auto elem : v) { umap[elem.first] = elem.second; } @@ -77,9 +77,12 @@ BOOST_AUTO_TEST_CASE(EncodeDecode) BOOST_AUTO_TEST_CASE(DecodeStatic) { // Hex: CA0A070508036F6E65CC0101CA0A0705080374776FCC0102 - constexpr std::string_view encoded{"\xCA\x0A\x07\x05\x08\x03\x6F\x6E\x65\xCC\x01\x01" - "\xCA\x0A\x07\x05\x08\x03\x74\x77\x6F\xCC\x01\x02"}; - VersionVector dv(ndn::encoding::makeStringBlock(svs::tlv::StateVector, encoded)); + constexpr std::string_view encoded{ + "\xCA\x0A\x07\x05\x08\x03\x6F\x6E\x65\xCC\x01\x01" + "\xCA\x0A\x07\x05\x08\x03\x74\x77\x6F\xCC\x01\x02" + }; + VersionVector dv( + ndn::encoding::makeStringBlock(svs::tlv::StateVector, encoded)); BOOST_CHECK_EQUAL(dv.get("one"), 1); BOOST_CHECK_EQUAL(dv.get("two"), 2); } @@ -96,8 +99,10 @@ BOOST_AUTO_TEST_CASE(Ordering) Block v1e = v1.encode(); Block v2e = v2.encode(); - std::string v1str(reinterpret_cast(v1e.value()), v1e.value_size()); - std::string v2str(reinterpret_cast(v2e.value()), v2e.value_size()); + std::string v1str(reinterpret_cast(v1e.value()), + v1e.value_size()); + std::string v2str(reinterpret_cast(v2e.value()), + v2e.value_size()); BOOST_CHECK_EQUAL(v1str, v2str); }