' +
label.replace("\r", " ") + " |
" + graph.create_dot() + "") response = flask.Response(graph.create_png(), mimetype='image/png') response.headers['Last-Modified'] = datetime.now() response.headers['Cache-Control'] = ('no-store, no-cache, ' 'must-revalidate, ' 'post-check=0, pre-check=0, ' 'max-age=0') response.headers['Pragma'] = 'no-cache' response.headers['Expires'] = '-1' return response def build_as_tree_from_raw_bird_ouput(host, proto, text): """Extract the as path from the raw bird "show route all" command""" path = None paths = [] net_dest = None for line in text: line = line.strip() expr = re.search(r'(.*)via\s+([0-9a-fA-F:\.]+)\s+on.*\[(\w+)\s+', line) if expr: if path: path.append(net_dest) paths.append(path) path = None if expr.group(1).strip(): net_dest = expr.group(1).strip() peer_ip = expr.group(2).strip() peer_protocol_name = expr.group(3).strip() # Check if via line is a internal route for rt_host, rt_ips in app.config["ROUTER_IP"].iteritems(): # Special case for internal routing if peer_ip in rt_ips: path = [rt_host] break else: # ugly hack for good printing path = [peer_protocol_name] # path = ["%s\r%s" % (peer_protocol_name, # get_as_name(get_as_number_from_protocol_name( # host, proto, peer_protocol_name)))] expr2 = re.search(r'(.*)unreachable\s+\[(\w+)\s+', line) if expr2: if path: path.append(net_dest) paths.append(path) path = None if expr2.group(1).strip(): net_dest = expr2.group(1).strip() if line.startswith("BGP.as_path:"): path.extend(line.replace("BGP.as_path:", "").strip().split(" ")) if path: path.append(net_dest) paths.append(path) return paths def show_route(request_type, hosts, proto): expression = get_query() if not expression: flask.abort(400) set_session(request_type, hosts, proto, expression) bgpmap = request_type.endswith("bgpmap") all = (request_type.endswith("detail") and " all" or "") if bgpmap: all = " all" if request_type.startswith("adv"): command = "show route " + expression.strip() if bgpmap and not command.endswith("all"): command = command + " all" elif request_type.startswith("where"): command = "show route where net ~ [ " + expression + " ]" + all else: mask = "" if len(expression.split("/")) == 2: expression, mask = (expression.split("/")) if not mask and proto == "ipv4": mask = "32" if not mask and proto == "ipv6": mask = "128" if not toolbox.mask_is_valid(mask): return error_page("mask %s is invalid" % mask) if proto == "ipv6" and not toolbox.ipv6_is_valid(expression): qtype = "AAAA" elif proto == "ipv4" and not toolbox.ipv4_is_valid(expression): qtype = "A" try: expression = toolbox.resolve(expression, qtype) except dns_exc.DNSException: return error_page("%s is unresolvable or invalid for %s" % (expression, proto)) if mask: expression += "/" + mask command = "show route for " + expression + all detail = {} errors = [] for host in hosts.split("+"): ret, res = bird_command(host, proto, command) res = res.split("\n") if ret is False: errors.append("%s" % res) continue if len(res) <= 1: errors.append("%s: bird command failed with error, %s" % (host, "\n".join(res))) continue if bgpmap: detail[host] = build_as_tree_from_raw_bird_ouput(host, proto, res) else: detail[host] = add_links(res) if bgpmap: detail = json.dumps(detail) return flask.render_template((bgpmap and 'bgpmap.html' or 'route.html'), detail=detail, command=command, expression=expression, errors=errors) if __name__ == "__main__": app.run(app.config.get("BIND_IP", "0.0.0.0"), app.config.get("BIND_PORT", 5000))