-
Notifications
You must be signed in to change notification settings - Fork 2
/
office1.py
112 lines (86 loc) · 3.25 KB
/
office1.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
from ryu.base import app_manager
from ryu.controller import ofp_event
from ryu.controller.handler import MAIN_DISPATCHER
from ryu.controller.handler import set_ev_cls
from ryu.ofproto import ofproto_v1_0
from ryu.lib.mac import haddr_to_bin
from ryu.lib.packet import packet
from ryu.lib.packet import ethernet, udp
from ryu.lib.packet import ether_types
"""
Controller Office 1 - "office1"
- agisce come un normale switch di livello 2
"""
class Office1(app_manager.RyuApp):
OFP_VERSIONS = [ofproto_v1_0.OFP_VERSION]
def __init__(self, *args, **kwargs):
super(Office1, self).__init__(*args, **kwargs)
self.mac_to_port = {2:{},3:{}}
def add_flow(self, datapath, priority, match, actions):
ofproto = datapath.ofproto
parser = datapath.ofproto_parser
# construct flow_mod message and send it.
mod = parser.OFPFlowMod(
datapath=datapath,
match=match,
cookie=0,
command=ofproto.OFPFC_ADD,
idle_timeout=20,
hard_timeout=120,
priority=priority,
flags=ofproto.OFPFF_SEND_FLOW_REM,
actions=actions,
)
datapath.send_msg(mod)
self.logger.info("O1 Flow added")
def _send_package(self, msg, datapath, in_port, actions):
data = None
ofproto = datapath.ofproto
if msg.buffer_id == ofproto.OFP_NO_BUFFER:
data = msg.data
out = datapath.ofproto_parser.OFPPacketOut(
datapath=datapath,
buffer_id=msg.buffer_id,
in_port=in_port,
actions=actions,
data=data,
)
datapath.send_msg(out)
# Callback gestione dei pacchetti
@set_ev_cls(ofp_event.EventOFPPacketIn, MAIN_DISPATCHER)
def _packet_in_handler(self, ev):
# Variabili
msg = ev.msg
datapath = msg.datapath
ofproto = datapath.ofproto
in_port = msg.in_port
dpid = datapath.id
pkt = packet.Packet(msg.data)
eth = pkt.get_protocol(ethernet.ethernet)
if eth.ethertype == ether_types.ETH_TYPE_LLDP:
return
dst = eth.dst
src = eth.src
self.logger.info("INFO packet arrived in s%s (in_port=%s)", dpid, in_port)
# === REGOLE === #
# Controllo se conosco le porte
if dpid in self.mac_to_port:
self.mac_to_port[dpid][src] = in_port
if dst in self.mac_to_port[dpid]:
out_port = self.mac_to_port[dpid][dst]
self.logger.info(
"INFO sending packet from s%s (out_port=%s) w/ mac-to-port rule",
dpid,
out_port,
)
else:
# altrimenti devo fare flooding
out_port = ofproto.OFPP_FLOOD
actions = [datapath.ofproto_parser.OFPActionOutput(out_port)]
if out_port != ofproto.OFPP_FLOOD:
match = datapath.ofproto_parser.OFPMatch(
in_port=in_port,
dl_dst=dst,
dl_src=src
)
self._send_package(msg, datapath, in_port, actions)