Ting Xu | ce4b645 | 2022-04-24 06:14:25 +0000 | [diff] [blame] | 1 | # Copyright (c) 2022 Intel and/or its affiliates. |
| 2 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 3 | # you may not use this file except in compliance with the License. |
| 4 | # You may obtain a copy of the License at: |
| 5 | # |
| 6 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 7 | # |
| 8 | # Unless required by applicable law or agreed to in writing, software |
| 9 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 10 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 11 | # See the License for the specific language governing permissions and |
| 12 | # limitations under the License. |
| 13 | |
| 14 | from vpp_papi.vpp_papi import VppEnum |
| 15 | from ParseGraph import * |
| 16 | from Path import * |
| 17 | import json |
| 18 | import re |
| 19 | import os |
| 20 | |
| 21 | parsegraph_path = os.getcwd() + "/parsegraph" |
| 22 | |
| 23 | |
| 24 | def Forge(pattern, actions, file_flag): |
| 25 | pg = ParseGraph.Create(parsegraph_path) |
| 26 | if pg == None: |
| 27 | print("error: create parsegraph failed") |
| 28 | return None |
| 29 | |
| 30 | if not file_flag: |
| 31 | token = ParsePattern(pattern) |
| 32 | if token == None: |
| 33 | return None |
| 34 | else: |
| 35 | if not os.path.exists(pattern): |
| 36 | print("error: file not exist '%s' " % (pattern)) |
| 37 | return |
| 38 | f = open(pattern, "r", encoding="utf-8") |
| 39 | token = json.load(f) |
| 40 | if "actions" in token: |
| 41 | actions = token["actions"] |
| 42 | |
| 43 | path = Path.Create(token) |
| 44 | if path == None: |
| 45 | print("error: path not exit") |
| 46 | return None |
| 47 | |
| 48 | result = pg.Forge(path) |
| 49 | if result == None: |
| 50 | print("error: result not available") |
| 51 | return None |
| 52 | |
| 53 | spec, mask = GetBinary(result.ToJSON()) |
| 54 | |
| 55 | # create generic flow |
| 56 | my_flow = { |
| 57 | "type": VppEnum.vl_api_flow_type_v2_t.FLOW_TYPE_GENERIC_V2, |
| 58 | "flow": { |
| 59 | "generic": { |
| 60 | "pattern": {"spec": bytes(spec.encode()), "mask": bytes(mask.encode())} |
| 61 | } |
| 62 | }, |
| 63 | } |
| 64 | |
| 65 | # update actions entry |
| 66 | my_flow = GetAction(actions, my_flow) |
| 67 | |
| 68 | return my_flow |
| 69 | |
| 70 | |
| 71 | def GetAction(actions, flow): |
| 72 | if len(actions.split(" ")) > 1: |
| 73 | type = actions.split(" ")[0] |
| 74 | else: |
| 75 | type = actions |
| 76 | |
| 77 | if type == "mark": |
| 78 | flow.update( |
| 79 | { |
| 80 | "actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_MARK_V2, |
| 81 | "mark_flow_id": int(actions.split(" ")[1]), |
| 82 | } |
| 83 | ) |
| 84 | elif type == "next-node": |
| 85 | flow.update( |
| 86 | { |
| 87 | "actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_REDIRECT_TO_NODE_V2, |
| 88 | "redirect_node_index": int(actions.split(" ")[1]), |
| 89 | } |
| 90 | ) |
| 91 | elif type == "buffer-advance": |
| 92 | flow.update( |
| 93 | { |
| 94 | "actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_BUFFER_ADVANCE_V2, |
| 95 | "buffer_advance": int(actions.split(" ")[1]), |
| 96 | } |
| 97 | ) |
| 98 | elif type == "redirect-to-queue": |
| 99 | flow.update( |
| 100 | { |
| 101 | "actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_REDIRECT_TO_QUEUE_V2, |
| 102 | "redirect_queue": int(actions.split(" ")[1]), |
| 103 | } |
| 104 | ) |
| 105 | elif type == "rss": |
| 106 | flow.update({"actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_RSS_V2}) |
| 107 | elif type == "rss-queues": |
| 108 | queue_end = int(actions.split(" ")[-1]) |
| 109 | queue_start = int(actions.split(" ")[-3]) |
| 110 | flow.update( |
| 111 | { |
| 112 | "actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_RSS_V2, |
| 113 | "queue_index": queue_start, |
| 114 | "queue_num": queue_end - queue_start + 1, |
| 115 | } |
| 116 | ) |
| 117 | elif type == "drop": |
| 118 | flow.update({"actions": VppEnum.vl_api_flow_action_v2_t.FLOW_ACTION_DROP_V2}) |
| 119 | |
| 120 | return flow |
| 121 | |
| 122 | |
| 123 | def GetBinary(flow_info): |
| 124 | spec = "".join(flow_info["Packet"]) |
| 125 | mask = "".join(flow_info["Mask"]) |
| 126 | return spec, mask |
| 127 | |
| 128 | |
| 129 | def ParseFields(item): |
| 130 | # get protocol name |
| 131 | prot = item.split("(")[0] |
| 132 | stack = {"header": prot} |
| 133 | # get fields contents |
| 134 | fields = re.findall(r"[(](.*?)[)]", item) |
| 135 | if not fields: |
| 136 | print("error: invalid pattern") |
| 137 | return None |
| 138 | if fields == [""]: |
| 139 | return stack |
| 140 | stack.update({"fields": []}) |
| 141 | return ParseStack(stack, fields[0].split(",")) |
| 142 | |
| 143 | |
| 144 | def GetMask(item): |
| 145 | if "format" in item: |
| 146 | format = item["format"] |
| 147 | if format == "mac": |
Ting Xu | c9d916c | 2022-09-29 13:50:55 +0800 | [diff] [blame^] | 148 | mask = "ff:ff:ff:ff:ff:ff" |
Ting Xu | ce4b645 | 2022-04-24 06:14:25 +0000 | [diff] [blame] | 149 | elif format == "ipv4": |
| 150 | mask = "255.255.255.255" |
| 151 | elif format == "ipv6": |
| 152 | mask = "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff" |
| 153 | return mask |
| 154 | if "size" in item: |
| 155 | mask = str((1 << int(item["size"])) - 1) |
| 156 | else: |
| 157 | print("mask error") |
| 158 | return mask |
| 159 | |
| 160 | |
| 161 | # parse protocol headers and its fields. Available fields are defined in corresponding nodes. |
| 162 | def ParseStack(stack, fields): |
| 163 | prot = stack["header"] |
| 164 | node_path = parsegraph_path + "/nodes/" + prot + ".json" |
| 165 | if not os.path.exists(node_path): |
| 166 | print("error file not exist '%s' " % (node_path)) |
| 167 | return None |
| 168 | f = open(node_path, "r", encoding="utf-8") |
| 169 | nodeinfo = json.load(f) |
| 170 | for field in fields: |
| 171 | fld_name = field.split("=")[0].strip() |
| 172 | fld_value = ( |
| 173 | field.split("=")[-1].strip() if (len(field.split("=")) >= 2) else None |
| 174 | ) |
| 175 | for item in nodeinfo["layout"]: |
| 176 | if fld_name == item["name"]: |
| 177 | mask = GetMask(item) |
| 178 | stack["fields"].append( |
| 179 | {"name": fld_name, "value": fld_value, "mask": mask} |
| 180 | ) |
| 181 | break |
| 182 | if not stack["fields"]: |
| 183 | print("warning: invalid field '%s'" % (fld_name)) |
| 184 | return None |
| 185 | |
| 186 | return stack |
| 187 | |
| 188 | |
| 189 | def ParsePattern(pattern): |
| 190 | # create json template |
| 191 | json_tmp = {"type": "path", "stack": []} |
| 192 | |
| 193 | items = pattern.split("/") |
| 194 | for item in items: |
| 195 | stack = ParseFields(item) |
| 196 | if stack == None: |
| 197 | return None |
| 198 | json_tmp["stack"].append(stack) |
| 199 | |
| 200 | return json_tmp |