|
|
楼主 |
发表于 2021-6-1 08:51:37
|
显示全部楼层
一、OVS概念
# D; b8 @/ |0 K- Y) b使用OpenStack neutron+vxlan部署模式下网络节点OVS网桥作为例子0 U# ~% B. N3 q
ovs-vsctl show. N7 u, D- a. U6 |
e44abab7-2f65-4efd-ab52-36e92d9f0200
5 k8 u. O3 ]' m) K% W Manager "ptcp:6640:127.0.0.1"6 l& e3 [1 g: c" Q8 x8 D
is_connected: true; f6 N, V6 G1 } B& b9 f$ m
Bridge br-ext; C" o: }$ ?# ]
Controller "tcp:127.0.0.1:6633"
$ f: J7 I+ E3 h is_connected: true
; U' X! j0 i6 x fail_mode: secure3 Q! \+ x: ~& N0 \. Y
Port br-ext, z: t0 q! q T( P3 d4 E; B, a
Interface br-ext @& m& a% h- u7 ?! |( P+ {8 m
type: internal
% q1 u" U$ r# f& y Port "eth1"& t2 g5 t! G, X+ C6 q j! i) D
Interface "eth1"' {/ U I3 j$ K6 T
Port phy-br-ext2 [; f+ f; y9 Q# p* s5 Q
Interface phy-br-ext
& y r, T3 \$ u type: patch
, D& i2 I% I. h+ H8 i& d5 L options: {peer=int-br-ext}* c7 Y* ~; ~+ |- H8 p' _0 n5 H
Bridge br-tun( G6 K! v) W2 ~% p. R0 c* P
Controller "tcp:127.0.0.1:6633": }9 t3 X) A8 _
is_connected: true
s; M: a* @. G6 U1 |6 `. u fail_mode: secure0 c! k2 q+ r, I4 E$ F/ A' P" B1 s
Port br-tun9 b) G. w. s& i: O2 v2 U! i
Interface br-tun
/ J. y3 k% F, ~, d# D0 A5 z% Z" H; N type: internal! _9 P7 N" ?. g' i! F
Port patch-int
8 J( T0 K2 R( t Interface patch-int
0 ?$ ?6 V: a9 m type: patch9 p$ R( b k1 }: c
options: {peer=patch-tun}
5 c' ?' I; F2 J5 H Port "vxlan-080058ca"
7 f) k2 e9 K* `3 | Interface "vxlan-080058ca"
9 Z7 K$ S8 ~: [0 ~# K& b9 O type: vxlan6 j2 A4 i$ W9 b5 Y4 \4 F. w. J
options: {df_default="true", in_key=flow, local_ip="8.0.88.201", out_key=flow, remote_ip="8.0.88.202"}
+ p9 q8 N8 G$ y0 A. g* K Bridge br-int9 K/ W |* r; K- p1 N# T
Controller "tcp:127.0.0.1:6633"' T/ m$ Z( B& u. d
is_connected: true
, m* \$ u+ r, h fail_mode: secure
. x& I9 h) J, x7 A- ?. s0 N% \ Port "qr-11591618-c4"
$ z& \' I+ y2 C/ Q; X' L" I tag: 3
$ ?' H& U, a2 Q Interface "qr-11591618-c4"5 n" N# r& m8 i3 y1 I4 a# q8 e1 ^
type: internal
' D6 B3 l0 x3 F8 a Port patch-tun
5 ], b3 K; B$ \# |2 k0 U; }& s Interface patch-tun
" `) t( d% i6 c- H1 X7 H type: patch6 V1 l8 Y9 U# N" v# c
options: {peer=patch-int}
: H* p- M2 |$ l+ R Port int-br-ext
4 o( C& d- G5 R! | Interface int-br-ext
3 X3 q9 I( |$ N& d2 `* c! O type: patch
3 K, D. u1 k$ p; }7 C options: {peer=phy-br-ext}/ d% q% n1 v1 a( }# F; Y; Y1 f0 E' Q
3 P8 ^ x5 O5 |( B6 K5 m! `3 `3 S
1、Bridge; {' |1 t/ Y5 b" n
Bridge代表一个以太网交换机(Switch),一个主机中可以创建一个或者多个Bridge。Bridge的功能是根据一定规则,把从端口收到的数据包转发到另一个或多个端口,上面例子中有三个Bridge,br-tun,br-int,br-ext
8 r: C8 A6 g) i) ?/ N5 m" R8 ]添加一个网桥br0
- b5 _8 J! T! [6 E! g! @7 C# oovs-vsctl add-br br0
9 G$ d( T; C C1
6 k) S5 x8 N5 Y' {: y" B网桥的fail-mode设置
- f& M) c8 u& ^6 }+ Sstandalone:该模式下,没有运行controller的情况下,OVS会自动调整为普通交换机模式,像物理交换机一样,此时可能会有环路产生的问题。如果有controller的情况,OVS会自动变成openflow交换机
7 t6 m. K' t+ D, z5 u0 Y9 V# J" ^secure: 该模式下,无论有没有controller,OVS都会作为openflow交换机运行,也就是所有网桥上的端口,等待controller或者用户添加flow到OVS来进行通信。/ w; `' l) w" W& `, F
2、Port" x0 T5 V$ Q" |2 O2 w4 h. T9 O
端口Port与物理交换机的端口概念类似,Port是OVS Bridge上创建的一个虚拟端口,每个Port都隶属于一个Bridge。Port有以下几种类型
K. e4 ?; x3 {& x6 [" i, w1)Normal
, @5 i/ i3 o! T0 m$ l$ y可以把操作系统中已有的网卡(物理网卡em1/eth0,或虚拟机的虚拟网卡tapxxx)挂载到ovs上,ovs会生成一个同名Port处理这块网卡进出的数据包。此时端口类型为Normal。
: B; f3 n# t! U. ^6 r如下,主机中有一块物理网卡eth1,把其挂载到OVS网桥br-ext上,OVS会自动创建同名Port eth1。
( a F: u0 J; N* X# b( y* x; Iovs-vsctl add-port br-ext eth1 //Bridge br-ext中出现port “eth1”6 R$ \' X( Z' ^( W( H
12 X9 ^0 H* N2 U; ~4 E) H; U
有一点要注意的是,挂载到OVS上的网卡设备不支持分配IP地址,因此若之前eth1配置有IP地址,挂载到OVS之后IP地址将不可访问。这里的网卡设备不只包括物理网卡,也包括主机上创建的虚拟网卡。
- `- y: `% L+ x3 z$ X( x2)Internal; ]1 i6 v& E5 a# J! |4 x, V
Internal类型是OVS内部创建的虚拟网卡接口,每创建一个Port,OVS会自动创建一个同名接口(Interface)挂载到新创建的Port上。接口的概念下面会提到。 O+ { d) a4 f* _
下面创建一个网桥br0,并创建一个internal类型的Port p0
3 g* k- U S/ A/ novs-vsctl add-br br06 ?/ v* d* V2 q
ovs-vsctl add-port br0 p0 -- set Interface p0 type=internal
& i; R% t3 Z, T4 [- {) J0 p1 B6 m) U# w: q, ~0 f
2! t4 i" O5 R1 w# w6 Q; g9 K! {" X+ e. p
查看结果2 N y' [9 ?% \0 C2 |+ k
//查看网桥br09 h: ?0 Y. {' n
ovs-vsctl show br0
2 m4 y9 U6 y- r* ~" l" q& ~ Bridge "br0"0 @2 G( G- S7 `9 `
fail_mode:secure3 i- F# D7 ^$ u( s, m/ q1 @
Port "p0"8 n5 i; s$ {/ G8 ~. }; U' g4 `& M
Interface "p0": {* j, B. r( r( Z
type:internal
& n+ H7 }! p5 N: s2 \! {6 v Port "br0"
- _7 C w7 @" v: x Interface "br0"
$ ? M) R: x8 w- g type:internal
& E, H; C$ i3 Q9 O2 L3 H: q% j+ d* i, S0 E& `
可以看到有两个Port。当ovs创建一个新网桥时,默认会创建一个与网桥同名的Internal Port。在OVS中,只有”internal”类型的设备才支持配置IP地址信息,因此我们可以为br0接口配置一个IP地址,当然p0也可以配置IP地址5 B$ S4 G) ]8 K F" K2 q
ip addr add 192.168.10.11/24 dev br0, K4 v. \- H2 X1 v& |8 |
ip link set br0 up$ E2 {5 T- N$ r$ I+ x+ Z6 _6 {
//添加默认路由
9 a$ k- }8 L& ]2 A9 _ip route add default via 192.168.10.1 dev br0/ P- p. Y7 u$ `( W- \8 F
6 |: I# B! D' f) e* V上面两种Port类型区别在于,Internal类型会自动创建接口(Interface),而Normal类型是把主机中已有的网卡接口添加到OVS中3 z3 Y, E o& `' d- E4 b* n& |
3)Patch
; o! v9 r7 C m0 G2 ?8 L- p当主机中有多个ovs网桥时,可以使用Patch Port把两个网桥连起来。Patch Port总是成对出现,分别连接在两个网桥上,从一个Patch Port收到的数据包会被转发到另一个Patch Port,类似于Linux系统中的veth。使用Patch连接的两个网桥跟一个网桥没什么区别,OpenStack Neutron中使用到了Patch Port。上面网桥br-ext中的Port phy-br-ext与br-int中的Port int-br-ext是一对Patch Port: }: K7 R4 z7 a$ i+ u9 k$ A; }
可以使用ovs-vsctl创建patch设备,如下创建两个网桥br0,br1,然后使用一对Patch Port连接它们3 s7 ~; i3 A- G: z: N7 V
ovs-vsctl add-br br0: Y3 s# r5 ^, _& }/ g; Z6 m! t9 D
ovs-vsctl add-br br1
3 f, I: t! W- N8 W" e8 D; kovs-vsctl \1 I4 Z' N8 h- H C+ b2 K
-- add-port br0 patch0 --set interface patch0 type=patch options:peer=patch1 \' i p- [& P8 N$ G
-- add-port br1 patch1 --set interface patch1 type=patch options:peer=patch04 F. ^1 F' s9 ^) @
. [1 ]( o& Z% T7 v. c
结果如下& p4 u [2 \' H- U+ G
ovs-vsctl show" F! F; G ^% D2 _% K' _* N. z6 T
Bridge "br0", D& }4 ]% f& p8 b5 c( E) w5 W
Port "br0"/ N3 W& u1 W. Z0 ]4 y' v( q) m
Interface "br0"
% W& C5 r) Q: j" z4 k type:internal
7 S5 ]4 S; s5 R8 T1 E" l. w Port "patch0"% E7 P& H6 N( R6 x( S& a
Interface "patch0"* t! l0 X' \# f( c+ d6 g8 _
type:patch
" U/ ^1 M3 p' @ options:{peer="patch1"}6 \) D/ E6 C5 @2 F" U. `1 K) \
Bridge "br1", V, c6 u( w' b
Port "br1"
; e; A( H7 t1 I5 M( g- }6 b; |. c Interface "br1"
& z1 I4 d4 ]" ~7 ]& B! G- l type:internal' v% p( m6 E4 y9 k$ \2 z3 @6 F
Port "patch1"
& g6 Y2 O3 l7 [1 X2 M+ B6 } |& V Interface "patch1"- A: d; V* D8 Y; b' H1 @0 d4 L
type:patch3 Q# ` \' Z* v' G/ X/ M
options:{peer="patch0"}
) L# [/ N- @! ~- r6 |" p; \3 u/ j3 f9 y/ W: j/ l" ^
4)Tunnel
& d+ j3 L8 v( ~# X9 bOVS中支持添加隧道(Tunnel)端口,常见隧道技术有两种gre或vxlan。隧道技术是在现有的物理网络之上构建一层虚拟网络,上层应用只与虚拟网络相关,以此实现的虚拟网络比物理网络配置更加灵活,并能够实现跨主机的L2通信以及必要的租户隔离。不同隧道技术其大体思路均是将以太网报文使用隧道协议封装,然后使用底层IP网络转发封装后的数据包,其差异性在于选择和构造隧道的协议不同。Tunnel在OpenStack中用作实现大二层网络以及租户隔离,以应对公有云大规模,多租户的复杂网络环境。
8 r- G. {9 ]% `- i' p, G$ Q9 rOpenStack是多节点结构,同一子网的虚拟机可能被调度到不同计算节点上,因此需要有隧道技术来保证这些同子网不同节点上的虚拟机能够二层互通,就像他们连接在同一个交换机上,同时也要保证能与其它子网隔离。/ [# M4 o0 p4 k" V! Y; z
OVS在计算和网络节点上建立隧道Port来连接各节点上的网桥br-int,这样所有网络和计算节点上的br-int互联形成了一个大的虚拟的跨所有节点的逻辑网桥(内部靠tunnel id或VNI隔离不同子网),这个逻辑网桥对虚拟机和qrouter是透明的,它们觉得自己连接到了一个大的br-int上。从某个计算节点虚拟机发出的数据包会被封装进隧道通过底层网络传输到目的主机然后解封装。; ?. x0 @* k9 @% m) c0 u& D
上面网桥br-tun中Port "vxlan-080058ca"就是一个vxlan类型tunnel端口。下面使用两台主机测试创建vxlan隧道
) y# k& u7 ]3 Y- |2 B3 |//主机192.168.7.21上
9 Z- o- Y/ {+ Jovs-vsctl add-br br-vxlan4 A1 v' U: K- t3 P
//主机192.168.7.23上
. d& ~3 I1 A9 `ovs-vsctl add-br br-vxlan
' e2 ?$ `! ^: k6 T* o6 }, g; N//主机192.168.7.21上添加连接到7.23的Tunnel Port" I. k! b0 g. Q0 }, n% n4 ?. |
ovs-vsctl add-port br-vxlan tun0 --set Interface tun0 type=vxlan options:remote_ip=192.168.7.23
; e8 s4 W- _) [9 ^. {0 A8 u//主机192.168.7.23上添加连接到7.21的Tunnel Port. v7 N4 d+ q$ Y. @% _
ovs-vsctl add-port br-vxlan tun0 --set Interface tun0 type=vxlan options:remote_ip=192.168.7.21, n( O6 K \; I7 N, R4 @. e
" R3 ~5 G; t0 D$ s4 X1 E; F8 a然后,两个主机上桥接到br-vxlan的虚拟机就像连接到同一个交换机一样,可以实现跨主机的L2连接,同时又完全与物理网络隔离。
$ ~" Z* Q+ T2 z/ a4 J. X3、Interface
6 g' _' |9 [' _, B& tInterface是连接到Port的网络接口设备,是OVS与外部交换数据包的组件,在通常情况下,Port和Interface是一对一的关系,只有在配置Port为 bond模式后,Port和Interface是一对多的关系。这个网络接口设备可能是创建Internal类型Port时OVS自动生成的虚拟网卡,也可能是系统的物理网卡或虚拟网卡(TUN/TAP)挂载在ovs上。 OVS中只有”Internal”类型的网卡接口才支持配置IP地址( C4 b* g r* H k& Z
Interface是一块网络接口设备,负责接收或发送数据包,Port是OVS网桥上建立的一个虚拟端口,Interface挂载在Port上。一个接口就是操作系统的一块网卡6 \$ B, h0 V3 k% J$ q
当Open vSwitch创建一个新网桥时,默认会创建一个与网桥同名的Internal Port,同时也创建一个与Port同名的Interface。三位一体,所以操作系统里就多了一块网卡,但是状态是down的。
. u# v) `3 q' _" k& Q6 u8 I4、Controller
2 ~- u* `6 }% u. _OpenFlow控制器。OVS可以同时接受一个或者多个OpenFlow控制器的管理。主要作用是下发流表(Flow Tables)到OVS,控制OVS数据包转发规则。控制器与OVS通过网络连接,不一定要在同一主机上3 n: o" @: N) n8 d. i d
可以看到上面实例中三个网桥br-int,br-ext,br-tun都连接到控制器Controller "tcp:127.0.0.1:6633上7 }5 w+ N" o- E
5、Datapath3 s) N5 e" O' r/ w
OVS内核模块,负责执行数据交换。由于流可能非常复杂,对每个进来的数据包都去尝试匹配所有流,效率会非常低,所以有了datapath这个东西。Datapath是流的一个缓存,会把流的执行结果保存起来,当下次遇到匹配到同一条流的数据包,直接通过datapath处理。考虑到转发效率,datapath完全是在内核态实现的,并且默认的超时时间非常短,大概只有3秒左右。) }; T9 V: ^3 q. X" \- E( f
二、open vswitch常用操作
! `3 c- F" e& |0 i1、ovs-vsctl9 ?- t+ B- b# Y' R9 B+ T' U- Y
ovs-vsctl是一个管理或配置ovs-vswitchd的高级命令行工具,高级是说其操作对用户友好,封装了对数据库的操作细节。它是管理OVS最常用的命令,除了配置flows之外,其它大部分操作比如Bridge/Port/Interface/Controller/Database/Vlan等都可以完成
& W' `' b9 r P+ ^% Z添加网桥br09 }% b' K$ w5 L+ r1 m- ^
ovs-vsctl add-br br01 q' k1 |' c- C; h3 @
列出所有网桥$ [& ]" { R6 @) D
ovs-vsctl list-br q$ V- a5 u( M$ X% b
添加一个Port p1到网桥br0
7 E$ m% X6 V% kovs-vsctl add-port br0 p1
# g% o* \! ^2 B( c# ~( G* L' N查看网络br0上所有Port
5 G8 K e; }, w5 g, K! g% b: jovs-vsctl list-ports br09 D* O; J B: h$ \) H9 |: h
获取br0网桥的Openflow控制器地址,没有控制器则返回空
; g8 }2 \( K) u4 H. j% |ovs-vsctl get-controller br0
7 `8 m1 s" l8 A4 e2 X& a1 U设置OpenFlow控制器,控制器地址为192.168.1.10,端口为6633
3 l5 U! X/ u F6 w" O' \- rovs-vsctl set-controller br0 tcp:192.168.1.10:6633* L* @1 F' L4 A% Y! T3 O9 I3 p
移除controller& t* C) @! V4 e" H
ovs-vsctl del-controller br0
& Q- I4 W1 L8 A( i$ @1 \删除网桥br0
- \- d& [5 J2 t2 \; Eovs-vsctl del-br br02 n4 q9 c3 K) D' k" u
设置端口p1的vlan tag为100
# b+ u: F! H/ n6 Sovs-vsctl set Port p1 tag=100
' P7 j. H7 f% P3 X! v" N设置端口p0类型为internal
2 }9 Q9 W8 ^& U5 eovs-vsctl set Interface p0 type=internal! Z( S" W: O2 m) p; m5 `
添加vlan10端口,并设置vlan tag为10,Port类型为Internal
. N7 M2 F& r3 \0 K! Govs-vsctl add-port br0 vlan10 tag=10 --set Interface vlan10 type=internal
) x$ L; {/ P9 V7 W" x/ U添加隧道端口gre0,类型为gre,远端IP为1.2.3.4, ^0 s% K( H' c( _; B) Y$ N) q
ovs-vsctl add-port br0 gre0 --set Interface gre0 type=gre options:remote_ip=1.2.3.4( }/ q- E& t/ c6 O0 ~% i$ _" z
- K' p9 @8 q* I2、ovsdb-tool
3 @ u1 ~* d9 N- S6 `- jovsdb-tool是一个专门管理OVS数据库文件的工具,不常用,它不直接与ovsdb-server进程通信 S' f7 e6 Q H4 J9 o. c3 N
//可以使用此工具创建并初始化database文件: A+ p! s2 T. m9 V+ U
ovsdb-tool create [db][schema]% d3 |8 k) l7 r' B
//可以使用ovsdb-client get-schema [database]获取某个数据库的schema(json格式)4 R) ] O- |' p& C1 ^$ P U9 n) `
//可以查看数据库更改记录,具体到操作命令,这个比较有用
; {* ~. D5 i5 m- X- |ovsdb-tool show-log -m
) P/ A* o7 ?& u: F. [# U Vrecord 48: 2017-01-07 03:34:15.147 "ovs-vsctl: ovs-vsctl --timeout=5 -- --if-exists del-port tapcea211ae-10"! S5 ?' J* Q" v/ L( d& }- l
table Interface row "tapcea211ae-10" (151f66b6):& z% _- Z' _% c. }1 U
delete row
+ B1 m, i, C/ [# x. D: u table Port row "tapcea211ae-10" (cc9898cd):0 n/ d# A/ x* p5 O; u
delete row0 j* I# U& q' B1 H) s% ]
table Bridge row "br-int" (fddd5e27):
5 i4 d# S) E3 m" ]# W6 p/ K table Open_vSwitch row a9fc1666 (a9fc1666):
& d6 `& H, W1 F% p9 O* P# rrecord 49: 2017-01-07 04:18:23.671 "ovs-vsctl: ovs-vsctl --timeout=5 -- --if-exists del-port tap5b4345ea-d5 -- add-port br-int tap5b4345ea-d5 -- set Interface tap5b4345ea-d5 "external-ids:attached-mac=\"fa:16:3e:50:1b:5b\"" -- set Interface tap5b4345ea-d5 "external-ids:iface-id=\"5b4345ea-d5ea-4285-be99-0e4cadf1600a\"" -- set Interface tap5b4345ea-d5 "external-ids:vm-id=\"0aa2d71e-9b41-4c88-9038-e4d042b6502a\"" -- set Interface tap5b4345ea-d5 external-ids:iface-status=active"
/ k1 M; k! ~5 ~3 P% G! W. g3 `5 { table Port insert row "tap5b4345ea-d5" (4befd532):
' E$ e$ x! g& i table Interface insert row "tap5b4345ea-d5" (b8a5e830):
0 V9 \" C, c; i- \: q1 N0 t table Bridge row "br-int" (fddd5e27):
3 Z+ ]$ B% i2 W table Open_vSwitch row a9fc1666 (a9fc1666):
+ R3 C4 @! V' J) E( R& I$ u7 }- O...
$ @6 r; w; x9 `. j, n' a5 u0 l7 |+ ?" U6 y
3、ovsdb-client
6 j5 S2 y' u% `' A: `4 E1 govsdb-client是ovsdb-server进程的命令行工具,主要是从正在运行的ovsdb-server中查询信息,操作的是数据库相关
! L i' @4 ~+ F) h//列出主机上的所有databases,默认只有一个库Open_vSwitch0 B# n- c& A/ s! f- K$ g& m8 K
ovsdb-client list-dbs
- B& y1 u4 R/ M4 A0 I: Y//获取指定数据库的schema信息/ @4 [3 k1 A3 _& \
ovsdb-client get-schema [DATABASE]$ r# T% c( X! I7 q5 X# S. x
//列出指定数据库的所有表" R& n, H, C' {% [3 E9 ^# V
ovsdb-client list-tables [DATABASE]* e% y* s- @4 F* f& k1 n" ~/ y
//dump指定数据库所有数据,默认dump所有table数据,如果指定table,只dump指定table数据 8 F& Z" `* v6 x! O. R3 e
ovsdb-client dump [DATABASE] [TABLE]2 x3 f7 Q- L$ V% Q' Q; ^% G; g
//监控指定数据库中的指定表记录改变
- ~" k0 ^+ S- m, ~ovsdb-client monitor DATABASE TABLE- s3 p: y# \) Y7 c
9 o; E; K8 X9 M8 E
4、ovs-ofctl
5 {4 A7 ?4 ]8 s3 J9 K( L( @1 ]ovs-ofctl是专门管理配置OpenFlow交换机的命令行工具,我们可以用它手动配置OVS中的OpenFlow flows,注意其不能操作datapath flows和”hidden” flows
8 ?7 Q5 P, F" X0 L! i//查看br-tun中OpenFlow flows
4 i {$ e: F3 M7 \ovs-ofctl dump-flows br-tun$ Q* A1 e# k4 n6 g. T" I: V
//查看br-tun端口信息
" z5 u8 K9 L5 m$ e9 v9 x# P5 rovs-ofctl show br-tun1 G0 e! b4 }0 g N! `/ Y/ H
//添加新的flow:对于从端口p0进入交换机的数据包,如果它不包含任何VLAN tag,则自动为它添加VLAN tag 1012 e: e% E6 ]. p/ [ M7 O) `* N& L
ovs-ofctl add-flow br0 "priority=3,in_port=100,dl_vlan=0xffff,actions=mod_vlan_vid:101,normal"1 L7 `' L7 V" v4 M7 A/ \/ j
//对于从端口3进入的数据包,若其vlan tag为100,去掉其vlan tag,并从端口1发出
. O) p! c% q" v/ \& j0 p; b) Jovs-ofctl add-flow br0 in_port=3,dl_vlan=101,actions=strip_vlan,output:1
4 ~2 v; L. \ Z' l' f//添加新的flow: 修改从端口p1收到的数据包的源地址为9.181.137.1,show 查看p1端口ID为100
3 h! @' z9 V8 E* _: L0 K0 }ovs-ofctl add-flow br0 "priority=1 idle_timeout=0,in_port=100,actions=mod_nw_src:9.181.137.1,normal"- [3 f7 F" ~' ~+ y
//添加新的flow: 重定向所有的ICMP数据包到端口 p2
9 a& M! x* i5 q& p' Y( l! Jovs-ofctl add-flow br0 idle_timeout=0,dl_type=0x0800,nw_proto=1,actions=output:102: |, C- i( f+ ]4 m8 X* g, V- v3 k0 t
//删除编号为 100 的端口上的所有流表项 $ ^, L9 Q7 W2 U! m2 F' D+ H# M: {% e
ovs-ofctl del-flows br0 "in_port=100" - U8 ^% H5 m3 f+ K$ O: h4 ~
) z/ U0 ^9 l M三、使用open vswitch构建虚拟网络
5 G4 Z# S+ p9 x1、构建物理机和物理机相互连接的网络
5 H3 ~9 K4 B, `8 Q在安装open vswitch的主机上有两块网卡,分别为eth0、eth1,把这两块网卡挂接到open vswitch的网桥上,然后有两台物理机host1、host2分别连接到eth0和eth1上,实现这两台物理机的通信。构建结果图如下:& k4 P& d2 g$ c3 b2 o. x+ P
在这里插入图片描述
4 F* `9 C! K7 h: U执行以下命令:
( N+ w' B+ h% S. E2 w# Q" xovs-vsctl add-br br0 //建立一个名为br0的open vswitch网桥
1 |6 G/ _% Q7 Z+ povs-vsctl add-port br0 eth0 //把eth0挂接到br0中
9 Q9 n" [, e/ R) ?$ {ovs-vsctl add-port br0 eth1 //把eth1挂接到br0中* ]+ G3 q/ h9 e% [
# G" g) o c4 Y2、构建虚拟机与虚拟机相连的网络6 \( p- u+ S2 z& w
在安装open vswitch的主机上安装两个虚拟机,把两个虚拟机的网卡都挂接在open vswitch的网桥上,实现两台虚拟机的通信,构建结果图如下:
1 ?9 _) m& Y+ W$ W在这里插入图片描述
4 \( u) n' _9 y" `- X# m执行以下命令:, u7 B2 G: @: z
ovs-vsctl add-br br0 //建立一个名为br0的open vswitch网桥' K8 M' |7 @' s
1
4 G t% I' V% y1 `. e0 B. |如果使用vbox或virt-manager把bridge设置为br0即可,如果使用cli kvm则先创建两个文件,用于虚拟网卡的添加与删除。假设这两个文件分别为/etc/ovs-ifup和/etc/ovs-ifdown,则向这两个文件中写入以下内容:
; _0 S* Z& G& m' g. B/etc/ovs-ifup
* L8 I8 s2 E# R* @% n* {#!/bin/sh d7 H3 \ Q. A
switch='br0'
2 ?- ]" l4 D5 u8 L/sbin/ifconfig $1 0.0.0.0 up% P$ u5 r! [* v
ovs-vsctl add-port $(switch) $1, l* }" B7 X! m! ]4 k, Y
. P' d% p9 H& ^" c( F$ U/etc/ovs-ifdown
- w( b m3 m# A8 X#!/bin/sh
9 [7 x9 |/ M& B+ D/ V. aswitch='br0'& C3 d. m+ u5 S5 @% s' }' y4 E
/sbin/ifconfig $1 0.0.0.0 down
! c9 V; W) a. O7 r! jovs-vsctl del-port $(switch) $1: [4 [( [4 p0 ^0 A `& p' f
+ k% O% G( H7 }, {/ X使用以下命令建立虚拟机( M, G0 A$ G8 F* e
kvm -m 512 -net nic,macaddr=00:11:22:33:44:55 \
! H: a* f7 K5 p" j-net tap,script=/etc/ovs-ifup,downscript=/etc/ovs-ifdown \
7 s4 z$ s7 c6 z$ {2 O$ w-drive file=/path/to/disk-image,boot=on 8 v o! X0 z9 n; N! p
kvm -m 512 -net nic,macaddr=11:22:33:44:55:66 \
1 G3 X6 I3 N8 o/ [! k-net tap,script=/etc/ovs-ifup,downscript=/etc/ovs-ifdown-drive \
* F4 g; m2 K! K2 {file=/path/to/disk-image,boot=on6 K4 g" F- ]$ ^" a1 N, @7 ?
& h e8 ]& S* r( C. j, Q2 Z
3、构建虚拟机与物理机相连的网络1 g t) y! \$ U8 F6 ^
在装有open vswitch的主机上有一个物理网卡eth0,一台主机通过网线和eth0相连,在open vswitch的主机上还装有一台虚拟机,把此虚拟机和连接到eth0的主机挂接到同一个网桥上,实现两者之间的通信,构建结果图如下:* E5 C: ~; [3 U, T2 U
在这里插入图片描述
7 j- l$ @9 t: y t: b执行命令:
) g0 Z# X6 Y+ i5 M/ b5 Bovs-vsctl add-br br0 //建立一个名为br0的open vswitch网桥) b5 z. N G$ t) X' _. e, c3 H
ovs-vsctl add-port br0 eth0 //把eth0挂接到br0中
/ s+ w$ n0 X! m2 x9 _1 okvm -m 512 -net nic,macaddr=00.11.22.33.44.55-net \
% V! u7 J: c& l$ s: r% x4 o8 J- Wtap,script=/etc/ovs-ifup,downscript=/etc/ovs-ifdown-drive \
: i4 M9 I: t4 L/ T3 [) jfile=/path/to/disk-image,boot=on //ovs-ifup和ovs-ifdown和上一节相同
8 X% p) Z4 d! F1 g% H# }- D* }- J3 s' \1 h: W# n+ h4 s" s2 p
4、构建网桥和网桥相连的网络' G" S( g! j' H3 s* ]; {$ l; G
以上操作都是将多个主机(物理机或虚拟机)连接到同一个网桥上,实现它们之间的通信,但是要构建复杂的网络,就需要多个网桥,在装有open vswitch的主机上建立两个网桥,实现它们之间的连接,构建结果如下:
5 r1 L1 j7 y3 K' a9 \在这里插入图片描述, Q' s5 i, T4 {( h
执行命令:
4 }. B' J+ R3 z# M+ |! p* Fovs-vsctl add-br br0 //添加一个名为br0的网桥
+ C! B" U) s& \8 u8 m6 Jovs-vsctl add-br br1 //添加一个名为br1的网桥1 K, f& O( C7 P* @; K% x; g
ovs-vsctl add-port br0 patch-to-br1 //为br0添加一个虚拟端口
5 F) G1 h, S7 e( a0 Hovs-vsctl set interface patch-to-br1 type=patch //把patch-to-br1的类型设置为patch
* A6 v V( D; S% \. covs-vsctl set interface patch-to-br1 options:peer=patch-to-br0 //把对端网桥和此网桥连接的端口名称设置为patch-to-br0
1 A4 y! P- Y4 g1 j5 J6 Xovs-vsctl add-port br1 patch-to-br0 //为br0添加一个虚拟端口7 j0 `. x. M9 W( f' @
ovs-vsctl set interface patch-to-br0 type=patch //把patch-to-br0的类型设置为patch
. U4 ^+ \' j! @6 j0 v0 kovs-vsctl set interface patch-to-br0 options:peer=patch-to-br1 //把对端网桥和此网桥连接的端口名称设置为patch-to-br17 s( o- v6 R8 Z) }9 h; q9 n. J
1# E, u- ^( b8 H, m
& `8 E( z+ } G5、在不同的主机之间构建网桥之间的连接2 W0 K" Y3 m5 |; d; d
在两台机器上分别安装上open vswitch并创建网桥,分别为两个网桥添加物理网卡,然后通过网线连接两个网桥,实现两个网桥之间的互通。构建结果图如下:5 J# h! k9 I: \0 g2 l6 e& B, {
在这里插入图片描述
" R8 P3 }4 [6 J2 R: j执行命令:
+ k3 ]. x, C! ]% X; |7 x0 X9 Bhost1! Z$ U5 A* O6 _. H/ @) }2 ?7 z
ovs-vsctl add-br br0 //添加名为br0的网桥
5 x5 ~* F6 U' Eovs-vsctl add-port br0 eth0 //把eth0挂接到br0上5 \* @, o! a M
" Y, q: Z% P! E9 f3 s
: x! S) V! ]! T m% i4 phost2
* i/ b& H* c1 S' N4 y) ]ovs-vsctl add-br br0 //添加名为br0的网桥
7 u5 Q& A( x1 d2 A6 B2 y/ u2 q" Tovs-vsctl add-port br0 eth0 //把eth0挂接到br0上
. A6 m d5 h8 G+ |/ C3 J
y% |+ ^9 H- h* s5 D! M然后使用网线把host1的eth0和host2的eth0相连即可。
z) T1 F. M" g# K1 q* D6 v使用上边五种方法的组合就可以构建出各种复杂的网络,为各种实验提供网络的支持。) D! b/ L. q+ o$ {6 h
四、OVS中的各种流(flows)
6 q; d1 R6 ^: f" {. J! T% Eflows是OVS进行数据转发策略控制的核心数据结构,区别于Linux Bridge是个单纯基于MAC地址学习的二层交换机,flows的存在使OVS作为一款SDN交换机成为云平台网络虚拟机化主要组件1 D6 X1 h8 S, i0 M$ y
OVS中有多种flows存在,用于不同目的,但最主要的还是OpenFlow flows这种,文中未明确说明的flows都是指OpenFlow flows! H) S6 ], Z% e3 `
1、OpenFlow flows
: g" f& [0 c1 E, S6 NOVS中最重要的一种flows,Controller控制器下发的就是这种flows,OVS架构部分已经简单介绍过,关于OpenFlow 的具体使用,会在另一篇文章中说明% L7 y+ a. d' I2 I
2、“hidden” flows0 a7 ^* V9 X- p! c9 ~0 A
OVS在使用OpenFlow flows时,需要与OpenFlow控制器建立TCP连接,若此TCP连接不依赖OVS,即没有OVS依然可以建立连接,此时就是out-of-band control模式,这种模式下不需要"hidden" flows。
' E. j- A; t' o( i但是在in-band control模式下,TCP连接的建立依赖OVS控制的网络,但此时OVS依赖OpenFLow控制器下发的flows才能正常工作,没法建立TCP连接也就无法下发flows,这就产生矛盾了,因此需要存在一些"hidden" flows,这些"hidden" flows保证了TCP连接能够正常建立。关于in-band control详细介绍,参考OVS官方文档Design Decisions In Open vSwitch 中In-Band Control部分,“hidden” flows优先级高于OpenFlow flows,它们不需要手动设置。可以使用ovs-appctl查看这些flows,下面命令输出内容包括OpenFlow flows,"hidden" flows6 R) m" A: b3 Q' `( `$ O- @
ovs-appctl bridge/dump-flows<br>, r+ [8 E8 D+ Z+ d4 S( G* B% Q
- `1 f; j0 b# Z
3、datapath flows
% F7 Q+ ~) c: ], Mdatapath flows是datapath内核模块维护的flows,由内核模块维护意味着我们并不需要去修改管理它。与OpenFlow flows不同的是,它不支持优先级,并且只有一个表,这些特点使它非常适合做缓存。与OpenFlow一样的是它支持通配符,也支持指令集(多个action)
+ o5 h# S$ ?% Q! j0 K4 M" wdatapath flows可以来自用户空间ovs-vswitchd缓存,也可以是datapath内核模块进行MAC地址学习到的flows,这取决与OVS是作为SDN交换机,还是像Linux Bridge那样只是一个简单基于MAC地址学习的二层交换机/ {# S( |9 u3 S0 j
4、管理flows的命令行工具9 k- l5 b- f( N
我们可以修改和配置的是OpenFlow flows。datapath flow和"hidden" flows由OVS自身管理,我们不必去修改它。当然,调试场景下还是可以使用工具修改的
6 M( _1 g6 f- E9 [$ U- n介绍下上面三种flows管理工具,不具体说明,具体使用可以查看相关man手册& N* ~# x) s$ X! A7 x
ovs-ofctl dump-flows <br> 打印指定网桥内的所有OpenFlow flows,可以存在多个流表(flow tables),按表顺序显示。不包括"hidden" flows。这是最常用的查看flows命令,当然这条命令对所有OpenFlow交换机都有效,不单单是OVS6 x5 B- |6 W1 E8 s7 p* m- A& t( t
ovs-appctl bridge/dump-flows <br> 打印指定网桥内所有OpenFlow flows,包括"hidden" flows,in-band control模式下排错可以用到# B9 K2 V7 O. P9 x
ovs-dpctl dump-flows [dp] 打印内核模块中datapath flows,[dp]可以省略,默认主机中只有一个datapath system@ovs-system man手册可以找到非常详细的用法说明,注意ovs-ofctl管理的是OpenFlow flows* o, C# z% z0 I- s- n7 w3 y
|
|