mirror of
https://git.proxmox.com/git/mirror_frr
synced 2025-08-12 12:32:41 +00:00
Merge pull request #16374 from opensourcerouting/fix/bgpd_import_vrf_multiple_instances
bgpd: Mark VRF instance as auto created if import vrf is configured for this instance
This commit is contained in:
commit
05b65b46cb
@ -1701,6 +1701,10 @@ DEFUN (no_router_bgp,
|
|||||||
for (ALL_LIST_ELEMENTS_RO(bm->bgp, node, tmp_bgp)) {
|
for (ALL_LIST_ELEMENTS_RO(bm->bgp, node, tmp_bgp)) {
|
||||||
if (tmp_bgp->inst_type != BGP_INSTANCE_TYPE_VRF)
|
if (tmp_bgp->inst_type != BGP_INSTANCE_TYPE_VRF)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (CHECK_FLAG(tmp_bgp->vrf_flags, BGP_VRF_AUTO))
|
||||||
|
continue;
|
||||||
|
|
||||||
if (CHECK_FLAG(
|
if (CHECK_FLAG(
|
||||||
tmp_bgp->af_flags[AFI_IP]
|
tmp_bgp->af_flags[AFI_IP]
|
||||||
[SAFI_UNICAST],
|
[SAFI_UNICAST],
|
||||||
@ -10567,12 +10571,20 @@ DEFPY(bgp_imexport_vrf, bgp_imexport_vrf_cmd,
|
|||||||
|
|
||||||
vrf_bgp = bgp_lookup_by_name(import_name);
|
vrf_bgp = bgp_lookup_by_name(import_name);
|
||||||
if (!vrf_bgp) {
|
if (!vrf_bgp) {
|
||||||
if (strcmp(import_name, VRF_DEFAULT_NAME) == 0)
|
if (strcmp(import_name, VRF_DEFAULT_NAME) == 0) {
|
||||||
vrf_bgp = bgp_default;
|
vrf_bgp = bgp_default;
|
||||||
else
|
} else {
|
||||||
/* Auto-create assuming the same AS */
|
/* Auto-create assuming the same AS */
|
||||||
ret = bgp_get_vty(&vrf_bgp, &as, import_name, bgp_type,
|
ret = bgp_get_vty(&vrf_bgp, &as, import_name, bgp_type,
|
||||||
NULL, ASNOTATION_UNDEFINED);
|
NULL, ASNOTATION_UNDEFINED);
|
||||||
|
|
||||||
|
/* Auto created VRF instances should be marked
|
||||||
|
* properly, otherwise we have a state after bgpd
|
||||||
|
* restart where VRF instance has default VRF's ASN.
|
||||||
|
*/
|
||||||
|
SET_FLAG(vrf_bgp->vrf_flags, BGP_VRF_AUTO);
|
||||||
|
}
|
||||||
|
|
||||||
if (ret) {
|
if (ret) {
|
||||||
vty_out(vty,
|
vty_out(vty,
|
||||||
"VRF %s is not configured as a bgp instance\n",
|
"VRF %s is not configured as a bgp instance\n",
|
||||||
@ -12745,6 +12757,9 @@ static void bgp_show_all_instances_summary_vty(struct vty *vty, afi_t afi,
|
|||||||
vty_out(vty, "{\n");
|
vty_out(vty, "{\n");
|
||||||
|
|
||||||
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
||||||
|
if (CHECK_FLAG(bgp->vrf_flags, BGP_VRF_AUTO))
|
||||||
|
continue;
|
||||||
|
|
||||||
nbr_output = true;
|
nbr_output = true;
|
||||||
if (use_json) {
|
if (use_json) {
|
||||||
if (!is_first)
|
if (!is_first)
|
||||||
@ -16130,6 +16145,9 @@ static void bgp_show_all_instances_neighbors_vty(struct vty *vty,
|
|||||||
vty_out(vty, "{\n");
|
vty_out(vty, "{\n");
|
||||||
|
|
||||||
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
||||||
|
if (CHECK_FLAG(bgp->vrf_flags, BGP_VRF_AUTO))
|
||||||
|
continue;
|
||||||
|
|
||||||
nbr_output = true;
|
nbr_output = true;
|
||||||
if (use_json) {
|
if (use_json) {
|
||||||
if (!(json = json_object_new_object())) {
|
if (!(json = json_object_new_object())) {
|
||||||
@ -16689,6 +16707,9 @@ static int bgp_show_all_instance_route_leak_vty(struct vty *vty, afi_t afi,
|
|||||||
if (bgp->inst_type != BGP_INSTANCE_TYPE_DEFAULT)
|
if (bgp->inst_type != BGP_INSTANCE_TYPE_DEFAULT)
|
||||||
vrf_name = bgp->name;
|
vrf_name = bgp->name;
|
||||||
|
|
||||||
|
if (CHECK_FLAG(bgp->vrf_flags, BGP_VRF_AUTO))
|
||||||
|
continue;
|
||||||
|
|
||||||
if (use_json) {
|
if (use_json) {
|
||||||
json_vrf = json_object_new_object();
|
json_vrf = json_object_new_object();
|
||||||
} else {
|
} else {
|
||||||
@ -16779,6 +16800,9 @@ static void bgp_show_all_instances_updgrps_vty(struct vty *vty, afi_t afi,
|
|||||||
struct bgp *bgp;
|
struct bgp *bgp;
|
||||||
|
|
||||||
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
for (ALL_LIST_ELEMENTS(bm->bgp, node, nnode, bgp)) {
|
||||||
|
if (CHECK_FLAG(bgp->vrf_flags, BGP_VRF_AUTO))
|
||||||
|
continue;
|
||||||
|
|
||||||
if (!uj)
|
if (!uj)
|
||||||
vty_out(vty, "\nInstance %s:\n",
|
vty_out(vty, "\nInstance %s:\n",
|
||||||
(bgp->inst_type == BGP_INSTANCE_TYPE_DEFAULT)
|
(bgp->inst_type == BGP_INSTANCE_TYPE_DEFAULT)
|
||||||
|
0
tests/topotests/bgp_vrf_different_asn/__init__.py
Normal file
0
tests/topotests/bgp_vrf_different_asn/__init__.py
Normal file
18
tests/topotests/bgp_vrf_different_asn/r1/frr.conf
Normal file
18
tests/topotests/bgp_vrf_different_asn/r1/frr.conf
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
!
|
||||||
|
vrf vrf100
|
||||||
|
vni 10100
|
||||||
|
exit-vrf
|
||||||
|
!
|
||||||
|
interface r1-eth0 vrf vrf100
|
||||||
|
ip address 192.168.1.1/24
|
||||||
|
!
|
||||||
|
router bgp 65000
|
||||||
|
address-family ipv4 unicast
|
||||||
|
import vrf vrf100
|
||||||
|
exit-address-family
|
||||||
|
!
|
||||||
|
router bgp 65100 vrf vrf100
|
||||||
|
address-family ipv4 unicast
|
||||||
|
redistribute connected
|
||||||
|
exit-address-family
|
||||||
|
!
|
@ -0,0 +1,107 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# SPDX-License-Identifier: ISC
|
||||||
|
|
||||||
|
#
|
||||||
|
# Copyright (c) 2024 by
|
||||||
|
# Donatas Abraitis <donatas@opensourcerouting.org>
|
||||||
|
#
|
||||||
|
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import json
|
||||||
|
import pytest
|
||||||
|
import functools
|
||||||
|
|
||||||
|
CWD = os.path.dirname(os.path.realpath(__file__))
|
||||||
|
sys.path.append(os.path.join(CWD, "../"))
|
||||||
|
|
||||||
|
# pylint: disable=C0413
|
||||||
|
from lib import topotest
|
||||||
|
from lib.topogen import Topogen, TopoRouter, get_topogen
|
||||||
|
|
||||||
|
pytestmark = [pytest.mark.bgpd]
|
||||||
|
|
||||||
|
|
||||||
|
def build_topo(tgen):
|
||||||
|
for routern in range(1, 2):
|
||||||
|
tgen.add_router("r{}".format(routern))
|
||||||
|
|
||||||
|
switch = tgen.add_switch("s1")
|
||||||
|
switch.add_link(tgen.gears["r1"])
|
||||||
|
|
||||||
|
|
||||||
|
def setup_module(mod):
|
||||||
|
tgen = Topogen(build_topo, mod.__name__)
|
||||||
|
tgen.start_topology()
|
||||||
|
|
||||||
|
r1 = tgen.gears["r1"]
|
||||||
|
r1.run("ip link add vrf100 type vrf table 1001")
|
||||||
|
r1.run("ip link set up dev vrf100")
|
||||||
|
r1.run("ip link set r1-eth0 master vrf100")
|
||||||
|
|
||||||
|
router_list = tgen.routers()
|
||||||
|
|
||||||
|
for _, (rname, router) in enumerate(router_list.items(), 1):
|
||||||
|
router.load_frr_config(os.path.join(CWD, "{}/frr.conf".format(rname)))
|
||||||
|
|
||||||
|
tgen.start_router()
|
||||||
|
|
||||||
|
|
||||||
|
def teardown_module(mod):
|
||||||
|
tgen = get_topogen()
|
||||||
|
tgen.stop_topology()
|
||||||
|
|
||||||
|
|
||||||
|
def test_bgp_vrf_different_asn():
|
||||||
|
tgen = get_topogen()
|
||||||
|
|
||||||
|
if tgen.routers_have_failure():
|
||||||
|
pytest.skip(tgen.errors)
|
||||||
|
|
||||||
|
def _bgp_check_instances():
|
||||||
|
output = json.loads(tgen.gears["r1"].vtysh_cmd("show bgp vrf all json"))
|
||||||
|
expected = {
|
||||||
|
"default": {
|
||||||
|
"vrfName": "default",
|
||||||
|
"localAS": 65000,
|
||||||
|
},
|
||||||
|
"vrf100": {
|
||||||
|
"vrfName": "vrf100",
|
||||||
|
"localAS": 65100,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
return topotest.json_cmp(output, expected)
|
||||||
|
|
||||||
|
test_func = functools.partial(_bgp_check_instances)
|
||||||
|
_, result = topotest.run_and_expect(test_func, None, count=30, wait=1)
|
||||||
|
assert result is None, "Can't see vrf100 to be under 65100 ASN"
|
||||||
|
|
||||||
|
def _bgp_check_imported_route():
|
||||||
|
output = json.loads(
|
||||||
|
tgen.gears["r1"].vtysh_cmd("show ip route 192.168.1.0/24 json")
|
||||||
|
)
|
||||||
|
expected = {
|
||||||
|
"192.168.1.0/24": [
|
||||||
|
{
|
||||||
|
"installed": True,
|
||||||
|
"selected": True,
|
||||||
|
"nexthops": [
|
||||||
|
{
|
||||||
|
"interfaceName": "vrf100",
|
||||||
|
"vrf": "vrf100",
|
||||||
|
"active": True,
|
||||||
|
}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
return topotest.json_cmp(output, expected)
|
||||||
|
|
||||||
|
test_func = functools.partial(_bgp_check_imported_route)
|
||||||
|
_, result = topotest.run_and_expect(test_func, None, count=30, wait=1)
|
||||||
|
assert result is None, "Can't see 192.168.1.0/24 being imported into a default VRF"
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
args = ["-s"] + sys.argv[1:]
|
||||||
|
sys.exit(pytest.main(args))
|
Loading…
Reference in New Issue
Block a user