POX 控制器

默认拓扑

如果我们不指定控制器直接运行 Mininet 提供给我们的默认拓扑:

sudo mn

你会发现节点间是 ping 得通的,因为默认的控制器进行 l2 层学习获得了流表。

要使用 POX 的远程交换机,你可以这样做:

$ cd ~/pox
$ ./pox.py forwarding.l2_learning

在另一个窗口,启动 Mininet 连接到“远程”控制器(这实际上是在本地运行,但 Mininet 的控制范围之外):

$ sudo mn --controller=remote, ip=127.0.0.1, port=6633

注意,这些其实都是默认的 IP 地址和端口值。

要进入命令行模式的话:

python pox.py openflow.of_01 py

下发流表(因为我想让两边 ping 通,所以应该配正反两条流表规则,以下只是单方向的):

POX> from pox.lib.addresses import IPAddr
POX> from pox.lib.addresses import EthAddr
POX> import pox.openflow.libopenflow_01 as of
POX> core.openflow.connections.keys()
[1]
POX> msg=of.ofp_flow_mod()
POX> msg.priority=3    # 数字越大, 优先级越高
POX> msg.match.nw_src="10.0.0.1"
POX> msg.match.in_port=2
POX> msg.actions.append(of.ofp_action_output(port=1))
POX> core.openflow.connections[1].send(msg)    # 这个 1 就是 connections.keys() 返回的 1

反方向的流表只要交换一下出入端口就 OK 了。

看一看 Mininet 这边,在配流表之前,ping 不通:

mininet> dpctl dump-flows
*** s1 ------------------------------------------------------------------------
NXST_FLOW reply (xid=0x4):
mininet> pingall
*** Ping: testing ping reachability
h1 -> X 
h2 -> X 
*** Results: 100% dropped (0/2 received)

配了单方向流表以后,依然 ping 不通:

mininet> dpctl dump-flows
*** s1 ------------------------------------------------------------------------
NXST_FLOW reply (xid=0x4):
 cookie=0x0, duration=4.827s, table=0, n_packets=0, n_bytes=0, idle_age=4, priority=5,in_port=1 actions=output:2
mininet> h1 ping h2
PING 10.0.0.2 (10.0.0.2) 56(84) bytes of data.
From 10.0.0.1 icmp_seq=1 Destination Host Unreachable
From 10.0.0.1 icmp_seq=2 Destination Host Unreachable
From 10.0.0.1 icmp_seq=3 Destination Host Unreachable
^C
--- 10.0.0.2 ping statistics ---
5 packets transmitted, 0 received, +3 errors, 100% packet loss, time 4000ms

配了双向的以后:

mininet> dpctl dump-flows
*** s1 ------------------------------------------------------------------------
NXST_FLOW reply (xid=0x4):
 cookie=0x0, duration=151.371s, table=0, n_packets=15, n_bytes=630, idle_age=22, priority=5,in_port=1 actions=output:2
 cookie=0x0, duration=4.423s, table=0, n_packets=0, n_bytes=0, idle_age=4, priority=5,in_port=2 actions=output:1
mininet> h1 ping h2
PING 10.0.0.2 (10.0.0.2) 56(84) bytes of data.
64 bytes from 10.0.0.2: icmp_seq=1 ttl=64 time=0.208 ms
64 bytes from 10.0.0.2: icmp_seq=2 ttl=64 time=0.035 ms
^C
--- 10.0.0.2 ping statistics ---
2 packets transmitted, 2 received, 0% packet loss, time 999ms
rtt min/avg/max/mdev = 0.035/0.121/0.208/0.087 ms
mininet> pingall
*** Ping: testing ping reachability
h1 -> h2 
h2 -> h1 
*** Results: 0% dropped (2/2 received)

成功了!

【注】交换机的端口号一般就是从 1 按顺序的,可以在启动 Mininet 以后新打开一个窗口用下面的命令查看都有哪些端口:

$ sudo ovs-vsctl show
0072de5e-fdb2-4e99-91e7-4f5d33acfa10
    Bridge "s1"
        Controller "tcp:127.0.0.1:6633"
            is_connected: true
        Controller "ptcp:6654"
        fail_mode: secure
        Port "s1-eth2"    # 端口 2
            Interface "s1-eth2"
        Port "s1"
            Interface "s1"
                type: internal
        Port "s1-eth1"    # 端口 1
            Interface "s1-eth1"
    ovs_version: "2.5.4"

POX 控制器里指定端口时不能用字符串,只能用数字,"s1-eth1" 和 "s1-eth2" 写成 1 和 2 就好。

(至于端口对应着哪台主机,可以从网页上看,不知道有没有什么直观的从命令行上就可以查看的方式?或者由 API 指定?)
问题解决,这个顺序是按照 AddLink 的顺序,比如 h1 和 h2 先后与 s1 连,那么 h1 就是端口 1 上的,h2 是端口 2 上的。当然也可以直接在 API 中指定,如:

# specify a different port to connect host 5 to switch 1
net.addLink( h5, s1, 1, 9 )    # h5 的 port 1 连接 s1 的 port 9 (注意如果没定义端口 1 就去定义端口 2 会报错)

另外如果不正常退出 Mininet 或 POX 的话,会对下一次运行产生影响,所以最好用 exit 退出 Mininet,用 Ctrl-D 退出 POX。不然就要 sudo mn -c 来清理 Mininet 的缓存,并手动 kill POX 的进程。

POX 的 API 可以查看这里,是相对很全的了。

自定义拓扑

利用 Mininet 的 Python API 可以很方便地开发各种拓扑以及连接远程控制器。给出一个示例:

#!/usr/bin/python
#coding=utf-8 
from mininet.topo import Topo
from mininet.net import Mininet
from mininet.util import dumpNodeConnections
from mininet.log import setLogLevel
from mininet.cli import CLI
from mininet.node import RemoteController

class SingleSwitchTopo(Topo):
        def __init__(self, **opts):
                Topo.__init__(self, **opts)

                switch1 = self.addSwitch( 's1' )    # 添加交换机到拓扑中
                switch2 = self.addSwitch( 's2' )
                switch3 = self.addSwitch( 's3' )
                host1 = self.addHost( 'h1' )    # 添加主机到拓扑中
                host2 = self.addHost( 'h2' )    # 添加主机到拓扑中

                self.addLink( host1, switch1 )
                self.addLink( host2, switch2 )    # 添加双向连接
                self.addLink( switch2, switch1 )
                self.addLink( switch2, switch3 )
                self.addLink( switch3, switch1 )

def simpleTest():
        topo = SingleSwitchTopo()
        net = Mininet(topo)    # 主要类来创建和管理网络
        mycontroller = RemoteController("c0", ip = "127.0.0.1", port=6633)    # 创建远程控制器 (控制器的命名最好是 "c数字", 否则可能报错)
        net.controllers = [mycontroller]
        net.start()    # 启动拓扑网络
        print "Dumping host connections"
        dumpNodeConnections(net.hosts)    # 输出指定的网络信息
        print "Testing network connectivity"
        net.pingAll()    # 所有节点彼此测试互连
        # CLI(net)    # 进入 mininet> 提示符 
        net.stop()    # 停止网络


if __name__ == '__main__':
        setLogLevel('info')  # 设置 Mininet 默认输出级别,设置 info 它将提供一些有用的信息
        simpleTest()

更多 API 参见:http://mininet.org/api/annotated.html

你可能感兴趣的:(POX 控制器)