blob: c1b7a3fd0eaf2cb65d8b49e2fb069599fcd4727e [file] [log] [blame]
awlane21acd052024-06-13 21:12:51 -05001# -*- Mode:python; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
2#
awlane26724712024-07-05 16:18:27 -05003# Copyright (C) 2015-2024, The University of Memphis,
awlane21acd052024-06-13 21:12:51 -05004# Arizona Board of Regents,
5# Regents of the University of California.
6#
7# This file is part of Mini-NDN.
8# See AUTHORS.md for a complete list of Mini-NDN authors and contributors.
9#
10# Mini-NDN is free software: you can redistribute it and/or modify
11# it under the terms of the GNU General Public License as published by
12# the Free Software Foundation, either version 3 of the License, or
13# (at your option) any later version.
14#
15# Mini-NDN is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
19#
20# You should have received a copy of the GNU General Public License
21# along with Mini-NDN, e.g., in COPYING.md file.
22# If not, see <http://www.gnu.org/licenses/>.
23
24from subprocess import PIPE
25from time import sleep
26
27from mininet.log import setLogLevel, info, debug
28from mininet.topo import Topo
29
30from minindn.minindn import Minindn
31from minindn.apps.app_manager import AppManager
32from minindn.util import MiniNDNCLI, getPopen
33from minindn.apps.nfd import Nfd
34from minindn.apps.nlsr import Nlsr
35from minindn.helpers.experiment import Experiment
36from minindn.helpers.nfdc import Nfdc
37from minindn.helpers.ndn_routing_helper import NdnRoutingHelper
38
39PREFIX = "/example"
40
41def printOutput(output):
42 _out = output.decode("utf-8").split("\n")
43 for _line in _out:
44 info(_line + "\n")
45
46def udp_run():
47 Minindn.cleanUp()
48 Minindn.verifyDependencies()
49
50 # Topology can be created/modified using Mininet topo object
51 topo = Topo()
52 info("Setup\n")
53 # add hosts
54 a = topo.addHost('a')
55 b = topo.addHost('b')
56 c = topo.addHost('c')
57
58 # add links
59 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
60 topo.addLink(b, c, delay='10ms', bw=10)
61 topo.addLink(a, c, delay='10ms', bw=10)
62
63 ndn = Minindn(topo=topo)
64 ndn.start()
65
66 # configure and start nfd on each node
67 info("Configuring NFD\n")
68 AppManager(ndn, ndn.net.hosts, Nfd, logLevel="DEBUG")
69
70 """
71 There are multiple ways of setting up routes in Mini-NDN
72 refer: https://minindn.memphis.edu/experiment.html#routing-options
73 It can also be set manually as follows. The important bit to note here
74 is the use of the Nfdc command
75 """
awlane26724712024-07-05 16:18:27 -050076 links = {"a": ["b", "c"], "b": ["c"]}
awlane21acd052024-06-13 21:12:51 -050077 nfdc_batches = dict()
awlane26724712024-07-05 16:18:27 -050078 info("Setting up routes and strategies...\n")
awlane21acd052024-06-13 21:12:51 -050079 for first in links:
80 for second in links[first]:
81 host1 = ndn.net[first]
82 host2 = ndn.net[second]
83 interface = host2.connectionsTo(host1)[0][0]
84 interface_ip = interface.IP()
85 Nfdc.createFace(host1, interface_ip)
86 Nfdc.registerRoute(host1, PREFIX, interface_ip, cost=0)
87 Nfdc.setStrategy(host1, PREFIX, Nfdc.STRATEGY_ASF)
awlane26724712024-07-05 16:18:27 -050088 sleep(1)
89 info(ndn.net["a"].cmd("nfdc face list"))
90 info(ndn.net["a"].cmd("nfdc fib list"))
91 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -050092
93 # Start ping server
94 info("Starting pings...\n")
awlane26724712024-07-05 16:18:27 -050095 pingserver_log = open(f"{ndn.workDir}/c/ndnpingserver.log", "w")
96 getPopen(ndn.net["c"], f"ndnpingserver {PREFIX}", stdout=pingserver_log,\
awlane21acd052024-06-13 21:12:51 -050097 stderr=pingserver_log)
98
99 # start ping client
awlane26724712024-07-05 16:18:27 -0500100 ping1 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500101 ping1.wait()
102 printOutput(ping1.stdout.read())
103
awlane26724712024-07-05 16:18:27 -0500104 info("Bringing down routes and strategies...\n")
105 links = {"a": ["b", "c"], "b": ["c"]}
awlane21acd052024-06-13 21:12:51 -0500106 for first in links:
107 for second in links[first]:
108 host1 = ndn.net[first]
109 host2 = ndn.net[second]
110 interface = host2.connectionsTo(host1)[0][0]
111 interface_ip = interface.IP()
112 Nfdc.unregisterRoute(host1, PREFIX, interface_ip)
113 Nfdc.destroyFace(host1, interface_ip)
114 Nfdc.unsetStrategy(host1, PREFIX)
115 sleep(1)
awlane26724712024-07-05 16:18:27 -0500116 info(ndn.net["a"].cmd("nfdc face list"))
117 info(ndn.net["a"].cmd("nfdc fib list"))
118 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500119
awlane26724712024-07-05 16:18:27 -0500120 ping2 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500121 ping2.wait()
122 printOutput(ping2.stdout.read())
123
124 info("\nExperiment Completed!\n")
125 # MiniNDNCLI(ndn.net)
126 ndn.stop()
127
128def eth_run():
129 Minindn.cleanUp()
130 Minindn.verifyDependencies()
131
132 # Topology can be created/modified using Mininet topo object
133 topo = Topo()
134 info("Setup\n")
135 # add hosts
136 a = topo.addHost('a')
137 b = topo.addHost('b')
138 c = topo.addHost('c')
139
140 # add links
141 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
142 topo.addLink(b, c, delay='10ms', bw=10)
143 topo.addLink(a, c, delay='10ms', bw=10)
144
145 ndn = Minindn(topo=topo)
146 ndn.start()
147
148 # configure and start nfd on each node
149 info("Configuring NFD\n")
150 AppManager(ndn, ndn.net.hosts, Nfd, logLevel="DEBUG")
151
152 """
153 There are multiple ways of setting up routes in Mini-NDN
154 refer: https://minindn.memphis.edu/experiment.html#routing-options
155 It can also be set manually as follows. The important bit to note here
156 is the use of the Nfdc command
157 """
awlane26724712024-07-05 16:18:27 -0500158 info("Setting up routes and strategies...\n")
159 links = {"a": ["b", "c"], "b": ["c"]}
awlane21acd052024-06-13 21:12:51 -0500160 nfdc_batches = dict()
161 for first in links:
162 for second in links[first]:
163 host1 = ndn.net[first]
164 host2 = ndn.net[second]
165 sender_interface = host1.connectionsTo(host2)[0][0]
166 interface = host2.connectionsTo(host1)[0][0]
167 interface_addr = interface.MAC()
168 Nfdc.createFace(host1, interface_addr, protocol=Nfdc.PROTOCOL_ETHER, localInterface=sender_interface)
169 Nfdc.registerRoute(host1, PREFIX, interface_addr, cost=0, protocol=Nfdc.PROTOCOL_ETHER)
170 Nfdc.setStrategy(host1, PREFIX, Nfdc.STRATEGY_ASF)
awlane26724712024-07-05 16:18:27 -0500171 sleep(1)
172 info(ndn.net["a"].cmd("nfdc face list"))
173 info(ndn.net["a"].cmd("nfdc fib list"))
174 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500175
176 # Start ping server
177 info("Starting pings...\n")
awlane26724712024-07-05 16:18:27 -0500178 pingserver_log = open(f"{ndn.workDir}/c/ndnpingserver.log", "w")
179 getPopen(ndn.net["c"], f"ndnpingserver {PREFIX}", stdout=pingserver_log,\
awlane21acd052024-06-13 21:12:51 -0500180 stderr=pingserver_log)
181
182 # start ping client
awlane26724712024-07-05 16:18:27 -0500183 ping1 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500184 ping1.wait()
185 printOutput(ping1.stdout.read())
186
awlane26724712024-07-05 16:18:27 -0500187 info("Bringing down routes and strategies...\n")
188
awlane21acd052024-06-13 21:12:51 -0500189 links = {"a":["b", "c"], "b":["c"]}
190 for first in links:
191 for second in links[first]:
192 host1 = ndn.net[first]
193 host2 = ndn.net[second]
194 interface = host2.connectionsTo(host1)[0][0]
195 interface_addr = interface.MAC()
196 Nfdc.unregisterRoute(host1, PREFIX, interface_addr, protocol=Nfdc.PROTOCOL_ETHER)
197 Nfdc.destroyFace(host1, interface_addr, protocol=Nfdc.PROTOCOL_ETHER)
198 Nfdc.unsetStrategy(host1, PREFIX)
199 sleep(1)
awlane26724712024-07-05 16:18:27 -0500200 info(ndn.net["a"].cmd("nfdc face list"))
201 info(ndn.net["a"].cmd("nfdc fib list"))
202 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500203
awlane26724712024-07-05 16:18:27 -0500204 ping2 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500205 ping2.wait()
206 printOutput(ping2.stdout.read())
207
208 info("\nExperiment Completed!\n")
209 # MiniNDNCLI(ndn.net)
210 ndn.stop()
211
212def udp_nlsr_run():
213 Minindn.cleanUp()
214 Minindn.verifyDependencies()
215
216 # Topology can be created/modified using Mininet topo object
217 topo = Topo()
218 info("Setup\n")
219 # add hosts
220 a = topo.addHost('a')
221 b = topo.addHost('b')
222 c = topo.addHost('c')
223
224 # add links
225 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
226 topo.addLink(b, c, delay='10ms', bw=10)
227 topo.addLink(a, c, delay='10ms', bw=10)
228
229 ndn = Minindn(topo=topo)
230 ndn.start()
231
232 nfds = AppManager(ndn, ndn.net.hosts, Nfd)
233 nlsrs = AppManager(ndn, ndn.net.hosts, Nlsr, faceType=Nfdc.PROTOCOL_UDP,
234 logLevel='ndn.*=TRACE:nlsr.*=TRACE')
235
236 Experiment.checkConvergence(ndn, ndn.net.hosts, 60, quit=False)
237
238 Experiment.setupPing(ndn.net.hosts, Nfdc.STRATEGY_BEST_ROUTE)
239 Experiment.startPctPings(ndn.net, 60)
240
241 sleep(70)
awlane26724712024-07-05 16:18:27 -0500242
243 info(ndn.net["a"].cmd("nfdc face list"))
244 info(ndn.net["a"].cmd("nfdc fib list"))
245 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500246
247 ndn.stop()
248
249def eth_nlsr_run():
250 Minindn.cleanUp()
251 Minindn.verifyDependencies()
252
253 # Topology can be created/modified using Mininet topo object
254 topo = Topo()
255 info("Setup\n")
256 # add hosts
257 a = topo.addHost('a')
258 b = topo.addHost('b')
259 c = topo.addHost('c')
260
261 # add links
262 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
263 topo.addLink(b, c, delay='10ms', bw=10)
264 topo.addLink(a, c, delay='10ms', bw=10)
265
266 ndn = Minindn(topo=topo)
267 ndn.start()
268
269 nfds = AppManager(ndn, ndn.net.hosts, Nfd)
270 nlsrs = AppManager(ndn, ndn.net.hosts, Nlsr, faceType=Nfdc.PROTOCOL_ETHER,
271 logLevel='ndn.*=TRACE:nlsr.*=TRACE')
272
273 Experiment.checkConvergence(ndn, ndn.net.hosts, 60, quit=False)
274
275 Experiment.setupPing(ndn.net.hosts, Nfdc.STRATEGY_BEST_ROUTE)
276 Experiment.startPctPings(ndn.net, 60)
277
278 sleep(70)
awlane26724712024-07-05 16:18:27 -0500279
280 info(ndn.net["a"].cmd("nfdc face list"))
281 info(ndn.net["a"].cmd("nfdc fib list"))
282 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500283
284 ndn.stop()
285
286def udp_static_run():
287 Minindn.cleanUp()
288 Minindn.verifyDependencies()
289
290 # Topology can be created/modified using Mininet topo object
291 topo = Topo()
292 info("Setup\n")
293 # add hosts
294 a = topo.addHost('a')
295 b = topo.addHost('b')
296 c = topo.addHost('c')
297
298 # add links
299 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
300 topo.addLink(b, c, delay='10ms', bw=10)
301 topo.addLink(a, c, delay='10ms', bw=10)
302
303 ndn = Minindn(topo=topo)
304 ndn.start()
305
306 nfds = AppManager(ndn, ndn.net.hosts, Nfd)
307 info('Adding static routes to NFD\n')
308 grh = NdnRoutingHelper(ndn.net, Nfdc.PROTOCOL_UDP)
309 # For all host, pass ndn.net.hosts or a list, [ndn.net['a'], ..] or [ndn.net.hosts[0],.]
310 grh.addOrigin([ndn.net['c']], ["/example"])
311 grh.calculateNPossibleRoutes()
312
313 '''
314 prefix "/abc" is advertise from node A, it should be reachable from all other nodes.
315 '''
316 routesFromA = ndn.net['a'].cmd("nfdc route | grep -v '/localhost/nfd'")
317 if '/ndn/b-site/b' not in routesFromA or '/ndn/c-site/c' not in routesFromA:
318 info("Route addition failed\n")
319
320 routesToPrefix = ndn.net['b'].cmd("nfdc fib | grep '/example'")
321 if '/example' not in routesToPrefix:
322 info("Missing route to advertised prefix, Route addition failed\n")
323 ndn.net.stop()
324
325 info('Route addition to NFD completed succesfully\n')
326
awlane26724712024-07-05 16:18:27 -0500327 info(ndn.net["a"].cmd("nfdc face list"))
328 info(ndn.net["a"].cmd("nfdc fib list"))
329 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500330
331 # Start ping server
332 info("Starting pings...\n")
awlane26724712024-07-05 16:18:27 -0500333 pingserver_log = open(f"{ndn.workDir}/c/ndnpingserver.log", "w")
334 getPopen(ndn.net["c"], f"ndnpingserver {PREFIX}", stdout=pingserver_log,\
awlane21acd052024-06-13 21:12:51 -0500335 stderr=pingserver_log)
336
337 # start ping client
awlane26724712024-07-05 16:18:27 -0500338 ping1 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500339 ping1.wait()
340 printOutput(ping1.stdout.read())
341
342 ndn.stop()
343
344def eth_static_run():
345 Minindn.cleanUp()
346 Minindn.verifyDependencies()
347
348 # Topology can be created/modified using Mininet topo object
349 topo = Topo()
350 info("Setup\n")
351 # add hosts
352 a = topo.addHost('a')
353 b = topo.addHost('b')
354 c = topo.addHost('c')
355
356 # add links
357 topo.addLink(a, b, delay='10ms', bw=10) # bw = bandwidth
358 topo.addLink(b, c, delay='10ms', bw=10)
359 topo.addLink(a, c, delay='10ms', bw=10)
360
361 ndn = Minindn(topo=topo)
362 ndn.start()
363
364 nfds = AppManager(ndn, ndn.net.hosts, Nfd)
365 info('Adding static routes to NFD\n')
366 grh = NdnRoutingHelper(ndn.net, Nfdc.PROTOCOL_ETHER)
367 # For all host, pass ndn.net.hosts or a list, [ndn.net['a'], ..] or [ndn.net.hosts[0],.]
368 grh.addOrigin([ndn.net['c']], ["/example"])
369 grh.calculateNPossibleRoutes()
370
371 '''
372 prefix "/abc" is advertise from node A, it should be reachable from all other nodes.
373 '''
374 routesFromA = ndn.net['a'].cmd("nfdc route | grep -v '/localhost/nfd'")
375 if '/ndn/b-site/b' not in routesFromA or '/ndn/c-site/c' not in routesFromA:
376 info("Route addition failed\n")
377
378 routesToPrefix = ndn.net['b'].cmd("nfdc fib | grep '/example'")
379 if '/example' not in routesToPrefix:
380 info("Missing route to advertised prefix, Route addition failed\n")
381 ndn.net.stop()
382
383 info('Route addition to NFD completed succesfully\n')
384
awlane26724712024-07-05 16:18:27 -0500385 info(ndn.net["a"].cmd("nfdc face list"))
386 info(ndn.net["a"].cmd("nfdc fib list"))
387 info(ndn.net["a"].cmd("nfdc strategy show /example"))
awlane21acd052024-06-13 21:12:51 -0500388
389 # Start ping server
390 info("Starting pings...\n")
awlane26724712024-07-05 16:18:27 -0500391 pingserver_log = open(f"{ndn.workDir}/c/ndnpingserver.log", "w")
392 getPopen(ndn.net["c"], f"ndnpingserver {PREFIX}", stdout=pingserver_log,\
awlane21acd052024-06-13 21:12:51 -0500393 stderr=pingserver_log)
394
395 # start ping client
awlane26724712024-07-05 16:18:27 -0500396 ping1 = getPopen(ndn.net["a"], f"ndnping {PREFIX} -c 5", stdout=PIPE, stderr=PIPE)
awlane21acd052024-06-13 21:12:51 -0500397 ping1.wait()
398 printOutput(ping1.stdout.read())
399
400 ndn.stop()
401
402if __name__ == '__main__':
awlane26724712024-07-05 16:18:27 -0500403 setLogLevel("info")
404 info("\n\nUDP FACES\n")
awlane21acd052024-06-13 21:12:51 -0500405 udp_run()
awlane26724712024-07-05 16:18:27 -0500406 info("\n\nETHERNET FACES\n")
awlane21acd052024-06-13 21:12:51 -0500407 eth_run()
awlane26724712024-07-05 16:18:27 -0500408 info("\n\nNLSR WITH UDP FACES\n")
awlane21acd052024-06-13 21:12:51 -0500409 udp_nlsr_run()
awlane26724712024-07-05 16:18:27 -0500410 info("\n\nNLSR WITH ETHERNET FACES\n")
awlane21acd052024-06-13 21:12:51 -0500411 eth_nlsr_run()
awlane26724712024-07-05 16:18:27 -0500412 info("\n\nSTATIC ROUTING HELPER WITH UDP FACES\n")
awlane21acd052024-06-13 21:12:51 -0500413 udp_static_run()
awlane26724712024-07-05 16:18:27 -0500414 info("\n\nSTATIC ROUTING HELPER WITH ETHERNET FACES\n")
awlane21acd052024-06-13 21:12:51 -0500415 eth_static_run()