annet 3.13.0__tar.gz → 3.14.0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of annet might be problematic. Click here for more details.
- {annet-3.13.0/annet.egg-info → annet-3.14.0}/PKG-INFO +1 -1
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/devdb/data/devdb.json +2 -2
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/match_builder.py +10 -2
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/aspath.py +22 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/community.py +73 -1
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/entities.py +50 -3
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/policy.py +367 -2
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/prefix_lists.py +67 -1
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/rd.py +15 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/juniper.rul +6 -2
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/tabparser.py +1 -1
- {annet-3.13.0 → annet-3.14.0/annet.egg-info}/PKG-INFO +1 -1
- {annet-3.13.0 → annet-3.14.0}/AUTHORS +0 -0
- {annet-3.13.0 → annet-3.14.0}/LICENSE +0 -0
- {annet-3.13.0 → annet-3.14.0}/MANIFEST.in +0 -0
- {annet-3.13.0 → annet-3.14.0}/README.md +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/fetchers/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/fetchers/stub/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/fetchers/stub/fetcher.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/file/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/file/provider.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/adapter.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/client.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/manufacturer.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/query.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/status_client.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/storage_base.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/common/storage_opts.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/provider.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v24/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v24/models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v24/storage.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v37/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v37/models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v37/storage.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v41/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v41/models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v41/storage.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v42/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v42/models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/adapters/netbox/v42/storage.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annet.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/command.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/diff.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/errors.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/filter_acl.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/jsontools.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/lib.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/db.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/devdb/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/views/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/views/dump.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/netdev/views/hardware.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/output.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/patching.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/acl.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/deploying.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/ordering.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/platform.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rbparser/syntax.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rulebook/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/rulebook/common.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/annlib/types.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/api/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/argparse.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/bgp_models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/cli.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/cli_args.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/configs/context.yml +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/configs/logging.yaml +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/connectors.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/deploy.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/deploy_ui.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/diff.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/filtering.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/gen.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/base.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/common/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/common/initial.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/entire.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/exceptions.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/jsonfragment.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/partial.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/perf.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/ref.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/generators/result.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/hardware.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/implicit.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/lib.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/basemodel.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/device_models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/executor.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/match_args.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/models_converter.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/peer_models.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/port_processor.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/mesh/registry.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/output.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/parallel.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/patching.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/reference.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/action.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/condition.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/policy.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/result.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/routemap.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl/statement_builder.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/cumulus_frr.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rpl_generators/execute.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/arista/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/arista/aaa.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/arista/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/aruba/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/aruba/ap_env.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/aruba/misc.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/b4com/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/b4com/file.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/b4com/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/cisco/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/cisco/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/cisco/misc.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/cisco/vlandb.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/common.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/deploying.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/generic/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/generic/misc.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/aaa.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/bgp.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/misc.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/huawei/vlandb.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/juniper/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/juniper/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/nexus/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/nexus/iface.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/patching.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/routeros/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/routeros/file.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/arista.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/arista.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/arista.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/aruba.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/aruba.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/aruba.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/b4com.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/b4com.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/b4com.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/cisco.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/cisco.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/cisco.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/huawei.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/huawei.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/huawei.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/iosxr.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/iosxr.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/iosxr.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/juniper.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/nexus.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/nexus.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/nexus.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/nokia.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/optixtrans.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/optixtrans.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/optixtrans.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/pc.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/pc.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/pc.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/ribbon.deploy +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/ribbon.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/routeros.order +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/rulebook/texts/routeros.rul +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/storage.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/text_term_format.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/tracing.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/types.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/base.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/arista.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/aruba.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/b4com.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/cisco.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/h3c.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/huawei.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/iosxr.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/juniper.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/nexus.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/nokia.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/optixtrans.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/pc.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/ribbon.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/library/routeros.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet/vendors/registry.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet.egg-info/SOURCES.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet.egg-info/dependency_links.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet.egg-info/entry_points.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet.egg-info/requires.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet.egg-info/top_level.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/example/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/example/hostname.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/example/lldp.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/mesh_example/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/mesh_example/bgp.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/mesh_example/mesh_logic.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/rpl_example/__init__.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/rpl_example/generator.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/rpl_example/items.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/rpl_example/mesh.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/annet_generators/rpl_example/route_policy.py +0 -0
- {annet-3.13.0 → annet-3.14.0}/requirements.txt +0 -0
- {annet-3.13.0 → annet-3.14.0}/setup.cfg +0 -0
- {annet-3.13.0 → annet-3.14.0}/setup.py +0 -0
|
@@ -144,7 +144,7 @@
|
|
|
144
144
|
"PC.Whitebox.Ufispace.S.S9300.S9301_32DB": " S9301-32DB",
|
|
145
145
|
"PC.Whitebox.Ufispace.S.S9300.S9321_64EO": " S9321-64EO",
|
|
146
146
|
"PC.Nebius": "^Nebius",
|
|
147
|
-
"PC.Nebius.
|
|
147
|
+
"PC.Nebius.Octoport": " NB-D-M-CSM-R",
|
|
148
148
|
"PC.Avocent": " [Aa]vocent",
|
|
149
149
|
"PC.Avocent.ACS8000": " (ACS|acs)8000",
|
|
150
150
|
|
|
@@ -189,4 +189,4 @@
|
|
|
189
189
|
"B4com.CS4132U": "^[Bb]4com B4T-CS4132U.*",
|
|
190
190
|
"B4com.CS4148Q": "^[Bb]4com B4T-CS4148Q.*",
|
|
191
191
|
"B4com.CS4164U": "^[Bb]4com B4T-CS4164U.*"
|
|
192
|
-
}
|
|
192
|
+
}
|
|
@@ -95,8 +95,12 @@ class Checkable:
|
|
|
95
95
|
def match_v6(
|
|
96
96
|
self,
|
|
97
97
|
*names: str,
|
|
98
|
-
or_longer: OrLonger = (None, None),
|
|
98
|
+
or_longer: bool | OrLonger = (None, None),
|
|
99
99
|
) -> SingleCondition[PrefixMatchValue]:
|
|
100
|
+
if or_longer is True:
|
|
101
|
+
or_longer = (None, 128)
|
|
102
|
+
if or_longer is False:
|
|
103
|
+
or_longer = (None, None)
|
|
100
104
|
return SingleCondition(
|
|
101
105
|
MatchField.ipv6_prefix,
|
|
102
106
|
ConditionOperator.CUSTOM,
|
|
@@ -106,8 +110,12 @@ class Checkable:
|
|
|
106
110
|
def match_v4(
|
|
107
111
|
self,
|
|
108
112
|
*names: str,
|
|
109
|
-
or_longer: OrLonger = (None, None),
|
|
113
|
+
or_longer: bool | OrLonger = (None, None),
|
|
110
114
|
) -> SingleCondition[PrefixMatchValue]:
|
|
115
|
+
if or_longer is True:
|
|
116
|
+
or_longer = (None, 32)
|
|
117
|
+
if or_longer is False:
|
|
118
|
+
or_longer = (None, None)
|
|
111
119
|
return SingleCondition(
|
|
112
120
|
MatchField.ip_prefix,
|
|
113
121
|
ConditionOperator.CUSTOM,
|
|
@@ -71,3 +71,25 @@ class AsPathFilterGenerator(PartialGenerator, ABC):
|
|
|
71
71
|
else:
|
|
72
72
|
comma = ""
|
|
73
73
|
yield "ios-regex", f"'{filter_item}'{comma}"
|
|
74
|
+
|
|
75
|
+
def acl_juniper(self, _):
|
|
76
|
+
return r"""
|
|
77
|
+
policy-options %cant_delete
|
|
78
|
+
as-path ~
|
|
79
|
+
"""
|
|
80
|
+
|
|
81
|
+
def _juniper_as_path(self, name: str, as_path_member: str):
|
|
82
|
+
if not as_path_member.isnumeric():
|
|
83
|
+
as_path_member = f'"{as_path_member}"'
|
|
84
|
+
yield "as-path", name, as_path_member
|
|
85
|
+
|
|
86
|
+
def run_juniper(self, device: Any):
|
|
87
|
+
for as_path_filter in self.get_used_as_path_filters(device):
|
|
88
|
+
# TODO could be implemented via as-path-groups
|
|
89
|
+
# But we need to provide as_path_filters to policy generator
|
|
90
|
+
# To select between regular as-path and as-path-groups
|
|
91
|
+
if len(as_path_filter.filters) > 1:
|
|
92
|
+
raise NotImplementedError(f"Multiple elements in as_path_filter {as_path_filter.name} is not supported for Juniper")
|
|
93
|
+
|
|
94
|
+
with self.block("policy-options"):
|
|
95
|
+
yield from self._juniper_as_path(as_path_filter.name, as_path_filter.filters[0])
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
from abc import ABC, abstractmethod
|
|
2
|
-
from collections
|
|
2
|
+
from collections import defaultdict
|
|
3
|
+
from collections.abc import Sequence, Collection, Iterator, Mapping
|
|
3
4
|
from typing import Any
|
|
4
5
|
|
|
5
6
|
from annet.generators import PartialGenerator
|
|
@@ -12,6 +13,7 @@ from .entities import (
|
|
|
12
13
|
def get_used_community_lists(
|
|
13
14
|
communities: Collection[CommunityList], policies: Collection[RoutingPolicy],
|
|
14
15
|
) -> list[CommunityList]:
|
|
16
|
+
assert_unique_names(communities)
|
|
15
17
|
communities_dict = {c.name: c for c in communities}
|
|
16
18
|
used_communities: set[str] = set()
|
|
17
19
|
for policy in policies:
|
|
@@ -44,6 +46,7 @@ def get_used_united_community_lists(
|
|
|
44
46
|
"""
|
|
45
47
|
Return communities united into groups according to HAS_ANY policy
|
|
46
48
|
"""
|
|
49
|
+
assert_unique_names(communities)
|
|
47
50
|
communities_dict = {c.name: c for c in communities}
|
|
48
51
|
used_communities: dict[str, list[CommunityList]] = {}
|
|
49
52
|
for policy in policies:
|
|
@@ -93,6 +96,17 @@ def get_used_united_community_lists(
|
|
|
93
96
|
]
|
|
94
97
|
|
|
95
98
|
|
|
99
|
+
def assert_unique_names(communities: Collection[CommunityList]) -> None:
|
|
100
|
+
duplicated: list[str] = []
|
|
101
|
+
seen_names: set[str] = set()
|
|
102
|
+
for c in communities:
|
|
103
|
+
if c.name in seen_names:
|
|
104
|
+
duplicated.append(c.name)
|
|
105
|
+
seen_names.add(c.name)
|
|
106
|
+
if duplicated:
|
|
107
|
+
raise NotImplementedError(f"Non-unique community-list names are not supported: {duplicated}")
|
|
108
|
+
|
|
109
|
+
|
|
96
110
|
class CommunityListGenerator(PartialGenerator, ABC):
|
|
97
111
|
TAGS = ["policy", "rpl", "routing"]
|
|
98
112
|
|
|
@@ -274,3 +288,61 @@ class CommunityListGenerator(PartialGenerator, ABC):
|
|
|
274
288
|
def run_iosxr(self, device):
|
|
275
289
|
for community_list in self.get_used_community_lists(device):
|
|
276
290
|
yield from self._iosxr_community_list(community_list)
|
|
291
|
+
|
|
292
|
+
def acl_juniper(self, _) -> str:
|
|
293
|
+
return r"""
|
|
294
|
+
policy-options %cant_delete
|
|
295
|
+
community ~
|
|
296
|
+
"""
|
|
297
|
+
|
|
298
|
+
def _juniper_community_list(self, name: str, community_lists: list[CommunityList]) -> Iterator[Sequence[str]]:
|
|
299
|
+
members: list[str] = []
|
|
300
|
+
logic: set[CommunityLogic] = set()
|
|
301
|
+
for community_list in community_lists:
|
|
302
|
+
prefix: str
|
|
303
|
+
if community_list.type is CommunityType.BASIC:
|
|
304
|
+
prefix = ""
|
|
305
|
+
elif community_list.type is CommunityType.RT:
|
|
306
|
+
prefix = "target:"
|
|
307
|
+
elif community_list.type is CommunityType.SOO:
|
|
308
|
+
prefix = "origin:"
|
|
309
|
+
elif community_list.type is CommunityType.LARGE:
|
|
310
|
+
prefix = "large:"
|
|
311
|
+
else:
|
|
312
|
+
raise NotImplementedError(f"CommunityList {name}: type {community_list.type} not implemented for Juniper")
|
|
313
|
+
|
|
314
|
+
logic.add(community_list.logic)
|
|
315
|
+
for community in community_list.members:
|
|
316
|
+
members.append(prefix + community)
|
|
317
|
+
|
|
318
|
+
if len(members) > 1 and logic != {CommunityLogic.AND}:
|
|
319
|
+
raise NotImplementedError(f"CommunityList {name}: only AND logic between members is implemeted for Juniper")
|
|
320
|
+
|
|
321
|
+
definition = ["community", name, "members"]
|
|
322
|
+
with self.block("policy-options"):
|
|
323
|
+
if len(members) == 1:
|
|
324
|
+
yield *definition, *members
|
|
325
|
+
if len(members) > 1:
|
|
326
|
+
yield *definition, "[", *members, "]"
|
|
327
|
+
|
|
328
|
+
def run_juniper(self, device):
|
|
329
|
+
# Juniper allows different community types
|
|
330
|
+
# so we write generator in a generic way to reflect that.
|
|
331
|
+
#
|
|
332
|
+
# But get_used_community_lists DOES NOT allow multiple names
|
|
333
|
+
# This is in part because juniper does not have a type-aware match
|
|
334
|
+
# It would mean that there is no way to describe a following config via rpl.py:
|
|
335
|
+
#
|
|
336
|
+
# CommunityList("COMM_LIST", ["65000:4000"], CommunityType.BASIC),
|
|
337
|
+
# CommunityList("COMM_LIST", ["65000:4000"], CommunityType.RT),
|
|
338
|
+
#
|
|
339
|
+
# # match only route-target but not basic one
|
|
340
|
+
# with route(R.extcommunity_rt.has("COMM_LIST")) as rule:
|
|
341
|
+
# ...
|
|
342
|
+
used = self.get_used_community_lists(device)
|
|
343
|
+
by_name: Mapping[str, list[CommunityList]] = defaultdict(list)
|
|
344
|
+
for community_list in used:
|
|
345
|
+
by_name[community_list.name].append(community_list)
|
|
346
|
+
|
|
347
|
+
for name, community_lists in by_name.items():
|
|
348
|
+
yield from self._juniper_community_list(name, community_lists)
|
|
@@ -3,7 +3,7 @@ from collections import defaultdict
|
|
|
3
3
|
from collections.abc import Sequence
|
|
4
4
|
from dataclasses import dataclass
|
|
5
5
|
from enum import Enum
|
|
6
|
-
from typing import Optional, List
|
|
6
|
+
from typing import Optional, List, Literal
|
|
7
7
|
|
|
8
8
|
from annet.rpl import RoutingPolicy, PrefixMatchValue, OrLonger
|
|
9
9
|
|
|
@@ -103,12 +103,12 @@ class PrefixListNameGenerator:
|
|
|
103
103
|
self._prefix_lists = {x.name: x for x in prefix_lists}
|
|
104
104
|
self._policies = {x.name: x for x in policies} # this is here for a later use ~azryve@
|
|
105
105
|
|
|
106
|
-
def get_prefix(self, name: str, match: PrefixMatchValue) -> IpPrefixList:
|
|
106
|
+
def get_prefix(self, name: str, match: PrefixMatchValue | None) -> IpPrefixList:
|
|
107
107
|
orig_prefix = self._prefix_lists[name]
|
|
108
108
|
override_name: Optional[str] = None
|
|
109
109
|
override_orlonger: Optional[OrLonger] = None
|
|
110
110
|
|
|
111
|
-
if
|
|
111
|
+
if match and match.or_longer != (None, None):
|
|
112
112
|
ge, le = match.or_longer
|
|
113
113
|
ge_str = "unset" if ge is None else str(ge)
|
|
114
114
|
le_str = "unset" if le is None else str(le)
|
|
@@ -135,3 +135,50 @@ def group_community_members(
|
|
|
135
135
|
community = all_communities[community_name]
|
|
136
136
|
members[community.type].extend(community.members)
|
|
137
137
|
return members
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
class JuniperPrefixListNameGenerator(PrefixListNameGenerator):
|
|
141
|
+
def get_prefix(self, name: str, match: PrefixMatchValue | None) -> IpPrefixList:
|
|
142
|
+
plist = super().get_prefix(name, match)
|
|
143
|
+
flavour = self.get_plist_flavour(plist)
|
|
144
|
+
# keep the orginal name for an orlonger match
|
|
145
|
+
if flavour != "custom":
|
|
146
|
+
plist.name = name
|
|
147
|
+
return plist
|
|
148
|
+
|
|
149
|
+
def get_type(self, name: str, match: PrefixMatchValue | None) -> Literal["prefix-list", "route-filter"]:
|
|
150
|
+
orig_plist = self.get_prefix(name, None)
|
|
151
|
+
plist = self.get_prefix(name, match)
|
|
152
|
+
orig_flavour = self.get_plist_flavour(orig_plist)
|
|
153
|
+
flavour = self.get_plist_flavour(plist)
|
|
154
|
+
if orig_flavour == "custom" or flavour == "custom":
|
|
155
|
+
return "route-filter"
|
|
156
|
+
else:
|
|
157
|
+
return "prefix-list"
|
|
158
|
+
|
|
159
|
+
def get_plist_flavour(self, plist: IpPrefixList) -> Literal["simple", "orlonger", "custom"]:
|
|
160
|
+
is_orlonger: bool = False
|
|
161
|
+
is_custom: bool = False
|
|
162
|
+
for m in plist.members:
|
|
163
|
+
ge, le = m.or_longer
|
|
164
|
+
|
|
165
|
+
# or_longer != (None, None)
|
|
166
|
+
if ge is not None or le is not None:
|
|
167
|
+
is_orlonger = True
|
|
168
|
+
|
|
169
|
+
# orlonger != (n, None), where n is .../n
|
|
170
|
+
if ge is not None and ge != m.prefix.prefixlen:
|
|
171
|
+
is_custom = True
|
|
172
|
+
break
|
|
173
|
+
|
|
174
|
+
# orlonger != (None, n), where n is 32 or 128 (ipv4/ipv6)
|
|
175
|
+
if le is not None and le != m.prefix.max_prefixlen:
|
|
176
|
+
is_custom = True
|
|
177
|
+
break
|
|
178
|
+
|
|
179
|
+
if is_custom:
|
|
180
|
+
return "custom"
|
|
181
|
+
elif is_orlonger:
|
|
182
|
+
return "orlonger"
|
|
183
|
+
else:
|
|
184
|
+
return "simple"
|
|
@@ -6,13 +6,13 @@ from annet.generators import PartialGenerator
|
|
|
6
6
|
from annet.rpl import (
|
|
7
7
|
CommunityActionValue,
|
|
8
8
|
ResultType, RoutingPolicyStatement, RoutingPolicy, ConditionOperator, SingleCondition, SingleAction, ActionType,
|
|
9
|
-
MatchField, PrefixMatchValue,
|
|
9
|
+
MatchField, PrefixMatchValue, AndCondition, Action,
|
|
10
10
|
)
|
|
11
11
|
from annet.rpl.statement_builder import AsPathActionValue, NextHopActionValue, ThenField
|
|
12
12
|
from annet.rpl_generators.entities import (
|
|
13
13
|
arista_well_known_community,
|
|
14
14
|
CommunityList, RDFilter, PrefixListNameGenerator, CommunityLogic, mangle_united_community_list_name,
|
|
15
|
-
IpPrefixList, group_community_members, CommunityType,
|
|
15
|
+
IpPrefixList, group_community_members, CommunityType, JuniperPrefixListNameGenerator
|
|
16
16
|
)
|
|
17
17
|
|
|
18
18
|
HUAWEI_MATCH_COMMAND_MAP: dict[str, str] = {
|
|
@@ -79,6 +79,31 @@ IOSXR_RESULT_MAP = {
|
|
|
79
79
|
ResultType.DENY: "drop",
|
|
80
80
|
ResultType.NEXT: "pass"
|
|
81
81
|
}
|
|
82
|
+
JUNIPER_MATCH_COMMAND_MAP: dict[str, str] = {
|
|
83
|
+
MatchField.protocol: "protocol {option_value}",
|
|
84
|
+
MatchField.metric: "metric {option_value}",
|
|
85
|
+
MatchField.as_path_filter: "as-path {option_value}",
|
|
86
|
+
MatchField.local_pref: "local-preference {option_value}",
|
|
87
|
+
# unsupported: rd
|
|
88
|
+
# unsupported: interface
|
|
89
|
+
# unsupported: net_len
|
|
90
|
+
# unsupported: family
|
|
91
|
+
}
|
|
92
|
+
JUNIPER_THEN_COMMAND_MAP: dict[str, str] = {
|
|
93
|
+
ThenField.local_pref: "local-preference {option_value}",
|
|
94
|
+
ThenField.origin: "origin {option_value}",
|
|
95
|
+
ThenField.tag: "tag {option_value}",
|
|
96
|
+
ThenField.metric: "metric {option_value}",
|
|
97
|
+
# unsupported: rpki_valid_state
|
|
98
|
+
# unsupported: resolution
|
|
99
|
+
# unsupported: mpls_label
|
|
100
|
+
# unsupported: metric_type
|
|
101
|
+
}
|
|
102
|
+
JUNIPER_RESULT_MAP = {
|
|
103
|
+
ResultType.ALLOW: "accept",
|
|
104
|
+
ResultType.DENY: "reject",
|
|
105
|
+
ResultType.NEXT: "next term"
|
|
106
|
+
}
|
|
82
107
|
|
|
83
108
|
|
|
84
109
|
class RoutingPolicyGenerator(PartialGenerator, ABC):
|
|
@@ -1119,3 +1144,343 @@ class RoutingPolicyGenerator(PartialGenerator, ABC):
|
|
|
1119
1144
|
yield from self._iosxr_statement(
|
|
1120
1145
|
communities, rd_filters, device, policy, statement, prefix_name_generator,
|
|
1121
1146
|
)
|
|
1147
|
+
|
|
1148
|
+
# Juniper
|
|
1149
|
+
def acl_juniper(self, device):
|
|
1150
|
+
return r"""
|
|
1151
|
+
policy-options %cant_delete
|
|
1152
|
+
policy-statement
|
|
1153
|
+
~ %global
|
|
1154
|
+
"""
|
|
1155
|
+
|
|
1156
|
+
def _juniper_match_communities(
|
|
1157
|
+
self,
|
|
1158
|
+
section: Literal["", "from"],
|
|
1159
|
+
conditions: list[SingleCondition],
|
|
1160
|
+
) -> Iterator[Sequence[str]]:
|
|
1161
|
+
names: list[str] = [name for cond in conditions for name in cond.value]
|
|
1162
|
+
operators = {x.operator for x in conditions}
|
|
1163
|
+
if len(names) > 1 and operators != {ConditionOperator.HAS_ANY}:
|
|
1164
|
+
raise NotImplementedError(
|
|
1165
|
+
f"Multiple community match [{' '.join(names)}] without has_any is not supported for Juniper",
|
|
1166
|
+
)
|
|
1167
|
+
yield section, "community", self._juniper_list_bracket(names)
|
|
1168
|
+
|
|
1169
|
+
def _juniper_match_prefix_lists(
|
|
1170
|
+
self,
|
|
1171
|
+
section: Literal["", "from"],
|
|
1172
|
+
conditions: list[SingleCondition[PrefixMatchValue]],
|
|
1173
|
+
name_generator: JuniperPrefixListNameGenerator,
|
|
1174
|
+
) -> Iterator[Sequence[str]]:
|
|
1175
|
+
operators = {x.operator for x in conditions}
|
|
1176
|
+
supported = {ConditionOperator.HAS_ANY}
|
|
1177
|
+
not_supported = operators - supported
|
|
1178
|
+
if len(conditions) > 1 and not_supported:
|
|
1179
|
+
raise NotImplementedError(
|
|
1180
|
+
f"Multiple prefix match with ops {not_supported} is not supported for Juniper",
|
|
1181
|
+
)
|
|
1182
|
+
for cond in conditions:
|
|
1183
|
+
for name in cond.value.names:
|
|
1184
|
+
prefix_list = name_generator.get_prefix(name, cond.value)
|
|
1185
|
+
plist_type = name_generator.get_type(name, cond.value)
|
|
1186
|
+
flavour = name_generator.get_plist_flavour(prefix_list)
|
|
1187
|
+
if plist_type == "prefix-list" and flavour == "simple":
|
|
1188
|
+
yield section, "prefix-list", prefix_list.name
|
|
1189
|
+
elif plist_type == "prefix-list" and flavour == "orlonger":
|
|
1190
|
+
yield section, "prefix-list-filter", prefix_list.name, "orlonger"
|
|
1191
|
+
elif plist_type == "route-filter":
|
|
1192
|
+
yield section, "route-filter-list", prefix_list.name
|
|
1193
|
+
else:
|
|
1194
|
+
raise NotImplementedError(
|
|
1195
|
+
f"Prefix list {prefix_list.name} type {plist_type} flavour {flavour} is not supported for Juniper",
|
|
1196
|
+
)
|
|
1197
|
+
|
|
1198
|
+
def _juniper_match_as_path_length(
|
|
1199
|
+
self,
|
|
1200
|
+
section: Literal["", "from"],
|
|
1201
|
+
conditions: list[SingleCondition],
|
|
1202
|
+
) -> Iterator[Sequence[str]]:
|
|
1203
|
+
for condition in conditions:
|
|
1204
|
+
if condition.operator is ConditionOperator.EQ:
|
|
1205
|
+
yield section, "as-path-calc-length", str(condition.value), "equal"
|
|
1206
|
+
elif condition.operator is ConditionOperator.LE:
|
|
1207
|
+
yield section, "as-path-calc-length", str(condition.value), "orlower"
|
|
1208
|
+
elif condition.operator is ConditionOperator.GE:
|
|
1209
|
+
yield section, "as-path-calc-length", str(condition.value), "orhigher"
|
|
1210
|
+
elif condition.operator is ConditionOperator.BETWEEN_INCLUDED:
|
|
1211
|
+
yield section, "as-path-calc-length", str(condition.value[0]), "orhigher"
|
|
1212
|
+
yield section, "as-path-calc-length", str(condition.value[1]), "orlower"
|
|
1213
|
+
else:
|
|
1214
|
+
raise NotImplementedError(
|
|
1215
|
+
f"Operator {condition.operator} is not supported for {condition.field} on Juniper",
|
|
1216
|
+
)
|
|
1217
|
+
|
|
1218
|
+
def _juniper_match_rd_filter(
|
|
1219
|
+
self,
|
|
1220
|
+
section: Literal["", "from"],
|
|
1221
|
+
conditions: list[SingleCondition[Sequence[str]]],
|
|
1222
|
+
) -> Iterator[Sequence[str]]:
|
|
1223
|
+
names = [x for c in conditions for x in c.value]
|
|
1224
|
+
operators = {x.operator for x in conditions}
|
|
1225
|
+
supported = {ConditionOperator.HAS_ANY}
|
|
1226
|
+
not_supported = operators - supported
|
|
1227
|
+
if len(names) > 1 and not_supported:
|
|
1228
|
+
raise NotImplementedError(
|
|
1229
|
+
f"Multiple rd_filter matches with ops {not_supported} is not supported for Juniper",
|
|
1230
|
+
)
|
|
1231
|
+
yield section, "route-distinguisher", self._juniper_list_bracket(names)
|
|
1232
|
+
|
|
1233
|
+
def _juniper_match_community_fields(self) -> set[MatchField]:
|
|
1234
|
+
return {
|
|
1235
|
+
MatchField.community,
|
|
1236
|
+
MatchField.extcommunity_rt,
|
|
1237
|
+
MatchField.extcommunity_soo,
|
|
1238
|
+
MatchField.large_community,
|
|
1239
|
+
}
|
|
1240
|
+
|
|
1241
|
+
def _juniper_match_prefix_fields(self) -> set[MatchField]:
|
|
1242
|
+
return {
|
|
1243
|
+
MatchField.ip_prefix,
|
|
1244
|
+
MatchField.ipv6_prefix,
|
|
1245
|
+
}
|
|
1246
|
+
|
|
1247
|
+
def _juniper_is_match_inlined(self, conditions: AndCondition) -> bool:
|
|
1248
|
+
used_fields = {x.field for x in conditions}
|
|
1249
|
+
used_prefix_fields = used_fields & self._juniper_match_prefix_fields()
|
|
1250
|
+
used_community_fields = used_fields & self._juniper_match_community_fields()
|
|
1251
|
+
|
|
1252
|
+
# prefix-list match is never inlined
|
|
1253
|
+
if used_prefix_fields:
|
|
1254
|
+
return False
|
|
1255
|
+
|
|
1256
|
+
# as-path-calc-length is never inlined
|
|
1257
|
+
if MatchField.as_path_length in used_fields:
|
|
1258
|
+
return False
|
|
1259
|
+
|
|
1260
|
+
# only community matches and nothing more
|
|
1261
|
+
if used_community_fields and used_fields == used_community_fields:
|
|
1262
|
+
return True
|
|
1263
|
+
|
|
1264
|
+
# inline when empty or just one match
|
|
1265
|
+
if len(used_fields) <= 1:
|
|
1266
|
+
return True
|
|
1267
|
+
return False
|
|
1268
|
+
|
|
1269
|
+
def _juniper_match(
|
|
1270
|
+
self,
|
|
1271
|
+
policy: RoutingPolicy,
|
|
1272
|
+
section: Literal["", "from"],
|
|
1273
|
+
conditions: AndCondition,
|
|
1274
|
+
prefix_name_generator: JuniperPrefixListNameGenerator,
|
|
1275
|
+
) -> Iterator[Sequence[str]]:
|
|
1276
|
+
community_fields = self._juniper_match_community_fields()
|
|
1277
|
+
prefix_fields = self._juniper_match_prefix_fields()
|
|
1278
|
+
community_conditions: list[SingleCondition] = []
|
|
1279
|
+
prefix_conditions: list[SingleCondition] = []
|
|
1280
|
+
simple_conditions: list[SingleCondition] = []
|
|
1281
|
+
as_path_length_conditions: list[SingleCondition] = []
|
|
1282
|
+
rd_filter_conditions: list[SingleCondition] = []
|
|
1283
|
+
for condition in conditions:
|
|
1284
|
+
if condition.field in community_fields:
|
|
1285
|
+
community_conditions.append(condition)
|
|
1286
|
+
elif condition.field in prefix_fields:
|
|
1287
|
+
prefix_conditions.append(condition)
|
|
1288
|
+
elif condition.field == MatchField.as_path_length:
|
|
1289
|
+
as_path_length_conditions.append(condition)
|
|
1290
|
+
elif condition.field == MatchField.rd:
|
|
1291
|
+
rd_filter_conditions.append(condition)
|
|
1292
|
+
else:
|
|
1293
|
+
simple_conditions.append(condition)
|
|
1294
|
+
|
|
1295
|
+
if community_conditions:
|
|
1296
|
+
yield from self._juniper_match_communities(section, community_conditions)
|
|
1297
|
+
if prefix_conditions:
|
|
1298
|
+
yield from self._juniper_match_prefix_lists(section, prefix_conditions, prefix_name_generator)
|
|
1299
|
+
if as_path_length_conditions:
|
|
1300
|
+
yield from self._juniper_match_as_path_length(section, as_path_length_conditions)
|
|
1301
|
+
if rd_filter_conditions:
|
|
1302
|
+
yield from self._juniper_match_rd_filter(section, rd_filter_conditions)
|
|
1303
|
+
|
|
1304
|
+
for condition in simple_conditions:
|
|
1305
|
+
if condition.operator is not ConditionOperator.EQ:
|
|
1306
|
+
raise NotImplementedError(
|
|
1307
|
+
f"`{condition.field}` with operator {condition.operator} in {policy.name} is not supported for Juniper",
|
|
1308
|
+
)
|
|
1309
|
+
if condition.field not in JUNIPER_MATCH_COMMAND_MAP:
|
|
1310
|
+
raise NotImplementedError(f"Match using `{condition.field}` in {policy.name} is not supported for Juniper")
|
|
1311
|
+
yield section, JUNIPER_MATCH_COMMAND_MAP[condition.field].format(option_value=condition.value)
|
|
1312
|
+
|
|
1313
|
+
def _juniper_then_community(
|
|
1314
|
+
self,
|
|
1315
|
+
section: Literal["", "then"],
|
|
1316
|
+
actions: list[SingleAction[CommunityActionValue]]
|
|
1317
|
+
):
|
|
1318
|
+
# juniper community ops are ORDERED
|
|
1319
|
+
# since data model does not support it
|
|
1320
|
+
# we use the order that makes sense: delete, set, add
|
|
1321
|
+
for single_action in actions:
|
|
1322
|
+
action = single_action.value
|
|
1323
|
+
for name in action.removed:
|
|
1324
|
+
yield section, "community", "delete", name
|
|
1325
|
+
|
|
1326
|
+
if action.replaced is not None:
|
|
1327
|
+
if not action.replaced:
|
|
1328
|
+
raise NotImplementedError("Empty community.set() is not supported for Juniper")
|
|
1329
|
+
for name in action.replaced:
|
|
1330
|
+
yield section, "community", "set", name
|
|
1331
|
+
|
|
1332
|
+
for name in action.added:
|
|
1333
|
+
yield section, "community", "add", name
|
|
1334
|
+
|
|
1335
|
+
def _juniper_then_next_hop(
|
|
1336
|
+
self,
|
|
1337
|
+
section: Literal["", "then"],
|
|
1338
|
+
actions: list[SingleAction[NextHopActionValue]],
|
|
1339
|
+
):
|
|
1340
|
+
if len(actions) > 1:
|
|
1341
|
+
raise NotImplementedError("Only single next-hop action is supported for Juniper")
|
|
1342
|
+
|
|
1343
|
+
action = actions[0]
|
|
1344
|
+
if action.value.target == "self":
|
|
1345
|
+
yield section, "next-hop", "self"
|
|
1346
|
+
elif action.value.target == "discard":
|
|
1347
|
+
yield section, "next-hop", "discard"
|
|
1348
|
+
elif action.value.target == "peer":
|
|
1349
|
+
yield section, "next-hop", "peer-address"
|
|
1350
|
+
elif action.value.target == "ipv4_addr":
|
|
1351
|
+
yield section, "next-hop", action.value.addr
|
|
1352
|
+
elif action.value.target == "ipv6_addr":
|
|
1353
|
+
yield section, "next-hop", action.value.addr.lower()
|
|
1354
|
+
elif action.value.target == "mapped_ipv4":
|
|
1355
|
+
yield section, "next-hop", f"::ffff:{action.value.addr}"
|
|
1356
|
+
else:
|
|
1357
|
+
raise NotImplementedError(f"Next_hop target {action.value.target} is not supported for Juniper")
|
|
1358
|
+
|
|
1359
|
+
def _juniper_list_quote(self, items: list[str]) -> str:
|
|
1360
|
+
joined = " ".join(items)
|
|
1361
|
+
if len(items) > 1:
|
|
1362
|
+
joined = f'"{joined}"'
|
|
1363
|
+
return joined
|
|
1364
|
+
|
|
1365
|
+
def _juniper_list_bracket(self, items: list[str]) -> str:
|
|
1366
|
+
joined = " ".join(items)
|
|
1367
|
+
if len(items) > 1:
|
|
1368
|
+
joined = f"[ {joined} ]"
|
|
1369
|
+
return joined
|
|
1370
|
+
|
|
1371
|
+
def _juniper_then_as_path(
|
|
1372
|
+
self,
|
|
1373
|
+
section: Literal["", "then"],
|
|
1374
|
+
actions: list[SingleAction[AsPathActionValue]],
|
|
1375
|
+
):
|
|
1376
|
+
if len(actions) > 1:
|
|
1377
|
+
raise NotImplementedError("Only single next-hop action is supported for Juniper")
|
|
1378
|
+
|
|
1379
|
+
action = actions[0]
|
|
1380
|
+
if action.value.expand and action.value.expand_last_as:
|
|
1381
|
+
raise NotImplementedError("Setting both `as_path.expand` and `as_path.expand_last_as` is not supported for Juniper")
|
|
1382
|
+
|
|
1383
|
+
if action.value.prepend:
|
|
1384
|
+
yield section, "as-path-prepend", self._juniper_list_quote(action.value.prepend)
|
|
1385
|
+
if action.value.expand:
|
|
1386
|
+
yield section, "as-path-expand", self._juniper_list_quote(action.value.expand)
|
|
1387
|
+
if action.value.expand_last_as:
|
|
1388
|
+
yield section, "as-path-expand last-as count", action.value.expand_last_as
|
|
1389
|
+
if action.value.set is not None:
|
|
1390
|
+
raise RuntimeError("as_path.set is not supported for Juniper")
|
|
1391
|
+
if action.value.delete:
|
|
1392
|
+
raise RuntimeError("as_path.delete is not supported for Juniper")
|
|
1393
|
+
|
|
1394
|
+
def _juniper_is_then_inlined(self, action: Action) -> bool:
|
|
1395
|
+
used_fields = {x.field for x in action}
|
|
1396
|
+
# inline when no actions permormed
|
|
1397
|
+
if not used_fields:
|
|
1398
|
+
return True
|
|
1399
|
+
return False
|
|
1400
|
+
|
|
1401
|
+
def _juniper_then(
|
|
1402
|
+
self,
|
|
1403
|
+
policy: RoutingPolicy,
|
|
1404
|
+
section: Literal["", "then"],
|
|
1405
|
+
actions: Action,
|
|
1406
|
+
) -> Iterator[Sequence[str]]:
|
|
1407
|
+
community_actions: list[SingleAction] = []
|
|
1408
|
+
next_hop_actions: list[SingleAction] = []
|
|
1409
|
+
as_path_actions: list[SingleAction] = []
|
|
1410
|
+
simple_actions: list[SingleAction] = []
|
|
1411
|
+
for action in actions:
|
|
1412
|
+
if action.field == ThenField.community:
|
|
1413
|
+
community_actions.append(action)
|
|
1414
|
+
elif action.field == ThenField.extcommunity:
|
|
1415
|
+
community_actions.append(action)
|
|
1416
|
+
elif action.field == ThenField.extcommunity_rt:
|
|
1417
|
+
community_actions.append(action)
|
|
1418
|
+
elif action.field == ThenField.extcommunity_soo:
|
|
1419
|
+
community_actions.append(action)
|
|
1420
|
+
elif action.field == ThenField.large_community:
|
|
1421
|
+
community_actions.append(action)
|
|
1422
|
+
elif action.field == ThenField.next_hop:
|
|
1423
|
+
next_hop_actions.append(action)
|
|
1424
|
+
elif action.field == ThenField.as_path:
|
|
1425
|
+
as_path_actions.append(action)
|
|
1426
|
+
else:
|
|
1427
|
+
simple_actions.append(action)
|
|
1428
|
+
|
|
1429
|
+
if community_actions:
|
|
1430
|
+
yield from self._juniper_then_community(section, community_actions)
|
|
1431
|
+
if next_hop_actions:
|
|
1432
|
+
yield from self._juniper_then_next_hop(section, next_hop_actions)
|
|
1433
|
+
if as_path_actions:
|
|
1434
|
+
yield from self._juniper_then_as_path(section, as_path_actions)
|
|
1435
|
+
|
|
1436
|
+
for action in simple_actions:
|
|
1437
|
+
if action.type not in {ActionType.SET}:
|
|
1438
|
+
raise NotImplementedError(f"Action type {action.type} for `{action.field}` in {policy.name} is not supported for Juniper")
|
|
1439
|
+
if action.field not in JUNIPER_THEN_COMMAND_MAP:
|
|
1440
|
+
raise NotImplementedError(f"Then action using `{action.field}` in {policy.name} is not supported for Juniper")
|
|
1441
|
+
yield section, JUNIPER_THEN_COMMAND_MAP[action.field].format(option_value=action.value)
|
|
1442
|
+
|
|
1443
|
+
def _juniper_statements(
|
|
1444
|
+
self,
|
|
1445
|
+
device: Any,
|
|
1446
|
+
policy: RoutingPolicy,
|
|
1447
|
+
prefix_name_generator: JuniperPrefixListNameGenerator,
|
|
1448
|
+
) -> Iterator[Sequence[str]]:
|
|
1449
|
+
term_number = 0
|
|
1450
|
+
for statement in policy.statements:
|
|
1451
|
+
if statement.number is not None:
|
|
1452
|
+
term_number = statement.number
|
|
1453
|
+
term_name = statement.name
|
|
1454
|
+
if not term_name:
|
|
1455
|
+
term_name = f"{policy.name}_{term_number}"
|
|
1456
|
+
term_number += 1
|
|
1457
|
+
|
|
1458
|
+
with self.block("term", term_name):
|
|
1459
|
+
# see test_juniper_inline
|
|
1460
|
+
match_inlined = self._juniper_is_match_inlined(statement.match)
|
|
1461
|
+
then_inlined = self._juniper_is_then_inlined(statement.then)
|
|
1462
|
+
match_section: Literal["", "from"] = "from" if match_inlined else ""
|
|
1463
|
+
then_section: Literal["", "then"] = "then" if then_inlined else ""
|
|
1464
|
+
|
|
1465
|
+
if statement.match:
|
|
1466
|
+
with self.block_if("from", condition=not match_inlined):
|
|
1467
|
+
yield from self._juniper_match(policy, match_section, statement.match, prefix_name_generator)
|
|
1468
|
+
|
|
1469
|
+
if statement.then:
|
|
1470
|
+
with self.block_if("then", condition=not then_inlined):
|
|
1471
|
+
yield from self._juniper_then(policy, then_section, statement.then)
|
|
1472
|
+
|
|
1473
|
+
with self.block_if("then", condition=not then_inlined):
|
|
1474
|
+
yield then_section, JUNIPER_RESULT_MAP[statement.result]
|
|
1475
|
+
|
|
1476
|
+
def run_juniper(self, device):
|
|
1477
|
+
prefix_lists = self.get_prefix_lists(device)
|
|
1478
|
+
policies = self.get_policies(device)
|
|
1479
|
+
prefix_name_generator = JuniperPrefixListNameGenerator(prefix_lists, policies)
|
|
1480
|
+
|
|
1481
|
+
for policy in policies:
|
|
1482
|
+
with self.block("policy-options"):
|
|
1483
|
+
with self.block("policy-statement", policy.name):
|
|
1484
|
+
yield from self._juniper_statements(
|
|
1485
|
+
device, policy, prefix_name_generator,
|
|
1486
|
+
)
|