Created
January 6, 2015 09:27
-
-
Save yoxisem544/ec9c9cd02bbcbaec8b9d to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from operator import attrgetter | |
from ryu.app import simple_switch_13 | |
from ryu.controller import ofp_event | |
from ryu.controller.handler import MAIN_DISPATCHER, DEAD_DISPATCHER | |
from ryu.controller.handler import set_ev_cls | |
from ryu.lib import hub | |
class SimpleMonitor(simple_switch_13.SimpleSwitch13): | |
def __init__(self, *args, **kwargs): | |
super(SimpleMonitor, self).__init__(*args, **kwargs) | |
self.datapaths = {} | |
self.monitor_thread = hub.spawn(self._monitor) | |
@set_ev_cls(ofp_event.EventOFPStateChange, | |
[MAIN_DISPATCHER, DEAD_DISPATCHER]) | |
def _state_change_handler(self, ev): | |
datapath = ev.datapath | |
if ev.state == MAIN_DISPATCHER: | |
if not datapath.id in self.datapaths: | |
self.logger.debug('register datapath: %016x', datapath.id) | |
self.datapaths[datapath.id] = datapath | |
elif ev.state == DEAD_DISPATCHER: | |
if datapath.id in self.datapaths: | |
self.logger.debug('unregister datapath: %016x', datapath.id) | |
del self.datapaths[datapath.id] | |
def _monitor(self): | |
while True: | |
for dp in self.datapaths.values(): | |
self._request_stats(dp) | |
hub.sleep(10) | |
def _request_stats(self, datapath): | |
self.logger.debug('send stats request: %016x', datapath.id) | |
ofproto = datapath.ofproto | |
parser = datapath.ofproto_parser | |
req = parser.OFPFlowStatsRequest(datapath) | |
datapath.send_msg(req) | |
req = parser.OFPPortStatsRequest(datapath, 0, ofproto.OFPP_ANY) | |
datapath.send_msg(req) | |
@set_ev_cls(ofp_event.EventOFPFlowStatsReply, MAIN_DISPATCHER) | |
def _flow_stats_reply_handler(self, ev): | |
body = ev.msg.body | |
self.logger.info('datapath ' | |
'in-port eth-dst ' | |
'out-port packets bytes') | |
self.logger.info('---------------- ' | |
'-------- ----------------- ' | |
'-------- -------- --------') | |
for stat in sorted([flow for flow in body if flow.priority == 1], | |
key=lambda flow: (flow.match['in_port'], | |
flow.match['eth_dst'])): | |
self.logger.info('%016x %8x %17s %8x %8d %8d', | |
ev.msg.datapath.id, | |
stat.match['in_port'], stat.match['eth_dst'], | |
stat.instructions[0].actions[0].port, | |
stat.packet_count, stat.byte_count) | |
@set_ev_cls(ofp_event.EventOFPPortStatsReply, MAIN_DISPATCHER) | |
def _port_stats_reply_handler(self, ev): | |
body = ev.msg.body | |
self.logger.info('datapath port ' | |
'rx-pkts rx-bytes rx-error ' | |
'tx-pkts tx-bytes tx-error') | |
self.logger.info('---------------- -------- ' | |
'-------- -------- -------- ' | |
'-------- -------- --------') | |
for stat in sorted(body, key=attrgetter('port_no')): | |
self.logger.info('%016x %8x %8d %8d %8d %8d %8d %8d', | |
ev.msg.datapath.id, stat.port_no, | |
stat.rx_packets, stat.rx_bytes, stat.rx_errors, | |
stat.tx_packets, stat.tx_bytes, stat.tx_errors) | |
@set_ev_cls(ofp_event.EventOFPSwitchFeatures, CONFIG_DISPATCHER) | |
def switch_features_handler(self, ev): | |
datapath = ev.msg.datapath | |
ofproto = datapath.ofproto | |
parser = datapath.ofproto_parser | |
# install table-miss flow entry | |
# | |
# We specify NO BUFFER to max_len of the output action due to | |
# OVS bug. At this moment, if we specify a lesser number, e.g., | |
# 128, OVS will send Packet-In with invalid buffer_id and | |
# truncated packet data. In that case, we cannot output packets | |
# correctly. The bug has been fixed in OVS v2.1.0. | |
match = parser.OFPMatch() | |
actions = [parser.OFPActionOutput(ofproto.OFPP_CONTROLLER, | |
ofproto.OFPCML_NO_BUFFER)] | |
self.add_flow(datapath, 0, match, actions) | |
def add_flow(self, datapath, priority, match, actions, buffer_id=None): | |
ofproto = datapath.ofproto | |
parser = datapath.ofproto_parser | |
inst = [parser.OFPInstructionActions(ofproto.OFPIT_APPLY_ACTIONS, | |
actions)] | |
if buffer_id: | |
mod = parser.OFPFlowMod(datapath=datapath, buffer_id=buffer_id, | |
priority=priority, match=match, | |
instructions=inst) | |
else: | |
mod = parser.OFPFlowMod(datapath=datapath, priority=priority, | |
match=match, instructions=inst) | |
datapath.send_msg(mod) | |
@set_ev_cls(ofp_event.EventOFPPacketIn, MAIN_DISPATCHER) | |
def _packet_in_handler(self, ev): | |
# If you hit this you might want to increase | |
# the "miss_send_length" of your switch | |
if ev.msg.msg_len < ev.msg.total_len: | |
self.logger.debug("packet truncated: only %s of %s bytes", | |
ev.msg.msg_len, ev.msg.total_len) | |
msg = ev.msg | |
datapath = msg.datapath | |
ofproto = datapath.ofproto | |
parser = datapath.ofproto_parser | |
in_port = msg.match['in_port'] | |
pkt = packet.Packet(msg.data) | |
eth = pkt.get_protocols(ethernet.ethernet)[0] | |
dst = eth.dst | |
src = eth.src | |
dpid = datapath.id | |
self.mac_to_port.setdefault(dpid, {}) | |
self.logger.info("packet in %s %s %s %s", dpid, src, dst, in_port) | |
# learn a mac address to avoid FLOOD next time. | |
self.mac_to_port[dpid][src] = in_port | |
if dst in self.mac_to_port[dpid]: | |
out_port = self.mac_to_port[dpid][dst] | |
else: | |
out_port = ofproto.OFPP_FLOOD | |
actions = [parser.OFPActionOutput(out_port)] | |
# install a flow to avoid packet_in next time | |
if out_port != ofproto.OFPP_FLOOD: | |
match = parser.OFPMatch(in_port=in_port, eth_dst=dst) | |
# verify if we have a valid buffer_id, if yes avoid to send both | |
# flow_mod & packet_out | |
if msg.buffer_id != ofproto.OFP_NO_BUFFER: | |
self.add_flow(datapath, 1, match, actions, msg.buffer_id) | |
return | |
else: | |
self.add_flow(datapath, 1, match, actions) | |
data = None | |
if msg.buffer_id == ofproto.OFP_NO_BUFFER: | |
data = msg.data | |
out = parser.OFPPacketOut(datapath=datapath, buffer_id=msg.buffer_id, | |
in_port=in_port, actions=actions, data=data) | |
datapath.send_msg(out) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment