blob: e401603d2307dba254475ff8af0cdc484f8d28cf [file] [log] [blame]
Weiqi Shi68f2cf62014-08-26 12:35:45 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Davide Pesavento49f3a5f2017-09-23 01:36:33 -04002/*
Davide Pesavento8891c832019-03-20 23:20:35 -04003 * Copyright (c) 2014-2019, Regents of the University of California.
Weiqi Shi68f2cf62014-08-26 12:35:45 -07004 *
5 * This file is part of NDN repo-ng (Next generation of NDN repository).
6 * See AUTHORS.md for complete list of repo-ng authors and contributors.
7 *
8 * repo-ng 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,
10 * either version 3 of the License, or (at your option) any later version.
11 *
12 * repo-ng 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
Alexander Afanasyev42290b22017-03-09 12:58:29 -080017 * repo-ng, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
Weiqi Shi68f2cf62014-08-26 12:35:45 -070018 */
19
20/**
Davide Pesavento8891c832019-03-20 23:20:35 -040021 * @file This file demonstrates how to generate data to be stored in a repo using
22 * the repo watch protocol and repo insertion protocol.
23 *
24 * The details of the protocols can be found here:
Alexander Afanasyev42290b22017-03-09 12:58:29 -080025 * <https://redmine.named-data.net/projects/repo-ng/wiki/Watched_Prefix_Insertion_Protocol>
26 * <https://redmine.named-data.net/projects/repo-ng/wiki/Basic_Repo_Insertion_Protocol>
Weiqi Shi68f2cf62014-08-26 12:35:45 -070027 *
28 * This file is used for debugging purpose. There are two modes for users to assign
29 * names for the data.
30 * 1)read the data name from a specific file
31 * 2)input a prefix and a random version number will be added automatically
32 * Users need to run nfd and repo-ng and set up specific repo protocols mentioned
33 * above before running this program.
34 * The description of command parameter can be found in the function usage().
35 */
36
Davide Pesaventod45e62b2018-01-01 23:33:18 -050037#include <boost/asio/io_service.hpp>
Weiqi Shi68f2cf62014-08-26 12:35:45 -070038#include <boost/lexical_cast.hpp>
Davide Pesavento8891c832019-03-20 23:20:35 -040039
40#include <ndn-cxx/data.hpp>
41#include <ndn-cxx/face.hpp>
42#include <ndn-cxx/name.hpp>
43#include <ndn-cxx/util/random.hpp>
44#include <ndn-cxx/util/scheduler.hpp>
45#include <ndn-cxx/util/time.hpp>
46
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040047#include <fstream>
Weiqi Shi68f2cf62014-08-26 12:35:45 -070048#include <iostream>
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040049#include <random>
50#include <string>
Weiqi Shi68f2cf62014-08-26 12:35:45 -070051
52namespace repo {
53
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040054using ndn::time::milliseconds;
Weiqi Shi68f2cf62014-08-26 12:35:45 -070055
Davide Pesavento8891c832019-03-20 23:20:35 -040056const milliseconds DEFAULT_TIME_INTERVAL(2000);
Weiqi Shi68f2cf62014-08-26 12:35:45 -070057
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040058enum Mode {
Weiqi Shi68f2cf62014-08-26 12:35:45 -070059 AUTO,
60 READFILE
61};
62
63class Publisher
64{
65public:
66 class Error : public std::runtime_error
67 {
68 public:
Davide Pesavento8891c832019-03-20 23:20:35 -040069 using std::runtime_error::runtime_error;
Weiqi Shi68f2cf62014-08-26 12:35:45 -070070 };
71
72public:
73 Publisher()
74 : mode(AUTO)
Davide Pesavento8891c832019-03-20 23:20:35 -040075 , dataPrefix("/example/data")
Weiqi Shi68f2cf62014-08-26 12:35:45 -070076 , timeInterval(DEFAULT_TIME_INTERVAL)
77 , duration(0)
78 , m_scheduler(m_face.getIoService())
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040079 , m_randomDist(200, 1000)
Weiqi Shi68f2cf62014-08-26 12:35:45 -070080 {
81 }
82
83 void
84 run();
85
86 void
87 autoGenerate();
88
89 void
90 generateFromFile();
91
Davide Pesavento8891c832019-03-20 23:20:35 -040092 static std::shared_ptr<ndn::Data>
Weiqi Shi68f2cf62014-08-26 12:35:45 -070093 createData(const ndn::Name& name);
Davide Pesavento49f3a5f2017-09-23 01:36:33 -040094
Weiqi Shi68f2cf62014-08-26 12:35:45 -070095public:
96 std::ifstream insertStream;
97 Mode mode;
Davide Pesavento8891c832019-03-20 23:20:35 -040098 ndn::Name dataPrefix;
Weiqi Shi68f2cf62014-08-26 12:35:45 -070099 milliseconds timeInterval;
100 milliseconds duration;
101
102private:
103 ndn::Face m_face;
104 ndn::Scheduler m_scheduler;
Davide Pesavento8891c832019-03-20 23:20:35 -0400105 std::uniform_int_distribution<> m_randomDist;
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700106};
107
108void
109Publisher::run()
110{
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700111 if (mode == AUTO) {
Davide Pesavento8891c832019-03-20 23:20:35 -0400112 m_scheduler.schedule(timeInterval, [this] { autoGenerate(); });
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700113 }
114 else {
Davide Pesavento8891c832019-03-20 23:20:35 -0400115 m_scheduler.schedule(timeInterval, [this] { generateFromFile(); });
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700116 }
117 m_face.processEvents(duration);
118}
119
120void
121Publisher::autoGenerate()
122{
Davide Pesavento8891c832019-03-20 23:20:35 -0400123 ndn::Name name = dataPrefix;
124 name.appendNumber(m_randomDist(ndn::random::getRandomNumberEngine()));
125 auto data = createData(name);
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700126 m_face.put(*data);
Davide Pesavento8891c832019-03-20 23:20:35 -0400127
128 m_scheduler.schedule(timeInterval, [this] { autoGenerate(); });
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700129}
130
131void
132Publisher::generateFromFile()
133{
134 if (insertStream.eof()) {
135 m_face.getIoService().stop();
136 return;
Davide Pesavento8891c832019-03-20 23:20:35 -0400137 }
138
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700139 std::string name;
140 getline(insertStream, name);
Davide Pesavento8891c832019-03-20 23:20:35 -0400141 auto data = createData(ndn::Name(name));
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700142 m_face.put(*data);
Davide Pesavento8891c832019-03-20 23:20:35 -0400143
144 m_scheduler.schedule(timeInterval, [this] { generateFromFile(); });
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700145}
146
Davide Pesavento8891c832019-03-20 23:20:35 -0400147std::shared_ptr<ndn::Data>
148Publisher::createData(const ndn::Name& name)
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700149{
150 static ndn::KeyChain keyChain;
151 static std::vector<uint8_t> content(1500, '-');
152
Davide Pesavento8891c832019-03-20 23:20:35 -0400153 auto data = std::make_shared<ndn::Data>();
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700154 data->setName(name);
Davide Pesavento8891c832019-03-20 23:20:35 -0400155 data->setContent(content.data(), content.size());
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700156 keyChain.sign(*data);
157 return data;
158}
159
160static void
161usage()
162{
163 std::cerr
164 << " Publisher [-d dataPrefix] [-f filename] [-s duration time] [-t generate time interval] \n"
165 << " -d: specify the data prefix publisher generate\n"
166 << " -f: specify filename that publish would read from\n"
167 << " -s: specify the time duration of generate data\n"
168 << " -t: specify the time interval between two data generated\n"
169 << std::endl;
170 exit(1);
171}
172
Davide Pesavento49f3a5f2017-09-23 01:36:33 -0400173static int
Davide Pesavento8891c832019-03-20 23:20:35 -0400174main(int argc, char* argv[])
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700175{
176 Publisher generator;
177 bool isAuto = false;
178 bool isRead = false;
179 int opt;
180 while ((opt = getopt(argc, argv, "d:f:s:t:")) != -1) {
181 switch (opt) {
182 case 'd':
183 {
Davide Pesavento8891c832019-03-20 23:20:35 -0400184 generator.dataPrefix = ndn::Name(std::string(optarg));
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700185 generator.mode = AUTO;
186 isAuto = true;
187 }
188 break;
189 case 'f':
190 {
191 isRead = true;
192 generator.mode = READFILE;
193 std::string str = std::string(optarg);
194 generator.insertStream.open(str.c_str());
195 if (!generator.insertStream.is_open()) {
196 std::cerr << "ERROR: cannot open " << std::string(optarg) << std::endl;
197 return 1;
198 }
199 }
200 break;
201 case 's':
202 try {
203 generator.duration = milliseconds(boost::lexical_cast<uint64_t>(optarg));
204 }
Alexander Afanasyev42290b22017-03-09 12:58:29 -0800205 catch (const boost::bad_lexical_cast&) {
weijia yuan3aa8d2b2018-03-06 15:35:57 -0800206 std::cerr << "-s option should be an integer" << std::endl;;
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700207 return 1;
208 }
209 break;
210 case 't':
211 try {
212 generator.timeInterval = milliseconds(boost::lexical_cast<uint64_t>(optarg));
213 }
Alexander Afanasyev42290b22017-03-09 12:58:29 -0800214 catch (const boost::bad_lexical_cast&) {
weijia yuan3aa8d2b2018-03-06 15:35:57 -0800215 std::cerr << "-t option should be an integer" << std::endl;;
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700216 return 1;
217 }
218 break;
219 default:
220 usage();
221 break;
222 }
223 }
224
225 argc -= optind;
226 argv += optind;
227
228 if (argc != 0)
229 usage();
230
231 if (isAuto && isRead)
232 usage();
233
234 generator.run();
235 return 0;
236}
237
238} // namespace repo
239
240int
Davide Pesavento8891c832019-03-20 23:20:35 -0400241main(int argc, char* argv[])
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700242{
243 try {
244 return repo::main(argc, argv);
245 }
Alexander Afanasyev42290b22017-03-09 12:58:29 -0800246 catch (const std::exception& e) {
Weiqi Shi68f2cf62014-08-26 12:35:45 -0700247 std::cerr << "ERROR: " << e.what() << std::endl;
248 return 2;
249 }
250}