blob: 356b2120f8ffe17f3711cee87b2fc144d8d4d550 [file] [log] [blame]
Yingdi Yu31ad44c2014-08-28 14:55:42 -07001/* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil -*- */
2/*
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -08003 * Copyright (c) 2012-2017 University of California, Los Angeles
Yingdi Yu31ad44c2014-08-28 14:55:42 -07004 *
5 * This file is part of ChronoSync, synchronization library for distributed realtime
6 * applications for NDN.
7 *
8 * ChronoSync is free software: you can redistribute it and/or modify it under the terms
9 * of the GNU General Public License as published by the Free Software Foundation, either
10 * version 3 of the License, or (at your option) any later version.
11 *
12 * ChronoSync is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
13 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
14 * PURPOSE. See the GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along with
17 * ChronoSync, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
18 *
19 * @author Zhenkai Zhu <http://irl.cs.ucla.edu/~zhenkai/>
20 * @author Chaoyi Bian <bcy@pku.edu.cn>
21 * @author Alexander Afanasyev <http://lasr.cs.ucla.edu/afanasyev/index.html>
22 * @author Yingdi Yu <yingdi@cs.ucla.edu>
23 */
24
25#ifndef CHRONOSYNC_SOCKET_HPP
26#define CHRONOSYNC_SOCKET_HPP
27
28#include <ndn-cxx/face.hpp>
Qiuhan Dingfb8c9e02015-01-30 14:04:55 -080029#include <ndn-cxx/util/in-memory-storage-persistent.hpp>
30#include <unordered_map>
Yingdi Yu31ad44c2014-08-28 14:55:42 -070031
32#include "logic.hpp"
33
34namespace chronosync {
35
36/**
37 * @brief A simple interface to interact with client code
38 *
39 * Though it is called Socket, it is not a real socket. It just trying to provide
40 * a simplified interface for data publishing and fetching.
41 *
42 * This interface simplify data publishing. Client can simply dump raw data
43 * into this interface without handling the ChronoSync specific details, such
44 * as sequence number and session id.
45 *
46 * This interface also simplify data fetching. Client only needs to provide a
47 * data fetching strategy (through a updateCallback).
48 */
49class Socket : noncopyable
50{
51public:
52 class Error : public std::runtime_error
53 {
54 public:
55 explicit
56 Error(const std::string& what)
57 : std::runtime_error(what)
58 {
59 }
60 };
61
62 Socket(const Name& syncPrefix,
63 const Name& userPrefix,
64 ndn::Face& face,
Yingdi Yucd339022014-11-05 17:51:19 -080065 const UpdateCallback& updateCallback,
66 const Name& signingId = DEFAULT_NAME,
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -080067 std::shared_ptr<ndn::Validator> validator = DEFAULT_VALIDATOR);
Yingdi Yu31ad44c2014-08-28 14:55:42 -070068
Qiuhan Ding03b9de32015-01-30 14:03:12 -080069 ~Socket();
70
71 /**
72 * @brief Add a sync node under same logic
73 *
74 * This method will add a new sync node in logic and then register this prefix.
75 * If pass an empty prefix, it will return directly without doing anything
76 * If the prefix is already registered, return directly
77 *
78 * @param prefix Prefix of the new node
79 * @param signingId Signing ID for the packet sent out by the new node
80 */
Yingdi Yu31ad44c2014-08-28 14:55:42 -070081 void
Qiuhan Dinge246b622014-12-03 21:57:48 -080082 addSyncNode(const Name& prefix, const Name& signingId = DEFAULT_NAME);
Yingdi Yu31ad44c2014-08-28 14:55:42 -070083
84 /**
Qiuhan Ding03b9de32015-01-30 14:03:12 -080085 * @brief Remove a sync node under same logic
86 *
87 * This method will remove a sync node in logic, unregister this prefix
88 * and then clear the in memory storage about this prefix.
89 * logic will be reset after removal of this node
90 *
91 * @param prefix Prefix of the node to remove
92 */
93 void
94 removeSyncNode(const Name& prefix);
95
96 /**
Yingdi Yu31ad44c2014-08-28 14:55:42 -070097 * @brief Publish a data packet in the session and trigger synchronization updates
98 *
99 * This method will create a data packet with the supplied content.
100 * The packet name is the local session + seqNo.
101 * The seqNo is automatically maintained by internal Logic.
102 *
Qiuhan Dinge246b622014-12-03 21:57:48 -0800103 * @throws It will throw error, if the prefix does not exist in m_logic
104 *
105 * @param buf Pointer to the bytes in content
106 * @param len size of the bytes in content
107 * @param freshness FreshnessPeriod of the data packet.
Ashlesh Gawande097bb442017-05-31 13:38:00 -0500108 * @param prefix The user prefix that will be used to publish the data.
Qiuhan Dinge246b622014-12-03 21:57:48 -0800109 */
110 void
111 publishData(const uint8_t* buf, size_t len, const ndn::time::milliseconds& freshness,
112 const Name& prefix = DEFAULT_PREFIX);
113
114 /**
115 * @brief Publish a data packet in the session and trigger synchronization updates
116 *
117 * This method will create a data packet with the supplied content.
118 * The packet name is the local session + seqNo.
Ashlesh Gawande8d1347a2017-04-03 19:10:28 -0500119 * The seqNo is set by the application.
120 *
121 * @throws It will throw error, if the prefix does not exist in m_logic
122 *
123 * @param buf Pointer to the bytes in content
124 * @param len size of the bytes in content
125 * @param freshness FreshnessPeriod of the data packet.
126 * @param seqNo Sequence number of the data
Ashlesh Gawande097bb442017-05-31 13:38:00 -0500127 * @param prefix The user prefix that will be used to publish the data.
Ashlesh Gawande8d1347a2017-04-03 19:10:28 -0500128 */
129 void
130 publishData(const uint8_t* buf, size_t len, const ndn::time::milliseconds& freshness,
131 const uint64_t& seqNo, const Name& prefix = DEFAULT_PREFIX);
132
133 /**
134 * @brief Publish a data packet in the session and trigger synchronization updates
135 *
136 * This method will create a data packet with the supplied content.
137 * The packet name is the local session + seqNo.
Qiuhan Dinge246b622014-12-03 21:57:48 -0800138 * The seqNo is automatically maintained by internal Logic.
139 *
140 * @throws It will throw error, if the prefix does not exist in m_logic
141 *
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700142 * @param content Block that will be set as the content of the data packet.
143 * @param freshness FreshnessPeriod of the data packet.
Ashlesh Gawande097bb442017-05-31 13:38:00 -0500144 * @param prefix The user prefix that will be used to publish the data.
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700145 */
146 void
Qiuhan Dinge246b622014-12-03 21:57:48 -0800147 publishData(const Block& content, const ndn::time::milliseconds& freshness,
148 const Name& prefix = DEFAULT_PREFIX);
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700149
150 /**
Ashlesh Gawande8d1347a2017-04-03 19:10:28 -0500151 * @brief Publish a data packet in the session and trigger synchronization updates
152 *
153 * This method will create a data packet with the supplied content.
154 * The packet name is the local session + seqNo.
155 * The seqNo is set by the application.
156 *
157 * @throws It will throw error, if the prefix does not exist in m_logic
158 *
159 * @param content Block that will be set as the content of the data packet.
160 * @param freshness FreshnessPeriod of the data packet.
161 * @param seqNo Sequence number of the data
Ashlesh Gawande097bb442017-05-31 13:38:00 -0500162 * @param prefix The user prefix that will be used to publish the data.
Ashlesh Gawande8d1347a2017-04-03 19:10:28 -0500163 */
164 void
165 publishData(const Block& content, const ndn::time::milliseconds& freshness,
166 const uint64_t& seqNo, const Name& prefix = DEFAULT_PREFIX);
167
168 /**
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700169 * @brief Retrive a data packet with a particular seqNo from a session
170 *
171 * @param sessionName The name of the target session.
172 * @param seq The seqNo of the data packet.
173 * @param onValidated The callback when the retrieved packet has been validated.
174 * @param nRetries The number of retries.
175 */
176 void
177 fetchData(const Name& sessionName, const SeqNo& seq,
178 const ndn::OnDataValidated& onValidated,
179 int nRetries = 0);
180
181 /**
182 * @brief Retrive a data packet with a particular seqNo from a session
183 *
184 * @param sessionName The name of the target session.
185 * @param seq The seqNo of the data packet.
186 * @param onValidated The callback when the retrieved packet has been validated.
Ashlesh Gawande097bb442017-05-31 13:38:00 -0500187 * @param onValidationFailed The callback when the retrieved packet failed validation.
188 * @param onTimeout The callback when data is not retrieved.
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700189 * @param nRetries The number of retries.
190 */
191 void
192 fetchData(const Name& sessionName, const SeqNo& seq,
193 const ndn::OnDataValidated& onValidated,
194 const ndn::OnDataValidationFailed& onValidationFailed,
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -0800195 const ndn::TimeoutCallback& onTimeout,
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700196 int nRetries = 0);
197
198 /// @brief Get the root digest of current sync tree
199 ndn::ConstBufferPtr
200 getRootDigest() const;
201
202 Logic&
203 getLogic()
204 {
205 return m_logic;
206 }
207
208private:
209 void
Qiuhan Dingfb8c9e02015-01-30 14:04:55 -0800210 onInterest(const Name& prefix, const Interest& interest);
211
212 void
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -0800213 onData(const Interest& interest, const Data& data,
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700214 const ndn::OnDataValidated& dataCallback,
215 const ndn::OnDataValidationFailed& failCallback);
216
217 void
218 onDataTimeout(const Interest& interest, int nRetries,
219 const ndn::OnDataValidated& dataCallback,
220 const ndn::OnDataValidationFailed& failCallback);
221
222 void
223 onDataValidationFailed(const shared_ptr<const Data>& data,
224 const std::string& failureInfo);
225
Yingdi Yucd339022014-11-05 17:51:19 -0800226public:
227 static const ndn::Name DEFAULT_NAME;
Qiuhan Dinge246b622014-12-03 21:57:48 -0800228 static const ndn::Name DEFAULT_PREFIX;
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -0800229 static const std::shared_ptr<ndn::Validator> DEFAULT_VALIDATOR;
Yingdi Yucd339022014-11-05 17:51:19 -0800230
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700231private:
Qiuhan Dingfb8c9e02015-01-30 14:04:55 -0800232 typedef std::unordered_map<ndn::Name, const ndn::RegisteredPrefixId*> RegisteredPrefixList;
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700233
234 Name m_userPrefix;
235 ndn::Face& m_face;
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700236 Logic m_logic;
237
Yingdi Yucd339022014-11-05 17:51:19 -0800238 ndn::Name m_signingId;
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700239 ndn::KeyChain m_keyChain;
Alexander Afanasyeve9eda8a2017-03-09 14:40:03 -0800240 std::shared_ptr<ndn::Validator> m_validator;
Qiuhan Dingfb8c9e02015-01-30 14:04:55 -0800241
242 RegisteredPrefixList m_registeredPrefixList;
243 ndn::util::InMemoryStoragePersistent m_ims;
Yingdi Yu31ad44c2014-08-28 14:55:42 -0700244};
245
246} // namespace chronosync
247
248#endif // CHRONOSYNC_SOCKET_HPP