blob: 889bbdf5a85ec7c82ed036de51420bd45feb4ae0 [file] [log] [blame]
# Python OpenFlow message wrapper classes
import logging
from cstruct import *
from action_list import action_list
from error import *
# Define templates for documentation
class ofp_template_msg:
"""
Sample base class for template_msg; normally auto generated
This class should live in the of_header name space and provides the
base class for this type of message. It will be wrapped for the
high level API.
"""
def __init__(self):
"""
Constructor for base class
"""
# Additional base data members declared here
# Normally will define pack, unpack, __len__ functions
class template_msg(ofp_template_msg):
"""
Sample class wrapper for template_msg
This class should live in the of_message name space and provides the
high level API for an OpenFlow message object. These objects must
implement the functions indicated in this template.
"""
def __init__(self):
"""
Constructor
Must set the header type value appropriately for the message
"""
##@var header
# OpenFlow message header: length, version, xid, type
ofp_template_msg.__init__(self)
# For a real message, will be set to an integer
self.type = "TEMPLATE_MSG_VALUE"
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
pass
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return Typically returns the remainder of binary_string that
was not parsed. May give a warning if that string is non-empty
"""
pass
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
pass
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
pass
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
pass
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
pass
################################################################
#
# OpenFlow Message Definitions
#
################################################################
class barrier_reply(ofp_header):
"""
Wrapper class for barrier_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_BARRIER_REPLY=19)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_BARRIER_REPLY
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'barrier_reply (OFPT_BARRIER_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class barrier_request(ofp_header):
"""
Wrapper class for barrier_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_BARRIER_REQUEST=18)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_BARRIER_REQUEST
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'barrier_request (OFPT_BARRIER_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class echo_reply(ofp_header):
"""
Wrapper class for echo_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_ECHO_REPLY=3)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_ECHO_REPLY
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'echo_reply (OFPT_ECHO_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class echo_request(ofp_header):
"""
Wrapper class for echo_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_ECHO_REQUEST=2)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_ECHO_REQUEST
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'echo_request (OFPT_ECHO_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class error(ofp_error_msg):
"""
Wrapper class for error
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_ERROR=1)
Data members inherited from ofp_error_msg:
@arg version
@arg type
@arg length
@arg xid
@arg err_type
@arg code
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_error_msg.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_ERROR
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_error_msg.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_error_msg.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_error_msg.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'error (OFPT_ERROR)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_error_msg.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_error_msg.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class features_reply(ofp_switch_features):
"""
Wrapper class for features_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_FEATURES_REPLY=6)
Data members inherited from ofp_switch_features:
@arg version
@arg type
@arg length
@arg xid
@arg datapath_id
@arg n_buffers
@arg n_tables
@arg capabilities
@arg actions
@arg ports: Variable length array of TBD
"""
def __init__(self, **kwargs):
ofp_switch_features.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_FEATURES_REPLY
self.xid = None
self.ports = []
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_switch_features.pack(self)
for obj in self.ports:
packed += obj.pack()
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_switch_features.unpack(self, binary_string)
while len(binary_string) >= OFP_PHY_PORT_BYTES:
new_port = ofp_phy_port()
binary_string = new_port.unpack(binary_string)
self.ports.append(new_port)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_switch_features.__len__(self)
for obj in self.ports:
length += len(obj)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'features_reply (OFPT_FEATURES_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_switch_features.show(self, prefix)
outstr += prefix + "Array ports\n"
for obj in self.ports:
outstr += obj.show(prefix + ' ')
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_switch_features.__eq__(self, other): return False
if self.ports != other.ports: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class features_request(ofp_header):
"""
Wrapper class for features_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_FEATURES_REQUEST=5)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_FEATURES_REQUEST
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'features_request (OFPT_FEATURES_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class flow_mod(ofp_flow_mod):
"""
Wrapper class for flow_mod
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_FLOW_MOD=14)
Data members inherited from ofp_flow_mod:
@arg version
@arg type
@arg length
@arg xid
@arg match
@arg cookie
@arg command
@arg idle_timeout
@arg hard_timeout
@arg priority
@arg buffer_id
@arg out_port
@arg flags
@arg actions: Object of type action_list
"""
def __init__(self, **kwargs):
ofp_flow_mod.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_FLOW_MOD
self.xid = None
self.actions = []
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_flow_mod.pack(self)
packed += action_list(self.actions).pack()
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_flow_mod.unpack(self, binary_string)
ai_len = self.length - (OFP_FLOW_MOD_BYTES + OFP_HEADER_BYTES)
obj = action_list()
binary_string = obj.unpack(binary_string, bytes=ai_len)
self.actions = list(obj)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_flow_mod.__len__(self)
for obj in self.actions:
length += len(obj)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'flow_mod (OFPT_FLOW_MOD)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_flow_mod.show(self, prefix)
outstr += prefix + "List actions\n"
for obj in self.actions:
outstr += obj.show(prefix + " ")
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_flow_mod.__eq__(self, other): return False
if self.actions != other.actions: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class flow_removed(ofp_flow_removed):
"""
Wrapper class for flow_removed
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_FLOW_REMOVED=11)
Data members inherited from ofp_flow_removed:
@arg version
@arg type
@arg length
@arg xid
@arg match
@arg cookie
@arg priority
@arg reason
@arg duration_sec
@arg duration_nsec
@arg idle_timeout
@arg packet_count
@arg byte_count
"""
def __init__(self, **kwargs):
ofp_flow_removed.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_FLOW_REMOVED
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_flow_removed.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_flow_removed.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_flow_removed.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'flow_removed (OFPT_FLOW_REMOVED)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_flow_removed.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_flow_removed.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class get_config_reply(ofp_switch_config):
"""
Wrapper class for get_config_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_GET_CONFIG_REPLY=8)
Data members inherited from ofp_switch_config:
@arg version
@arg type
@arg length
@arg xid
@arg flags
@arg miss_send_len
"""
def __init__(self, **kwargs):
ofp_switch_config.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_GET_CONFIG_REPLY
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_switch_config.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_switch_config.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_switch_config.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'get_config_reply (OFPT_GET_CONFIG_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_switch_config.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_switch_config.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class get_config_request(ofp_header):
"""
Wrapper class for get_config_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_GET_CONFIG_REQUEST=7)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_GET_CONFIG_REQUEST
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'get_config_request (OFPT_GET_CONFIG_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class hello(ofp_header):
"""
Wrapper class for hello
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_HELLO=0)
Data members inherited from ofp_header:
@arg version
@arg type
@arg length
@arg xid
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_HELLO
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_header.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_header.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_header.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'hello (OFPT_HELLO)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_header.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_header.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class packet_in(ofp_packet_in):
"""
Wrapper class for packet_in
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_PACKET_IN=10)
Data members inherited from ofp_packet_in:
@arg version
@arg type
@arg length
@arg xid
@arg buffer_id
@arg total_len
@arg in_port
@arg reason
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_packet_in.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_PACKET_IN
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_packet_in.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_packet_in.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_packet_in.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'packet_in (OFPT_PACKET_IN)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_packet_in.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_packet_in.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class packet_out(ofp_packet_out):
"""
Wrapper class for packet_out
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_PACKET_OUT=13)
Data members inherited from ofp_packet_out:
@arg version
@arg type
@arg length
@arg xid
@arg buffer_id
@arg in_port
@arg actions_len
@arg actions: Object of type action_list
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_packet_out.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_PACKET_OUT
self.xid = None
self.actions = []
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
self.actions_len = 0
for obj in self.actions:
self.actions_len += len(obj)
packed += ofp_packet_out.pack(self)
packed += action_list(self.actions).pack()
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_packet_out.unpack(self, binary_string)
obj = action_list()
binary_string = obj.unpack(binary_string, bytes=self.actions_len)
self.actions = list(obj)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_packet_out.__len__(self)
for obj in self.actions:
length += len(obj)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'packet_out (OFPT_PACKET_OUT)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_packet_out.show(self, prefix)
outstr += prefix + "List actions\n"
for obj in self.actions:
outstr += obj.show(prefix + " ")
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_packet_out.__eq__(self, other): return False
if self.data != other.data: return False
if self.actions != other.actions: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class port_mod(ofp_port_mod):
"""
Wrapper class for port_mod
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_PORT_MOD=15)
Data members inherited from ofp_port_mod:
@arg version
@arg type
@arg length
@arg xid
@arg port_no
@arg hw_addr
@arg config
@arg mask
@arg advertise
"""
def __init__(self, **kwargs):
ofp_port_mod.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_PORT_MOD
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_port_mod.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_port_mod.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_port_mod.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'port_mod (OFPT_PORT_MOD)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_port_mod.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_port_mod.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class port_status(ofp_port_status):
"""
Wrapper class for port_status
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_PORT_STATUS=12)
Data members inherited from ofp_port_status:
@arg version
@arg type
@arg length
@arg xid
@arg reason
@arg desc
"""
def __init__(self, **kwargs):
ofp_port_status.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_PORT_STATUS
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_port_status.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_port_status.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_port_status.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'port_status (OFPT_PORT_STATUS)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_port_status.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_port_status.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class queue_get_config_reply(ofp_queue_get_config_reply):
"""
Wrapper class for queue_get_config_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_QUEUE_GET_CONFIG_REPLY=21)
Data members inherited from ofp_queue_get_config_reply:
@arg version
@arg type
@arg length
@arg xid
@arg port
@arg queues: Variable length array of TBD
"""
def __init__(self, **kwargs):
ofp_queue_get_config_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_QUEUE_GET_CONFIG_REPLY
self.xid = None
self.queues = []
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_queue_get_config_reply.pack(self)
for obj in self.queues:
packed += obj.pack()
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_queue_get_config_reply.unpack(self, binary_string)
for obj in self.queues:
binary_string = obj.unpack(binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_queue_get_config_reply.__len__(self)
for obj in self.queues:
length += len(obj)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'queue_get_config_reply (OFPT_QUEUE_GET_CONFIG_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_queue_get_config_reply.show(self, prefix)
outstr += prefix + "Array queues\n"
for obj in self.queues:
outstr += obj.show(prefix + ' ')
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_queue_get_config_reply.__eq__(self, other): return False
if self.queues != other.queues: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class queue_get_config_request(ofp_queue_get_config_request):
"""
Wrapper class for queue_get_config_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_QUEUE_GET_CONFIG_REQUEST=20)
Data members inherited from ofp_queue_get_config_request:
@arg version
@arg type
@arg length
@arg xid
@arg port
"""
def __init__(self, **kwargs):
ofp_queue_get_config_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_QUEUE_GET_CONFIG_REQUEST
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_queue_get_config_request.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_queue_get_config_request.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_queue_get_config_request.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'queue_get_config_request (OFPT_QUEUE_GET_CONFIG_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_queue_get_config_request.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_queue_get_config_request.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class set_config(ofp_switch_config):
"""
Wrapper class for set_config
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_SET_CONFIG=9)
Data members inherited from ofp_switch_config:
@arg version
@arg type
@arg length
@arg xid
@arg flags
@arg miss_send_len
"""
def __init__(self, **kwargs):
ofp_switch_config.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_SET_CONFIG
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_switch_config.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_switch_config.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_switch_config.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'set_config (OFPT_SET_CONFIG)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_switch_config.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_switch_config.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class stats_reply(ofp_stats_reply):
"""
Wrapper class for stats_reply
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_STATS_REPLY=17)
Data members inherited from ofp_stats_reply:
@arg version
@arg type
@arg length
@arg xid
@arg stats_type
@arg flags
"""
def __init__(self, **kwargs):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_stats_reply.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_stats_reply.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'stats_reply (OFPT_STATS_REPLY)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_stats_reply.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_stats_reply.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class stats_request(ofp_stats_request):
"""
Wrapper class for stats_request
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_STATS_REQUEST=16)
Data members inherited from ofp_stats_request:
@arg version
@arg type
@arg length
@arg xid
@arg stats_type
@arg flags
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_stats_request.unpack(self, binary_string)
# Fixme: If no self.data, add check for data remaining
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_stats_request.__len__(self)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'stats_request (OFPT_STATS_REQUEST)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_stats_request.show(self, prefix)
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_stats_request.__eq__(self, other): return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
class vendor(ofp_vendor_header):
"""
Wrapper class for vendor
OpenFlow message header: length, version, xid, type
@arg length: The total length of the message
@arg version: The OpenFlow version (1)
@arg xid: The transaction ID
@arg type: The message type (OFPT_VENDOR=4)
Data members inherited from ofp_vendor_header:
@arg version
@arg type
@arg length
@arg xid
@arg vendor
@arg data: Binary string following message members
"""
def __init__(self, **kwargs):
ofp_vendor_header.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_VENDOR
self.xid = None
self.data = ""
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self):
"""
Pack object into string
@return The packed string which can go on the wire
"""
self.length = len(self)
packed = ""
packed += ofp_vendor_header.pack(self)
packed += self.data
return packed
def unpack(self, binary_string):
"""
Unpack object from a binary string
@param binary_string The wire protocol byte string holding the object
represented as an array of bytes.
@return The remainder of binary_string that was not parsed.
"""
binary_string = ofp_vendor_header.unpack(self, binary_string)
self.data = binary_string
binary_string = ''
return binary_string
def __len__(self):
"""
Return the length of this object once packed into a string
@return An integer representing the number bytes in the packed
string.
"""
length = 0
length += ofp_vendor_header.__len__(self)
length += len(self.data)
return length
def show(self, prefix=''):
"""
Generate a string (with multiple lines) describing the contents
of the object in a readable manner
@param prefix Pre-pended at the beginning of each line.
"""
outstr = prefix + 'vendor (OFPT_VENDOR)\n'
prefix += ' '
outstr += prefix + 'ofp header\n'
outstr += ofp_vendor_header.show(self, prefix)
outstr += prefix + 'data is of length ' + str(len(self.data)) + '\n'
##@todo Fix this circular reference
# if len(self.data) > 0:
# obj = of_message_parse(self.data)
# if obj != None:
# outstr += obj.show(prefix)
# else:
# outstr += prefix + "Unable to parse data\n"
return outstr
def __eq__(self, other):
"""
Return True if self and other hold the same data
@param other Other object in comparison
"""
if type(self) != type(other): return False
if not ofp_vendor_header.__eq__(self, other): return False
if self.data != other.data: return False
return True
def __ne__(self, other):
"""
Return True if self and other do not hold the same data
@param other Other object in comparison
"""
return not self.__eq__(other)
################################################################
#
# Stats request and reply subclass definitions
#
################################################################
# Stats request bodies for desc and table stats are not defined in the
# OpenFlow header; We define them here. They are empty classes, really
class ofp_desc_stats_request:
"""
Forced definition of ofp_desc_stats_request (empty class)
"""
def __init__(self):
pass
def pack(self, assertstruct=True):
return ""
def unpack(self, binary_string):
return binary_string
def __len__(self):
return 0
def show(self, prefix=''):
return prefix + "ofp_desc_stats_request (empty)\n"
def __eq__(self, other):
return type(self) == type(other)
def __ne__(self, other):
return type(self) != type(other)
OFP_DESC_STATS_REQUEST_BYTES = 0
class ofp_table_stats_request:
"""
Forced definition of ofp_table_stats_request (empty class)
"""
def __init__(self):
pass
def pack(self, assertstruct=True):
return ""
def unpack(self, binary_string):
return binary_string
def __len__(self):
return 0
def show(self, prefix=''):
return prefix + "ofp_table_stats_request (empty)\n"
def __eq__(self, other):
return type(self) == type(other)
def __ne__(self, other):
return type(self) != type(other)
OFP_TABLE_STATS_REQUEST_BYTES = 0
# Stats entries define the content of one element in a stats
# reply for the indicated type; define _entry for consistency
aggregate_stats_entry = ofp_aggregate_stats_reply
desc_stats_entry = ofp_desc_stats
port_stats_entry = ofp_port_stats
queue_stats_entry = ofp_queue_stats
table_stats_entry = ofp_table_stats
#
# Flow stats entry contains an action list of variable length, so
# it is done by hand
#
class flow_stats_entry(ofp_flow_stats):
"""
Special case flow stats entry to handle action list object
"""
def __init__(self):
ofp_flow_stats.__init__(self)
self.actions = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ofp_flow_stats.pack(self, assertstruct)
packed += action_list(self.actions).pack()
if len(packed) != self.length:
print("ERROR: flow_stats_entry pack length not equal",
self.length, len(packed))
return packed
def unpack(self, binary_string):
binary_string = ofp_flow_stats.unpack(self, binary_string)
ai_len = self.length - OFP_FLOW_STATS_BYTES
if ai_len < 0:
print("ERROR: flow_stats_entry unpack length too small",
self.length)
obj = action_list()
binary_string = obj.unpack(binary_string, bytes=ai_len)
self.actions = list(obj)
return binary_string
def __len__(self):
return OFP_FLOW_STATS_BYTES + len(self.actions)
def show(self, prefix=''):
outstr = prefix + "flow_stats_entry\n"
outstr += ofp_flow_stats.show(self, prefix + ' ')
outstr += prefix + "List actions\n"
for obj in self.actions:
outstr += obj.show(prefix + ' ')
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_flow_stats.__eq__(self, other) and
self.actions == other.actions)
def __ne__(self, other): return not self.__eq__(other)
class aggregate_stats_request(ofp_stats_request, ofp_aggregate_stats_request):
"""
Wrapper class for aggregate stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_aggregate_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_AGGREGATE
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_aggregate_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_aggregate_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking aggregate: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_AGGREGATE_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "aggregate_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_aggregate_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_aggregate_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class aggregate_stats_reply(ofp_stats_reply):
"""
Wrapper class for aggregate stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_AGGREGATE
# stats: Array of type aggregate_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = aggregate_stats_entry()
while len(binary_string) >= len(dummy):
obj = aggregate_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking aggregate stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "aggregate_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
class desc_stats_request(ofp_stats_request, ofp_desc_stats_request):
"""
Wrapper class for desc stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_desc_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_DESC
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_desc_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_desc_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking desc: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_DESC_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "desc_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_desc_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_desc_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class desc_stats_reply(ofp_stats_reply):
"""
Wrapper class for desc stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_DESC
# stats: Array of type desc_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = desc_stats_entry()
while len(binary_string) >= len(dummy):
obj = desc_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking desc stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "desc_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
class flow_stats_request(ofp_stats_request, ofp_flow_stats_request):
"""
Wrapper class for flow stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_flow_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_FLOW
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_flow_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_flow_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking flow: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_FLOW_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "flow_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_flow_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_flow_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class flow_stats_reply(ofp_stats_reply):
"""
Wrapper class for flow stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_FLOW
# stats: Array of type flow_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = flow_stats_entry()
while len(binary_string) >= len(dummy):
obj = flow_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking flow stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "flow_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
class port_stats_request(ofp_stats_request, ofp_port_stats_request):
"""
Wrapper class for port stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_port_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_PORT
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_port_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_port_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking port: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_PORT_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "port_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_port_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_port_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class port_stats_reply(ofp_stats_reply):
"""
Wrapper class for port stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_PORT
# stats: Array of type port_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = port_stats_entry()
while len(binary_string) >= len(dummy):
obj = port_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking port stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "port_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
class queue_stats_request(ofp_stats_request, ofp_queue_stats_request):
"""
Wrapper class for queue stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_queue_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_QUEUE
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_queue_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_queue_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking queue: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_QUEUE_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "queue_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_queue_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_queue_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class queue_stats_reply(ofp_stats_reply):
"""
Wrapper class for queue stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_QUEUE
# stats: Array of type queue_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = queue_stats_entry()
while len(binary_string) >= len(dummy):
obj = queue_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking queue stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "queue_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
class table_stats_request(ofp_stats_request, ofp_table_stats_request):
"""
Wrapper class for table stats request message
"""
def __init__(self, **kwargs):
ofp_stats_request.__init__(self)
ofp_table_stats_request.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REQUEST
self.xid = None
self.stats_type = OFPST_TABLE
for (k, v) in kwargs.items():
if hasattr(self, k):
setattr(self, k, v)
else:
raise NameError("field %s does not exist in %s" % (k, self.__class__))
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_request.pack(self)
packed += ofp_table_stats_request.pack(self)
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_request.unpack(self, binary_string)
binary_string = ofp_table_stats_request.unpack(self, binary_string)
if len(binary_string) != 0:
print "ERROR unpacking table: extra data"
return binary_string
def __len__(self):
return OFP_STATS_REQUEST_BYTES + \
OFP_TABLE_STATS_REQUEST_BYTES
def show(self, prefix=''):
outstr = prefix + "table_stats_request\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_request.show(self)
outstr += ofp_table_stats_request.show(self)
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_request.__eq__(self, other) and
ofp_table_stats_request.__eq__(self, other))
def __ne__(self, other): return not self.__eq__(other)
class table_stats_reply(ofp_stats_reply):
"""
Wrapper class for table stats reply
"""
def __init__(self):
ofp_stats_reply.__init__(self)
self.version = OFP_VERSION
self.type = OFPT_STATS_REPLY
self.xid = None
self.stats_type = OFPST_TABLE
# stats: Array of type table_stats_entry
self.entries = []
def pack(self, assertstruct=True):
self.length = len(self)
packed = ""
packed += ofp_stats_reply.pack(self)
for obj in self.entries:
packed += obj.pack()
return packed
def unpack(self, binary_string):
binary_string = ofp_stats_reply.unpack(self, binary_string)
dummy = table_stats_entry()
while len(binary_string) >= len(dummy):
obj = table_stats_entry()
binary_string = obj.unpack(binary_string)
self.entries.append(obj)
if len(binary_string) != 0:
print "ERROR unpacking table stats string: extra bytes"
return binary_string
def __len__(self):
length = OFP_STATS_REPLY_BYTES
for obj in self.entries:
length += len(obj)
return length
def show(self, prefix=''):
outstr = prefix + "table_stats_reply\n"
outstr += prefix + "ofp header:\n"
outstr += ofp_stats_reply.show(self)
outstr += prefix + "Stats array of length " + str(len(self.entries)) + '\n'
for obj in self.entries:
outstr += obj.show()
return outstr
def __eq__(self, other):
if type(self) != type(other): return False
return (ofp_stats_reply.__eq__(self, other) and
self.entries == other.entries)
def __ne__(self, other): return not self.__eq__(other)
message_type_list = (
aggregate_stats_reply,
aggregate_stats_request,
bad_action_error_msg,
bad_request_error_msg,
barrier_reply,
barrier_request,
desc_stats_reply,
desc_stats_request,
echo_reply,
echo_request,
features_reply,
features_request,
flow_mod,
flow_mod_failed_error_msg,
flow_removed,
flow_stats_reply,
flow_stats_request,
get_config_reply,
get_config_request,
hello,
hello_failed_error_msg,
packet_in,
packet_out,
port_mod,
port_mod_failed_error_msg,
port_stats_reply,
port_stats_request,
port_status,
queue_get_config_reply,
queue_get_config_request,
queue_op_failed_error_msg,
queue_stats_reply,
queue_stats_request,
set_config,
table_stats_reply,
table_stats_request,
vendor
)
_flow_mod = flow_mod
flow_mod = None
class flow_add(_flow_mod):
def __init__(self, **kwargs):
_flow_mod.__init__(self, **kwargs)
self.command = OFPFC_ADD
class flow_modify(_flow_mod):
def __init__(self, **kwargs):
_flow_mod.__init__(self, **kwargs)
self.command = OFPFC_MODIFY
class flow_modify_strict(_flow_mod):
def __init__(self, **kwargs):
_flow_mod.__init__(self, **kwargs)
self.command = OFPFC_MODIFY_STRICT
class flow_delete(_flow_mod):
def __init__(self, **kwargs):
_flow_mod.__init__(self, **kwargs)
self.command = OFPFC_DELETE
class flow_delete_strict(_flow_mod):
def __init__(self, **kwargs):
_flow_mod.__init__(self, **kwargs)
self.command = OFPFC_DELETE_STRICT
# These message types are subclassed
msg_type_subclassed = [
OFPT_STATS_REQUEST,
OFPT_STATS_REPLY,
OFPT_ERROR
]
# Maps from sub-types to classes
stats_reply_to_class_map = {
OFPST_DESC : desc_stats_reply,
OFPST_AGGREGATE : aggregate_stats_reply,
OFPST_FLOW : flow_stats_reply,
OFPST_TABLE : table_stats_reply,
OFPST_PORT : port_stats_reply,
OFPST_QUEUE : queue_stats_reply
}
stats_request_to_class_map = {
OFPST_DESC : desc_stats_request,
OFPST_AGGREGATE : aggregate_stats_request,
OFPST_FLOW : flow_stats_request,
OFPST_TABLE : table_stats_request,
OFPST_PORT : port_stats_request,
OFPST_QUEUE : queue_stats_request
}
error_to_class_map = {
OFPET_HELLO_FAILED : hello_failed_error_msg,
OFPET_BAD_REQUEST : bad_request_error_msg,
OFPET_BAD_ACTION : bad_action_error_msg,
OFPET_FLOW_MOD_FAILED : flow_mod_failed_error_msg,
OFPET_PORT_MOD_FAILED : port_mod_failed_error_msg,
OFPET_QUEUE_OP_FAILED : queue_op_failed_error_msg
}
# Map from header type value to the underlieing message class
msg_type_to_class_map = {
OFPT_HELLO : hello,
OFPT_ERROR : error,
OFPT_ECHO_REQUEST : echo_request,
OFPT_ECHO_REPLY : echo_reply,
OFPT_VENDOR : vendor,
OFPT_FEATURES_REQUEST : features_request,
OFPT_FEATURES_REPLY : features_reply,
OFPT_GET_CONFIG_REQUEST : get_config_request,
OFPT_GET_CONFIG_REPLY : get_config_reply,
OFPT_SET_CONFIG : set_config,
OFPT_PACKET_IN : packet_in,
OFPT_FLOW_REMOVED : flow_removed,
OFPT_PORT_STATUS : port_status,
OFPT_PACKET_OUT : packet_out,
OFPT_FLOW_MOD : flow_mod,
OFPT_PORT_MOD : port_mod,
OFPT_STATS_REQUEST : stats_request,
OFPT_STATS_REPLY : stats_reply,
OFPT_BARRIER_REQUEST : barrier_request,
OFPT_BARRIER_REPLY : barrier_reply,
OFPT_QUEUE_GET_CONFIG_REQUEST : queue_get_config_request,
OFPT_QUEUE_GET_CONFIG_REPLY : queue_get_config_reply
}
def _of_message_to_object(binary_string):
"""
Map a binary string to the corresponding class.
Appropriately resolves subclasses
"""
hdr = ofp_header()
hdr.unpack(binary_string)
logging.info(hdr.show())
# FIXME: Add error detection
if not hdr.type in msg_type_subclassed:
return msg_type_to_class_map[hdr.type]()
if hdr.type == OFPT_STATS_REQUEST:
sub_hdr = ofp_stats_request()
sub_hdr.unpack(binary_string)
try:
obj = stats_request_to_class_map[sub_hdr.stats_type]()
except KeyError:
obj = None
return obj
elif hdr.type == OFPT_STATS_REPLY:
sub_hdr = ofp_stats_reply()
sub_hdr.unpack(binary_string)
try:
obj = stats_reply_to_class_map[sub_hdr.stats_type]()
except KeyError:
obj = None
return obj
elif hdr.type == OFPT_ERROR:
sub_hdr = ofp_error_msg()
sub_hdr.unpack(binary_string)
return error_to_class_map[sub_hdr.err_type]()
else:
logging.error("Cannot parse pkt to message")
return None
def parse_message(binary_string):
"""
Parse an OpenFlow packet
Parses a raw OpenFlow packet into a Python class, with class
members fully populated.
@param binary_string The packet (string) to be parsed
@param raw If true, interpret the packet as an L2 packet. Not
yet supported.
@return An object of some message class or None if fails
Note that any data beyond that parsed is not returned
"""
obj = _of_message_to_object(binary_string)
if obj:
obj.unpack(binary_string)
return obj
def parse_header(binary_string):
"""
Parse only the header from an OpenFlow packet
Parses the header from a raw OpenFlow packet into a
an ofp_header Python class.
@param binary_string The packet (string) to be parsed
@return Tuple of (verison, type, length, xid)
"""
return struct.unpack_from("!BBHL", binary_string)