ndn: Add GUI
Change-Id: I7336bbb04121166d064b76cbad9f760c3dd16c2d
diff --git a/bin/minindnedit b/bin/minindnedit
index c0ff708..70e0dce 100755
--- a/bin/minindnedit
+++ b/bin/minindnedit
@@ -1,7 +1,7 @@
#!/usr/bin/python
"""
-MiniCCNxEdit: a simple network editor for MiniCCNx
+MiniNDNEdit: a simple network editor for MiniNDN
Based on miniedit by:
Bob Lantz, April 2010
@@ -18,7 +18,7 @@
from Tkinter import *
from ttk import Notebook
from tkMessageBox import showinfo, showerror, showwarning
-from subprocess import call
+from subprocess import call, Popen
import tkFont
import csv
import tkFileDialog
@@ -28,6 +28,7 @@
from distutils.version import StrictVersion
import os
import sys
+import threading
from functools import partial
import pdb
@@ -52,11 +53,14 @@
from mininet.topo import SingleSwitchTopo, LinearTopo, SingleSwitchReversedTopo
from mininet.topolib import TreeTopo
-print 'MiniCCNxEdit running...' #+VERSION
+print 'MiniNDNEdit running...' #+VERSION
MININET_VERSION = re.sub(r'[^\d\.]', '', VERSION)
if StrictVersion(MININET_VERSION) > StrictVersion('2.0'):
from mininet.node import IVSSwitch
+
+from ndn.gui import NfdFrame, NlsrFrame
+
TOPODEF = 'none'
TOPOS = { 'minimal': lambda: SingleSwitchTopo( k=2 ),
'linear': LinearTopo,
@@ -72,6 +76,16 @@
'rt': custom( CPULimitedHost, sched='rt' ),
'cfs': custom( CPULimitedHost, sched='cfs' ) }
+def runMiniNdn(window, template):
+ # Hide window
+ window.withdraw()
+
+ proc = Popen("sudo minindn %s" % template, shell=True)
+ proc.wait()
+
+ # Restore window
+ window.deiconify()
+
class LegacyRouter( Node ):
def __init__( self, name, inNamespace=True, **params ):
@@ -135,8 +149,17 @@
n = Notebook(self.rootFrame)
self.propFrame = Frame(n)
self.fibFrame = Frame(n)
+
+ # NDN
+ self.nfdFrame = NfdFrame(n)
+ self.nlsrFrame = NlsrFrame(n)
+
n.add(self.propFrame, text='Properties')
n.add(self.fibFrame, text='FIB Entries')
+
+ n.add(self.nfdFrame, text=self.nfdFrame.frameLabel)
+ n.add(self.nlsrFrame, text=self.nlsrFrame.frameLabel)
+
n.pack()
### TAB 1
@@ -221,7 +244,10 @@
'mem': self.memEntry.get(),
'hostname':self.hostnameEntry.get(),
'startCommand':self.startEntry.get(),
- 'fibEntries':fibEntries}
+ 'fibEntries':fibEntries,
+ 'nlsr': self.nlsrFrame.getValues()
+ }
+
self.result = results
class VerticalScrolledTable(LabelFrame):
@@ -391,15 +417,15 @@
class MiniEdit( Frame ):
- "A simple network editor for MiniCCNx."
+ "A simple network editor for MiniNDN."
- def __init__( self, parent=None, cheight=600, cwidth=1000, template_file='miniccnx.conf' ):
+ def __init__( self, parent=None, cheight=600, cwidth=1000, template_file='minindn.conf' ):
self.template_file = template_file
Frame.__init__( self, parent )
self.action = None
- self.appName = 'MiniccnxEdit'
+ self.appName = 'MiniNDNEdit'
self.fixedFont = tkFont.Font ( family="DejaVu Sans Mono", size="14" )
# Style
@@ -425,8 +451,8 @@
self.images = miniEditImages()
self.buttons = {}
self.active = None
- self.tools = ( 'Select', 'Host', 'LegacyRouter', 'NetLink' )
- self.customColors = { 'LegacyRouter': 'darkGreen', 'Host': 'blue' }
+ self.tools = ( 'Select', 'Host', 'NetLink' )
+ #self.customColors = { 'LegacyRouter': 'darkGreen', 'Host': 'blue' }
self.toolbar = self.createToolbar()
# Layout
@@ -508,6 +534,7 @@
fileMenu.add_command( label="Open", font=font, command=self.loadTopology )
fileMenu.add_command( label="Save", font=font, command=self.saveTopology )
fileMenu.add_command( label="Generate", font=font, command=self.doGenerate )
+ fileMenu.add_command( label="Run", font=font, command=self.doRun )
fileMenu.add_separator()
fileMenu.add_command( label='Quit', command=self.quit, font=font )
@@ -519,7 +546,7 @@
# Application menu
appMenu = Menu( mbar, tearoff=False )
mbar.add_cascade( label=self.appName, font=font, menu=appMenu )
- appMenu.add_command( label='About Mini-CCNx', command=self.about,
+ appMenu.add_command( label='About Mini-NDN', command=self.about,
font=font)
#appMenu.add_separator()
#appMenu.add_command( label='Quit', command=self.quit, font=font )
@@ -618,7 +645,7 @@
# Spacer
Label( toolbar, text='' ).pack()
- # abaixo copiado Mini-CCNx para criar botao Generate
+ # abaixo copiado Mini-NDN para criar botao Generate
for cmd, color in [ ( 'Generate', 'darkGreen' ) ]:
doCmd = getattr( self, 'do' + cmd )
@@ -628,7 +655,7 @@
return toolbar
- def doGenerate( self ): #COPIA Mini-CCNx - GERA TEMPLATE
+ def doGenerate( self ): #COPIA Mini-NDN - GERA TEMPLATE
"Generate template."
self.activate( 'Select' )
for tool in self.tools:
@@ -645,6 +672,19 @@
b=Button(toplevel, text="Ok", width=5, command=toplevel.destroy)
b.pack(side='bottom', padx=0,pady=0)
+ def doRun( self ):
+ "Use current configuration to generate a template and run the topology"
+
+ # Generate temporary template file
+ old_template_file = self.template_file
+ self.template_file = "/tmp/tmp.conf"
+ self.doGenerate()
+
+ thread = threading.Thread(target=runMiniNdn, args=(self.master, self.template_file))
+ thread.start()
+
+ self.template_file = old_template_file
+
def parseFibEntries ( self, fibEntries ):
"Parse FIB Entries for write"
result=''
@@ -655,17 +695,17 @@
return result
- def buildTemplate( self ): #COPIA Mini-CCNx para criar Template
+ def buildTemplate( self ): #COPIA Mini-NDN para criar Template
"Generate template"
template = open(self.template_file, 'w')
# hosts
- template.write('[hosts]\n')
+ template.write('[nodes]\n')
for widget in self.widgetToItem:
name = widget[ 'text' ]
tags = self.canvas.gettags( self.widgetToItem[ widget ] )
- #print self.hostOpts[name]
+ print self.hostOpts[name]
if 'Host' in tags:
hOpts=self.hostOpts[name]
template.write(name + ': ')
@@ -684,10 +724,22 @@
if 'fibEntries' in hOpts:
customFib = self.parseFibEntries(hOpts['fibEntries'])
template.write(customFib)
+ if 'nlsr' in hOpts:
+ values = hOpts['nlsr']
+
+ template.write('hyperbolic-state=' + values['hyperbolic-state'] + ' ')
+ template.write('radius=' + values['radius'] + ' ')
+ template.write('angle=' + values['angle'] + ' ')
+ template.write('network=' + values['network'] + ' ')
+ template.write('router=' + values['router'] + ' ')
+ template.write('site=' + values['site'] + ' ')
+ template.write('log-level=' + values['log-level'] + ' ')
+ template.write('max-faces-per-prefix=' + values['max-faces-per-prefix'] + ' ')
+
template.write('\n')
# switches/routers
- template.write('[routers]\n')
+ #template.write('[routers]\n')
for router in self.routerOpts.values():
@@ -735,7 +787,7 @@
if 'loss' in linkopts:
template.write('loss=' + repr(linkopts['loss']) + ' ' )
if 'delay' in linkopts:
- template.write('delay=' + str(linkopts['delay']))
+ template.write('delay=' + str(linkopts['delay'])+ 'ms' )
template.write('\n')
@@ -778,7 +830,7 @@
c = self.canvas
myFormats = [
- ('Miniccnx Topology','*.mnccnx'),
+ ('MiniNDN Topology','*.mnndn'),
('All Files','*'),
]
f = tkFileDialog.askopenfile(filetypes=myFormats, mode='rb')
@@ -788,7 +840,7 @@
loadedTopology = self.convertJsonUnicode(json.load(f))
# Load hosts
- hosts = loadedTopology['hosts']
+ hosts = loadedTopology['nodes']
for host in hosts:
nodeNum = host['number']
hostname = 'h'+nodeNum
@@ -800,6 +852,7 @@
host['opts']['nodeNum'] = int(nodeNum)
x = host['x']
y = host['y']
+
self.addNode('Host', nodeNum, float(x), float(y), name=hostname)
# Fix JSON converting tuple to list when saving
@@ -875,7 +928,7 @@
def saveTopology( self ):
"Save command."
myFormats = [
- ('Miniccnx Topology','*.mnccnx'),
+ ('MiniNDN Topology','*.mnndn'),
('All Files','*'),
]
@@ -1004,11 +1057,10 @@
def clearPopups(self):
print 'Entrou funcao clear_popups'
-
- if isHostPopup == True:
- print 'Hostpopup = true'
- self.hostPopup.unpost
- isHostPopup = False
+ if isHostPopup == True:
+ print 'Hostpopup = true'
+ self.hostPopup.unpost
+ isHostPopup = False
#if isRouterPopup == True
#if isLinkPopup == True
@@ -1211,9 +1263,9 @@
# For now, don't allow hosts to be directly linked
stags = self.canvas.gettags( self.widgetToItem[ source ] )
dtags = self.canvas.gettags( target )
- if (('Host' in stags and 'Host' in dtags)):
- self.releaseNetLink( event )
- return
+ #if (('Host' in stags and 'Host' in dtags)):
+ #self.releaseNetLink( event )
+ #return
# Set link type
linkType='data'
@@ -1237,12 +1289,14 @@
bg = 'white'
about = Toplevel( bg='white' )
about.title( 'About' )
- info = self.appName + ': a simple network editor for MiniCCNx - based on Miniedit'
+ info = self.appName + ': a simple network editor for MiniNDN - based on Miniedit'
warning = 'Development version - not entirely functional!'
#version = 'MiniEdit '+MINIEDIT_VERSION
- author = 'Carlos Cabral, Jan 2013'
- author2 = 'Caio Elias, Nov 2014'
- author3 = 'Originally by: Bob Lantz <rlantz@cs>, April 2010'
+ author = 'Vince Lehman, Jan 2015'
+ author2 = 'Ashlesh Gawande, Jan 2015'
+ author3 = 'Carlos Cabral, Jan 2013'
+ author4 = 'Caio Elias, Nov 2014'
+ author5 = 'Originally by: Bob Lantz <rlantz@cs>, April 2010'
enhancements = 'Enhancements by: Gregory Gee, Since July 2013'
www = 'http://gregorygee.wordpress.com/category/miniedit/'
line1 = Label( about, text=info, font='Helvetica 10 bold', bg=bg )
@@ -1250,11 +1304,14 @@
line3 = Label( about, text=author, font='Helvetica 9', bg=bg )
line4 = Label( about, text=author2, font='Helvetica 9', bg=bg )
line5 = Label( about, text=author3, font='Helvetica 9', bg=bg )
- line6 = Label( about, text=enhancements, font='Helvetica 9', bg=bg )
- line7 = Entry( about, font='Helvetica 9', bg=bg, width=len(www), justify=CENTER )
+ line6 = Label( about, text=author4, font='Helvetica 9', bg=bg )
+ line7 = Label( about, text=author5, font='Helvetica 9', bg=bg )
+ line8 = Label( about, text=enhancements, font='Helvetica 9', bg=bg )
+ line9 = Entry( about, font='Helvetica 9', bg=bg, width=len(www), justify=CENTER )
- line7.insert(0, www)
- line7.configure(state='readonly')
+
+ line9.insert(0, www)
+ line9.configure(state='readonly')
line1.pack( padx=20, pady=10 )
line2.pack(pady=10 )
line3.pack(pady=10 )
@@ -1262,6 +1319,8 @@
line5.pack(pady=10 )
line6.pack(pady=10 )
line7.pack(pady=10 )
+ line8.pack(pady=10 )
+ line9.pack(pady=10 )
hide = ( lambda about=about: about.withdraw() )
self.aboutBox = about
# Hide on close rather than destroying window
@@ -1304,6 +1363,9 @@
newHostOpts['cache'] = hostBox.result['cache']
if len(hostBox.result['fibEntries']) > 0:
newHostOpts['fibEntries'] = hostBox.result['fibEntries']
+
+ newHostOpts['nlsr'] = hostBox.nlsrFrame.getValues()
+
self.hostOpts[name] = newHostOpts
print 'New host details for ' + name + ' = ' + str(newHostOpts)
@@ -1456,12 +1518,12 @@
else:
raise Exception( 'Custom file name not found' )
- desc = ( "The %prog utility creates Miniccnx network from the\n"
+ desc = ( "The %prog utility creates Minindn network from the\n"
"command line. It can create parametrized topologies,\n"
- "invoke the Miniccnx CLI, and run tests." )
+ "invoke the Minindn CLI, and run tests." )
usage = ( '%prog [options] [template_file]\n'
- '\nIf no template_file is given, generated template will be written to the file miniccnx.conf in the current directory.\n'
+ '\nIf no template_file is given, generated template will be written to the file minindn.conf in the current directory.\n'
'Type %prog -h for details)' )
opts = OptionParser( description=desc, usage=usage )
diff --git a/ndn/gui.py b/ndn/gui.py
new file mode 100644
index 0000000..c4097e4
--- /dev/null
+++ b/ndn/gui.py
@@ -0,0 +1,109 @@
+#!/usr/bin/env python
+
+from Tkinter import *
+
+LOG_LEVELS = [
+ "NONE",
+ "ERROR",
+ "WARN",
+ "INFO",
+ "DEBUG",
+ "TRACE",
+ "ALL"
+]
+
+class GuiFrame(Frame):
+ def __init__(self, notebook):
+ Frame.__init__(self, notebook)
+
+ self.row = 0
+ self.column = 0
+
+ def addEntryBox(self, label, variable, defaultValue=""):
+ variable.set(defaultValue)
+
+ Label(self, text=label).grid(row=self.row, sticky=E)
+ entry = Entry(self, textvariable=variable)
+ entry.grid(row=self.row, column=1)
+
+ self.row += 1
+
+ def addDropDown(self, label, variable, values, defaultValue=""):
+ variable.set(defaultValue)
+
+ Label(self, text=label).grid(row=self.row, sticky=E)
+
+ self.entry = apply(OptionMenu, (self, variable) + tuple(values))
+ self.entry.grid(row=self.row, column=1)
+
+ self.row += 1
+
+class NfdFrame(GuiFrame):
+ def __init__(self, notebook):
+ GuiFrame.__init__(self, notebook)
+
+ self.frameLabel = "NFD"
+
+ # log-level
+ self.logLevel = StringVar(self)
+ self.addDropDown("Log level:", self.logLevel, LOG_LEVELS, LOG_LEVELS[3])
+
+class NlsrFrame(GuiFrame):
+
+ HYPERBOLIC_STATES = [
+ "off",
+ "on",
+ "dry-run"
+ ]
+
+ def __init__(self, notebook):
+ GuiFrame.__init__(self, notebook)
+
+ self.frameLabel = "NLSR"
+
+ # general: network
+ self.network = StringVar(self)
+ self.addEntryBox("Network:", self.network, "/ndn/")
+
+ # general: site
+ self.site = StringVar(self)
+ self.addEntryBox("Site:", self.site, "/edu/site")
+
+ # general: router
+ self.router = StringVar(self)
+ self.addEntryBox("Router:", self.router, "/%C1.Router/cs/host")
+
+ # general: log-level
+ self.logLevel = StringVar(self)
+ self.addDropDown("Log level:", self.logLevel, LOG_LEVELS, LOG_LEVELS[3])
+
+ # hyperbolic: state
+ self.hyperbolicState = StringVar(self)
+ self.addDropDown("Hyperbolic routing:", self.hyperbolicState,
+ self.HYPERBOLIC_STATES, self.HYPERBOLIC_STATES[0])
+
+ # hyperbolic: angle
+ self.angle = StringVar(self)
+ self.addEntryBox("Angle:", self.angle, "0.0")
+
+ # hyperbolic: radius
+ self.radius = StringVar(self)
+ self.addEntryBox("Radius:", self.radius, "0.0")
+
+ # fib: max-faces-per-prefix
+ self.maxFaces = StringVar(self)
+ self.addEntryBox("Max faces per prefix:", self.maxFaces, "0")
+
+
+ def getValues(self):
+ return {
+ "network": self.network.get(),
+ "site": self.site.get(),
+ "router": self.router.get(),
+ "log-level": self.logLevel.get(),
+ "hyperbolic-state": self.hyperbolicState.get(),
+ "angle": self.angle.get(),
+ "radius": self.radius.get(),
+ "max-faces-per-prefix": self.maxFaces.get()
+ }
+
diff --git a/ndn/nlsr.py b/ndn/nlsr.py
index 03719e5..cda91ea 100644
--- a/ndn/nlsr.py
+++ b/ndn/nlsr.py
@@ -113,7 +113,7 @@
other = node1
ip = other.IP(str(link.intf1))
- linkCost = intf.params.get("delay", "0ms").replace("ms", "")
+ linkCost = intf.params.get("delay", "10ms").replace("ms", "")
neighbors += "neighbor\n"
neighbors += "{\n"