
1. 理解Mininet与OpenDaylight本地连接的挑战
在软件定义网络(sdn)的实践中,mininet作为轻量级网络模拟器,常与opendaylight(odl)等控制器配合使用,以构建和测试网络拓扑及控制逻辑。当mininet和odl都在同一台本地机器上运行时,我们通常期望通过python脚本自动化拓扑的创建和连接。然而,一个常见的问题是,尽管通过命令行直接运行sudo mn --controller=remote,ip=127.0.0.1,port=6633可以成功连接控制器,但使用自定义python脚本时,却可能无法建立连接。
以下是一个典型的Mininet脚本示例,它尝试连接一个远程控制器:
from mininet.net import Mininet
from mininet.node import RemoteController
from mininet.cli import CLI
from mininet.log import info, setLogLevel
setLogLevel('info')
def create_topology():
# 初始尝试:将控制器添加到已创建的网络中
net = Mininet(controller=None) # 此时Mininet尚未指定默认控制器类型
info('*** Adding controller\n')
# 显式添加一个远程控制器实例
c0 = net.addController('c0', controller=RemoteController, ip='127.0.0.1', port=6633)
info('*** Adding hosts and switches\n')
s1 = net.addSwitch('s1')
s2 = net.addSwitch('s2')
h1 = net.addHost('h1')
h2 = net.addHost('h2')
h3 = net.addHost('h3')
h4 = net.addHost('h4')
info('*** Creating links\n')
net.addLink(h1, s1)
net.addLink(h2, s2)
net.addLink(h3, s1)
net.addLink(h4, s2)
# 链接交换机到控制器 (虽然通过addController已指定,但这里是逻辑上的拓扑连接)
# 注意:Mininet的addLink通常用于主机和交换机之间,或交换机之间。
# 交换机到控制器的连接是由交换机自身配置的OpenFlow协议完成。
# net.addLink(s1, c0) # 这行在Mininet中通常是不必要的,因为交换机通过OpenFlow协议连接控制器
# net.addLink(s2, c0) # 同上
return net
if __name__ == '__main__':
topo = create_topology()
info('*** Starting network\n')
# 启动控制器实例
topo.controllers[0].start()
topo.start()
info('*** Running CLI\n')
CLI(topo)
info('*** Stopping network\n')
topo.controllers[0].stop()
topo.stop()上述脚本的问题在于,尽管我们通过net.addController添加了一个RemoteController实例,但在Mininet(controller=None)初始化时,Mininet并没有被告知其拓扑中的交换机应该默认连接哪种类型的控制器。这可能导致交换机在启动时无法正确地与远程控制器建立OpenFlow会话。
2. 问题根源分析:Mininet默认行为与控制器类型
Mininet在初始化时,如果未明确指定controller参数,它会默认使用DefaultController(通常是一个内部的简单控制器)或不预设任何控制器类型。当后续通过net.addController()添加一个RemoteController实例时,这个实例本身会被创建并启动,但Mininet拓扑中的交换机(尤其是Open vSwitch, OVS)在启动时,可能并不知道要连接到哪个控制器,或者它们尝试连接到错误的默认控制器类型。
与此不同的是,sudo mn --controller=remote,ip=127.0.0.1,port=6633这样的命令行参数,会指示Mininet在构建拓扑时,将所有创建的交换机默认配置为连接到指定的远程控制器。这种“全局”的默认配置确保了交换机能够正确地发现并连接到远程ODL实例。
3. 解决方案:显式配置Mininet构造器
解决此问题的关键在于,在Mininet构造函数中显式地指定默认的控制器类型和交换机类型。通过将controller=RemoteController和switch=OVSSwitch作为参数传递给Mininet类,我们可以确保Mininet在创建拓扑时,其内部的交换机能够正确地被配置为连接到远程OpenDaylight控制器。
citySHOP是一款集CMS、网店、商品、分类信息、论坛等为一体的城市多用户商城系统,已完美整合目前流行的Discuz! 6.0论坛,采用最新的5.0版PHP+MYSQL技术。面向对象的数据库连接机制,缓存及80%静态化处理,使它能最大程度减轻服务器负担,为您节约建设成本。多级店铺区分及联盟商户地图标注,实体店与虚拟完美结合。个性化的店铺系统,会员后台一体化管理。后台登陆初始网站密匙:LOVES
修改后的Mininet初始化代码如下:
from mininet.net import Mininet
from mininet.node import RemoteController, OVSSwitch # 引入OVSSwitch
from mininet.cli import CLI
from mininet.log import info, setLogLevel
setLogLevel('info')
def create_topology_fixed():
# 关键修改:在Mininet初始化时,显式指定默认控制器类型和交换机类型
net = Mininet(controller=RemoteController, switch=OVSSwitch)
info('*** Adding controller\n')
# 此时,Mininet会自动创建并管理一个RemoteController实例
# 如果需要自定义IP/Port,可以通过addController方法,但通常Mininet会根据默认参数进行配置
# 更好的做法是让Mininet自动处理,或者在Mininet构造器中直接指定控制器参数
# 如果Mininet构造器中已指定controller=RemoteController,则可以省略addController,
# 或者用addController来覆盖默认行为或添加多个控制器。
# 对于单个本地控制器,通常Mininet会自动将其配置为127.0.0.1:6653 (或6633)
# 为确保与ODL的6633端口匹配,我们仍可显式添加或确保Mininet的默认端口设置。
# 这里的addController仍然有效,它会添加一个控制器实例并可能覆盖Mininet的默认配置
c0 = net.addController('c0', controller=RemoteController, ip='127.0.0.1', port=6633)
info('*** Adding hosts and switches\n')
s1 = net.addSwitch('s1')
s2 = net.addSwitch('s2')
h1 = net.addHost('h1')
h2 = net.addHost('h2')
h3 = net.addHost('h3')
h4 = net.addHost('h4')
info('*** Creating links\n')
net.addLink(h1, s1)
net.addLink(h2, s2)
net.addLink(h3, s1)
net.addLink(h4, s2)
# 交换机到控制器的连接由OpenFlow协议处理,无需在Mininet中显式添加链接
# net.addLink(s1, c0) # 移除或注释掉
# net.addLink(s2, c0) # 移除或注释掉
return net
if __name__ == '__main__':
topo = create_topology_fixed()
info('*** Starting network\n')
# 启动控制器实例
topo.controllers[0].start() # 确保RemoteController实例被启动
topo.start()
info('*** Running CLI\n')
CLI(topo)
info('*** Stopping network\n')
topo.controllers[0].stop()
topo.stop()为什么这个修改有效?
- controller=RemoteController: 当Mininet在初始化时被告知所有交换机都应连接到RemoteController类型时,它会相应地配置这些交换机。具体来说,它会指示Open vSwitch(OVS)实例尝试连接到RemoteController的默认地址和端口(通常是127.0.0.1:6633或6653,取决于Mininet版本和配置)。
- switch=OVSSwitch: 明确指定使用OVSSwitch作为交换机类型,确保Mininet使用Open vSwitch,这是与OpenDaylight控制器交互的标准交换机。虽然Mininet通常默认使用OVS,但显式声明可以避免潜在的兼容性问题。
- net.addController('c0', controller=RemoteController, ip='127.0.0.1', port=6633):这一行仍然很重要,它定义了具体要连接的远程控制器的IP地址和端口。即使在Mininet构造器中指定了RemoteController,仍需要此行来提供控制器的具体连接参数。
4. OpenDaylight控制器准备
在运行Mininet脚本之前,请确保您的OpenDaylight控制器已正确启动并运行。
-
启动OpenDaylight Karaf:
导航到您的OpenDaylight安装目录,并运行Karaf容器:
./bin/karaf
-
安装必要功能:
在Karaf命令行中,安装SDN控制器所需的核心功能,例如:
feature:install odl-l2switch-switch odl-restconf odl-dlux-all
- odl-l2switch-switch: 提供了基本的二层交换功能,允许控制器学习MAC地址并转发数据包。
- odl-restconf: 提供RESTful API接口,用于管理和配置控制器。
- odl-dlux-all: 提供DLUX Web UI,方便可视化管理和监控网络。
- 确认端口: OpenDaylight默认监听OpenFlow端口6633(或6653,取决于版本和配置)。请确保此端口没有被其他应用程序占用。您可以通过查看ODL的日志或配置来确认。
5. 操作步骤与验证
- 启动OpenDaylight: 按照上述步骤启动ODL Karaf并安装所需功能。等待ODL完全启动,这可能需要几分钟。
-
运行Mininet脚本:
打开一个新的终端,导航到您的Mininet脚本目录,并运行:
sudo python your_mininet_script.py
-
验证连接:
- Mininet CLI: 在Mininet CLI中,输入dpctl show s1或dpctl show s2。您应该能看到交换机已连接到控制器,例如输出中包含is_connected:true和控制器的IP/端口信息。
- OpenDaylight Karaf日志: 观察ODL Karaf的日志输出,您应该能看到类似“OpenFlow connection received from ...”或“Switch s1 connected”等消息。
- OpenDaylight DLUX UI: 如果您安装了odl-dlux-all,可以在浏览器中访问http://localhost:8181/dlux/index.html(默认凭据:admin/admin),在“Topology”或“Nodes”视图中,您应该能看到Mininet创建的交换机和主机。
- Ping测试: 在Mininet CLI中,运行h1 ping h2或net.pingAll()。如果连接成功,ping操作应该能够正常进行,表明控制器正在正确地处理数据包转发。
6. 注意事项与最佳实践
- 防火墙设置: 确保您的系统防火墙允许OpenDaylight控制器监听的端口(通常是6633)的入站连接。
- IP地址: 对于本地连接,使用127.0.0.1作为控制器IP地址是推荐的做法。如果Mininet和ODL运行在不同的虚拟机或容器中,则需要使用ODL实例的实际IP地址。
- Mininet版本: 确保您的Mininet版本与OpenDaylight版本兼容。虽然上述解决方案在大多数Mininet 2.x版本中都适用,但更新的Mininet或ODL版本可能引入新的特性或行为。
- OpenDaylight功能: 根据您的需求,可能需要安装额外的ODL功能。例如,如果您需要使用特定的路由协议或高级策略,请查阅ODL文档以了解所需的功能。
- 日志分析: 在遇到连接问题时,仔细检查OpenDaylight Karaf的日志输出和Mininet的日志(使用setLogLevel('debug')获取更详细信息)是诊断问题的最有效方法。
7. 总结
通过在Mininet构造函数中显式地指定controller=RemoteController和switch=OVSSwitch,我们可以确保Mininet模拟器中的交换机在启动时能够正确地与本地运行的OpenDaylight控制器建立OpenFlow连接。这种方法提供了比简单地添加控制器实例更健壮和明确的配置,解决了自定义Python脚本在本地集成Mininet与OpenDaylight时常见的连接问题,为SDN实验和开发奠定了坚实的基础。









