From dc5e461ae4a9ee48f487b6d8fb6e81d1ad4a91de Mon Sep 17 00:00:00 2001 From: Johann Wagner Date: Tue, 10 Dec 2024 10:55:18 +0100 Subject: [PATCH 1/2] Added serialization error for invalid loopnacks --- cosmo/serializer.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/cosmo/serializer.py b/cosmo/serializer.py index fe6ee08..d379e1b 100644 --- a/cosmo/serializer.py +++ b/cosmo/serializer.py @@ -168,7 +168,7 @@ def _get_vrf_rib(self, routes, vrf): rib[table]["static"][route["prefix"]["prefix"]] = r return rib - def _get_unit(self, iface): + def _get_unit(self, iface, is_loopback=False): unit_stub = {} name = iface['name'].split(".")[1] @@ -190,6 +190,10 @@ def _get_unit(self, iface): if not iface["vrf"] and not ipa.is_global and not is_mgmt and not self.cfg.allow_private_ips: raise InterfaceSerializationError(f"Private IP {ipa} used on interface {iface['name']} in default VRF for device {self.device['name']}. Did you forget to configure a VRF?") + # We only want /32 on loopback interfaces. + if is_loopback and not ipa.network.prefixlen == ipa.max_prefixlen: + raise InterfaceSerializationError(f"IP {ipa} is not a valid loopback IP address.") + if ipa.version == 4: ipv4Family[ipa.network].add_ip(ipa, is_secondary) elif ipa.version == 6: @@ -479,7 +483,7 @@ def serialize(self): if not si['enabled']: continue - name, unit = self._get_unit(si) + name, unit = self._get_unit(si, is_loopback=is_loopback) sub_num = int(name or '0') if not is_loopback and sub_num != 0 and not unit.get("vlan", None): warnings.warn(f"Sub interface {si['name']} does not have a access VLAN configured, skipping...") From a28301381167f0b781b1288f24a47d0108aa6f04 Mon Sep 17 00:00:00 2001 From: Johann Wagner Date: Tue, 10 Dec 2024 11:15:11 +0100 Subject: [PATCH 2/2] Fixed up test_case_ips case and fixed loopbacks for all test cases --- cosmo/tests/test_case_1.yaml | 4 +-- cosmo/tests/test_case_2.yaml | 4 +-- cosmo/tests/test_case_bgpcpe.yml | 4 +-- cosmo/tests/test_case_fec.yaml | 4 +-- cosmo/tests/test_case_ips.yaml | 28 +++------------------ cosmo/tests/test_case_l2x_err_template.yaml | 2 +- cosmo/tests/test_case_l3vpn.yml | 6 ++--- cosmo/tests/test_case_lag.yaml | 4 +-- cosmo/tests/test_case_local_l2x.yaml | 2 +- cosmo/tests/test_case_mpls_evpn.yaml | 8 +++--- cosmo/tests/test_case_no_manuf_slug.yaml | 2 +- cosmo/tests/test_case_vendor_unknown.yaml | 2 +- cosmo/tests/test_case_vpws.yaml | 6 ++--- cosmo/tests/test_case_vrf_staticroute.yaml | 2 +- cosmo/tests/test_serializer.py | 23 +++++++++-------- 15 files changed, 41 insertions(+), 60 deletions(-) diff --git a/cosmo/tests/test_case_1.yaml b/cosmo/tests/test_case_1.yaml index 651cb19..121d2d8 100644 --- a/cosmo/tests/test_case_1.yaml +++ b/cosmo/tests/test_case_1.yaml @@ -48,11 +48,11 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [] l2vpn_list: [] vrf_list: [] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_2.yaml b/cosmo/tests/test_case_2.yaml index 1f33431..4b86136 100644 --- a/cosmo/tests/test_case_2.yaml +++ b/cosmo/tests/test_case_2.yaml @@ -74,11 +74,11 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [] l2vpn_list: [] vrf_list: [] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_bgpcpe.yml b/cosmo/tests/test_case_bgpcpe.yml index 687aa6c..554bdc2 100644 --- a/cosmo/tests/test_case_bgpcpe.yml +++ b/cosmo/tests/test_case_bgpcpe.yml @@ -122,7 +122,7 @@ device_list: enabled: true id: '192047' ip_addresses: - - address: 45.139.136.10/24 + - address: 45.139.136.10/32 lag: null mac_address: null mode: null @@ -155,4 +155,4 @@ vrf_list: rd: null loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_fec.yaml b/cosmo/tests/test_case_fec.yaml index 87c4a71..183354e 100644 --- a/cosmo/tests/test_case_fec.yaml +++ b/cosmo/tests/test_case_fec.yaml @@ -66,11 +66,11 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [ ] l2vpn_list: [ ] vrf_list: [ ] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_ips.yaml b/cosmo/tests/test_case_ips.yaml index b393532..8dd5c5d 100644 --- a/cosmo/tests/test_case_ips.yaml +++ b/cosmo/tests/test_case_ips.yaml @@ -3,26 +3,6 @@ device_list: slug: mx204 id: '1746' interfaces: - - custom_fields: - bpdufilter: false - bpduguard: false - inner_tag: null - ipv6_ra: false - outer_tag: null - description: '' - enabled: true - id: '3646' - ip_addresses: [ ] - lag: null - mac_address: null - mode: null - mtu: null - name: lo-0/0/0 - tagged_vlans: [ ] - tags: [ ] - type: LOOPBACK - untagged_vlan: null - vrf: null - custom_fields: inner_tag: null outer_tag: null @@ -40,7 +20,7 @@ device_list: mac_address: 94:BF:41:41:41:F3 mode: null mtu: null - name: lo-0/0/0.14 + name: ifp-0/0/2.0 tagged_vlans: [ ] tags: [ ] type: A_100GBASE_X_QSFP28 @@ -63,7 +43,7 @@ device_list: mac_address: 94:BF:41:41:41:F4 mode: null mtu: null - name: lo-0/0/0.16 + name: ifp-0/0/3.0 tagged_vlans: [ ] tags: [ ] type: A_100GBASE_X_QSFP28 @@ -75,11 +55,11 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [ ] l2vpn_list: [ ] vrf_list: [ ] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_l2x_err_template.yaml b/cosmo/tests/test_case_l2x_err_template.yaml index c4e9c99..6464a2a 100644 --- a/cosmo/tests/test_case_l2x_err_template.yaml +++ b/cosmo/tests/test_case_l2x_err_template.yaml @@ -16,4 +16,4 @@ l2vpn_list: [] vrf_list: [] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_l3vpn.yml b/cosmo/tests/test_case_l3vpn.yml index d113eca..2a79e95 100644 --- a/cosmo/tests/test_case_l3vpn.yml +++ b/cosmo/tests/test_case_l3vpn.yml @@ -81,7 +81,7 @@ device_list: enabled: true id: '192047' ip_addresses: - - address: 45.139.136.10/24 + - address: 45.139.136.10/32 lag: null mac_address: null mode: null @@ -121,9 +121,9 @@ loopback_interface_list: - name: "lo0.0" vrf: null ip_addresses: - - address: 45.139.136.10/24 + - address: 45.139.136.10/32 family: value: 4 loopbacks: TEST0001: - ipv4: 45.139.136.10/24 + ipv4: 45.139.136.10/32 diff --git a/cosmo/tests/test_case_lag.yaml b/cosmo/tests/test_case_lag.yaml index 574e331..1f53949 100644 --- a/cosmo/tests/test_case_lag.yaml +++ b/cosmo/tests/test_case_lag.yaml @@ -66,11 +66,11 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [ ] l2vpn_list: [ ] vrf_list: [ ] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_local_l2x.yaml b/cosmo/tests/test_case_local_l2x.yaml index cc3843e..a396e2b 100644 --- a/cosmo/tests/test_case_local_l2x.yaml +++ b/cosmo/tests/test_case_local_l2x.yaml @@ -109,4 +109,4 @@ l2vpn_list: vrf_list: [] loopbacks: TEST0001: - ipv4: 45.139.136.10/24 \ No newline at end of file + ipv4: 45.139.136.10/32 \ No newline at end of file diff --git a/cosmo/tests/test_case_mpls_evpn.yaml b/cosmo/tests/test_case_mpls_evpn.yaml index e5b87ea..11c456f 100644 --- a/cosmo/tests/test_case_mpls_evpn.yaml +++ b/cosmo/tests/test_case_mpls_evpn.yaml @@ -90,7 +90,7 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.11/24 + address: 45.139.136.11/32 serial: CT702 staticroute_set: [] - device_type: @@ -184,7 +184,7 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: '' staticroute_set: [] l2vpn_list: @@ -211,6 +211,6 @@ l2vpn_list: vrf_list: [] loopbacks: TEST0001: - ipv4: "45.139.136.11/24" + ipv4: "45.139.136.11/32" TEST0002: - ipv4: "45.139.136.10/24" + ipv4: "45.139.136.10/32" diff --git a/cosmo/tests/test_case_no_manuf_slug.yaml b/cosmo/tests/test_case_no_manuf_slug.yaml index 67e761a..15fe92e 100644 --- a/cosmo/tests/test_case_no_manuf_slug.yaml +++ b/cosmo/tests/test_case_no_manuf_slug.yaml @@ -9,7 +9,7 @@ device_list: raw: ACME raw: coyote-0-2-1-rc1 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [] l2vpn_list: [] diff --git a/cosmo/tests/test_case_vendor_unknown.yaml b/cosmo/tests/test_case_vendor_unknown.yaml index 8249462..02ebae0 100644 --- a/cosmo/tests/test_case_vendor_unknown.yaml +++ b/cosmo/tests/test_case_vendor_unknown.yaml @@ -9,7 +9,7 @@ device_list: slug: ACME slug: coyote-0-2-1-rc1 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: 4242 staticroute_set: [] l2vpn_list: [] diff --git a/cosmo/tests/test_case_vpws.yaml b/cosmo/tests/test_case_vpws.yaml index 84b150a..7cb76bd 100644 --- a/cosmo/tests/test_case_vpws.yaml +++ b/cosmo/tests/test_case_vpws.yaml @@ -106,7 +106,7 @@ device_list: slug: juniper slug: junos-21-4r3-s5-4 primary_ip4: - address: 45.139.136.10/24 + address: 45.139.136.10/32 serial: '' staticroute_set: [] l2vpn_list: @@ -132,6 +132,6 @@ l2vpn_list: vrf_list: [] loopbacks: TEST0002: - ipv4: 45.139.136.10/24 + ipv4: 45.139.136.10/32 TEST0001: - ipv4: 45.139.136.11/24 + ipv4: 45.139.136.11/32 diff --git a/cosmo/tests/test_case_vrf_staticroute.yaml b/cosmo/tests/test_case_vrf_staticroute.yaml index 3947dff..f82a02a 100644 --- a/cosmo/tests/test_case_vrf_staticroute.yaml +++ b/cosmo/tests/test_case_vrf_staticroute.yaml @@ -139,4 +139,4 @@ vrf_list: rd: null loopbacks: TEST0001: - ipv4: 45.139.136.10/24 + ipv4: 45.139.136.10/32 diff --git a/cosmo/tests/test_serializer.py b/cosmo/tests/test_serializer.py index f3f5f24..90b08ed 100644 --- a/cosmo/tests/test_serializer.py +++ b/cosmo/tests/test_serializer.py @@ -189,20 +189,21 @@ def test_router_vrf_rib(): def test_router_ips(): [sd] = get_router_sd_from_path("./test_case_ips.yaml") - assert 'lo-0/0/0' in sd['interfaces'] - assert 14 in sd['interfaces']['lo-0/0/0']['units'] - assert 16 in sd['interfaces']['lo-0/0/0']['units'] + assert 'ifp-0/0/2' in sd['interfaces'] + assert 'ifp-0/0/3' in sd['interfaces'] + assert 0 in sd['interfaces']['ifp-0/0/2']['units'] + assert 0 in sd['interfaces']['ifp-0/0/3']['units'] - unit_14 = sd['interfaces']['lo-0/0/0']['units'][14] - unit_16 = sd['interfaces']['lo-0/0/0']['units'][16] + unit_v4 = sd['interfaces']['ifp-0/0/2']['units'][0] + unit_v6 = sd['interfaces']['ifp-0/0/3']['units'][0] - assert unit_14['families']['inet']['address']['45.139.138.1/29'] == {} - assert unit_14['families']['inet']['address']['45.139.138.8/29'] == {"primary": True} - assert unit_14['families']['inet']['address']['45.139.138.9/29'] == {"secondary": True} + assert unit_v4['families']['inet']['address']['45.139.138.1/29'] == {} + assert unit_v4['families']['inet']['address']['45.139.138.8/29'] == {"primary": True} + assert unit_v4['families']['inet']['address']['45.139.138.9/29'] == {"secondary": True} - assert unit_16['families']['inet6']['address']['2a0e:b941:2::/122'] == {} - assert unit_16['families']['inet6']['address']['2a0e:b941:2::40/122'] == {"primary": True} - assert unit_16['families']['inet6']['address']['2a0e:b941:2::41/122'] == {"secondary": True} + assert unit_v6['families']['inet6']['address']['2a0e:b941:2::/122'] == {} + assert unit_v6['families']['inet6']['address']['2a0e:b941:2::40/122'] == {"primary": True} + assert unit_v6['families']['inet6']['address']['2a0e:b941:2::41/122'] == {"secondary": True} def test_router_case_mpls_evpn():