blob: 83dc85084c4e7de423aa3bcbf7bb642daf072c29 [file] [log] [blame]
"""
OpenFlow actions list class
"""
from action import *
from ofp import ofp_header
# # Map OFP action identifiers to the actual structures used on the wire
# action_object_map = {
# OFPAT_OUTPUT : ofp_action_output,
# OFPAT_SET_VLAN_VID : ofp_action_vlan_vid,
# OFPAT_SET_VLAN_PCP : ofp_action_vlan_pcp,
# OFPAT_STRIP_VLAN : ofp_action_header,
# OFPAT_SET_DL_SRC : ofp_action_dl_addr,
# OFPAT_SET_DL_DST : ofp_action_dl_addr,
# OFPAT_SET_NW_SRC : ofp_action_nw_addr,
# OFPAT_SET_NW_DST : ofp_action_nw_addr,
# OFPAT_SET_NW_TOS : ofp_action_nw_tos,
# OFPAT_SET_TP_SRC : ofp_action_tp_port,
# OFPAT_SET_TP_DST : ofp_action_tp_port,
# OFPAT_ENQUEUE : ofp_action_enqueue
# }
# For debugging
action_object_map = {
OFPAT_OUTPUT : action_output,
OFPAT_SET_VLAN_VID : action_set_vlan_vid,
OFPAT_SET_VLAN_PCP : action_set_vlan_pcp,
OFPAT_STRIP_VLAN : action_strip_vlan,
OFPAT_SET_DL_SRC : action_set_dl_src,
OFPAT_SET_DL_DST : action_set_dl_dst,
OFPAT_SET_NW_SRC : action_set_nw_src,
OFPAT_SET_NW_DST : action_set_nw_dst,
OFPAT_SET_NW_TOS : action_set_nw_tos,
OFPAT_SET_TP_SRC : action_set_tp_src,
OFPAT_SET_TP_DST : action_set_tp_dst,
OFPAT_ENQUEUE : action_enqueue
}
class action_list(object):
"""
Maintain a list of actions
Data members:
@arg actions: An array of action objects such as action_output, etc.
Methods:
@arg pack: Pack the structure into a string
@arg unpack: Unpack a string to objects, with proper typing
@arg add: Add an action to the list; you can directly access
the action member, but add will validate that the added object
is an action.
"""
def __init__(self):
self.actions = []
def pack(self):
"""
Pack a list of actions
Returns the packed string
"""
packed = ""
for act in self.actions:
packed += act.pack()
return packed
def unpack(self, binary_string, bytes=None):
"""
Unpack a list of actions
Unpack actions from a binary string, creating an array
of objects of the appropriate type
@param binary_string The string to be unpacked
@param bytes The total length of the action list in bytes.
Ignored if decode is True. If None and decode is false, the
list is assumed to extend through the entire string.
@return The remainder of binary_string that was not parsed
"""
if bytes == None:
bytes = len(binary_string)
bytes_done = 0
count = 0
cur_string = binary_string
while bytes_done < bytes:
hdr = ofp_action_header()
hdr.unpack(cur_string)
if not hdr.type in action_object_map.keys():
print "WARNING: Skipping unknown action ", hdr.type
else:
print "DEBUG: Found action of type ", hdr.type
self.actions.append(action_object_map[hdr.type]())
self.actions[count].unpack(binary_string)
count += 1
cur_string = cur_string[hdr.len:]
bytes_done += hdr.len
return cur_string
def add(self, action):
"""
Add an action to an action list
@param action The action to add
@return True if successful, False if not an action object
"""
if isinstance(action, action_class_list):
self.actions.append(action)
return True
return False
def __len__(self):
length = 0
for act in self.actions:
length += act.__len__()
return length
def __eq__(self, other):
if type(self) != type(other): return False
if self.actions != other.actions: return False
return True
def __ne__(self, other): return not self.__eq__(other)
def show(self, prefix=''):
print prefix + "Action List with " + str(len(self.actions)) + \
" actions"
count = 0
for obj in self.actions:
count += 1
print " Action " + str(count) + ": "
obj.show(prefix + ' ')