summaryrefslogtreecommitdiffstats
path: root/collections-debian-merged/ansible_collections/arista/eos/plugins
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-14 20:03:01 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-14 20:03:01 +0000
commita453ac31f3428614cceb99027f8efbdb9258a40b (patch)
treef61f87408f32a8511cbd91799f9cececb53e0374 /collections-debian-merged/ansible_collections/arista/eos/plugins
parentInitial commit. (diff)
downloadansible-upstream.tar.xz
ansible-upstream.zip
Adding upstream version 2.10.7+merged+base+2.10.8+dfsg.upstream/2.10.7+merged+base+2.10.8+dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'collections-debian-merged/ansible_collections/arista/eos/plugins')
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py194
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py374
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py118
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py207
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py85
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py416
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py195
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py999
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py25
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py72
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py77
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py75
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py63
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py65
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py73
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py75
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py65
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py178
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py329
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py476
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py97
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py64
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py475
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py605
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py272
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py379
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py305
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py361
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py322
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py197
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py254
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py271
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py203
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py260
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py202
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py796
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py285
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py364
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py258
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py751
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py146
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py387
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py116
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py119
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py127
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py110
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py117
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py118
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py102
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py104
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py119
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py181
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py96
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py103
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py103
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py496
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py115
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py237
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py120
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py143
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py193
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py185
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py72
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py128
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py661
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py2881
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py1082
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py1055
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py85
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py421
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py931
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py219
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py465
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py1332
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py2312
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py260
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py593
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py452
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py207
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py601
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py413
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py461
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py428
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py374
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py406
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py245
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py336
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py340
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py432
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py122
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py345
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py344
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py505
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py1228
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py1545
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py1528
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py372
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py967
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py376
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py477
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py463
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py327
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py432
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py0
-rw-r--r--collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py108
177 files changed, 40750 insertions, 0 deletions
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py
new file mode 100644
index 00000000..03d317f0
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py
@@ -0,0 +1,194 @@
+#
+# (c) 2016 Red Hat Inc.
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import sys
+import copy
+
+from ansible import constants as C
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_provider_spec,
+)
+from ansible_collections.ansible.netcommon.plugins.action.network import (
+ ActionModule as ActionNetworkModule,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ load_provider,
+)
+from ansible.utils.display import Display
+
+display = Display()
+
+
+class ActionModule(ActionNetworkModule):
+ def run(self, tmp=None, task_vars=None):
+ del tmp # tmp no longer has any effect
+
+ module_name = self._task.action.split(".")[-1]
+ self._config_module = (
+ True if module_name in ["eos_config", "config"] else False
+ )
+ persistent_connection = self._play_context.connection.split(".")[-1]
+ warnings = []
+
+ if persistent_connection in ("network_cli", "httpapi"):
+ provider = self._task.args.get("provider", {})
+ if any(provider.values()):
+ display.warning(
+ "provider is unnecessary when using %s and will be ignored"
+ % self._play_context.connection
+ )
+ del self._task.args["provider"]
+ if self._task.args.get("transport"):
+ display.warning(
+ "transport is unnecessary when using %s and will be ignored"
+ % self._play_context.connection
+ )
+ del self._task.args["transport"]
+ elif self._play_context.connection == "local":
+ provider = load_provider(eos_provider_spec, self._task.args)
+ transport = provider["transport"] or "cli"
+
+ display.vvvv(
+ "connection transport is %s" % transport,
+ self._play_context.remote_addr,
+ )
+
+ if transport == "cli":
+ pc = copy.deepcopy(self._play_context)
+ pc.connection = "ansible.netcommon.network_cli"
+ pc.network_os = "arista.eos.eos"
+ pc.remote_addr = (
+ provider["host"] or self._play_context.remote_addr
+ )
+ pc.port = int(
+ provider["port"] or self._play_context.port or 22
+ )
+ pc.remote_user = (
+ provider["username"] or self._play_context.connection_user
+ )
+ pc.password = (
+ provider["password"] or self._play_context.password
+ )
+ pc.private_key_file = (
+ provider["ssh_keyfile"]
+ or self._play_context.private_key_file
+ )
+ pc.become = provider["authorize"] or False
+ if pc.become:
+ pc.become_method = "enable"
+ pc.become_pass = provider["auth_pass"]
+
+ connection = self._shared_loader_obj.connection_loader.get(
+ "ansible.netcommon.persistent",
+ pc,
+ sys.stdin,
+ task_uuid=self._task._uuid,
+ )
+
+ # TODO: Remove below code after ansible minimal is cut out
+ if connection is None:
+ pc.connection = "network_cli"
+ pc.network_os = "eos"
+ connection = self._shared_loader_obj.connection_loader.get(
+ "persistent", pc, sys.stdin, task_uuid=self._task._uuid
+ )
+
+ display.vvv(
+ "using connection plugin %s (was local)" % pc.connection,
+ pc.remote_addr,
+ )
+
+ command_timeout = (
+ int(provider["timeout"])
+ if provider["timeout"]
+ else connection.get_option("persistent_command_timeout")
+ )
+ connection.set_options(
+ direct={"persistent_command_timeout": command_timeout}
+ )
+
+ socket_path = connection.run()
+ display.vvvv("socket_path: %s" % socket_path, pc.remote_addr)
+ if not socket_path:
+ return {
+ "failed": True,
+ "msg": "unable to open shell. Please see: "
+ + "https://docs.ansible.com/ansible/network_debug_troubleshooting.html#unable-to-open-shell",
+ }
+
+ task_vars["ansible_socket"] = socket_path
+ warnings.append(
+ [
+ "connection local support for this module is deprecated and will be removed in version 2.14,"
+ " use connection %s" % pc.connection
+ ]
+ )
+ else:
+ self._task.args["provider"] = ActionModule.eapi_implementation(
+ provider, self._play_context
+ )
+ warnings.append(
+ [
+ "connection local support for this module is deprecated and will be removed in version 2.14,"
+ " use connection either httpapi or ansible.netcommon.httpapi (whichever is applicable)"
+ ]
+ )
+ else:
+ return {
+ "failed": True,
+ "msg": "Connection type %s is not valid for this module"
+ % self._play_context.connection,
+ }
+
+ result = super(ActionModule, self).run(task_vars=task_vars)
+ if warnings:
+ if "warnings" in result:
+ result["warnings"].extend(warnings)
+ else:
+ result["warnings"] = warnings
+ return result
+
+ @staticmethod
+ def eapi_implementation(provider, play_context):
+ provider["transport"] = "eapi"
+
+ if provider.get("host") is None:
+ provider["host"] = play_context.remote_addr
+
+ if provider.get("port") is None:
+ default_port = 443 if provider["use_ssl"] else 80
+ provider["port"] = int(play_context.port or default_port)
+
+ if provider.get("timeout") is None:
+ provider["timeout"] = C.PERSISTENT_COMMAND_TIMEOUT
+
+ if provider.get("username") is None:
+ provider["username"] = play_context.connection_user
+
+ if provider.get("password") is None:
+ provider["password"] = play_context.password
+
+ if provider.get("authorize") is None:
+ provider["authorize"] = False
+
+ return provider
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py
new file mode 100644
index 00000000..0e62fc03
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py
@@ -0,0 +1,374 @@
+#
+# (c) 2017 Red Hat Inc.
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+author: Ansible Networking Team
+cliconf: eos
+short_description: Use eos cliconf to run command on Arista EOS platform
+description:
+- This eos plugin provides low level abstraction apis for sending and receiving CLI
+ commands from Arista EOS network devices.
+version_added: 1.0.0
+options:
+ eos_use_sessions:
+ type: boolean
+ default: true
+ description:
+ - Specifies if sessions should be used on remote host or not
+ env:
+ - name: ANSIBLE_EOS_USE_SESSIONS
+ vars:
+ - name: ansible_eos_use_sessions
+"""
+
+import json
+import time
+import re
+
+from ansible.errors import AnsibleConnectionFailure
+from ansible.module_utils._text import to_text
+from ansible.module_utils.common._collections_compat import Mapping
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+ dumps,
+)
+from ansible.plugins.cliconf import CliconfBase, enable_mode
+
+
+class Cliconf(CliconfBase):
+ def __init__(self, *args, **kwargs):
+ super(Cliconf, self).__init__(*args, **kwargs)
+ self._session_support = None
+
+ @enable_mode
+ def get_config(self, source="running", format="text", flags=None):
+ options_values = self.get_option_values()
+ if format not in options_values["format"]:
+ raise ValueError(
+ "'format' value %s is invalid. Valid values are %s"
+ % (format, ",".join(options_values["format"]))
+ )
+
+ lookup = {"running": "running-config", "startup": "startup-config"}
+ if source not in lookup:
+ raise ValueError(
+ "fetching configuration from %s is not supported" % source
+ )
+
+ cmd = "show %s " % lookup[source]
+ if format and format != "text":
+ cmd += "| %s " % format
+
+ cmd += " ".join(to_list(flags))
+ cmd = cmd.strip()
+ return self.send_command(cmd)
+
+ @enable_mode
+ def edit_config(
+ self, candidate=None, commit=True, replace=None, comment=None
+ ):
+
+ operations = self.get_device_operations()
+ self.check_edit_config_capability(
+ operations, candidate, commit, replace, comment
+ )
+
+ if (commit is False) and (not self.supports_sessions()):
+ raise ValueError(
+ "check mode is not supported without configuration session"
+ )
+
+ resp = {}
+ session = None
+ if self.supports_sessions():
+ session = "ansible_%s" % int(time.time())
+ resp.update({"session": session})
+ self.send_command("configure session %s" % session)
+ if replace:
+ self.send_command("rollback clean-config")
+ else:
+ self.send_command("configure")
+
+ results = []
+ requests = []
+ multiline = False
+ for line in to_list(candidate):
+ if not isinstance(line, Mapping):
+ line = {"command": line}
+
+ cmd = line["command"]
+ if cmd == "end":
+ continue
+ if cmd.startswith("banner") or multiline:
+ multiline = True
+ elif cmd == "EOF" and multiline:
+ multiline = False
+
+ if multiline:
+ line["sendonly"] = True
+
+ if cmd != "end" and not cmd.startswith("!"):
+ try:
+ results.append(self.send_command(**line))
+ requests.append(cmd)
+ except AnsibleConnectionFailure as e:
+ self.discard_changes(session)
+ raise AnsibleConnectionFailure(e.message)
+
+ resp["request"] = requests
+ resp["response"] = results
+ if self.supports_sessions():
+ out = self.send_command("show session-config diffs")
+ if out:
+ resp["diff"] = out.strip()
+
+ if commit:
+ self.commit()
+ else:
+ self.discard_changes(session)
+ else:
+ self.send_command("end")
+ return resp
+
+ def get(
+ self,
+ command,
+ prompt=None,
+ answer=None,
+ sendonly=False,
+ output=None,
+ newline=True,
+ check_all=False,
+ ):
+ if output:
+ command = self._get_command_with_output(command, output)
+ return self.send_command(
+ command=command,
+ prompt=prompt,
+ answer=answer,
+ sendonly=sendonly,
+ newline=newline,
+ check_all=check_all,
+ )
+
+ def commit(self):
+ self.send_command("commit")
+
+ def discard_changes(self, session=None):
+ commands = ["end"]
+ if self.supports_sessions():
+ # to close session gracefully execute abort in top level session prompt.
+ commands.extend(["configure session %s" % session, "abort"])
+
+ for cmd in commands:
+ self.send_command(cmd)
+
+ def run_commands(self, commands=None, check_rc=True):
+ if commands is None:
+ raise ValueError("'commands' value is required")
+ responses = list()
+ for cmd in to_list(commands):
+ if not isinstance(cmd, Mapping):
+ cmd = {"command": cmd}
+
+ output = cmd.pop("output", None)
+ if output:
+ cmd["command"] = self._get_command_with_output(
+ cmd["command"], output
+ )
+
+ try:
+ out = self.send_command(**cmd)
+ except AnsibleConnectionFailure as e:
+ if check_rc:
+ raise
+ out = getattr(e, "err", e)
+ out = to_text(out, errors="surrogate_or_strict")
+
+ if out is not None:
+ try:
+ out = json.loads(out)
+ except ValueError:
+ out = out.strip()
+
+ responses.append(out)
+ return responses
+
+ def get_diff(
+ self,
+ candidate=None,
+ running=None,
+ diff_match="line",
+ diff_ignore_lines=None,
+ path=None,
+ diff_replace="line",
+ ):
+ diff = {}
+ device_operations = self.get_device_operations()
+ option_values = self.get_option_values()
+
+ if candidate is None and device_operations["supports_generate_diff"]:
+ raise ValueError(
+ "candidate configuration is required to generate diff"
+ )
+
+ if diff_match not in option_values["diff_match"]:
+ raise ValueError(
+ "'match' value %s in invalid, valid values are %s"
+ % (diff_match, ", ".join(option_values["diff_match"]))
+ )
+
+ if diff_replace not in option_values["diff_replace"]:
+ raise ValueError(
+ "'replace' value %s in invalid, valid values are %s"
+ % (diff_replace, ", ".join(option_values["diff_replace"]))
+ )
+
+ # prepare candidate configuration
+ candidate_obj = NetworkConfig(indent=3)
+ candidate_obj.load(candidate)
+
+ if running and diff_match != "none" and diff_replace != "config":
+ # running configuration
+ running_obj = NetworkConfig(
+ indent=3, contents=running, ignore_lines=diff_ignore_lines
+ )
+ configdiffobjs = candidate_obj.difference(
+ running_obj, path=path, match=diff_match, replace=diff_replace
+ )
+
+ else:
+ configdiffobjs = candidate_obj.items
+
+ diff["config_diff"] = (
+ dumps(configdiffobjs, "commands") if configdiffobjs else ""
+ )
+ return diff
+
+ def supports_sessions(self):
+ if not self.get_option("eos_use_sessions"):
+ self._session_support = False
+ else:
+ if self._session_support:
+ return self._session_support
+
+ try:
+ self.get("show configuration sessions")
+ self._session_support = True
+ except AnsibleConnectionFailure:
+ self._session_support = False
+
+ return self._session_support
+
+ def get_device_info(self):
+ device_info = {}
+
+ device_info["network_os"] = "eos"
+ reply = self.get("show version | json")
+ data = json.loads(reply)
+
+ device_info["network_os_version"] = data["version"]
+ device_info["network_os_model"] = data["modelName"]
+
+ reply = self.get("show hostname | json")
+ data = json.loads(reply)
+
+ device_info["network_os_hostname"] = data["hostname"]
+
+ try:
+ reply = self.get("bash timeout 5 cat /mnt/flash/boot-config")
+
+ match = re.search(r"SWI=(.+)$", reply, re.M)
+ if match:
+ device_info["network_os_image"] = match.group(1)
+ except AnsibleConnectionFailure:
+ # This requires enable mode to run
+ self._connection.queue_message(
+ "vvv", "Unable to gather network_os_image without enable mode"
+ )
+
+ return device_info
+
+ def get_device_operations(self):
+ return {
+ "supports_diff_replace": True,
+ "supports_commit": bool(self.supports_sessions()),
+ "supports_rollback": False,
+ "supports_defaults": False,
+ "supports_onbox_diff": bool(self.supports_sessions()),
+ "supports_commit_comment": False,
+ "supports_multiline_delimiter": False,
+ "supports_diff_match": True,
+ "supports_diff_ignore_lines": True,
+ "supports_generate_diff": not bool(self.supports_sessions()),
+ "supports_replace": bool(self.supports_sessions()),
+ }
+
+ def get_option_values(self):
+ return {
+ "format": ["text", "json"],
+ "diff_match": ["line", "strict", "exact", "none"],
+ "diff_replace": ["line", "block", "config"],
+ "output": ["text", "json"],
+ }
+
+ def get_capabilities(self):
+ result = super(Cliconf, self).get_capabilities()
+ result["rpc"] += [
+ "commit",
+ "discard_changes",
+ "get_diff",
+ "run_commands",
+ "supports_sessions",
+ ]
+ result["device_operations"] = self.get_device_operations()
+ result.update(self.get_option_values())
+
+ return json.dumps(result)
+
+ def set_cli_prompt_context(self):
+ """
+ Make sure we are in the operational cli mode
+ :return: None
+ """
+ if self._connection.connected:
+ self._update_cli_prompt_context(
+ config_context="(config", exit_command="abort"
+ )
+
+ def _get_command_with_output(self, command, output):
+ options_values = self.get_option_values()
+ if output not in options_values["output"]:
+ raise ValueError(
+ "'output' value %s is invalid. Valid values are %s"
+ % (output, ",".join(options_values["output"]))
+ )
+
+ if output == "json" and not command.endswith("| json"):
+ cmd = "%s | json" % command
+ else:
+ cmd = command
+ return cmd
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py
new file mode 100644
index 00000000..a9eeccd7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py
@@ -0,0 +1,118 @@
+# -*- coding: utf-8 -*-
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+# Copyright: (c) 2015, Peter Sprygada <psprygada@ansible.com>
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+
+class ModuleDocFragment(object):
+
+ # Standard files documentation fragment
+ DOCUMENTATION = r"""options:
+ provider:
+ description:
+ - B(Deprecated)
+ - 'Starting with Ansible 2.5 we recommend using C(connection: network_cli).'
+ - 'Starting with Ansible 2.6 we recommend using C(connection: httpapi) for eAPI.'
+ - This option will be removed in a release after 2022-06-01.
+ - For more information please see the L(EOS Platform Options guide, ../network/user_guide/platform_eos.html).
+ - HORIZONTALLINE
+ - A dict object containing connection details.
+ type: dict
+ suboptions:
+ host:
+ description:
+ - Specifies the DNS host name or address for connecting to the remote device
+ over the specified transport. The value of host is used as the destination
+ address for the transport.
+ type: str
+ port:
+ description:
+ - Specifies the port to use when building the connection to the remote device. This
+ value applies to either I(cli) or I(eapi).
+ - The port value will default to the appropriate transport common port if
+ none is provided in the task (cli=22, http=80, https=443).
+ type: int
+ default: 0
+ username:
+ description:
+ - Configures the username to use to authenticate the connection to the remote
+ device. This value is used to authenticate either the CLI login or the
+ eAPI authentication depending on which transport is used. If the value is
+ not specified in the task, the value of environment variable C(ANSIBLE_NET_USERNAME)
+ will be used instead.
+ type: str
+ password:
+ description:
+ - Specifies the password to use to authenticate the connection to the remote
+ device. This is a common argument used for either I(cli) or I(eapi) transports.
+ If the value is not specified in the task, the value of environment variable
+ C(ANSIBLE_NET_PASSWORD) will be used instead.
+ type: str
+ timeout:
+ description:
+ - Specifies the timeout in seconds for communicating with the network device
+ for either connecting or sending commands. If the timeout is exceeded before
+ the operation is completed, the module will error.
+ type: int
+ ssh_keyfile:
+ description:
+ - Specifies the SSH keyfile to use to authenticate the connection to the remote
+ device. This argument is only used for I(cli) transports. If the value
+ is not specified in the task, the value of environment variable C(ANSIBLE_NET_SSH_KEYFILE)
+ will be used instead.
+ type: path
+ authorize:
+ description:
+ - Instructs the module to enter privileged mode on the remote device before
+ sending any commands. If not specified, the device will attempt to execute
+ all commands in non-privileged mode. If the value is not specified in the
+ task, the value of environment variable C(ANSIBLE_NET_AUTHORIZE) will be
+ used instead.
+ type: bool
+ default: false
+ auth_pass:
+ description:
+ - Specifies the password to use if required to enter privileged mode on the
+ remote device. If I(authorize) is false, then this argument does nothing.
+ If the value is not specified in the task, the value of environment variable
+ C(ANSIBLE_NET_AUTH_PASS) will be used instead.
+ type: str
+ transport:
+ description:
+ - Configures the transport connection to use when connecting to the remote
+ device.
+ type: str
+ choices:
+ - cli
+ - eapi
+ default: cli
+ use_ssl:
+ description:
+ - Configures the I(transport) to use SSL if set to C(yes) only when the C(transport=eapi). If
+ the transport argument is not eapi, this value is ignored.
+ type: bool
+ default: true
+ validate_certs:
+ description:
+ - If C(no), SSL certificates will not be validated. This should only be used
+ on personally controlled sites using self-signed certificates. If the transport
+ argument is not eapi, this value is ignored.
+ type: bool
+ default: true
+ use_proxy:
+ description:
+ - If C(no), the environment variables C(http_proxy) and C(https_proxy) will
+ be ignored.
+ type: bool
+ default: true
+notes:
+- For information on using CLI, eAPI and privileged mode see the :ref:`EOS Platform
+ Options guide <eos_platform_options>`
+- For more information on using Ansible to manage network devices see the :ref:`Ansible
+ Network Guide <network_guide>`
+- For more information on using Ansible to manage Arista EOS devices see the `Arista
+ integration page <https://www.ansible.com/ansible-arista-networks>`_.
+"""
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py
new file mode 100644
index 00000000..ae4203db
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py
@@ -0,0 +1,207 @@
+# (c) 2018 Red Hat Inc.
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+author: Ansible Networking Team
+httpapi: eos
+short_description: Use eAPI to run command on eos platform
+description:
+- This eos plugin provides low level abstraction api's for sending and receiving CLI
+ commands with eos network devices.
+version_added: 1.0.0
+options:
+ eos_use_sessions:
+ type: int
+ default: 1
+ description:
+ - Specifies if sessions should be used on remote host or not
+ env:
+ - name: ANSIBLE_EOS_USE_SESSIONS
+ vars:
+ - name: ansible_eos_use_sessions
+"""
+
+import json
+import time
+
+from ansible.module_utils._text import to_text
+from ansible.module_utils.connection import ConnectionError
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible.plugins.httpapi import HttpApiBase
+
+
+OPTIONS = {
+ "format": ["text", "json"],
+ "diff_match": ["line", "strict", "exact", "none"],
+ "diff_replace": ["line", "block", "config"],
+ "output": ["text", "json"],
+}
+
+
+class HttpApi(HttpApiBase):
+ def __init__(self, *args, **kwargs):
+ super(HttpApi, self).__init__(*args, **kwargs)
+ self._device_info = None
+ self._session_support = None
+
+ def supports_sessions(self):
+ use_session = self.get_option("eos_use_sessions")
+ try:
+ use_session = int(use_session)
+ except ValueError:
+ pass
+
+ if not bool(use_session):
+ self._session_support = False
+ else:
+ if self._session_support:
+ return self._session_support
+
+ response = self.send_request("show configuration sessions")
+ self._session_support = "error" not in response
+
+ return self._session_support
+
+ def send_request(self, data, **message_kwargs):
+ data = to_list(data)
+ become = self._become
+ if become:
+ self.connection.queue_message("vvvv", "firing event: on_become")
+ data.insert(0, {"cmd": "enable", "input": self._become_pass})
+
+ output = message_kwargs.get("output") or "text"
+ request = request_builder(data, output)
+ headers = {"Content-Type": "application/json-rpc"}
+
+ _response, response_data = self.connection.send(
+ "/command-api", request, headers=headers, method="POST"
+ )
+
+ try:
+ response_data = json.loads(to_text(response_data.getvalue()))
+ except ValueError:
+ raise ConnectionError(
+ "Response was not valid JSON, got {0}".format(
+ to_text(response_data.getvalue())
+ )
+ )
+
+ results = handle_response(response_data)
+
+ if become:
+ results = results[1:]
+ if len(results) == 1:
+ results = results[0]
+
+ return results
+
+ def get_device_info(self):
+ if self._device_info:
+ return self._device_info
+
+ device_info = {}
+
+ device_info["network_os"] = "eos"
+ reply = self.send_request("show version", output="json")
+ data = json.loads(reply)
+
+ device_info["network_os_version"] = data["version"]
+ device_info["network_os_model"] = data["modelName"]
+
+ reply = self.send_request("show hostname", output="json")
+ data = json.loads(reply)
+
+ device_info["network_os_hostname"] = data["hostname"]
+
+ self._device_info = device_info
+ return self._device_info
+
+ def get_device_operations(self):
+ return {
+ "supports_diff_replace": True,
+ "supports_commit": bool(self.supports_sessions()),
+ "supports_rollback": False,
+ "supports_defaults": False,
+ "supports_onbox_diff": bool(self.supports_sessions()),
+ "supports_commit_comment": False,
+ "supports_multiline_delimiter": False,
+ "supports_diff_match": True,
+ "supports_diff_ignore_lines": True,
+ "supports_generate_diff": not bool(self.supports_sessions()),
+ "supports_replace": bool(self.supports_sessions()),
+ }
+
+ def get_capabilities(self):
+ result = {}
+ result["rpc"] = []
+ result["device_info"] = self.get_device_info()
+ result["device_operations"] = self.get_device_operations()
+ result.update(OPTIONS)
+ result["network_api"] = "eapi"
+
+ return json.dumps(result)
+
+ # Shims for resource module support
+ def get(self, command, output=None):
+ # This method is ONLY here to support resource modules. Therefore most
+ # arguments are unsupported and not present.
+
+ return self.send_request(data=command, output=output)
+
+ def edit_config(self, candidate):
+ # This method is ONLY here to support resource modules. Therefore most
+ # arguments are unsupported and not present.
+
+ session = None
+ if self.supports_sessions():
+ session = "ansible_%d" % int(time.time())
+ candidate = ["configure session %s" % session] + candidate
+ else:
+ candidate = ["configure"] + candidate
+ candidate.append("commit")
+
+ try:
+ responses = self.send_request(candidate)
+ except ConnectionError:
+ if session:
+ self.send_request(["configure session %s" % session, "abort"])
+ raise
+
+ return [resp for resp in to_list(responses) if resp != "{}"]
+
+
+def handle_response(response):
+ if "error" in response:
+ error = response["error"]
+
+ error_text = []
+ for data in error.get("data", []):
+ error_text.extend(data.get("errors", []))
+ error_text = "\n".join(error_text) or error["message"]
+
+ raise ConnectionError(error_text, code=error["code"])
+
+ results = []
+
+ for result in response["result"]:
+ if "messages" in result:
+ results.append(result["messages"][0])
+ elif "output" in result:
+ results.append(result["output"].strip())
+ else:
+ results.append(json.dumps(result))
+
+ return results
+
+
+def request_builder(commands, output, reqid=None):
+ params = dict(version=1, cmds=commands, format=output)
+ return json.dumps(
+ dict(jsonrpc="2.0", id=reqid, method="runCmds", params=params)
+ )
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py
new file mode 100644
index 00000000..ae20ee94
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py
@@ -0,0 +1,85 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+"""
+The arg spec for the eos_acl_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Acl_interfacesArgs(object): # pylint: disable=R0903
+ """The arg spec for the eos_acl_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "access_groups": {
+ "elements": "dict",
+ "options": {
+ "acls": {
+ "elements": "dict",
+ "options": {
+ "direction": {
+ "required": True,
+ "choices": ["in", "out"],
+ "type": "str",
+ },
+ "name": {"required": True, "type": "str"},
+ },
+ "type": "list",
+ },
+ "afi": {
+ "required": True,
+ "choices": ["ipv4", "ipv6"],
+ "type": "str",
+ },
+ },
+ "type": "list",
+ },
+ "name": {"required": True, "type": "str"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "parsed",
+ "rendered",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py
new file mode 100644
index 00000000..b6d09a38
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py
@@ -0,0 +1,416 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+"""
+The arg spec for the eos_acls module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class AclsArgs(object): # pylint: disable=R0903
+ """The arg spec for the eos_acls module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "acls": {
+ "elements": "dict",
+ "options": {
+ "aces": {
+ "elements": "dict",
+ "options": {
+ "destination": {
+ "mutually_exclusive": [
+ [
+ "address",
+ "subnet_address",
+ "any",
+ "host",
+ ],
+ [
+ "wildcard_bits",
+ "subnet_address",
+ "any",
+ "host",
+ ],
+ ],
+ "options": {
+ "address": {"type": "str"},
+ "any": {"type": "bool"},
+ "host": {"type": "str"},
+ "port_protocol": {"type": "dict"},
+ "subnet_address": {"type": "str"},
+ "wildcard_bits": {"type": "str"},
+ },
+ "required_together": [
+ ["address", "wildcard_bits"]
+ ],
+ "type": "dict",
+ },
+ "fragment_rules": {"type": "bool"},
+ "fragments": {"type": "bool"},
+ "grant": {
+ "choices": ["permit", "deny"],
+ "type": "str",
+ },
+ "line": {"type": "str", "aliases": ["ace"]},
+ "hop_limit": {"type": "dict"},
+ "log": {"type": "bool"},
+ "protocol": {"type": "str"},
+ "protocol_options": {
+ "options": {
+ "icmp": {
+ "options": {
+ "administratively_prohibited": {
+ "type": "bool"
+ },
+ "alternate_address": {
+ "type": "bool"
+ },
+ "conversion_error": {
+ "type": "bool"
+ },
+ "dod_host_prohibited": {
+ "type": "bool"
+ },
+ "dod_net_prohibited": {
+ "type": "bool"
+ },
+ "echo": {"type": "bool"},
+ "echo_reply": {"type": "bool"},
+ "general_parameter_problem": {
+ "type": "bool"
+ },
+ "host_isolated": {
+ "type": "bool"
+ },
+ "host_precedence_unreachable": {
+ "type": "bool"
+ },
+ "host_redirect": {
+ "type": "bool"
+ },
+ "host_tos_redirect": {
+ "type": "bool"
+ },
+ "host_tos_unreachable": {
+ "type": "bool"
+ },
+ "host_unknown": {
+ "type": "bool"
+ },
+ "host_unreachable": {
+ "type": "bool"
+ },
+ "information_reply": {
+ "type": "bool"
+ },
+ "information_request": {
+ "type": "bool"
+ },
+ "mask_reply": {"type": "bool"},
+ "mask_request": {
+ "type": "bool"
+ },
+ "message_code": {
+ "type": "int"
+ },
+ "message_num": {"type": "int"},
+ "message_type": {
+ "type": "int"
+ },
+ "mobile_redirect": {
+ "type": "bool"
+ },
+ "net_redirect": {
+ "type": "bool"
+ },
+ "net_tos_redirect": {
+ "type": "bool"
+ },
+ "net_tos_unreachable": {
+ "type": "bool"
+ },
+ "net_unreachable": {
+ "type": "bool"
+ },
+ "network_unknown": {
+ "type": "bool"
+ },
+ "no_room_for_option": {
+ "type": "bool"
+ },
+ "option_missing": {
+ "type": "bool"
+ },
+ "packet_too_big": {
+ "type": "bool"
+ },
+ "parameter_problem": {
+ "type": "bool"
+ },
+ "port_unreachable": {
+ "type": "bool"
+ },
+ "precedence_unreachable": {
+ "type": "bool"
+ },
+ "protocol_unreachable": {
+ "type": "bool"
+ },
+ "reassembly_timeout": {
+ "type": "bool"
+ },
+ "redirect": {"type": "bool"},
+ "router_advertisement": {
+ "type": "bool"
+ },
+ "router_solicitation": {
+ "type": "bool"
+ },
+ "source_quench": {
+ "type": "bool"
+ },
+ "source_route_failed": {
+ "type": "bool"
+ },
+ "time_exceeded": {
+ "type": "bool"
+ },
+ "timestamp_reply": {
+ "type": "bool"
+ },
+ "timestamp_request": {
+ "type": "bool"
+ },
+ "traceroute": {"type": "bool"},
+ "ttl_exceeded": {
+ "type": "bool"
+ },
+ "unreachable": {
+ "type": "bool"
+ },
+ },
+ "type": "dict",
+ },
+ "icmpv6": {
+ "options": {
+ "address_unreachable": {
+ "type": "bool"
+ },
+ "beyond_scope": {
+ "type": "bool"
+ },
+ "echo_reply": {"type": "bool"},
+ "echo_request": {
+ "type": "bool"
+ },
+ "erroneous_header": {
+ "type": "bool"
+ },
+ "fragment_reassembly_exceeded": {
+ "type": "bool"
+ },
+ "hop_limit_exceeded": {
+ "type": "bool"
+ },
+ "neighbor_advertisement": {
+ "type": "bool"
+ },
+ "neighbor_solicitation": {
+ "type": "bool"
+ },
+ "no_admin": {"type": "bool"},
+ "no_route": {"type": "bool"},
+ "packet_too_big": {
+ "type": "bool"
+ },
+ "parameter_problem": {
+ "type": "bool"
+ },
+ "port_unreachable": {
+ "type": "bool"
+ },
+ "redirect_message": {
+ "type": "bool"
+ },
+ "reject_route": {
+ "type": "bool"
+ },
+ "router_advertisement": {
+ "type": "bool"
+ },
+ "router_solicitation": {
+ "type": "bool"
+ },
+ "source_address_failed": {
+ "type": "bool"
+ },
+ "source_routing_error": {
+ "type": "bool"
+ },
+ "time_exceeded": {
+ "type": "bool"
+ },
+ "unreachable": {
+ "type": "bool"
+ },
+ "unrecognized_ipv6_option": {
+ "type": "bool"
+ },
+ "unrecognized_next_header": {
+ "type": "bool"
+ },
+ },
+ "type": "dict",
+ },
+ "ip": {
+ "options": {
+ "nexthop_group": {
+ "type": "str"
+ }
+ },
+ "type": "dict",
+ },
+ "ipv6": {
+ "options": {
+ "nexthop_group": {
+ "type": "str"
+ }
+ },
+ "type": "dict",
+ },
+ "tcp": {
+ "options": {
+ "flags": {
+ "options": {
+ "ack": {
+ "type": "bool"
+ },
+ "established": {
+ "type": "bool"
+ },
+ "fin": {
+ "type": "bool"
+ },
+ "psh": {
+ "type": "bool"
+ },
+ "rst": {
+ "type": "bool"
+ },
+ "syn": {
+ "type": "bool"
+ },
+ "urg": {
+ "type": "bool"
+ },
+ },
+ "type": "dict",
+ }
+ },
+ "type": "dict",
+ },
+ },
+ "type": "dict",
+ },
+ "remark": {"type": "str"},
+ "sequence": {"type": "int"},
+ "source": {
+ "mutually_exclusive": [
+ [
+ "address",
+ "subnet_address",
+ "any",
+ "host",
+ ],
+ [
+ "wildcard_bits",
+ "subnet_address",
+ "any",
+ "host",
+ ],
+ ],
+ "options": {
+ "address": {"type": "str"},
+ "any": {"type": "bool"},
+ "host": {"type": "str"},
+ "port_protocol": {"type": "dict"},
+ "subnet_address": {"type": "str"},
+ "wildcard_bits": {"type": "str"},
+ },
+ "required_together": [
+ ["address", "wildcard_bits"]
+ ],
+ "type": "dict",
+ },
+ "tracked": {"type": "bool"},
+ "ttl": {
+ "options": {
+ "eq": {"type": "int"},
+ "gt": {"type": "int"},
+ "lt": {"type": "int"},
+ "neq": {"type": "int"},
+ },
+ "type": "dict",
+ },
+ "vlan": {"type": "str"},
+ },
+ "type": "list",
+ },
+ "name": {"required": True, "type": "str"},
+ "standard": {"type": "bool"},
+ },
+ "type": "list",
+ },
+ "afi": {
+ "choices": ["ipv4", "ipv6"],
+ "required": True,
+ "type": "str",
+ },
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "deleted",
+ "merged",
+ "overridden",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py
new file mode 100644
index 00000000..aa488920
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py
@@ -0,0 +1,195 @@
+# -*- coding: utf-8 -*-
+# Copyright 2021 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_bgp_address_family module
+"""
+
+
+class Bgp_afArgs(object): # pylint: disable=R0903
+ """The arg spec for the eos_bgp_address_family module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "type": "str",
+ "choices": [
+ "deleted",
+ "merged",
+ "overridden",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ },
+ "config": {
+ "type": "dict",
+ "options": {
+ "as_number": {"type": "str"},
+ "address_family": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "network": {
+ "type": "list",
+ "elements": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "address": {"type": "str"},
+ },
+ },
+ "redistribute": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "ospf_route": {
+ "type": "str",
+ "choices": [
+ "internal",
+ "external",
+ "nssa_external_1",
+ "nssa_external_2",
+ ],
+ },
+ "route_map": {"type": "str"},
+ "protocol": {
+ "type": "str",
+ "choices": ["isis", "ospf3", "dhcp"],
+ },
+ "isis_level": {
+ "type": "str",
+ "choices": [
+ "level-1",
+ "level-2",
+ "level-1-2",
+ ],
+ },
+ },
+ },
+ "route_target": {
+ "type": "dict",
+ "options": {
+ "mode": {
+ "type": "str",
+ "choices": ["both", "import", "export"],
+ },
+ "target": {"type": "str"},
+ },
+ },
+ "graceful_restart": {"type": "bool"},
+ "bgp_params": {
+ "type": "dict",
+ "options": {
+ "next_hop_address_family": {
+ "type": "str",
+ "choices": ["ipv6"],
+ },
+ "redistribute_internal": {"type": "bool"},
+ "route": {"type": "str"},
+ "additional_paths": {
+ "type": "str",
+ "choices": ["install", "send", "receive"],
+ },
+ "next_hop_unchanged": {"type": "bool"},
+ },
+ },
+ "safi": {
+ "type": "str",
+ "choices": ["labeled-unicast", "multicast"],
+ },
+ "neighbor": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "activate": {"type": "bool"},
+ "graceful_restart": {"type": "bool"},
+ "weight": {"type": "int"},
+ "default_originate": {
+ "type": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "always": {"type": "bool"},
+ },
+ },
+ "route_map": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "next_hop_address_family": {
+ "type": "str",
+ "choices": ["ipv6"],
+ },
+ "additional_paths": {
+ "type": "str",
+ "choices": ["send", "receive"],
+ },
+ "peer": {"type": "str"},
+ "encapsulation": {
+ "type": "dict",
+ "options": {
+ "transport": {
+ "type": "str",
+ "choices": ["mpls", "vxlan"],
+ },
+ "source_interface": {"type": "str"},
+ },
+ },
+ "prefix_list": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "next_hop_unchanged": {"type": "bool"},
+ },
+ },
+ "afi": {
+ "type": "str",
+ "choices": ["ipv4", "ipv6", "evpn"],
+ },
+ "vrf": {"type": "str"},
+ },
+ },
+ },
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py
new file mode 100644
index 00000000..cd291d05
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py
@@ -0,0 +1,999 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_bgp_global module
+"""
+
+
+class Bgp_globalArgs(object): # pylint: disable=R0903
+ """The arg spec for the eos_bgp_global module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "type": "str",
+ "choices": [
+ "deleted",
+ "purged",
+ "merged",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ },
+ "config": {
+ "type": "dict",
+ "options": {
+ "router_id": {"type": "str"},
+ "as_number": {"type": "str"},
+ "redistribute": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "ospf_route": {
+ "type": "str",
+ "choices": [
+ "internal",
+ "external",
+ "nssa_external_1",
+ "nssa_external_2",
+ ],
+ },
+ "route_map": {"type": "str"},
+ "protocol": {
+ "type": "str",
+ "choices": [
+ "isis",
+ "ospf3",
+ "ospf",
+ "attached-host",
+ "connected",
+ "rip",
+ "static",
+ ],
+ },
+ "isis_level": {
+ "type": "str",
+ "choices": ["level-1", "level-2", "level-1-2"],
+ },
+ },
+ },
+ "monitoring": {
+ "type": "dict",
+ "options": {
+ "received": {
+ "type": "str",
+ "choices": ["post_policy", "pre_policy"],
+ },
+ "station": {"type": "str"},
+ "port": {"type": "int"},
+ "timestamp": {
+ "type": "str",
+ "choices": ["none", "send_time"],
+ },
+ },
+ },
+ "default_metric": {"type": "int"},
+ "bgp_params": {
+ "type": "dict",
+ "options": {
+ "labeled_unicast": {
+ "type": "str",
+ "choices": ["ip", "tunnel"],
+ },
+ "host_routes": {"type": "bool"},
+ "transport": {"type": "int"},
+ "next_hop_unchanged": {"type": "bool"},
+ "missing_policy": {
+ "type": "dict",
+ "options": {
+ "action": {
+ "type": "str",
+ "choices": [
+ "deny",
+ "permit",
+ "deny-in-out",
+ ],
+ },
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ },
+ },
+ "monitoring": {"type": "bool"},
+ "allowas_in": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "additional_paths": {
+ "type": "str",
+ "choices": ["install", "send", "receive"],
+ },
+ "advertise_inactive": {"type": "bool"},
+ "listen": {
+ "type": "dict",
+ "options": {
+ "range": {
+ "type": "dict",
+ "options": {
+ "peer_group": {
+ "type": "dict",
+ "options": {
+ "peer_filter": {"type": "str"},
+ "remote_as": {"type": "str"},
+ "name": {"type": "str"},
+ },
+ },
+ "address": {"type": "str"},
+ },
+ },
+ "limit": {"type": "int"},
+ },
+ },
+ "route_reflector": {
+ "type": "dict",
+ "options": {
+ "preserve": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ },
+ "always_compare_med": {"type": "bool"},
+ "client_to_client": {"type": "bool"},
+ "bestpath": {
+ "type": "dict",
+ "options": {
+ "ecmp_fast": {"type": "bool"},
+ "tie_break": {
+ "type": "str",
+ "choices": [
+ "cluster_list_length",
+ "router_id",
+ ],
+ },
+ "skip": {"type": "bool"},
+ "as_path": {
+ "type": "str",
+ "choices": ["ignore", "multipath_relax"],
+ },
+ "med": {
+ "type": "dict",
+ "options": {
+ "confed": {"type": "bool"},
+ "missing_as_worst": {"type": "bool"},
+ },
+ },
+ },
+ },
+ "convergence": {
+ "type": "dict",
+ "options": {
+ "slow_peer": {"type": "bool"},
+ "time": {"type": "int"},
+ },
+ },
+ "log_neighbor_changes": {"type": "bool"},
+ "asn": {
+ "type": "str",
+ "choices": ["asdot", "asplain"],
+ },
+ "default": {
+ "type": "str",
+ "choices": ["ipv4_unicast", "ipv6_unicast"],
+ },
+ "route": {"type": "str"},
+ "enforce_first_as": {"type": "bool"},
+ "auto_local_addr": {"type": "bool"},
+ "redistribute_internal": {"type": "bool"},
+ "cluster_id": {"type": "str"},
+ "control_plan_filter": {"type": "bool"},
+ "confederation": {
+ "type": "dict",
+ "options": {
+ "peers": {"type": "str"},
+ "identifier": {"type": "str"},
+ },
+ },
+ },
+ },
+ "vlan": {"type": "int"},
+ "update": {
+ "type": "dict",
+ "options": {
+ "wait_for": {
+ "type": "str",
+ "choices": [
+ "wait_for_convergence",
+ "wait_install",
+ ],
+ },
+ "batch_size": {"type": "int"},
+ },
+ },
+ "vlan_aware_bundle": {"type": "str"},
+ "aggregate_address": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "advertise_only": {"type": "bool"},
+ "match_map": {"type": "str"},
+ "attribute_map": {"type": "str"},
+ "as_set": {"type": "bool"},
+ "summary_only": {"type": "bool"},
+ "address": {"type": "str"},
+ },
+ },
+ "neighbor": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "weight": {"type": "int"},
+ "default_originate": {
+ "type": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "always": {"type": "bool"},
+ },
+ },
+ "next_hop_v6_address": {"type": "str"},
+ "route_reflector_client": {"type": "bool"},
+ "ttl": {"type": "int"},
+ "remove_private_as": {
+ "type": "dict",
+ "options": {
+ "all": {"type": "bool"},
+ "set": {"type": "bool"},
+ "replace_as": {"type": "bool"},
+ },
+ },
+ "local_v6_addr": {"type": "str"},
+ "transport": {
+ "type": "dict",
+ "options": {
+ "connection_mode": {"type": "str"},
+ "remote_port": {"type": "int"},
+ },
+ },
+ "next_hop_unchanged": {"type": "bool"},
+ "monitoring": {"type": "bool"},
+ "ebgp_multihop": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "ttl": {"type": "int"},
+ },
+ },
+ "shut_down": {"type": "bool"},
+ "fall_over": {"type": "bool"},
+ "idle_restart_timer": {"type": "int"},
+ "allowas_in": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "additional_paths": {
+ "type": "str",
+ "choices": ["send", "receive"],
+ },
+ "peer_group": {"type": "str"},
+ "out_delay": {"type": "int"},
+ "import_localpref": {"type": "int"},
+ "prefix_list": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "dont_capability_negotiate": {"type": "bool"},
+ "update_source": {"type": "str"},
+ "export_localpref": {"type": "int"},
+ "local_as": {
+ "type": "dict",
+ "options": {
+ "as_number": {"type": "str"},
+ "fallback": {"type": "bool"},
+ },
+ },
+ "maximum_received_routes": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "warning_limit": {
+ "type": "dict",
+ "options": {
+ "limit_count": {"type": "int"},
+ "limit_percent": {"type": "int"},
+ },
+ },
+ "warning_only": {"type": "bool"},
+ },
+ },
+ "encryption_password": {
+ "type": "dict",
+ "options": {
+ "password": {"type": "str"},
+ "type": {"type": "int", "choices": [0, 7]},
+ },
+ },
+ "link_bandwidth": {
+ "type": "dict",
+ "options": {
+ "default": {"type": "str"},
+ "auto": {"type": "bool"},
+ "set": {"type": "bool"},
+ "update_delay": {"type": "int"},
+ },
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "peer": {"type": "str"},
+ "next_hop_self": {"type": "bool"},
+ "route_to_peer": {"type": "bool"},
+ "soft_recognition": {
+ "type": "str",
+ "choices": ["all", "None"],
+ },
+ "graceful_restart": {"type": "bool"},
+ "enforce_first_as": {"type": "bool"},
+ "send_community": {
+ "type": "dict",
+ "options": {
+ "community_attribute": {"type": "str"},
+ "sub_attribute": {
+ "type": "str",
+ "choices": [
+ "extended",
+ "link-bandwidth",
+ "standard",
+ ],
+ },
+ "speed": {"type": "str"},
+ "divide": {
+ "type": "str",
+ "choices": ["equal", "ratio"],
+ },
+ "link_bandwidth_attribute": {
+ "type": "str",
+ "choices": ["aggregate", "divide"],
+ },
+ },
+ },
+ "description": {"type": "str"},
+ "maximum_accepted_routes": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "warning_limit": {"type": "int"},
+ },
+ },
+ "auto_local_addr": {"type": "bool"},
+ "metric_out": {"type": "int"},
+ "timers": {
+ "type": "dict",
+ "options": {
+ "holdtime": {"type": "int"},
+ "keepalive": {"type": "int"},
+ },
+ },
+ "route_map": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "remote_as": {"type": "str"},
+ },
+ },
+ "graceful_restart": {
+ "type": "dict",
+ "options": {
+ "stalepath_time": {"type": "int"},
+ "restart_time": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "timers": {
+ "type": "dict",
+ "options": {
+ "holdtime": {"type": "int"},
+ "keepalive": {"type": "int"},
+ },
+ },
+ "distance": {
+ "type": "dict",
+ "options": {
+ "internal": {"type": "int"},
+ "local": {"type": "int"},
+ "external": {"type": "int"},
+ },
+ },
+ "route_target": {
+ "type": "dict",
+ "options": {
+ "action": {
+ "type": "str",
+ "choices": ["both", "import", "export"],
+ },
+ "target": {"type": "str"},
+ },
+ },
+ "vrfs": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "router_id": {"type": "str"},
+ "vrf": {"type": "str"},
+ "route_target": {
+ "type": "dict",
+ "options": {
+ "action": {
+ "type": "str",
+ "choices": ["both", "import", "export"],
+ },
+ "target": {"type": "str"},
+ },
+ },
+ "redistribute": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "ospf_route": {
+ "type": "str",
+ "choices": [
+ "internal",
+ "external",
+ "nssa_external_1",
+ "nssa_external_2",
+ ],
+ },
+ "route_map": {"type": "str"},
+ "protocol": {
+ "type": "str",
+ "choices": [
+ "isis",
+ "ospf3",
+ "ospf",
+ "attached-host",
+ "connected",
+ "rip",
+ "static",
+ ],
+ },
+ "isis_level": {
+ "type": "str",
+ "choices": [
+ "level-1",
+ "level-2",
+ "level-1-2",
+ ],
+ },
+ },
+ },
+ "distance": {
+ "type": "dict",
+ "options": {
+ "internal": {"type": "int"},
+ "local": {"type": "int"},
+ "external": {"type": "int"},
+ },
+ },
+ "default_metric": {"type": "int"},
+ "bgp_params": {
+ "type": "dict",
+ "options": {
+ "control_plane_filter": {"type": "bool"},
+ "convergence": {
+ "type": "dict",
+ "options": {
+ "slow_peer": {"type": "bool"},
+ "time": {"type": "int"},
+ },
+ },
+ "host_routes": {"type": "bool"},
+ "transport": {"type": "int"},
+ "next_hop_unchanged": {"type": "bool"},
+ "missing_policy": {
+ "type": "dict",
+ "options": {
+ "action": {
+ "type": "str",
+ "choices": [
+ "deny",
+ "permit",
+ "deny-in-out",
+ ],
+ },
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ },
+ },
+ "monitoring": {"type": "bool"},
+ "allowas_in": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "additional_paths": {
+ "type": "str",
+ "choices": ["install", "send", "receive"],
+ },
+ "advertise_inactive": {"type": "bool"},
+ "listen": {
+ "type": "dict",
+ "options": {
+ "range": {
+ "type": "dict",
+ "options": {
+ "peer_group": {
+ "type": "dict",
+ "options": {
+ "peer_filter": {
+ "type": "str"
+ },
+ "remote_as": {
+ "type": "str"
+ },
+ "name": {
+ "type": "str"
+ },
+ },
+ },
+ "address": {"type": "str"},
+ },
+ },
+ "limit": {"type": "int"},
+ },
+ },
+ "route_reflector": {
+ "type": "dict",
+ "options": {
+ "preserve": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ },
+ "always_compare_med": {"type": "bool"},
+ "client_to_client": {"type": "bool"},
+ "bestpath": {
+ "type": "dict",
+ "options": {
+ "ecmp_fast": {"type": "bool"},
+ "tie_break": {
+ "type": "str",
+ "choices": [
+ "cluster_list_length",
+ "router_id",
+ ],
+ },
+ "skip": {"type": "bool"},
+ "as_path": {
+ "type": "str",
+ "choices": [
+ "ignore",
+ "multipath_relax",
+ ],
+ },
+ "med": {
+ "type": "dict",
+ "options": {
+ "confed": {"type": "bool"},
+ "missing_as_worst": {
+ "type": "bool"
+ },
+ },
+ },
+ },
+ },
+ "labeled_unicast": {
+ "type": "str",
+ "choices": ["ip", "tunnel"],
+ },
+ "log_neighbor_changes": {"type": "bool"},
+ "asn": {
+ "type": "str",
+ "choices": ["asdot", "asplain"],
+ },
+ "default": {
+ "type": "str",
+ "choices": [
+ "ipv4_unicast",
+ "ipv6_unicast",
+ ],
+ },
+ "route": {"type": "str"},
+ "enforce_first_as": {"type": "bool"},
+ "auto_local_addr": {"type": "bool"},
+ "redistribute_internal": {"type": "bool"},
+ "cluster_id": {"type": "str"},
+ "confederation": {
+ "type": "dict",
+ "options": {
+ "peers": {"type": "str"},
+ "identifier": {"type": "str"},
+ },
+ },
+ },
+ },
+ "update": {
+ "type": "dict",
+ "options": {
+ "wait_for": {
+ "type": "str",
+ "choices": [
+ "wait_for_convergence",
+ "wait_install",
+ ],
+ },
+ "batch_size": {"type": "int"},
+ },
+ },
+ "aggregate_address": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "advertise_only": {"type": "bool"},
+ "match_map": {"type": "str"},
+ "attribute_map": {"type": "str"},
+ "as_set": {"type": "bool"},
+ "summary_only": {"type": "bool"},
+ "address": {"type": "str"},
+ },
+ },
+ "neighbor": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "weight": {"type": "int"},
+ "default_originate": {
+ "type": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "always": {"type": "bool"},
+ },
+ },
+ "next_hop_v6_address": {"type": "str"},
+ "route_reflector_client": {"type": "bool"},
+ "ttl": {"type": "int"},
+ "remove_private_as": {
+ "type": "dict",
+ "options": {
+ "all": {"type": "bool"},
+ "set": {"type": "bool"},
+ "replace_as": {"type": "bool"},
+ },
+ },
+ "local_v6_addr": {"type": "str"},
+ "transport": {
+ "type": "dict",
+ "options": {
+ "connection_mode": {"type": "str"},
+ "remote_port": {"type": "int"},
+ },
+ },
+ "next_hop_unchanged": {"type": "bool"},
+ "monitoring": {"type": "bool"},
+ "ebgp_multihop": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "ttl": {"type": "int"},
+ },
+ },
+ "shut_down": {"type": "bool"},
+ "fall_over": {"type": "bool"},
+ "idle_restart_timer": {"type": "int"},
+ "allowas_in": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "additional_paths": {
+ "type": "str",
+ "choices": ["send", "receive"],
+ },
+ "peer_group": {"type": "str"},
+ "out_delay": {"type": "int"},
+ "import_localpref": {"type": "int"},
+ "prefix_list": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "dont_capability_negotiate": {"type": "bool"},
+ "update_source": {"type": "str"},
+ "export_localpref": {"type": "int"},
+ "local_as": {
+ "type": "dict",
+ "options": {
+ "as_number": {"type": "str"},
+ "fallback": {"type": "bool"},
+ },
+ },
+ "maximum_received_routes": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "warning_limit": {
+ "type": "dict",
+ "options": {
+ "limit_count": {"type": "int"},
+ "limit_percent": {
+ "type": "int"
+ },
+ },
+ },
+ "warning_only": {"type": "bool"},
+ },
+ },
+ "encryption_password": {
+ "type": "dict",
+ "options": {
+ "password": {"type": "str"},
+ "type": {
+ "type": "int",
+ "choices": [0, 7],
+ },
+ },
+ },
+ "link_bandwidth": {
+ "type": "dict",
+ "options": {
+ "default": {"type": "str"},
+ "auto": {"type": "bool"},
+ "set": {"type": "bool"},
+ "update_delay": {"type": "int"},
+ },
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "peer": {"type": "str"},
+ "next_hop_self": {"type": "bool"},
+ "route_to_peer": {"type": "bool"},
+ "soft_recognition": {
+ "type": "str",
+ "choices": ["all", "None"],
+ },
+ "graceful_restart": {"type": "bool"},
+ "enforce_first_as": {"type": "bool"},
+ "send_community": {
+ "type": "dict",
+ "options": {
+ "community_attribute": {"type": "str"},
+ "sub_attribute": {
+ "type": "str",
+ "choices": [
+ "extended",
+ "link-bandwidth",
+ "standard",
+ ],
+ },
+ "speed": {"type": "str"},
+ "divide": {
+ "type": "str",
+ "choices": ["equal", "ratio"],
+ },
+ "link_bandwidth_attribute": {
+ "type": "str",
+ "choices": ["aggregate", "divide"],
+ },
+ },
+ },
+ "description": {"type": "str"},
+ "maximum_accepted_routes": {
+ "type": "dict",
+ "options": {
+ "count": {"type": "int"},
+ "warning_limit": {"type": "int"},
+ },
+ },
+ "auto_local_addr": {"type": "bool"},
+ "metric_out": {"type": "int"},
+ "timers": {
+ "type": "dict",
+ "options": {
+ "holdtime": {"type": "int"},
+ "keepalive": {"type": "int"},
+ },
+ },
+ "route_map": {
+ "type": "dict",
+ "options": {
+ "direction": {
+ "type": "str",
+ "choices": ["in", "out"],
+ },
+ "name": {"type": "str"},
+ },
+ },
+ "remote_as": {"type": "str"},
+ },
+ },
+ "graceful_restart": {
+ "type": "dict",
+ "options": {
+ "stalepath_time": {"type": "int"},
+ "restart_time": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "timers": {
+ "type": "dict",
+ "options": {
+ "holdtime": {"type": "int"},
+ "keepalive": {"type": "int"},
+ },
+ },
+ "shutdown": {"type": "bool"},
+ "access_group": {
+ "type": "dict",
+ "options": {
+ "direction": {"type": "str"},
+ "afi": {
+ "type": "str",
+ "choices": ["ip", "ipv6"],
+ },
+ "acl_name": {"type": "str"},
+ },
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "ucmp": {
+ "type": "dict",
+ "options": {
+ "link_bandwidth": {
+ "type": "dict",
+ "options": {
+ "update_delay": {"type": "int"},
+ "mode": {
+ "type": "str",
+ "choices": [
+ "encoding_weighted",
+ "recursive",
+ ],
+ },
+ },
+ },
+ "fec": {
+ "type": "dict",
+ "options": {
+ "clear": {"type": "int"},
+ "trigger": {"type": "int"},
+ },
+ },
+ "mode": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "nexthops": {"type": "int"},
+ },
+ },
+ },
+ },
+ "maximum_paths": {
+ "type": "dict",
+ "options": {
+ "max_equal_cost_paths": {"type": "int"},
+ "max_installed_ecmp_paths": {"type": "int"},
+ },
+ },
+ "network": {
+ "type": "list",
+ "elements": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "address": {"type": "str"},
+ },
+ },
+ },
+ },
+ "access_group": {
+ "type": "dict",
+ "options": {
+ "direction": {"type": "str"},
+ "afi": {"type": "str", "choices": ["ipv4", "ipv6"]},
+ "acl_name": {"type": "str"},
+ },
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "ucmp": {
+ "type": "dict",
+ "options": {
+ "link_bandwidth": {
+ "type": "dict",
+ "options": {
+ "update_delay": {"type": "int"},
+ "mode": {
+ "type": "str",
+ "choices": [
+ "encoding_weighted",
+ "recursive",
+ ],
+ },
+ },
+ },
+ "fec": {
+ "type": "dict",
+ "options": {
+ "clear": {"type": "int"},
+ "trigger": {"type": "int"},
+ },
+ },
+ "mode": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "nexthops": {"type": "int"},
+ },
+ },
+ },
+ },
+ "shutdown": {"type": "bool"},
+ "maximum_paths": {
+ "type": "dict",
+ "options": {
+ "max_equal_cost_paths": {"type": "int"},
+ "max_installed_ecmp_paths": {"type": "int"},
+ },
+ },
+ "network": {
+ "type": "list",
+ "elements": "dict",
+ "options": {
+ "route_map": {"type": "str"},
+ "address": {"type": "str"},
+ },
+ },
+ },
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py
new file mode 100644
index 00000000..5d7566d9
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py
@@ -0,0 +1,25 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The arg spec for the eos facts module.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class FactsArgs(object):
+ """ The arg spec for the eos facts module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "gather_subset": dict(
+ default=["!config"], type="list", elements="str"
+ ),
+ "gather_network_resources": dict(type="list", elements="str"),
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py
new file mode 100644
index 00000000..00710be7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py
@@ -0,0 +1,72 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+##############################################
+# WARNING #
+##############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+##############################################
+
+"""
+The arg spec for the eos_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class InterfacesArgs(object):
+ """The arg spec for the eos_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "name": {"required": True, "type": "str"},
+ "description": {"required": False, "type": "str"},
+ "enabled": {
+ "default": True,
+ "required": False,
+ "type": "bool",
+ },
+ "mtu": {"required": False, "type": "int"},
+ "speed": {"required": False, "type": "str"},
+ "duplex": {"required": False, "type": "str"},
+ "mode": {"choices": ["layer2", "layer3"], "type": "str"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "required": False,
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py
new file mode 100644
index 00000000..2a8b7f86
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py
@@ -0,0 +1,77 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+##############################################
+# WARNING #
+##############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+##############################################
+
+"""
+The arg spec for the eos_l2_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class L2_interfacesArgs(object):
+ """The arg spec for the eos_l2_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "access": {
+ "options": {"vlan": {"type": "int"}},
+ "type": "dict",
+ },
+ "mode": {"type": "str", "choices": ["access", "trunk"]},
+ "name": {"required": True, "type": "str"},
+ "trunk": {
+ "options": {
+ "native_vlan": {"type": "int"},
+ "trunk_allowed_vlans": {
+ "type": "list",
+ "elements": "str",
+ },
+ },
+ "type": "dict",
+ },
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "required": False,
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py
new file mode 100644
index 00000000..ae8738fa
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py
@@ -0,0 +1,75 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_l3_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class L3_interfacesArgs(object):
+ """The arg spec for the eos_l3_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "name": {"required": True, "type": "str"},
+ "ipv4": {
+ "elements": "dict",
+ "options": {
+ "address": {"type": "str"},
+ "secondary": {"type": "bool"},
+ },
+ "type": "list",
+ },
+ "ipv6": {
+ "elements": "dict",
+ "options": {"address": {"type": "str"}},
+ "type": "list",
+ },
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py
new file mode 100644
index 00000000..6c913c01
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py
@@ -0,0 +1,63 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_lacp module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class LacpArgs(object):
+ """The arg spec for the eos_lacp module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "options": {
+ "system": {
+ "options": {"priority": {"type": "int"}},
+ "type": "dict",
+ }
+ },
+ "type": "dict",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py
new file mode 100644
index 00000000..802af365
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py
@@ -0,0 +1,65 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_lacp_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Lacp_interfacesArgs(object):
+ """The arg spec for the eos_lacp_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "name": {"type": "str"},
+ "port_priority": {"type": "int"},
+ "rate": {"choices": ["fast", "normal"], "type": "str"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py
new file mode 100644
index 00000000..f8792d5c
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py
@@ -0,0 +1,73 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_lag_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Lag_interfacesArgs(object):
+ """The arg spec for the eos_lag_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "name": {"required": True, "type": "str"},
+ "members": {
+ "elements": "dict",
+ "options": {
+ "member": {"type": "str"},
+ "mode": {
+ "choices": ["active", "on", "passive"],
+ "type": "str",
+ },
+ },
+ "type": "list",
+ },
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py
new file mode 100644
index 00000000..97094fac
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py
@@ -0,0 +1,75 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_lldp_global module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Lldp_globalArgs(object):
+ """The arg spec for the eos_lldp_global module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "options": {
+ "holdtime": {"type": "int"},
+ "reinit": {"type": "int"},
+ "timer": {"type": "int"},
+ "tlv_select": {
+ "options": {
+ "link_aggregation": {"type": "bool"},
+ "management_address": {"type": "bool"},
+ "max_frame_size": {"type": "bool"},
+ "port_description": {"type": "bool"},
+ "system_capabilities": {"type": "bool"},
+ "system_description": {"type": "bool"},
+ "system_name": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ },
+ "type": "dict",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py
new file mode 100644
index 00000000..01c00653
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py
@@ -0,0 +1,65 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_lldp_interfaces module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Lldp_interfacesArgs(object):
+ """The arg spec for the eos_lldp_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "name": {"type": "str"},
+ "receive": {"type": "bool"},
+ "transmit": {"type": "bool"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py
new file mode 100644
index 00000000..30a99536
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py
@@ -0,0 +1,178 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_ospf_interfaces module
+"""
+
+
+class Ospf_interfacesArgs(object): # pylint: disable=R0903
+ """The arg spec for the eos_ospf_interfaces module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "state": {
+ "default": "merged",
+ "type": "str",
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "parsed",
+ "rendered",
+ ],
+ },
+ "running_config": {"type": "str"},
+ "config": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "name": {"type": "str"},
+ "address_family": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "ip_params": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "retransmit_interval": {"type": "int"},
+ "cost": {"type": "int"},
+ "afi": {
+ "required": True,
+ "type": "str",
+ "choices": ["ipv4", "ipv6"],
+ },
+ "area": {
+ "type": "dict",
+ "options": {
+ "area_id": {
+ "required": True,
+ "type": "str",
+ }
+ },
+ },
+ "bfd": {"type": "bool"},
+ "mtu_ignore": {"type": "bool"},
+ "priority": {"type": "int"},
+ "dead_interval": {"type": "int"},
+ "hello_interval": {"type": "int"},
+ "passive_interface": {"type": "bool"},
+ "transmit_delay": {"type": "int"},
+ "network": {"type": "str"},
+ },
+ },
+ "encryption_v3": {
+ "type": "dict",
+ "options": {
+ "key": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["md5", "sha1"],
+ },
+ "encryption": {
+ "type": "str",
+ "choices": [
+ "3des-cbc",
+ "aes-128-cbc",
+ "aes-192-cbc",
+ "aes-256-cbc",
+ "null",
+ ],
+ },
+ "keytype": {"type": "str"},
+ "spi": {"type": "int"},
+ "passphrase": {"type": "str"},
+ },
+ },
+ "cost": {"type": "int"},
+ "afi": {
+ "required": True,
+ "type": "str",
+ "choices": ["ipv4", "ipv6"],
+ },
+ "authentication_v2": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "message_digest": {"type": "bool"},
+ },
+ },
+ "bfd": {"type": "bool"},
+ "authentication_v3": {
+ "type": "dict",
+ "options": {
+ "key": {"type": "str"},
+ "spi": {"type": "int"},
+ "keytype": {"type": "str"},
+ "passphrase": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["md5", "sha1"],
+ },
+ },
+ },
+ "retransmit_interval": {"type": "int"},
+ "message_digest_key": {
+ "type": "dict",
+ "options": {
+ "key_id": {"type": "int"},
+ "key": {"type": "str"},
+ "encryption": {"type": "str"},
+ },
+ },
+ "mtu_ignore": {"type": "bool"},
+ "priority": {"type": "int"},
+ "area": {
+ "type": "dict",
+ "options": {
+ "area_id": {"required": True, "type": "str"}
+ },
+ },
+ "dead_interval": {"type": "int"},
+ "shutdown": {"type": "bool"},
+ "passive_interface": {"type": "bool"},
+ "authentication_key": {
+ "type": "dict",
+ "options": {
+ "encryption": {"type": "str"},
+ "key": {"type": "str"},
+ },
+ },
+ "hello_interval": {"type": "int"},
+ "transmit_delay": {"type": "int"},
+ "network": {"type": "str"},
+ },
+ },
+ },
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py
new file mode 100644
index 00000000..44c0f4ca
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py
@@ -0,0 +1,329 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+"""
+The arg spec for the eos_ospfv2 module
+"""
+
+
+class Ospfv2Args(object): # pylint: disable=R0903
+ """The arg spec for the eos_ospfv2 module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "options": {
+ "processes": {
+ "elements": "dict",
+ "options": {
+ "process_id": {"type": "int"},
+ "vrf": {"type": "str"},
+ "traffic_engineering": {"type": "bool"},
+ "adjacency": {
+ "options": {
+ "exchange_start": {
+ "options": {"threshold": {"type": "int"}},
+ "type": "dict",
+ }
+ },
+ "type": "dict",
+ },
+ "areas": {
+ "elements": "dict",
+ "options": {
+ "default_cost": {"type": "int"},
+ "filter": {
+ "options": {
+ "address": {"type": "str"},
+ "prefix_list": {"type": "str"},
+ "subnet_address": {"type": "str"},
+ "subnet_mask": {"type": "str"},
+ },
+ "type": "dict",
+ },
+ "not_so_stubby": {
+ "options": {
+ "default_information_originate": {
+ "options": {
+ "metric": {"type": "int"},
+ "metric_type": {"type": "int"},
+ "nssa_only": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "no_summary": {"type": "bool"},
+ "nssa_only": {"type": "bool"},
+ "lsa": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "nssa": {
+ "options": {
+ "default_information_originate": {
+ "options": {
+ "metric": {"type": "int"},
+ "metric_type": {"type": "int"},
+ "nssa_only": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "no_summary": {"type": "bool"},
+ "nssa_only": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "area_id": {"type": "str"},
+ "range": {
+ "options": {
+ "address": {"type": "str"},
+ "advertise": {"type": "bool"},
+ "cost": {"type": "int"},
+ "subnet_address": {"type": "str"},
+ "subnet_mask": {"type": "str"},
+ },
+ "type": "dict",
+ },
+ "stub": {
+ "options": {
+ "no_summary": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ },
+ "type": "list",
+ },
+ "auto_cost": {
+ "options": {
+ "reference_bandwidth": {"type": "int"}
+ },
+ "type": "dict",
+ },
+ "bfd": {
+ "options": {"all_interfaces": {"type": "bool"}},
+ "type": "dict",
+ },
+ "default_information": {
+ "options": {
+ "always": {"type": "bool"},
+ "metric": {"type": "int"},
+ "metric_type": {"type": "int"},
+ "originate": {"type": "bool"},
+ "route_map": {"type": "str"},
+ },
+ "type": "dict",
+ },
+ "default_metric": {"type": "int"},
+ "distance": {
+ "options": {
+ "external": {"type": "int"},
+ "inter_area": {"type": "int"},
+ "intra_area": {"type": "int"},
+ },
+ "type": "dict",
+ },
+ "distribute_list": {
+ "options": {
+ "prefix_list": {"type": "str"},
+ "route_map": {"type": "str"},
+ },
+ "type": "dict",
+ },
+ "dn_bit_ignore": {"type": "bool"},
+ "fips_restrictions": {"type": "str"},
+ "graceful_restart": {
+ "options": {
+ "grace_period": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "log_adjacency_changes": {
+ "options": {"detail": {"type": "bool"}},
+ "type": "dict",
+ },
+ "max_lsa": {
+ "options": {
+ "count": {"type": "int"},
+ "ignore_count": {"type": "int"},
+ "ignore_time": {"type": "int"},
+ "reset_time": {"type": "int"},
+ "threshold": {"type": "int"},
+ "warning": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "max_metric": {
+ "options": {
+ "router_lsa": {
+ "options": {
+ "set": {"type": "bool"},
+ "include_stub": {"type": "bool"},
+ "on_startup": {
+ "options": {
+ "wait_period": {"type": "int"}
+ },
+ "type": "dict",
+ },
+ "summary_lsa": {
+ "options": {
+ "max_metric_value": {
+ "type": "int"
+ },
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ "external_lsa": {
+ "options": {
+ "max_metric_value": {
+ "type": "int"
+ },
+ "set": {"type": "bool"},
+ },
+ "type": "dict",
+ },
+ },
+ "type": "dict",
+ }
+ },
+ "type": "dict",
+ },
+ "maximum_paths": {"type": "int"},
+ "mpls_ldp": {"type": "bool"},
+ "networks": {
+ "elements": "dict",
+ "options": {
+ "area": {"type": "str"},
+ "mask": {"type": "str"},
+ "network_address": {"type": "str"},
+ "prefix": {"type": "str"},
+ },
+ "type": "list",
+ },
+ "passive_interface": {
+ "type": "dict",
+ "options": {
+ "interface_list": {"type": "str"},
+ "default": {"type": "bool"},
+ },
+ },
+ "point_to_point": {"type": "bool"},
+ "redistribute": {
+ "elements": "dict",
+ "options": {
+ "isis_level": {"type": "str"},
+ "route_map": {"type": "str"},
+ "routes": {"type": "str"},
+ },
+ "type": "list",
+ },
+ "retransmission_threshold": {"type": "int"},
+ "rfc1583compatibility": {"type": "bool"},
+ "router_id": {"type": "str"},
+ "shutdown": {"type": "bool"},
+ "summary_address": {
+ "options": {
+ "address": {"type": "str"},
+ "attribute_map": {"type": "str"},
+ "mask": {"type": "str"},
+ "not_advertise": {"type": "bool"},
+ "prefix": {"type": "str"},
+ "tag": {"type": "int"},
+ },
+ "type": "dict",
+ },
+ "timers": {
+ "elements": "dict",
+ "options": {
+ "lsa": {
+ "options": {
+ "rx": {
+ "options": {
+ "min_interval": {"type": "int"}
+ },
+ "type": "dict",
+ },
+ "tx": {
+ "options": {
+ "delay": {
+ "options": {
+ "initial": {
+ "type": "int"
+ },
+ "max": {"type": "int"},
+ "min": {"type": "int"},
+ },
+ "type": "dict",
+ }
+ },
+ "type": "dict",
+ },
+ },
+ "type": "dict",
+ },
+ "out_delay": {"type": "int"},
+ "pacing": {"type": "int"},
+ "spf": {
+ "options": {
+ "initial": {"type": "int"},
+ "max": {"type": "int"},
+ "min": {"type": "int"},
+ "seconds": {"type": "int"},
+ },
+ "type": "dict",
+ },
+ },
+ "type": "list",
+ },
+ },
+ "type": "list",
+ }
+ },
+ "type": "dict",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "deleted",
+ "merged",
+ "overridden",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py
new file mode 100644
index 00000000..f68afded
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py
@@ -0,0 +1,476 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_ospfv3 module
+"""
+
+
+class Ospfv3Args(object): # pylint: disable=R0903
+ """The arg spec for the eos_ospfv3 module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "running_config": {"type": "str"},
+ "state": {
+ "default": "merged",
+ "type": "str",
+ "choices": [
+ "deleted",
+ "merged",
+ "overridden",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ },
+ "config": {
+ "type": "dict",
+ "options": {
+ "processes": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "router_id": {"type": "str"},
+ "shutdown": {"type": "bool"},
+ "fips_restrictions": {"type": "bool"},
+ "graceful_restart_helper": {"type": "bool"},
+ "adjacency": {
+ "type": "dict",
+ "options": {
+ "exchange_start": {
+ "type": "dict",
+ "options": {"threshold": {"type": "int"}},
+ }
+ },
+ },
+ "max_metric": {
+ "type": "dict",
+ "options": {
+ "router_lsa": {
+ "type": "dict",
+ "options": {
+ "external_lsa": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "max_metric_value": {
+ "type": "int"
+ },
+ },
+ },
+ "summary_lsa": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "max_metric_value": {
+ "type": "int"
+ },
+ },
+ },
+ "set": {"type": "bool"},
+ "on_startup": {
+ "type": "dict",
+ "options": {
+ "wait_for_bgp": {
+ "type": "bool"
+ },
+ "wait_period": {"type": "int"},
+ },
+ },
+ "include_stub": {"type": "bool"},
+ },
+ }
+ },
+ },
+ "log_adjacency_changes": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "detail": {"type": "bool"},
+ },
+ },
+ "graceful_restart": {
+ "type": "dict",
+ "options": {
+ "grace_period": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ "timers": {
+ "type": "dict",
+ "options": {
+ "throttle": {
+ "type": "dict",
+ "options": {
+ "max": {"type": "int"},
+ "initial": {"type": "int"},
+ "min": {"type": "int"},
+ "spf": {"type": "bool"},
+ "lsa": {"type": "bool"},
+ },
+ },
+ "out_delay": {"type": "int"},
+ "pacing": {"type": "int"},
+ "lsa": {"type": "int"},
+ },
+ },
+ "vrf": {"type": "str"},
+ "auto_cost": {
+ "type": "dict",
+ "options": {
+ "reference_bandwidth": {"type": "int"}
+ },
+ },
+ "passive_interface": {"type": "bool"},
+ "bfd": {
+ "type": "dict",
+ "options": {"all_interfaces": {"type": "bool"}},
+ },
+ "areas": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "area_id": {"type": "str"},
+ "encryption": {
+ "type": "dict",
+ "options": {
+ "hidden_key": {"type": "bool"},
+ "key": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["sha1", "md5"],
+ },
+ "encrypt_key": {"type": "bool"},
+ "encryption": {
+ "type": "str",
+ "choices": [
+ "3des-cbc",
+ "aes-128-cbc",
+ "aes-192-cbc",
+ "aes-256-cbc",
+ "null",
+ ],
+ },
+ "spi": {"type": "int"},
+ "passphrase": {"type": "str"},
+ },
+ },
+ "nssa": {
+ "type": "dict",
+ "options": {
+ "translate": {"type": "bool"},
+ "default_information_originate": {
+ "type": "dict",
+ "options": {
+ "metric_type": {"type": "int"},
+ "metric": {"type": "int"},
+ "nssa_only": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ },
+ "nssa_only": {"type": "bool"},
+ "set": {"type": "bool"},
+ "no_summary": {"type": "bool"},
+ },
+ },
+ "stub": {
+ "type": "dict",
+ "options": {
+ "summary_lsa": {"type": "bool"},
+ "set": {"type": "bool"},
+ },
+ },
+ "default_cost": {"type": "int"},
+ "authentication": {
+ "type": "dict",
+ "options": {
+ "hidden_key": {"type": "bool"},
+ "key": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["md5", "sha1"],
+ },
+ "encrypt_key": {"type": "bool"},
+ "spi": {"type": "int"},
+ "passphrase": {"type": "str"},
+ },
+ },
+ },
+ },
+ "address_family": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "router_id": {"type": "str"},
+ "distance": {"type": "int"},
+ "redistribute": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "routes": {
+ "type": "str",
+ "choices": [
+ "bgp",
+ "connected",
+ "static",
+ ],
+ },
+ "route_map": {"type": "str"},
+ },
+ },
+ "default_information": {
+ "type": "dict",
+ "options": {
+ "metric_type": {"type": "int"},
+ "always": {"type": "bool"},
+ "metric": {"type": "int"},
+ "originate": {"type": "bool"},
+ "route_map": {"type": "str"},
+ },
+ },
+ "afi": {
+ "choices": ["ipv4", "ipv6"],
+ "type": "str",
+ },
+ "fips_restrictions": {"type": "bool"},
+ "default_metric": {"type": "int"},
+ "maximum_paths": {"type": "int"},
+ "adjacency": {
+ "type": "dict",
+ "options": {
+ "exchange_start": {
+ "type": "dict",
+ "options": {
+ "threshold": {"type": "int"}
+ },
+ }
+ },
+ },
+ "max_metric": {
+ "type": "dict",
+ "options": {
+ "router_lsa": {
+ "type": "dict",
+ "options": {
+ "external_lsa": {
+ "type": "dict",
+ "options": {
+ "set": {
+ "type": "bool"
+ },
+ "max_metric_value": {
+ "type": "int"
+ },
+ },
+ },
+ "summary_lsa": {
+ "type": "dict",
+ "options": {
+ "set": {
+ "type": "bool"
+ },
+ "max_metric_value": {
+ "type": "int"
+ },
+ },
+ },
+ "set": {"type": "bool"},
+ "on_startup": {
+ "type": "dict",
+ "options": {
+ "wait_for_bgp": {
+ "type": "bool"
+ },
+ "wait_period": {
+ "type": "int"
+ },
+ },
+ },
+ "include_stub": {
+ "type": "bool"
+ },
+ },
+ }
+ },
+ },
+ "log_adjacency_changes": {
+ "type": "dict",
+ "options": {
+ "set": {"type": "bool"},
+ "detail": {"type": "bool"},
+ },
+ },
+ "timers": {
+ "type": "dict",
+ "options": {
+ "throttle": {
+ "type": "dict",
+ "options": {
+ "max": {"type": "int"},
+ "initial": {"type": "int"},
+ "min": {"type": "int"},
+ "spf": {"type": "bool"},
+ "lsa": {"type": "bool"},
+ },
+ },
+ "out_delay": {"type": "int"},
+ "pacing": {"type": "int"},
+ "lsa": {"type": "int"},
+ },
+ },
+ "shutdown": {"type": "bool"},
+ "auto_cost": {
+ "type": "dict",
+ "options": {
+ "reference_bandwidth": {"type": "int"}
+ },
+ },
+ "graceful_restart_helper": {"type": "bool"},
+ "passive_interface": {"type": "bool"},
+ "bfd": {
+ "type": "dict",
+ "options": {
+ "all_interfaces": {"type": "bool"}
+ },
+ },
+ "areas": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "ranges": {
+ "elements": "dict",
+ "type": "list",
+ "options": {
+ "subnet_mask": {"type": "str"},
+ "advertise": {"type": "bool"},
+ "cost": {"type": "int"},
+ "subnet_address": {
+ "type": "str"
+ },
+ "address": {"type": "str"},
+ },
+ },
+ "area_id": {"type": "str"},
+ "encryption": {
+ "type": "dict",
+ "options": {
+ "hidden_key": {"type": "bool"},
+ "key": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["sha1", "md5"],
+ },
+ "encrypt_key": {
+ "type": "bool"
+ },
+ "encryption": {
+ "type": "str",
+ "choices": [
+ "3des-cbc",
+ "aes-128-cbc",
+ "aes-192-cbc",
+ "aes-256-cbc",
+ "null",
+ ],
+ },
+ "spi": {"type": "int"},
+ "passphrase": {"type": "str"},
+ },
+ },
+ "nssa": {
+ "type": "dict",
+ "options": {
+ "translate": {"type": "bool"},
+ "default_information_originate": {
+ "type": "dict",
+ "options": {
+ "metric_type": {
+ "type": "int"
+ },
+ "metric": {
+ "type": "int"
+ },
+ "nssa_only": {
+ "type": "bool"
+ },
+ "set": {
+ "type": "bool"
+ },
+ },
+ },
+ "nssa_only": {"type": "bool"},
+ "set": {"type": "bool"},
+ "no_summary": {"type": "bool"},
+ },
+ },
+ "stub": {
+ "type": "dict",
+ "options": {
+ "summary_lsa": {
+ "type": "bool"
+ },
+ "set": {"type": "bool"},
+ },
+ },
+ "default_cost": {"type": "int"},
+ "authentication": {
+ "type": "dict",
+ "options": {
+ "hidden_key": {"type": "bool"},
+ "key": {"type": "str"},
+ "algorithm": {
+ "type": "str",
+ "choices": ["md5", "sha1"],
+ },
+ "encrypt_key": {
+ "type": "bool"
+ },
+ "spi": {"type": "int"},
+ "passphrase": {"type": "str"},
+ },
+ },
+ },
+ },
+ "graceful_restart": {
+ "type": "dict",
+ "options": {
+ "grace_period": {"type": "int"},
+ "set": {"type": "bool"},
+ },
+ },
+ },
+ },
+ },
+ }
+ },
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py
new file mode 100644
index 00000000..d28f8d7c
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py
@@ -0,0 +1,97 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+"""
+The arg spec for the eos_static_routes module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class Static_routesArgs(object):
+ """The arg spec for the eos_static_routes module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "address_families": {
+ "elements": "dict",
+ "options": {
+ "afi": {
+ "choices": ["ipv4", "ipv6"],
+ "required": True,
+ "type": "str",
+ },
+ "routes": {
+ "elements": "dict",
+ "options": {
+ "dest": {"required": True, "type": "str"},
+ "next_hops": {
+ "elements": "dict",
+ "options": {
+ "admin_distance": {"type": "int"},
+ "description": {"type": "str"},
+ "forward_router_address": {
+ "type": "str"
+ },
+ "interface": {"type": "str"},
+ "nexthop_grp": {"type": "str"},
+ "mpls_label": {"type": "int"},
+ "tag": {"type": "int"},
+ "track": {"type": "str"},
+ "vrf": {"type": "str"},
+ },
+ "type": "list",
+ },
+ },
+ "type": "list",
+ },
+ },
+ "type": "list",
+ },
+ "vrf": {"type": "str"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "deleted",
+ "merged",
+ "overridden",
+ "replaced",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ } # pylint: disable=C0301
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py
new file mode 100644
index 00000000..c4622093
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py
@@ -0,0 +1,64 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The arg spec for the eos_vlans module
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+class VlansArgs(object):
+ """The arg spec for the eos_vlans module
+ """
+
+ def __init__(self, **kwargs):
+ pass
+
+ argument_spec = {
+ "config": {
+ "elements": "dict",
+ "options": {
+ "vlan_id": {"required": True, "type": "int"},
+ "name": {"type": "str"},
+ "state": {"choices": ["active", "suspend"], "type": "str"},
+ },
+ "type": "list",
+ },
+ "running_config": {"type": "str"},
+ "state": {
+ "choices": [
+ "merged",
+ "replaced",
+ "overridden",
+ "deleted",
+ "gathered",
+ "rendered",
+ "parsed",
+ ],
+ "default": "merged",
+ "type": "str",
+ },
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py
new file mode 100644
index 00000000..0f9db622
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py
@@ -0,0 +1,475 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_acl_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import itertools
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ search_obj_in_list,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Acl_interfaces(ConfigBase):
+ """
+ The eos_acl_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["acl_interfaces"]
+
+ def __init__(self, module):
+ super(Acl_interfaces, self).__init__(module)
+
+ def get_acl_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ acl_interfaces_facts = facts["ansible_network_resources"].get(
+ "acl_interfaces"
+ )
+ if not acl_interfaces_facts:
+ return []
+ return acl_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+ changed = False
+
+ if self.state in self.ACTION_STATES:
+ existing_acl_interfaces_facts = self.get_acl_interfaces_facts()
+ else:
+ existing_acl_interfaces_facts = []
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_acl_interfaces_facts))
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ changed = True
+ if changed:
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_acl_interfaces_facts = self.get_acl_interfaces_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ if not self._module.params["running_config"]:
+ self._module.fail_json(
+ msg="Value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_acl_interfaces_facts(
+ data=self._module.params["running_config"]
+ )
+ else:
+ changed_acl_interfaces_facts = []
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_acl_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_acl_interfaces_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_acl_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_acl_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_acl_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ if self.state in ("merged", "replaced", "overridden") and not want:
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ self.state
+ )
+ )
+ state = self._module.params["state"]
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or self.state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ def _state_replaced(self, want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commandset = []
+ want_interface = []
+ for w in want:
+ commands = []
+ diff_access_group = []
+ want_interface.append(w["name"])
+ obj_in_have = search_obj_in_list(w["name"], have, "name")
+ if not obj_in_have or "access_groups" not in obj_in_have.keys():
+ commands.append(add_commands(w["access_groups"], w["name"]))
+ else:
+ if "access_groups" in obj_in_have.keys():
+ obj = self.get_acl_diff(obj_in_have, w)
+ if obj[0]:
+ to_delete = {
+ "access_groups": [{"acls": obj[0], "afi": "ipv4"}]
+ }
+ commands.append(remove_commands(to_delete, w["name"]))
+ if obj[1]:
+ to_delete = {
+ "access_groups": [{"acls": obj[1], "afi": "ipv6"}]
+ }
+ commands.append(remove_commands(to_delete, w["name"]))
+ diff = self.get_acl_diff(w, obj_in_have)
+ if diff[0]:
+ diff_access_group.append(
+ {"afi": "ipv4", "acls": diff[0]}
+ )
+ if diff[1]:
+ diff_access_group.append(
+ {"afi": "ipv6", "acls": diff[1]}
+ )
+ if diff_access_group:
+ commands.append(
+ add_commands(diff_access_group, w["name"])
+ )
+ if commands:
+ intf_command = ["interface " + w["name"]]
+ commands = list(itertools.chain(*commands))
+ commandset.append(intf_command)
+ commandset.append(commands)
+
+ if commandset:
+ commandset = list(itertools.chain(*commandset))
+ return commandset
+
+ def _state_overridden(self, want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commandset = []
+ want_interface = []
+ for w in want:
+ commands = []
+ diff_access_group = []
+ want_interface.append(w["name"])
+ obj_in_have = search_obj_in_list(w["name"], have, "name")
+ if not obj_in_have or "access_groups" not in obj_in_have.keys():
+ commands.append(add_commands(w["access_groups"], w["name"]))
+ else:
+ if "access_groups" in obj_in_have.keys():
+ obj = self.get_acl_diff(obj_in_have, w)
+ if obj[0]:
+ to_delete = {
+ "access_groups": [{"acls": obj[0], "afi": "ipv4"}]
+ }
+ commands.append(remove_commands(to_delete, w["name"]))
+ if obj[1]:
+ to_delete = {
+ "access_groups": [{"acls": obj[1], "afi": "ipv6"}]
+ }
+ commands.append(remove_commands(to_delete, w["name"]))
+ diff = self.get_acl_diff(w, obj_in_have)
+ if diff[0]:
+ diff_access_group.append(
+ {"afi": "ipv4", "acls": diff[0]}
+ )
+ if diff[1]:
+ diff_access_group.append(
+ {"afi": "ipv6", "acls": diff[1]}
+ )
+ if diff_access_group:
+ commands.append(
+ add_commands(diff_access_group, w["name"])
+ )
+ if commands:
+ intf_command = ["interface " + w["name"]]
+ commands = list(itertools.chain(*commands))
+ commandset.append(intf_command)
+ commandset.append(commands)
+ for h in have:
+ commands = []
+ if "access_groups" in h.keys() and h["access_groups"]:
+ if h["name"] not in want_interface:
+ for h_group in h["access_groups"]:
+ to_delete = {"access_groups": [h_group]}
+ commands.append(remove_commands(to_delete, h["name"]))
+ if commands:
+ intf_command = ["interface " + h["name"]]
+ commands = list(itertools.chain(*commands))
+ commandset.append(intf_command)
+ commandset.append(commands)
+
+ if commandset:
+ commandset = list(itertools.chain(*commandset))
+
+ return commandset
+
+ def _state_merged(self, want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commandset = []
+ for w in want:
+ commands = []
+ diff_access_group = []
+ obj_in_have = search_obj_in_list(w["name"], have, "name")
+ if not obj_in_have:
+ commands = add_commands(w["access_groups"], w["name"])
+ else:
+ if "access_groups" in obj_in_have.keys():
+ diff = self.get_acl_diff(w, obj_in_have)
+ if diff[0]:
+ diff_access_group.append(
+ {"afi": "ipv4", "acls": diff[0]}
+ )
+ if diff[1]:
+ diff_access_group.append(
+ {"afi": "ipv6", "acls": diff[1]}
+ )
+ if diff_access_group:
+ commands = add_commands(diff_access_group, w["name"])
+ else:
+ commands = add_commands(w["access_groups"], w["name"])
+ if commands:
+ intf_command = ["interface " + w["name"]]
+ commandset.append(intf_command)
+ commandset.append(commands)
+ if commandset:
+ commandset = list(itertools.chain(*commandset))
+ return commandset
+
+ def _state_deleted(self, want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commandset = []
+ for w in want:
+ commands = []
+ intf_command = ["interface " + w["name"]]
+ obj_in_have = search_obj_in_list(w["name"], have, "name")
+ if "access_groups" not in w.keys() or not w["access_groups"]:
+ commands = remove_commands(obj_in_have, w["name"])
+ if w["access_groups"]:
+ for w_grp in w["access_groups"]:
+ if "acls" not in w_grp.keys() or not w_grp["acls"]:
+ obj = self.get_acls_from_afi(
+ w["name"], w_grp["afi"], have
+ )
+ to_delete = {
+ "access_groups": [
+ {"acls": obj, "afi": w_grp["afi"]}
+ ]
+ }
+ commands = remove_commands(to_delete, w["name"])
+ else:
+ if (
+ "access_groups" not in obj_in_have.keys()
+ or not obj_in_have["access_groups"]
+ ):
+ continue
+ group = {"access_groups": [w_grp]}
+ obj = self.get_acl_diff(group, obj_in_have, True)
+ if obj[0]:
+ to_delete = {
+ "access_groups": [
+ {"acls": obj[0], "afi": "ipv4"}
+ ]
+ }
+ commands.append(
+ remove_commands(to_delete, w["name"])
+ )
+ if obj[1]:
+ to_delete = {
+ "access_groups": [
+ {"acls": obj[1], "afi": "ipv6"}
+ ]
+ }
+ commands.append(
+ remove_commands(to_delete, w["name"])
+ )
+ if commands:
+ commands = list(itertools.chain(*commands))
+ if commands:
+ commandset.append(intf_command)
+ commandset.append(commands)
+
+ if commandset:
+ commandset = list(itertools.chain(*commandset))
+ return commandset
+
+ def get_acl_diff(self, w, h, intersection=False):
+ diff_v4 = []
+ diff_v6 = []
+ w_acls_v4 = []
+ w_acls_v6 = []
+ h_acls_v4 = []
+ h_acls_v6 = []
+ for w_group in w["access_groups"]:
+ if w_group["afi"] == "ipv4":
+ w_acls_v4 = w_group["acls"]
+ if w_group["afi"] == "ipv6":
+ w_acls_v6 = w_group["acls"]
+ for h_group in h["access_groups"]:
+ if h_group["afi"] == "ipv4":
+ h_acls_v4 = h_group["acls"]
+ if h_group["afi"] == "ipv6":
+ h_acls_v6 = h_group["acls"]
+ for item in w_acls_v4:
+ match = list(
+ filter(lambda x: x["name"] == item["name"], h_acls_v4)
+ )
+ if match:
+ if item["direction"] == match[0]["direction"]:
+ if intersection:
+ diff_v4.append(item)
+ else:
+ if not intersection:
+ diff_v4.append(item)
+ else:
+ if not intersection:
+ diff_v4.append(item)
+ for item in w_acls_v6:
+ match = list(
+ filter(lambda x: x["name"] == item["name"], h_acls_v6)
+ )
+ if match:
+ if item["direction"] == match[0]["direction"]:
+ if intersection:
+ diff_v6.append(item)
+ else:
+ if not intersection:
+ diff_v6.append(item)
+ else:
+ if not intersection:
+ diff_v6.append(item)
+ return diff_v4, diff_v6
+
+ def get_acls_from_afi(self, interface, afi, have):
+ config = []
+ for h in have:
+ if h["name"] == interface:
+ if "access_groups" not in h.keys() or not h["access_groups"]:
+ continue
+ if h["access_groups"]:
+ for h_grp in h["access_groups"]:
+ if h_grp["afi"] == afi:
+ config = h_grp["acls"]
+ return config
+
+
+def add_commands(want, interface):
+ commands = []
+
+ for w in want:
+ # This module was verified on an ios device since vEOS doesnot support
+ # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as
+ # traffic-filter and in eos it is access-group
+
+ # a_cmd = "traffic-filter" if w['afi'] == 'ipv6' else "access-group"
+ a_cmd = "access-group"
+ afi = "ip" if w["afi"] == "ipv4" else w["afi"]
+ if "acls" in w.keys():
+ for acl in w["acls"]:
+ commands.append(
+ afi
+ + " "
+ + a_cmd
+ + " "
+ + acl["name"]
+ + " "
+ + acl["direction"]
+ )
+ return commands
+
+
+def remove_commands(want, interface):
+ commands = []
+ if "access_groups" not in want.keys() or not want["access_groups"]:
+ return commands
+ for w in want["access_groups"]:
+ # This module was verified on an ios device since vEOS doesnot support
+ # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as
+ # traffic-filter and in eos it is access-group
+
+ # a_cmd = "traffic-filter" if w['afi'] == 'ipv6' else "access-group"
+ a_cmd = "access-group"
+
+ afi = "ip" if w["afi"] == "ipv4" else w["afi"]
+ if "acls" in w.keys():
+ for acl in w["acls"]:
+ commands.append(
+ "no "
+ + afi
+ + " "
+ + a_cmd
+ + " "
+ + acl["name"]
+ + " "
+ + acl["direction"]
+ )
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py
new file mode 100644
index 00000000..3d1d8216
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py
@@ -0,0 +1,605 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_acls class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import socket
+import re
+import itertools
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_empties,
+ dict_diff,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Acls(ConfigBase):
+ """
+ The eos_acls class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["acls"]
+
+ def __init__(self, module):
+ super(Acls, self).__init__(module)
+
+ def get_acls_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ acls_facts = facts["ansible_network_resources"].get("acls")
+ if not acls_facts:
+ return []
+ return acls_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+ changed = False
+
+ if self.state in self.ACTION_STATES:
+ existing_acls_facts = self.get_acls_facts()
+ else:
+ existing_acls_facts = []
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_acls_facts))
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ changed = True
+ if changed:
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_acls_facts = self.get_acls_facts()
+ elif self.state == "rendered":
+ commands = list(itertools.chain(*commands))
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ if not self._module.params["running_config"]:
+ self._module.fail_json(
+ msg="Value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_acls_facts(
+ data=self._module.params["running_config"]
+ )
+ else:
+ changed_acls_facts = []
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_acls_facts
+ if result["changed"]:
+ result["after"] = changed_acls_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_acls_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_acls_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ config = self._module.params.get("config")
+ want = []
+ onbox_configs = []
+ for h in existing_acls_facts:
+ have_configs = add_commands(remove_empties(h))
+ onbox_configs.append(have_configs)
+ if config:
+ for w in config:
+ want.append(remove_empties(w))
+ have = existing_acls_facts
+ resp = self.set_state(want, have)
+ if self.state == "merged":
+ to_config = self.compare_configs(onbox_configs, to_list(resp))
+ else:
+ to_config = resp
+ return to_config
+
+ def compare_configs(self, have, want):
+ commands = []
+ want = list(itertools.chain(*want))
+ have = list(itertools.chain(*have))
+ h_index = 0
+ config = list(want)
+ for w in want:
+ access_list = re.findall(r"(ip.*) access-list (.*)", w)
+ if access_list:
+ if w in have:
+ h_index = have.index(w)
+ else:
+ for num, h in enumerate(have, start=h_index + 1):
+ if "access-list" not in h:
+ seq_num = re.search(r"(\d+) (.*)", w)
+ if seq_num:
+ have_seq_num = re.search(r"(\d+) (.*)", h)
+ if seq_num.group(1) == have_seq_num.group(
+ 1
+ ) and have_seq_num.group(2) != seq_num.group(2):
+ negate_cmd = "no " + seq_num.group(1)
+ config.insert(config.index(w), negate_cmd)
+ if w in h:
+ config.pop(config.index(w))
+ break
+ for c in config:
+ access_list = re.findall(r"(ip.*) access-list (.*)", c)
+ if access_list:
+ acl_index = config.index(c)
+ else:
+ if config[acl_index] not in commands:
+ commands.append(config[acl_index])
+ commands.append(c)
+ return commands
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ if (
+ self.state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ self.state
+ )
+ )
+ state = self._module.params["state"]
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or self.state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ def _state_replaced(self, want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ config_cmds = []
+ remove_cmds = []
+ diff = {}
+ for w in want:
+ afi = "ipv6" if w["afi"] == "ipv6" else "ipv4"
+ for acl in w["acls"]:
+ name = acl["name"]
+ want_ace = acl["aces"]
+ for h in have:
+ if h["afi"] == afi:
+ for h_acl in h["acls"]:
+ if h_acl["name"] == name:
+ h = {"afi": afi, "acls": [{"name": name}]}
+ for h_ace in h_acl["aces"]:
+ diff = get_ace_diff(h_ace, want_ace)
+ if diff:
+ h = {
+ "afi": afi,
+ "acls": [{"name": name, "aces": [h_ace]}],
+ }
+ remove_cmds.append(del_commands(h, have))
+ for w_ace in want_ace:
+ w_diff = get_ace_diff(w_ace, h_acl["aces"])
+ if w_diff:
+ w = [
+ {
+ "afi": afi,
+ "acls": [
+ {"name": name, "aces": [w_ace]}
+ ],
+ }
+ ]
+ config_cmds = set_commands(w, have)
+ config_cmds = list(
+ itertools.chain(*config_cmds)
+ )
+
+ if remove_cmds:
+ remove_cmds = list(itertools.chain(*remove_cmds))
+ commands.append(remove_cmds)
+ if config_cmds:
+ commands.append(config_cmds)
+ commands = list(itertools.chain(*commands))
+ commandset = []
+ [commandset.append(cmd) for cmd in commands if cmd not in commandset]
+ return commandset
+
+ def _state_overridden(self, want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ ace_diff = {}
+ h_afi_list = []
+ w_afi_list = []
+ for h in have:
+ h_afi_list.append(h["afi"])
+ for w in want:
+ w_afi_list.append(w["afi"])
+ for hafi in h_afi_list:
+ if hafi not in w_afi_list:
+ h = {"afi": hafi}
+ remove_cmds = del_commands(h, have)
+ commands.append(remove_cmds)
+ for w in want:
+ w_names = []
+ for h in have:
+ h_names = []
+ if w["afi"] == h["afi"]:
+ for w_acl in w["acls"]:
+ w_names.append(w_acl["name"])
+ for h_acl in h["acls"]:
+ h_names.append(h_acl["name"])
+ if h_acl["name"] == w_acl["name"]:
+ for h_ace in h_acl["aces"]:
+ ace_diff = get_ace_diff(
+ h_ace, w_acl["aces"]
+ )
+ if ace_diff:
+ h = {
+ "afi": h["afi"],
+ "acls": [
+ {
+ "name": h_acl["name"],
+ "aces": [h_ace],
+ }
+ ],
+ }
+ remove_cmds = del_commands(h, have)
+ commands.append(remove_cmds)
+ for w_ace in w_acl["aces"]:
+ w_ace_diff = get_ace_diff(
+ w_ace, h_acl["aces"]
+ )
+ if w_ace_diff:
+ w_diff = [
+ {
+ "afi": w["afi"],
+ "acls": [
+ {
+ "name": w_acl["name"],
+ "aces": [w_ace],
+ }
+ ],
+ }
+ ]
+ config_cmds = set_commands(
+ w_diff, have
+ )
+ config_cmds = list(
+ itertools.chain(*config_cmds)
+ )
+ commands.append(config_cmds)
+ for hname in h_names:
+ if hname not in w_names:
+ h = {"afi": h["afi"], "acls": [{"name": hname}]}
+ remove_cmds = del_commands(h, have)
+ if remove_cmds not in commands:
+ commands.append(remove_cmds)
+
+ if commands:
+ commands = list(itertools.chain(*commands))
+ commandset = []
+ for c in commands:
+ access_list = re.findall(r"(ip.*) access-list (.*)", c)
+ if access_list and "no" in c:
+ commandset.append(c)
+ continue
+ if access_list:
+ acl_index = commands.index(c)
+ else:
+ if commands[acl_index] not in commandset:
+ commandset.append(commands[acl_index])
+ commandset.append(c)
+
+ return commandset
+
+ def _state_merged(self, want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ return set_commands(want, have)
+
+ def _state_deleted(self, want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ if not want:
+ for h in have:
+ return_command = add_commands(h)
+ for command in return_command:
+ command = "no " + command
+ commands.append(command)
+ else:
+ for w in want:
+ return_command = del_commands(w, have, True)
+ if return_command[:5] == "Warn ":
+ self._module.warn(return_command[5:])
+ return commands
+ else:
+ commands.append(return_command)
+ commands = list(itertools.chain(*commands))
+ return commands
+
+
+def set_commands(want, have):
+ commands = []
+ for w in want:
+ wace_updated = []
+ for h in have:
+ if w["afi"] == h["afi"]:
+ for wacl in w["acls"]:
+ for hacl in h["acls"]:
+ if wacl["name"] == hacl["name"]:
+ want_aces = wacl["aces"]
+ for wace in wacl["aces"]:
+ for hace in hacl["aces"]:
+ if (
+ "sequence" in wace.keys()
+ and "sequence" in hace.keys()
+ ):
+ if (
+ wace["sequence"]
+ == hace["sequence"]
+ ):
+ wace_updated = get_updated_ace(
+ wace, hace
+ )
+ if wace_updated:
+ want_aces.pop(
+ want_aces.index(wace)
+ )
+ want_aces.append(wace_updated)
+ return_command = add_commands(w)
+ commands.append(return_command)
+ return commands
+
+
+def get_updated_ace(w, h):
+ # gives the ace to be updated in case of merge update.
+ if not dict_diff(w, h):
+ return
+ w_updated = w.copy()
+ for hkey in h.keys():
+ if hkey not in w.keys():
+ w_updated.update({hkey: h[hkey]})
+ else:
+ w_updated.update({hkey: w[hkey]})
+ return w_updated
+
+
+def add_commands(want):
+ commandset = []
+ protocol_name = {
+ "51": "ahp",
+ "47": "gre",
+ "1": "icmp",
+ "2": "igmp",
+ "4": "ip",
+ "89": "ospf",
+ "103": "pim",
+ "6": "tcp",
+ "17": "udp",
+ "112": "vrrp",
+ }
+ if not want:
+ return commandset
+ command = ""
+ afi = "ip" if want["afi"] == "ipv4" else "ipv6"
+ for acl in want["acls"]:
+ if "standard" in acl.keys() and acl["standard"]:
+ command = afi + " access-list standard " + acl["name"]
+ else:
+ command = afi + " access-list " + acl["name"]
+ commandset.append(command)
+ if "aces" not in acl.keys():
+ continue
+ for ace in acl["aces"]:
+ command = ""
+ if "sequence" in ace.keys():
+ command = str(ace["sequence"])
+ if "remark" in ace.keys():
+ command = command + " remark " + ace["remark"]
+ if "fragment_rules" in ace.keys() and ace["fragment_rules"]:
+ command = command + " fragment-rules"
+ if "grant" in ace.keys():
+ command = command + " " + ace["grant"]
+ if "vlan" in ace.keys():
+ command = command + " vlan " + ace["vlan"]
+ if "protocol" in ace.keys():
+ protocol = ace["protocol"]
+ if protocol.isdigit():
+ if protocol in protocol_name.keys():
+ protocol = protocol_name[protocol]
+ command = command + " " + protocol
+ if "source" in ace.keys():
+ if "any" in ace["source"].keys():
+ command = command + " any"
+ elif "subnet_address" in ace["source"].keys():
+ command = command + " " + ace["source"]["subnet_address"]
+ elif "host" in ace["source"].keys():
+ command = command + " host " + ace["source"]["host"]
+ elif "address" in ace["source"].keys():
+ command = (
+ command
+ + " "
+ + ace["source"]["address"]
+ + " "
+ + ace["source"]["wildcard_bits"]
+ )
+ if "port_protocol" in ace["source"].keys():
+ for op, val in ace["source"]["port_protocol"].items():
+ if val.isdigit():
+ val = socket.getservbyport(int(val))
+ command = command + " " + op + " " + val
+ if "destination" in ace.keys():
+ if "any" in ace["destination"].keys():
+ command = command + " any"
+ elif "subnet_address" in ace["destination"].keys():
+ command = (
+ command + " " + ace["destination"]["subnet_address"]
+ )
+ elif "host" in ace["destination"].keys():
+ command = command + " host " + ace["destination"]["host"]
+ elif "address" in ace["destination"].keys():
+ command = (
+ command
+ + " "
+ + ace["destination"]["address"]
+ + " "
+ + ace["destination"]["wildcard_bits"]
+ )
+ if "port_protocol" in ace["destination"].keys():
+ for op in ace["destination"]["port_protocol"].keys():
+ command = (
+ command
+ + " "
+ + op
+ + " "
+ + ace["destination"]["port_protocol"][op]
+ )
+ if "protocol_options" in ace.keys():
+ for proto in ace["protocol_options"].keys():
+ if proto == "icmp" or proto == "icmpv6":
+ for icmp_msg in ace["protocol_options"][proto].keys():
+ command = command + " " + icmp_msg
+ elif proto == "ip" or proto == "ipv6":
+ command = (
+ command
+ + " nexthop-group "
+ + ace["protocol_options"][proto]["nexthop_group"]
+ )
+ elif proto == "tcp":
+ for flag, val in ace["protocol_options"][proto][
+ "flags"
+ ].items():
+ if val:
+ command = command + " " + flag
+ if "hop_limit" in ace.keys():
+ for op, val in ace["hop_limit"].items():
+ command = command + " hop-limit " + op + " " + val
+ if "tracked" in ace.keys() and ace["tracked"]:
+ command = command + " tracked"
+ if "ttl" in ace.keys():
+ for op, val in ace["ttl"].items():
+ command = command + " ttl " + op + " " + str(val)
+ if "fragments" in ace.keys():
+ command = command + " fragments"
+ if "log" in ace.keys():
+ command = command + " log"
+ commandset.append(command.strip())
+ return commandset
+
+
+def del_commands(want, have, name_only=False):
+ commandset = []
+ command = ""
+ have_command = []
+ for h in have:
+ have_configs = add_commands(h)
+ have_command.append(have_configs)
+ have_command = list(itertools.chain(*have_command))
+ afi = "ip" if want["afi"] == "ipv4" else "ipv6"
+ if "acls" not in want.keys():
+ for have_cmd in have_command:
+ access_list = re.search(r"(ip.*)\s+access-list .*", have_cmd)
+ if access_list and access_list.group(1) == afi:
+ commandset.append("no " + have_cmd)
+ return commandset
+
+ for acl in want["acls"]:
+ ace_present = True
+ if "standard" in acl.keys() and acl["standard"]:
+ command = afi + " access-list standard " + acl["name"]
+ else:
+ command = afi + " access-list " + acl["name"]
+ if "aces" not in acl.keys():
+ ace_present = False
+ commandset.append("no " + command)
+ if ace_present:
+ if name_only:
+ msg = "Deleted operation allows deletion of access-list only and not the entries !!"
+ return "Warn " + msg
+ return_command = add_commands(want)
+ for cmd in return_command:
+ if "access-list" in cmd:
+ commandset.append(cmd)
+ continue
+ seq = re.search(
+ r"(\d+) (permit|deny|fragment-rules|remark) .*", cmd
+ )
+ if seq:
+ commandset.append("no " + seq.group(1))
+ else:
+ commandset.append("no " + cmd)
+ return commandset
+
+
+def get_ace_diff(want_ace, have_ace):
+ # gives the diff of the aces passed.
+ for h_a in have_ace:
+ d = dict_diff(want_ace, h_a)
+ if not d:
+ break
+ return d
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py
new file mode 100644
index 00000000..559cd624
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py
@@ -0,0 +1,272 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2021 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos_bgp_address_family config file.
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to its desired end-state is
+created.
+"""
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ dict_merge,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import (
+ ResourceModule,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_address_family import (
+ Bgp_afTemplate,
+)
+import re
+
+
+class Bgp_af(ResourceModule):
+ """
+ The eos_bgp_address_family config class
+ """
+
+ def __init__(self, module):
+ super(Bgp_af, self).__init__(
+ empty_fact_val={},
+ facts_module=Facts(module),
+ module=module,
+ resource="bgp_address_family",
+ tmplt=Bgp_afTemplate(),
+ )
+ self.parsers = [
+ "router",
+ "address_family",
+ "bgp_params_additional_paths",
+ "bgp_params.nexthop_address_family",
+ "bgp_params.nexthop_unchanged",
+ "bgp_params.redistribute_internal",
+ "bgp_params.route",
+ "graceful_restart",
+ "neighbor.activate",
+ "neighbor.additional_paths",
+ "neighbor.default_originate",
+ "neighbor.graceful_restart",
+ "neighbor.next_hop_unchanged",
+ "neighbor.next_hop_address_family",
+ "neighbor.prefix_list",
+ "neighbor.route_map",
+ "neighbor.weight",
+ "neighbor.encapsulation",
+ "network",
+ "redistribute",
+ "route_target",
+ ]
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ if self.state not in ["parsed", "gathered"]:
+ self.generate_commands()
+ self.run_commands()
+ return self.result
+
+ def generate_commands(self):
+ """ Generate configuration commands to send based on
+ want, have and desired state.
+ """
+ wantd = {}
+ haved = {}
+
+ if self.want:
+ wantd = {self.want["as_number"]: self.want}
+ if self.have:
+ haved = {self.have["as_number"]: self.have}
+
+ # turn all lists of dicts into dicts prior to merge
+ for entry in wantd, haved:
+ self._bgp_af_list_to_dict(entry)
+
+ # if state is merged, merge want onto have and then compare
+ if self.state == "merged":
+ wantd = dict_merge(haved, wantd)
+ if len(wantd.keys()) > 1:
+ self._module.fail_json(
+ msg="Only one bgp instance is allowed per device"
+ )
+ wantd = {}
+ # if state is deleted, empty out wantd and set haved to wantd
+ if self.state == "deleted":
+ haved = {
+ k: v for k, v in iteritems(haved) if k in wantd or not wantd
+ }
+ for wk, wv in iteritems(wantd):
+ self._compare(want=wv, have=haved.pop(wk, {}))
+
+ wantd = {}
+
+ # remove superfluous config for overridden
+ if self.state == "overridden":
+ for k, have in iteritems(haved):
+ if k not in wantd:
+ self._compare(want={}, have=have)
+ for k, want in iteritems(wantd):
+ self._compare(want=want, have=haved.pop(k, {}))
+
+ def _delete_af(self, want, have):
+ waf = want.get("address_family", {})
+ haf = have.get("address_family", {})
+ for hkey, entry in iteritems(haf):
+ if hkey in waf.keys():
+ af_no_command = self._tmplt.render(
+ entry, "address_family", True
+ ).split("\n")
+ if re.search(r"\S+_\S+", hkey):
+ af_no_command[0] = af_no_command[0][3:]
+ af_no_command[1] = "no " + af_no_command[1]
+ for cmd in af_no_command:
+ self.commands.append(cmd)
+ else:
+ self.addcmd(entry, "address_family", True)
+ have = {}
+
+ def _compare(self, want, have):
+ """Leverages the base class `compare()` method and
+ populates the list of commands to be run by comparing
+ the `want` and `have` data with the `parsers` defined
+ for the Bgp_af network resource.
+ """
+ for name, entry in iteritems(want):
+ if name != "as_number":
+ if self.state == "deleted":
+ self._delete_af(want, have)
+ else:
+ self._compare_af({name: entry}, {name: have.get(name, {})})
+
+ if self.commands and "router bgp" not in self.commands[0]:
+ self.commands.insert(
+ 0,
+ self._tmplt.render(
+ {"as_number": want.get("as_number") or have["as_number"]},
+ "router",
+ False,
+ ),
+ )
+
+ def _compare_af(self, want, have):
+ waf = want.get("address_family", {})
+ haf = have.get("address_family", {})
+ for name, entry in iteritems(waf):
+ begin = len(self.commands)
+ self._compare_lists(entry, have=haf.get(name, {}))
+ self._compare_neighbor(entry, have=haf.get(name, {}))
+ self.compare(
+ parsers=self.parsers, want=entry, have=haf.pop(name, {})
+ )
+ if len(self.commands) != begin:
+ af_command = self._tmplt.render(
+ entry, "address_family", False
+ ).split("\n")
+ for cmd in af_command:
+ self.commands.insert(begin, cmd)
+ self.commands.append("exit")
+ begin += 1
+ for name, entry in iteritems(haf):
+ # skip superfluous configs for replaced
+ if self.state in ["replaced"]:
+ if name in waf.keys():
+ self.addcmd(entry, "address_family", True)
+ else:
+ # overridden
+ # check if want has vrf or not
+ # if want doesnot have vrf, device's vrf config will not
+ # be touched.
+ vrf_present = False
+ for w_key in waf.keys():
+ if re.search(r"\S+_\S+", w_key):
+ vrf_present = True
+ break
+ if vrf_present:
+ if re.search(r"\S+_\S+", name):
+ af_no_command = self._tmplt.render(
+ entry, "address_family", True
+ ).split("\n")
+ if name not in waf.keys():
+ af_no_command[0] = af_no_command[0][3:]
+ af_no_command[1] = "no " + af_no_command[1]
+ for cmd in af_no_command:
+ self.commands.append(cmd)
+ else:
+ self.addcmd(entry, "address_family", True)
+ else:
+ if not re.search(r"\S+_\S+", name):
+ self.addcmd(entry, "address_family", True)
+
+ def _compare_neighbor(self, want, have):
+ parsers = [
+ "neighbor.activate",
+ "neighbor.additional_paths",
+ "neighbor.default_originate",
+ "neighbor.graceful_restart",
+ "neighbor.next_hop_unchanged",
+ "neighbor.next_hop_address_family",
+ "neighbor.prefix_list",
+ "neighbor.route_map",
+ "neighbor.weight",
+ "neighbor.encapsulation",
+ ]
+ wneigh = want.get("neighbor", {})
+ hneigh = have.get("neighbor", {})
+ for name, entry in iteritems(wneigh):
+ self.compare(
+ parsers=parsers,
+ want={"neighbor": entry},
+ have={"neighbor": hneigh.pop(name, {})},
+ )
+ for name, entry in iteritems(hneigh):
+ self.compare(parsers=parsers, want={}, have={"neighbor": entry})
+
+ def _compare_lists(self, want, have):
+ for attrib in ["redistribute", "network"]:
+ wdict = want.pop(attrib, {})
+ hdict = have.pop(attrib, {})
+ for key, entry in iteritems(wdict):
+ if entry != hdict.pop(key, {}):
+ self.addcmd(entry, attrib, False)
+ # remove remaining items in have for replaced
+ for entry in hdict.values():
+ self.addcmd(entry, attrib, True)
+
+ def _bgp_af_list_to_dict(self, entry):
+ for name, proc in iteritems(entry):
+ if "address_family" in proc:
+ proc["address_family"] = {
+ entry["afi"] + "_" + entry.get("vrf", ""): entry
+ for entry in proc.get("address_family", [])
+ }
+ self._bgp_af_list_to_dict(proc["address_family"])
+
+ if "neighbor" in proc:
+ proc["neighbor"] = {
+ entry["peer"]: entry for entry in proc.get("neighbor", [])
+ }
+ if "network" in proc:
+ proc["network"] = {
+ entry["address"]: entry
+ for entry in proc.get("network", [])
+ }
+ if "redistribute" in proc:
+ proc["redistribute"] = {
+ entry["protocol"]: entry
+ for entry in proc.get("redistribute", [])
+ }
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py
new file mode 100644
index 00000000..e2a3c973
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py
@@ -0,0 +1,379 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos_bgp_global config file.
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to its desired end-state is
+created.
+"""
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ dict_merge,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import (
+ ResourceModule,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_global import (
+ Bgp_globalTemplate,
+)
+
+
+class Bgp_global(ResourceModule):
+ """
+ The eos_bgp_global config class
+ """
+
+ def __init__(self, module):
+ super(Bgp_global, self).__init__(
+ empty_fact_val={},
+ facts_module=Facts(module),
+ module=module,
+ resource="bgp_global",
+ tmplt=Bgp_globalTemplate(),
+ )
+ self.parsers = [
+ "router",
+ "vrf",
+ "default_metric",
+ "distance",
+ "graceful_restart",
+ "graceful_restart_helper",
+ "acccess_group",
+ "maximum_paths",
+ "monitoring",
+ "route_target",
+ "router_id",
+ "shutdown",
+ "timers",
+ "ucmp_fec",
+ "ucmp_link_bandwidth",
+ "ucmp_mode",
+ "update",
+ "vlan",
+ "vlan_aware_bundle",
+ ]
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ if self.state not in ["parsed", "gathered"]:
+ self.generate_commands()
+ self.run_commands()
+ return self.result
+
+ def generate_commands(self):
+ """ Generate configuration commands to send based on
+ want, have and desired state.
+ """
+ wantd = {}
+ haved = {}
+ if (
+ self.want.get("as_number") == self.have.get("as_number")
+ or not self.have
+ ):
+ if self.want:
+ wantd = {self.want["as_number"]: self.want}
+ if self.have:
+ haved = {self.have["as_number"]: self.have}
+ else:
+ self._module.fail_json(
+ msg="Only one bgp instance is allowed per device"
+ )
+
+ # turn all lists of dicts into dicts prior to merge
+ for entry in wantd, haved:
+ self._bgp_global_list_to_dict(entry)
+
+ # if state is merged, merge want onto have and then compare
+ if self.state == "merged":
+ wantd = dict_merge(haved, wantd)
+
+ # if state is deleted, empty out wantd and set haved to wantd
+ if self.state in ["deleted", "purged"]:
+ haved = {
+ k: v for k, v in iteritems(haved) if k in wantd or not wantd
+ }
+ wantd = {}
+
+ if self.state == "deleted":
+ self._compare(want={}, have=self.have)
+
+ if self.state == "purged":
+ for num, entry in iteritems(haved):
+ self.commands.append(
+ self._tmplt.render({"as_number": num}, "router", True)
+ )
+
+ for k, want in iteritems(wantd):
+ self._compare(want=want, have=haved.pop(k, {}))
+
+ def _compare(self, want, have):
+ """Leverages the base class `compare()` method and
+ populates the list of commands to be run by comparing
+ the `want` and `have` data with the `parsers` defined
+ for the Bgp_global network resource.
+ """
+ self._compare_vrfs(want, have)
+ self._compare_neighbor(want, have)
+ self._compare_lists(want, have)
+ self._compare_bgp_params(want, have)
+ for name, entry in iteritems(want):
+ if name != "as_number":
+ self.compare(
+ parsers=self.parsers,
+ want={name: entry},
+ have={name: have.pop(name, {})},
+ )
+ for name, entry in iteritems(have):
+ if name != "as_number":
+ self.compare(
+ parsers=self.parsers, want={}, have={name: have.get(name)}
+ )
+
+ if self.commands and "router bgp" not in self.commands[0]:
+ self.commands.insert(
+ 0, self._tmplt.render(want or have, "router", False)
+ )
+
+ def _compare_bgp_params(self, want, have):
+ parsers = [
+ "bgp_params_additional_paths",
+ "bgp_params_advertise_inactive",
+ "bgp_params_allowas_in",
+ "bgp_params_always_compare_med",
+ "bgp_params_asn",
+ "bgp_params_auto_local_addr",
+ "bgp_params_bestpath_as_path",
+ "bgp_params_bestpath_ecmp_fast",
+ "bgp_params_bestpath_med",
+ "bgp_params_bestpath_skip",
+ "bgp_params_tie_break",
+ "bgp_params_client_to_client",
+ "bgp_params_cluster_id",
+ "bgp_params_confederation",
+ "bgp_params_control_plane_filter",
+ "bgp_params_convergence",
+ "bgp_params_default",
+ "bgp_params.enforce_first_as",
+ "bgp_params.host_routes",
+ "bgp_params.labelled_unicast",
+ "bgp_params.listen_limit",
+ "bgp_params.listen_range",
+ "bgp_params.log_neighbor_changes",
+ "bgp_params.missing_policy",
+ "bgp_params.monitoring",
+ "bgp_params.nexthop_unchanged",
+ "bgp_params.redistribute_internal",
+ "bgp_params.route",
+ "bgp_params.route_reflector",
+ "bgp_params.transport",
+ ]
+ wbgp = want.pop("bgp_params", {})
+ hbgp = have.pop("bgp_params", {})
+ for name, entry in iteritems(wbgp):
+ self.compare(
+ parsers=parsers,
+ want={"bgp_params": {name: entry}},
+ have={"bgp_params": {name: hbgp.pop(name, {})}},
+ )
+ for name, entry in iteritems(hbgp):
+ self.compare(
+ parsers=parsers, want={}, have={"bgp_params": {name: entry}}
+ )
+
+ def _compare_vrfs(self, want, have):
+ wvrf = want.pop("vrfs", {})
+ hvrf = have.pop("vrfs", {})
+ begin = len(self.commands)
+ for name, entry in iteritems(wvrf):
+ self._compare_neighbor(entry, hvrf.get(name, {}))
+ self._compare_lists(entry, hvrf.get(name, {}))
+ self._compare_bgp_params(entry, hvrf.get(name, {}))
+ for k, v in entry.items():
+ if hvrf.get(name):
+ h = {k: hvrf[name].pop(k, {})}
+ else:
+ h = {}
+ if k != "vrf":
+ self.compare(parsers=self.parsers, want={k: v}, have=h)
+
+ if len(self.commands) != begin:
+ self.commands.insert(
+ begin, self._tmplt.render({"vrf": name}, "vrf", False)
+ )
+ self.commands.append("exit")
+ begin_negate = len(self.commands)
+ for name, entry in iteritems(hvrf):
+ if name not in wvrf.keys():
+ if self._check_af(name):
+ self._module.fail_json(
+ msg="Use the _bgp_address_family module to delete the address_family under vrf, before replacing/deleting the vrf."
+ )
+ else:
+ self.commands.append(
+ self._tmplt.render({"vrf": name}, "vrf", True)
+ )
+ continue
+ self.compare(parsers=self.parsers, want={}, have=entry)
+ after_negate = len(self.commands)
+ if after_negate != begin_negate:
+ if "vrf " + name in self.commands:
+ index = self.commands.index("vrf " + name)
+ i = begin_negate
+ while i < after_negate:
+ cmd = self.commands.pop(i)
+ if cmd != "exit":
+ self.commands.insert(index + 1, cmd)
+ i += 1
+ else:
+ self.commands.insert(
+ begin_negate,
+ self._tmplt.render({"vrf": name}, "vrf", False),
+ )
+ self.commands.append("exit")
+
+ def _get_config(self, connection):
+ return connection.get("show running-config | section bgp ")
+
+ def _check_af(self, vrf):
+ af_present = False
+ if self._connection:
+ config_lines = self._get_config(self._connection).splitlines()
+ index = [i + 1 for i, el in enumerate(config_lines) if vrf in el]
+ if index:
+ # had to do this to escape flake8 and black errors
+ ind = index[0]
+ for line in config_lines[ind:]:
+ if "vrf" in line:
+ break
+ if "address-family" in line:
+ af_present = True
+ break
+ return af_present
+
+ def _compare_neighbor(self, want, have):
+ parsers = [
+ "neighbor.additional_paths",
+ "neighbor.allowas_in",
+ "neighbor.auto_local_addr",
+ "neighbor.default_originate",
+ "neighbor.description",
+ "neighbor.dont_capability_negotiate",
+ "neighbor.ebgp_multihop",
+ "neighbor.enforce_first_as",
+ "neighbor.export_localpref",
+ "neighbor.fall_over",
+ "neighbor.graceful_restart",
+ "neighbor.graceful_restart_helper",
+ "neighbor.idle_restart_timer",
+ "neighbor.import_localpref",
+ "neighbor.link_bandwidth",
+ "neighbor.local_as",
+ "neighbor.local_v6_addr",
+ "neighbor.maximum_accepted_routes",
+ "neighbor.maximum_received_routes",
+ "neighbor.metric_out",
+ "neighbor.monitoring",
+ "neighbor.next_hop_self",
+ "neighbor.next_hop_unchanged",
+ "neighbor.next_hop_v6_addr",
+ "neighbor.out_delay",
+ "neighbor.remote_as",
+ "neighbor.remove_private_as",
+ "neighbor.peer_group",
+ "neighbor.prefix_list",
+ "neighbor.route_map",
+ "neighbor.route_reflector_client",
+ "neighbor.route_to_peer",
+ "neighbor.send_community",
+ "neighbor.shutdown",
+ "neighbor.soft_reconfiguration",
+ "neighbor.transport",
+ "neighbor.timers",
+ "neighbor.ttl",
+ "neighbor.update_source",
+ "neighbor.weight",
+ ]
+ wneigh = want.pop("neighbor", {})
+ hneigh = have.pop("neighbor", {})
+ for name, entry in iteritems(wneigh):
+ for k, v in entry.items():
+ peer = entry["peer"]
+ if hneigh.get(name):
+ h = {"peer": peer, k: hneigh[name].pop(k, {})}
+ else:
+ h = {}
+ self.compare(
+ parsers=parsers,
+ want={"neighbor": {"peer": peer, k: v}},
+ have={"neighbor": h},
+ )
+ for name, entry in iteritems(hneigh):
+ if name not in wneigh.keys() and "peer_group" not in entry.keys():
+ self.commands.append("no neighbor " + name)
+ continue
+ for k, v in entry.items():
+ peer = entry["peer"]
+ self.compare(
+ parsers=parsers,
+ want={},
+ have={"neighbor": {"peer": peer, k: v}},
+ )
+
+ def _compare_lists(self, want, have):
+ for attrib in ["redistribute", "network", "aggregate_address"]:
+ wdict = want.pop(attrib, {})
+ hdict = have.pop(attrib, {})
+ for key, entry in iteritems(wdict):
+ if entry != hdict.pop(key, {}):
+ self.addcmd(entry, attrib, False)
+ # remove remaining items in have for replaced
+ for entry in hdict.values():
+ self.addcmd(entry, attrib, True)
+
+ def _bgp_global_list_to_dict(self, entry):
+ for name, proc in iteritems(entry):
+ if "neighbor" in proc:
+ proc["neighbor"] = {
+ entry["peer"]: entry for entry in proc.get("neighbor", [])
+ }
+ if "network" in proc:
+ proc["network"] = {
+ entry["address"]: entry
+ for entry in proc.get("network", [])
+ }
+
+ if "aggregate_address" in proc:
+ proc["aggregate_address"] = {
+ entry["address"]: entry
+ for entry in proc.get("aggregate_address", [])
+ }
+
+ if "redistribute" in proc:
+ proc["redistribute"] = {
+ entry["protocol"]: entry
+ for entry in proc.get("redistribute", [])
+ }
+
+ if "vrfs" in proc:
+ proc["vrfs"] = {
+ entry["vrf"]: entry for entry in proc.get("vrfs", [])
+ }
+ self._bgp_global_list_to_dict(proc["vrfs"])
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py
new file mode 100644
index 00000000..734ca311
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py
@@ -0,0 +1,305 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+)
+
+
+class Interfaces(ConfigBase):
+ """
+ The eos_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["interfaces"]
+
+ def get_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ interfaces_facts = facts["ansible_network_resources"].get("interfaces")
+ if not interfaces_facts:
+ return []
+ return interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ commands = list()
+ warnings = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_interfaces_facts = self.get_interfaces_facts()
+ else:
+ existing_interfaces_facts = []
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_interfaces_facts = self.get_interfaces_facts()
+
+ elif self.state == "rendered":
+ result["rendered"] = commands
+
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_interfaces_facts(data=running_config)
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_interfaces_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_interfaces_facts
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want)
+ have = param_list_to_dict(have)
+ commands = []
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ if (
+ "speed" in add_config.keys()
+ and "duplex" not in add_config.keys()
+ ):
+ add_config.update({"duplex": desired.get("duplex")})
+
+ commands.extend(generate_commands(key, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, extant in have.items():
+ if key in want:
+ desired = want[key]
+ else:
+ desired = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ if (
+ "speed" in add_config.keys()
+ and "duplex" not in add_config.keys()
+ ):
+ add_config.update({"duplex": desired.get("duplex")})
+
+ commands.extend(generate_commands(key, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+ if (
+ "speed" in add_config.keys()
+ and "duplex" not in add_config.keys()
+ ):
+ add_config.update({"duplex": desired.get("duplex")})
+ commands.extend(generate_commands(key, add_config, {}))
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for key in want:
+ desired = dict()
+ if key in have:
+ extant = have[key]
+ else:
+ continue
+
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(key, {}, del_config))
+
+ return commands
+
+
+def generate_commands(interface, to_set, to_remove):
+ commands = []
+ for key, value in to_set.items():
+ if value is None:
+ continue
+
+ if key == "enabled":
+ commands.append("{0}shutdown".format("no " if value else ""))
+ elif key == "speed":
+ if value == "auto":
+ commands.append("{0} {1}".format(key, value))
+ else:
+ commands.append("speed {0}{1}".format(value, to_set["duplex"]))
+ elif key == "duplex":
+ # duplex is handled with speed
+ continue
+ elif key == "mode":
+ if not re.search(r"(M|m)anagement.*", interface):
+ if value == "layer3":
+ # switching from default (layer2) mode to layer3
+ commands.append("no switchport")
+ else:
+ # setting to default (layer 2) mode
+ commands.append("switchport")
+ else:
+ commands.append("{0} {1}".format(key, value))
+
+ # Don't try to also remove the same key, if present in to_remove
+ to_remove.pop(key, None)
+
+ for key in to_remove.keys():
+ if key == "enabled":
+ commands.append("no shutdown")
+ elif key == "speed":
+ commands.append("speed auto")
+ elif key == "duplex":
+ # duplex is handled with speed
+ continue
+ elif key == "mode":
+ if not re.search(r"(M|m)anagement.*", interface):
+ commands.append("switchport")
+ else:
+ commands.append("no {0}".format(key))
+
+ if commands:
+ commands.insert(0, "interface {0}".format(interface))
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py
new file mode 100644
index 00000000..6c7d2ab9
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py
@@ -0,0 +1,361 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_l2_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+ vlan_range_to_list,
+)
+
+
+class L2_interfaces(ConfigBase):
+ """
+ The eos_l2_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["l2_interfaces"]
+
+ def get_l2_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ l2_interfaces_facts = facts["ansible_network_resources"].get(
+ "l2_interfaces"
+ )
+ if not l2_interfaces_facts:
+ return []
+ return l2_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ commands = list()
+ warnings = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_l2_interfaces_facts = self.get_l2_interfaces_facts()
+ else:
+ existing_l2_interfaces_facts = []
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_l2_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_l2_interfaces_facts = self.get_l2_interfaces_facts()
+
+ elif self.state == "rendered":
+ result["rendered"] = commands
+
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_l2_interfaces_facts(
+ data=running_config
+ )
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_l2_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_l2_interfaces_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_l2_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_l2_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_l2_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want)
+ have = param_list_to_dict(have)
+ commands = []
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ intf_commands = set_interface(desired, extant)
+ intf_commands.extend(clear_interface(desired, extant))
+
+ if intf_commands:
+ commands.append("interface {0}".format(interface_name))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, extant in have.items():
+ if key in want:
+ desired = want[key]
+ else:
+ desired = dict()
+
+ intf_commands = set_interface(desired, extant)
+ intf_commands.extend(clear_interface(desired, extant))
+
+ if intf_commands:
+ commands.append("interface {0}".format(key))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ intf_commands = set_interface(desired, extant)
+
+ if intf_commands:
+ commands.append("interface {0}".format(interface_name))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for key in want:
+ desired = dict()
+ if key in have:
+ extant = have[key]
+ else:
+ continue
+ intf_commands = clear_interface(desired, extant)
+
+ if intf_commands:
+ commands.append("interface {0}".format(key))
+ commands.extend(intf_commands)
+
+ return commands
+
+
+def set_interface(want, have):
+ commands = []
+
+ want_mode = want.get("mode")
+ if want_mode and want_mode != have.get("mode"):
+ commands.append("switchport mode {0}".format(want_mode))
+
+ wants_access = want.get("access")
+ if wants_access:
+ access_vlan = wants_access.get("vlan")
+ if access_vlan and access_vlan != have.get("access", {}).get("vlan"):
+ commands.append("switchport access vlan {0}".format(access_vlan))
+
+ wants_trunk = want.get("trunk")
+ if wants_trunk:
+ allowed_vlans = []
+ has_allowed_vlans = {}
+ want_allowed_vlans = {}
+ has_trunk = have.get("trunk", {})
+ native_vlan = wants_trunk.get("native_vlan")
+ if native_vlan and native_vlan != has_trunk.get("native_vlan"):
+ commands.append(
+ "switchport trunk native vlan {0}".format(native_vlan)
+ )
+ for con in [want, have]:
+ expand_trunk_allowed_vlans(con)
+ want_allowed_vlans = want["trunk"].get("trunk_allowed_vlans")
+ if has_trunk:
+ has_allowed_vlans = has_trunk.get("trunk_allowed_vlans")
+
+ if want_allowed_vlans and has_allowed_vlans:
+ allowed_vlans = list(
+ set(want_allowed_vlans.split(","))
+ - set(has_allowed_vlans.split(","))
+ )
+ elif want_allowed_vlans:
+ allowed_vlans = want_allowed_vlans.split(",")
+ if allowed_vlans:
+ allowed_vlans.sort()
+ allowed_vlans = ",".join(
+ ["{0}".format(vlan) for vlan in allowed_vlans]
+ )
+ if has_allowed_vlans:
+ commands.append(
+ "switchport trunk allowed vlan add {0}".format(
+ allowed_vlans
+ )
+ )
+ else:
+ commands.append(
+ "switchport trunk allowed vlan {0}".format(allowed_vlans)
+ )
+ return commands
+
+
+def expand_trunk_allowed_vlans(want):
+ if not want:
+ return None
+ if want.get("trunk"):
+ if "trunk_allowed_vlans" in want["trunk"]:
+ allowed_vlans = vlan_range_to_list(
+ want["trunk"]["trunk_allowed_vlans"]
+ )
+ vlans_list = [str(num) for num in sorted(allowed_vlans)]
+ want["trunk"]["trunk_allowed_vlans"] = ",".join(vlans_list)
+
+
+def clear_interface(want, have):
+ commands = []
+
+ if "mode" in have and want.get("mode") is None:
+ commands.append("no switchport mode")
+
+ if "access" in have and not want.get("access"):
+ commands.append("no switchport access vlan")
+
+ has_trunk = have.get("trunk") or {}
+ wants_trunk = want.get("trunk") or {}
+ if (
+ "trunk_allowed_vlans" in has_trunk
+ and "trunk_allowed_vlans" not in wants_trunk
+ ):
+ commands.append("no switchport trunk allowed vlan")
+ if (
+ "trunk_allowed_vlans" in has_trunk
+ and "trunk_allowed_vlans" in wants_trunk
+ ):
+ for con in [want, have]:
+ expand_trunk_allowed_vlans(con)
+ want_allowed_vlans = want["trunk"].get("trunk_allowed_vlans")
+ has_allowed_vlans = has_trunk.get("trunk_allowed_vlans")
+ allowed_vlans = list(
+ set(has_allowed_vlans.split(","))
+ - set(want_allowed_vlans.split(","))
+ )
+ if allowed_vlans:
+ allowed_vlans = ",".join(
+ ["{0}".format(vlan) for vlan in allowed_vlans]
+ )
+
+ commands.append(
+ "switchport trunk allowed vlan remove {0}".format(
+ allowed_vlans
+ )
+ )
+
+ if "native_vlan" in has_trunk and "native_vlan" not in wants_trunk:
+ commands.append("no switchport trunk native vlan")
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py
new file mode 100644
index 00000000..453332ea
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py
@@ -0,0 +1,322 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_l3_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+)
+
+
+class L3_interfaces(ConfigBase):
+ """
+ The eos_l3_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["l3_interfaces"]
+
+ def get_l3_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ l3_interfaces_facts = facts["ansible_network_resources"].get(
+ "l3_interfaces"
+ )
+ if not l3_interfaces_facts:
+ return []
+ return l3_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ commands = list()
+ warnings = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_l3_interfaces_facts = self.get_l3_interfaces_facts()
+ else:
+ existing_l3_interfaces_facts = []
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_l3_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_l3_interfaces_facts = self.get_l3_interfaces_facts()
+
+ elif self.state == "rendered":
+ result["rendered"] = commands
+
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_l3_interfaces_facts(
+ data=running_config
+ )
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_l3_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_l3_interfaces_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_l3_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_l3_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_l3_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want)
+ have = param_list_to_dict(have)
+ commands = []
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+ intf_commands = set_interface(desired, extant)
+ intf_commands.extend(clear_interface(desired, extant))
+
+ if intf_commands:
+ commands.append("interface {0}".format(interface_name))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, extant in have.items():
+ if key in want:
+ desired = want[key]
+ else:
+ desired = dict()
+ if desired.get("ipv4"):
+ for ipv4 in desired["ipv4"]:
+ if ipv4["secondary"] is None:
+ del ipv4["secondary"]
+
+ intf_commands = set_interface(desired, extant)
+ intf_commands.extend(clear_interface(desired, extant))
+
+ if intf_commands:
+ commands.append("interface {0}".format(key))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ intf_commands = set_interface(desired, extant)
+ if intf_commands:
+ commands.append("interface {0}".format(interface_name))
+ commands.extend(intf_commands)
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for key in want:
+ desired = dict()
+ if key in have:
+ extant = have[key]
+ else:
+ continue
+
+ intf_commands = clear_interface(desired, extant)
+
+ if intf_commands:
+ commands.append("interface {0}".format(key))
+ commands.extend(intf_commands)
+
+ return commands
+
+
+def set_interface(want, have):
+ commands = []
+
+ want_ipv4 = set(
+ tuple(sorted(address.items())) for address in want.get("ipv4") or []
+ )
+ have_ipv4 = set(
+ tuple(sorted(address.items())) for address in have.get("ipv4") or []
+ )
+ for address in want_ipv4 - have_ipv4:
+ address = dict(address)
+ if "secondary" in address and not address["secondary"]:
+ del address["secondary"]
+ if tuple(address.items()) in have_ipv4:
+ continue
+
+ address_cmd = "ip address {0}".format(address["address"])
+ if address.get("secondary"):
+ address_cmd += " secondary"
+ commands.append(address_cmd)
+
+ want_ipv6 = set(
+ tuple(sorted(address.items())) for address in want.get("ipv6") or []
+ )
+ have_ipv6 = set(
+ tuple(sorted(address.items())) for address in have.get("ipv6") or []
+ )
+ for address in want_ipv6 - have_ipv6:
+ address = dict(address)
+ commands.append("ipv6 address {0}".format(address["address"]))
+ return commands
+
+
+def clear_interface(want, have):
+ commands = []
+ want_ipv4 = set(
+ tuple(sorted(address.items())) for address in want.get("ipv4") or []
+ )
+ have_ipv4 = set(
+ tuple(sorted(address.items())) for address in have.get("ipv4") or []
+ )
+ if not want_ipv4 and have_ipv4:
+ commands.append("no ip address")
+ else:
+ for address in have_ipv4 - want_ipv4:
+ address = dict(address)
+ if "secondary" not in address:
+ address["secondary"] = False
+ if tuple(address.items()) in want_ipv4:
+ continue
+
+ if address.get("secondary"):
+ address_cmd = " {0} secondary".format(address["address"])
+ commands.append(address_cmd)
+
+ if "secondary" not in address:
+ # Removing non-secondary removes all other interfaces
+ break
+
+ want_ipv6 = set(
+ tuple(sorted(address.items())) for address in want.get("ipv6") or []
+ )
+ have_ipv6 = set(
+ tuple(sorted(address.items())) for address in have.get("ipv6") or []
+ )
+ for address in have_ipv6 - want_ipv6:
+ address = dict(address)
+ commands.append("no ipv6 address {0}".format(address["address"]))
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py
new file mode 100644
index 00000000..6d11ae3f
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py
@@ -0,0 +1,197 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_lacp class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Lacp(ConfigBase):
+ """
+ The eos_lacp class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["lacp"]
+
+ def get_lacp_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ lacp_facts = facts["ansible_network_resources"].get("lacp")
+ if not lacp_facts:
+ return {}
+ return lacp_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_lacp_facts = self.get_lacp_facts()
+ else:
+ existing_lacp_facts = {}
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_lacp_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_lacp_facts = self.get_lacp_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_lacp_facts(data=running_config)
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_lacp_facts
+ if result["changed"]:
+ result["after"] = changed_lacp_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_lacp_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_lacp_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"] or {}
+ have = existing_lacp_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if state in ("merged", "replaced", "rendered") and not want:
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ if state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ to_set = dict_diff(have, want)
+ if "system" in to_set:
+ system = to_set["system"]
+ if "priority" in system:
+ commands.append(
+ "lacp system-priority {0}".format(system["priority"])
+ )
+
+ to_del = dict_diff(want, have)
+ if "system" in to_del:
+ system = to_del["system"]
+ system_set = to_set.get("system", {})
+ if "priority" in system and "priority" not in system_set:
+ commands.append("no lacp system-priority")
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ to_set = dict_diff(have, want)
+ if "system" in to_set:
+ system = to_set["system"]
+ if "priority" in system:
+ commands.append(
+ "lacp system-priority {0}".format(system["priority"])
+ )
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ to_del = dict_diff(want, have)
+ if "system" in to_del:
+ system = to_del["system"]
+ if "priority" in system:
+ commands.append("no lacp system-priority")
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py
new file mode 100644
index 00000000..3f013824
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py
@@ -0,0 +1,254 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_lacp_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+)
+
+
+class Lacp_interfaces(ConfigBase):
+ """
+ The eos_lacp_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["lacp_interfaces"]
+
+ def get_lacp_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ lacp_interfaces_facts = facts["ansible_network_resources"].get(
+ "lacp_interfaces"
+ )
+ if not lacp_interfaces_facts:
+ return []
+ return lacp_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_lacp_interfaces_facts = self.get_lacp_interfaces_facts()
+ else:
+ existing_lacp_interfaces_facts = {}
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_lacp_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_lacp_interfaces_facts = self.get_lacp_interfaces_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_lacp_interfaces_facts(
+ data=running_config
+ )
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_lacp_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_lacp_interfaces_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_lacp_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_lacp_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_lacp_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want)
+ have = param_list_to_dict(have)
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(key, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, extant in have.items():
+ if key in want:
+ desired = want[key]
+ else:
+ desired = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(key, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+
+ commands.extend(generate_commands(key, add_config, {}))
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for key in want:
+ desired = dict()
+ if key in have:
+ extant = have[key]
+ else:
+ continue
+
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(key, {}, del_config))
+
+ return commands
+
+
+def generate_commands(interface, to_set, to_remove):
+ commands = []
+ for key in to_remove.keys():
+ commands.append("no lacp {0}".format(key.replace("_", "-")))
+
+ for key, value in to_set.items():
+ if value is None:
+ continue
+
+ commands.append("lacp {0} {1}".format(key.replace("_", "-"), value))
+
+ if commands:
+ commands.insert(0, "interface {0}".format(interface))
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py
new file mode 100644
index 00000000..87faf91f
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py
@@ -0,0 +1,271 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_lag_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+import re
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+)
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+)
+
+
+class Lag_interfaces(ConfigBase):
+ """
+ The eos_lag_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["lag_interfaces"]
+
+ def get_lag_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ lag_interfaces_facts = facts["ansible_network_resources"].get(
+ "lag_interfaces"
+ )
+ if not lag_interfaces_facts:
+ return []
+ return lag_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ commands = list()
+ warnings = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_lag_interfaces_facts = self.get_lag_interfaces_facts()
+ else:
+ existing_lag_interfaces_facts = {}
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_lag_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_lag_interfaces_facts = self.get_lag_interfaces_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_lag_interfaces_facts(
+ data=running_config
+ )
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_lag_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_lag_interfaces_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_lag_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_lag_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_lag_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for interface in want:
+ interface_name = normalize_interface(interface["name"])
+ for extant in have:
+ if extant["name"] == interface_name:
+ break
+ else:
+ extant = dict(name=interface_name)
+
+ commands.extend(set_config(interface, extant))
+ commands.extend(remove_config(interface, extant))
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for extant in have:
+ for interface in want:
+ if normalize_interface(interface["name"]) == extant["name"]:
+ break
+ else:
+ interface = dict(name=extant["name"])
+ commands.extend(remove_config(interface, extant))
+
+ for interface in want:
+ interface_name = normalize_interface(interface["name"])
+ for extant in have:
+ if extant["name"] == interface_name:
+ break
+ else:
+ extant = dict(name=interface_name)
+ commands.extend(set_config(interface, extant))
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for interface in want:
+ interface_name = normalize_interface(interface["name"])
+ for extant in have:
+ if extant["name"] == interface_name:
+ break
+ else:
+ extant = dict(name=interface_name)
+
+ commands.extend(set_config(interface, extant))
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for interface in want:
+ interface_name = normalize_interface(interface["name"])
+ for extant in have:
+ if extant["name"] == interface_name:
+ break
+ else:
+ extant = dict(name=interface_name)
+
+ # Clearing all args, send empty dictionary
+ interface = dict(name=interface_name)
+ commands.extend(remove_config(interface, extant))
+
+ return commands
+
+
+def set_config(want, have):
+ commands = []
+ to_set = dict_diff(have, want)
+ for member in to_set.get("members", []):
+ channel_id = re.search(r"\d.*", want["name"])
+ if channel_id:
+ commands.extend(
+ [
+ "interface {0}".format(member["member"]),
+ "channel-group {0} mode {1}".format(
+ channel_id.group(0), member["mode"]
+ ),
+ ]
+ )
+
+ return commands
+
+
+def remove_config(want, have):
+ commands = []
+ if not want.get("members"):
+ return ["no interface {0}".format(want["name"])]
+
+ to_remove = dict_diff(want, have)
+ for member in to_remove.get("members", []):
+ commands.extend(
+ ["interface {0}".format(member["member"]), "no channel-group"]
+ )
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py
new file mode 100644
index 00000000..caf1c409
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py
@@ -0,0 +1,203 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_lldp_global class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ dict_diff,
+ to_list,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Lldp_global(ConfigBase):
+ """
+ The eos_lldp_global class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["lldp_global"]
+
+ def __init__(self, module):
+ super(Lldp_global, self).__init__(module)
+
+ def get_lldp_global_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ lldp_global_facts = facts["ansible_network_resources"].get(
+ "lldp_global"
+ )
+ if not lldp_global_facts:
+ return {}
+ return lldp_global_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_lldp_global_facts = self.get_lldp_global_facts()
+ else:
+ existing_lldp_global_facts = {}
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_lldp_global_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_lldp_global_facts = self.get_lldp_global_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_lldp_global_facts(data=running_config)
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_lldp_global_facts
+ if result["changed"]:
+ result["after"] = changed_lldp_global_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_lldp_global_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_lldp_global_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"] or {}
+ have = existing_lldp_global_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ if state == "deleted":
+ commands = state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = state_merged(want, have)
+ elif state == "replaced":
+ commands = state_replaced(want, have)
+ return commands
+
+
+def state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = set()
+ # merged and deleted are likely to emit duplicate tlv-select commands
+ commands.update(state_merged(want, have))
+ commands.update(state_deleted(want, have))
+
+ return list(commands)
+
+
+def state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ to_set = dict_diff(have, want)
+ tlv_options = to_set.pop("tlv_select", {})
+ for key, value in to_set.items():
+ commands.append("lldp {0} {1}".format(key, value))
+ for key, value in tlv_options.items():
+ device_option = key.replace("_", "-")
+ if value is True:
+ commands.append("lldp tlv-select {0}".format(device_option))
+ elif value is False:
+ commands.append("no lldp tlv-select {0}".format(device_option))
+
+ return commands
+
+
+def state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ to_remove = dict_diff(want, have)
+ tlv_options = to_remove.pop("tlv_select", {})
+ for key in to_remove:
+ commands.append("no lldp {0}".format(key))
+ for key, value in tlv_options.items():
+ device_option = key.replace("_", "-")
+ if value is False:
+ commands.append("lldp tlv-select {0}".format(device_option))
+ elif value is True:
+ commands.append("no lldp tlv-select {0}".format(device_option))
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py
new file mode 100644
index 00000000..1a7354eb
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py
@@ -0,0 +1,260 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_lldp_interfaces class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import (
+ normalize_interface,
+)
+
+
+class Lldp_interfaces(ConfigBase):
+ """
+ The eos_lldp_interfaces class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["lldp_interfaces"]
+
+ def get_lldp_interfaces_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ lldp_interfaces_facts = facts["ansible_network_resources"].get(
+ "lldp_interfaces"
+ )
+ if not lldp_interfaces_facts:
+ return []
+ return lldp_interfaces_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_lldp_interfaces_facts = self.get_lldp_interfaces_facts()
+ else:
+ existing_lldp_interfaces_facts = []
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_lldp_interfaces_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_lldp_interfaces_facts = self.get_lldp_interfaces_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_lldp_interfaces_facts(
+ data=running_config
+ )
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_lldp_interfaces_facts
+ if result["changed"]:
+ result["after"] = changed_lldp_interfaces_facts
+
+ elif self.state == "gathered":
+ result["gathered"] = changed_lldp_interfaces_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_lldp_interfaces_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_lldp_interfaces_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want, remove_key=False)
+ have = param_list_to_dict(have, remove_key=False)
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict(name=interface_name)
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(
+ generate_commands(interface_name, add_config, del_config)
+ )
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for key, extant in have.items():
+ if key in want:
+ desired = want[key]
+ else:
+ desired = dict(name=key)
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(key, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for key, desired in want.items():
+ interface_name = normalize_interface(key)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ extant = dict(name=interface_name)
+
+ add_config = dict_diff(extant, desired)
+
+ commands.extend(generate_commands(interface_name, add_config, {}))
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for key in want.keys():
+ interface_name = normalize_interface(key)
+ desired = dict(name=interface_name)
+ if interface_name in have:
+ extant = have[interface_name]
+ else:
+ continue
+
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(interface_name, {}, del_config))
+
+ return commands
+
+
+def generate_commands(name, to_set, to_remove):
+ commands = []
+ for key, value in to_set.items():
+ if value is None:
+ continue
+
+ prefix = "" if value else "no "
+ commands.append("{0}lldp {1}".format(prefix, key))
+
+ for key in to_remove:
+ commands.append("lldp {0}".format(key))
+
+ if commands:
+ commands.insert(0, "interface {0}".format(name))
+
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py
new file mode 100644
index 00000000..766f0571
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py
@@ -0,0 +1,202 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos_ospf_interfaces config file.
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to its desired end-state is
+created.
+"""
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ dict_merge,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import (
+ ResourceModule,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospf_interfaces import (
+ Ospf_interfacesTemplate,
+)
+
+
+class Ospf_interfaces(ResourceModule):
+ """
+ The eos_ospf_interfaces config class
+ """
+
+ def __init__(self, module):
+ super(Ospf_interfaces, self).__init__(
+ empty_fact_val={},
+ facts_module=Facts(module),
+ module=module,
+ resource="ospf_interfaces",
+ tmplt=Ospf_interfacesTemplate(),
+ )
+ self.parsers = [
+ "interfaces",
+ "area",
+ "authentication_v2",
+ "authentication_v3",
+ "authentication_key",
+ "deadinterval",
+ "encryption",
+ "hellointerval",
+ "bfd",
+ "cost",
+ "ip_params_area",
+ "ip_params_bfd",
+ "ip_params_cost",
+ "ip_params_dead_interval",
+ "ip_params_hello_interval",
+ "ip_params_mtu_ignore",
+ "ip_params_network",
+ "ip_params_priority",
+ "ip_params_passive_interface",
+ "ip_params_retransmit_interval",
+ "ip_params_transmit_delay",
+ "mtu_ignore",
+ "network",
+ "priority",
+ "passive_interface",
+ "retransmit_interval",
+ "transmit_delay",
+ "message_digest_key",
+ ]
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ if self.state not in ["parsed", "gathered"]:
+ self.generate_commands()
+ self.run_commands()
+ return self.result
+
+ def generate_commands(self):
+ """ Generate configuration commands to send based on
+ want, have and desired state.
+ """
+
+ # convert list of dicts to dicts of dicts
+ wantd = {entry["name"]: entry for entry in self.want}
+ haved = {entry["name"]: entry for entry in self.have}
+
+ # turn all lists of dicts into dicts prior to merge
+ for entry in wantd, haved:
+ self._ospf_int_list_to_dict(entry)
+
+ # if state is merged, merge want onto have and then compare
+ if self.state == "merged":
+ wantd = dict_merge(haved, wantd)
+
+ # if state is deleted, empty out wantd and set haved to wantd
+ if self.state == "deleted":
+ haved = {
+ k: v for k, v in iteritems(haved) if k in wantd or not wantd
+ }
+ for k, have in iteritems(haved):
+ self._compare(want={}, have=have)
+ wantd = {}
+
+ # remove superfluous config for overridden
+ if self.state == "overridden":
+ for k, have in iteritems(haved):
+ if k not in wantd:
+ self._compare(want={}, have=have)
+
+ for k, want in iteritems(wantd):
+ self._compare(want=want, have=haved.pop(k, {}))
+
+ def _compare(self, want, have):
+ """Leverages the base class `compare()` method and
+ populates the list of commands to be run by comparing
+ the `want` and `have` data with the `parsers` defined
+ for the Ospf_interfaces network resource.
+ """
+ begin = len(self.commands)
+ self._compare_addr_family(want=want, have=have)
+ if len(self.commands) != begin:
+
+ tmp = want or have
+ tmp.pop("address_family", {})
+ self.commands.insert(
+ begin, self._tmplt.render(tmp, "interfaces", False)
+ )
+
+ def _compare_addr_family(self, want, have):
+ wdict = want.get("address_family", {})
+ hdict = have.get("address_family", {})
+ for afi in ["ipv4", "ipv6"]:
+ w_family = wdict.pop(afi, {})
+ h_family = hdict.pop(afi, {})
+ for k in w_family.keys():
+ if k == "afi":
+ continue
+ w = {"afi": afi, k: w_family[k]}
+ h = {"afi": afi, k: h_family.pop(k, {})}
+ if k == "ip_params":
+ self._compare_ip_params(want=w, have=h)
+ self.compare(parsers=self.parsers, want=w, have=h)
+ for k in h_family.keys():
+ if k in ["afi"]:
+ continue
+ w = {"afi": afi, k: None}
+ h = {"afi": afi, k: h_family[k]}
+ if k == "ip_params":
+ w = {"afi": afi, k: {}}
+ self._compare_ip_params(want=w, have=h)
+ self.compare(parsers=self.parsers, want=w, have=h)
+
+ def _compare_ip_params(self, want, have):
+ w_params = want.get("ip_params", {})
+ h_params = have.get("ip_params", {})
+ for afi in ["ipv4", "ipv6"]:
+ w_p = w_params.pop(afi, {})
+ h_p = h_params.pop(afi, {})
+ for k, params in iteritems(w_p):
+ if k == "afi":
+ continue
+ w = {"afi": afi, k: params}
+ h = {"afi": afi, k: h_p.pop(k, None)}
+ self.compare(
+ parsers=self.parsers,
+ want={"ip_params": w},
+ have={"ip_params": h},
+ )
+ for k, params in iteritems(h_p):
+ if k == "afi":
+ continue
+ w = {"afi": afi, k: None}
+ h = {"afi": afi, k: params}
+ self.compare(
+ parsers=self.parsers,
+ want={"ip_params": w},
+ have={"ip_params": h},
+ )
+
+ def _ospf_int_list_to_dict(self, entry):
+ for name, family in iteritems(entry):
+ if family.get("ip_params"):
+ family["ip_params"] = {
+ entry["afi"]: entry for entry in family["ip_params"]
+ }
+ if "address_family" in family:
+ family["address_family"] = {
+ entry["afi"]: entry
+ for entry in family.get("address_family", [])
+ }
+ self._ospf_int_list_to_dict(family["address_family"])
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py
new file mode 100644
index 00000000..5fe8836c
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py
@@ -0,0 +1,796 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_ospfv2 class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+ remove_empties,
+)
+
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Ospfv2(ConfigBase):
+ """
+ The eos_ospfv2 class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["ospfv2"]
+
+ def __init__(self, module):
+ super(Ospfv2, self).__init__(module)
+
+ def get_ospfv2_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+
+ ospfv2_facts = facts["ansible_network_resources"].get("ospfv2")
+ if not ospfv2_facts:
+ return []
+ return ospfv2_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_ospfv2_facts = self.get_ospfv2_facts()
+ else:
+ existing_ospfv2_facts = []
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_ospfv2_facts))
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_ospfv2_facts = self.get_ospfv2_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ if not self._module.params["running_config"]:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_ospfv2_facts(
+ data=self._module.params["running_config"]
+ )
+ else:
+ changed_ospfv2_facts = self.get_ospfv2_facts()
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_ospfv2_facts
+ if result["changed"]:
+ result["after"] = changed_ospfv2_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_ospfv2_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_ospfv2_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_ospfv2_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ if (
+ self.state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ self.state
+ )
+ )
+ if self.state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif self.state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif self.state == "merged" or self.state == "rendered":
+ commands = self._state_merged(want, have)
+ elif self.state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ def _state_replaced(self, want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for w in want["processes"]:
+ del_cmds = w.copy()
+ add_cmds = {}
+ for h in have["processes"]:
+ if h["process_id"] != w["process_id"]:
+ continue
+ if w.get("vrf"):
+ if w["vrf"] != h["vrf"]:
+ self._module.fail_json(
+ msg="Value of vrf and process_id does not match the config present in the device"
+ )
+ break
+ del_instance_list = self.compare_dicts(h, w)
+ if del_instance_list:
+ del_cmds = {"processes": del_instance_list}
+ add_instance_list = self.compare_dicts(w, h)
+ if add_instance_list:
+ add_cmds = {"processes": add_instance_list}
+
+ return_command = self.del_commands(del_cmds, have)
+ for command in return_command:
+ if "exit" not in command:
+ commands.append(command)
+ return_command = self.add_commands(add_cmds, have)
+ for command in return_command:
+ if "router ospf" in command:
+ if command not in commands:
+ commands.append(command)
+ else:
+ commands.append(command)
+ commandset = []
+ if commands:
+ commandset.append(commands[0])
+ for cmd in commands[1::]:
+ if "router ospf" in cmd and commandset[-1] != "exit":
+ commandset.append("exit")
+ commandset.append(cmd)
+ if commandset[-1] != "exit":
+ commandset.append("exit")
+ return commandset
+
+ def _state_overridden(self, want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for h in have["processes"]:
+ present = False
+ for w in want["processes"]:
+ if h["process_id"] == w["process_id"]:
+ present = True
+ break
+ if not present:
+ commands.append("no router ospf " + str(h["process_id"]))
+ replace_cmds = self._state_replaced(want, have)
+ for cmd in replace_cmds:
+ commands.append(cmd)
+ return commands
+
+ def _state_merged(self, want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+
+ return self.set_commands(want, have)
+
+ def _state_deleted(self, want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ return_command = self.del_commands(want, have)
+ if return_command:
+ for cmd in return_command:
+ if "no exit" in cmd:
+ cmd = "exit"
+ commands.append(cmd)
+ return commands
+
+ def set_commands(self, want, have):
+ commands = []
+ instance_list = []
+ for w in want["processes"]:
+ present = False
+ c = []
+ if have and not have.get("processes"):
+ instance_list = want["processes"]
+ break
+ if have:
+ for h in have["processes"]:
+ if w["process_id"] == h["process_id"]:
+ if w.get("vrf"):
+ if w["vrf"] != h["vrf"]:
+ self._module.fail_json(
+ msg="Value of vrf and process_id does not match the config present in the device"
+ )
+ continue
+ present = True
+ c = self.compare_dicts(w, h)
+ break
+ if c:
+ instance_list.append(c[0])
+ if not present:
+ if w["vrf"] in _get_vrf_list(have):
+ self._module.fail_json(
+ msg="Value of vrf and process_id does not match the config present in the device"
+ )
+ instance_list.append(w)
+ instance_dict = {"processes": instance_list}
+ return_command = self.add_commands(instance_dict, have)
+ for command in return_command:
+ commands.append(command)
+ return commands
+
+ def compare_dicts(self, want_inst, have_inst):
+ want_dict = remove_empties(want_inst)
+ have = have_inst
+ ospf_list = []
+ return_ospf_dict = {}
+ for w_key in want_dict.keys():
+ if not have.get(w_key):
+ return_ospf_dict.update({w_key: want_dict[w_key]})
+ elif (
+ isinstance(want_dict[w_key], str)
+ or isinstance(want_dict[w_key], bool)
+ or isinstance(want_dict[w_key], int)
+ ):
+ if want_dict[w_key] != have[w_key]:
+ return_ospf_dict.update({w_key: want_dict[w_key]})
+ elif isinstance(want_dict[w_key], dict):
+ diff = dict_diff(have.get(w_key, {}), want_dict[w_key])
+ if diff:
+ return_ospf_dict.update({w_key: diff})
+ elif isinstance(want_dict[w_key], list):
+ if have.get(w_key):
+ compare_list = self.compare_ospf_list(
+ want_dict[w_key], have.get(w_key), w_key
+ )
+ if compare_list:
+ return_ospf_dict.update({w_key: compare_list})
+ else:
+ if want_dict[w_key] != have.get(w_key):
+ return_ospf_dict.update({w_key: want_dict[w_key]})
+
+ if return_ospf_dict:
+ if want_dict.get("vrf"):
+ return_ospf_dict.update(
+ {
+ "process_id": want_dict["process_id"],
+ "vrf": want_dict["vrf"],
+ }
+ )
+ else:
+ return_ospf_dict.update(
+ {"process_id": want_dict["process_id"]}
+ )
+ ospf_list.append(return_ospf_dict)
+ return ospf_list
+
+ def compare_ospf_list(self, w_list, h_list, l_key):
+ return_list = []
+ for w in w_list:
+ present = False
+ for h in h_list:
+ diff = dict_diff(h, w)
+ if not diff:
+ present = True
+ break
+ if not present:
+ return_list.append(w)
+ return return_list
+
+ def add_commands(self, want, have):
+ commands = []
+ if not want:
+ return commands
+ for ospf_params in want["processes"]:
+ commands.append(_get_router_command(ospf_params))
+ if ospf_params.get("traffic_engineering"):
+ commands.append("traffic-engineering")
+ if ospf_params.get("adjacency"):
+ threshold = ospf_params["adjacency"]["exchange_start"][
+ "threshold"
+ ]
+ commands.append(
+ "adjacency exchange-start threshold " + str(threshold)
+ )
+ if ospf_params.get("areas"):
+ command_list = _parse_areas(ospf_params["areas"])
+ for c in command_list:
+ commands.append(c)
+ if ospf_params.get("auto_cost"):
+ commands.append(
+ "auto-cose reference-bandwidth " + ospf_params["auto_cost"]
+ )
+ if ospf_params.get("bfd"):
+ commands.append("bfd default")
+ if ospf_params.get("default_information"):
+ commands.append(
+ _parse_default_information(
+ ospf_params["default_information"]
+ )
+ )
+ if ospf_params.get("default_metric"):
+ commands.append(
+ "default-metric" + " " + str(ospf_params["default_metric"])
+ )
+ if ospf_params.get("distance"):
+ for k, v in ospf_params["distance"].items():
+ if v:
+ k = re.sub(r"_", "-", k)
+ commands.append("distance ospf " + k + " " + str(v))
+ if ospf_params.get("distribute_list"):
+ commands.append(
+ "distribute-list "
+ + ospf_params["distribute_list"].keys()[0]
+ + " "
+ + ospf_params["distribute_list"].values()[0]
+ + " in"
+ )
+ if ospf_params.get("dn_bit_ignore"):
+ commands.append("dn-bit-ignore")
+ if ospf_params.get("graceful_restart"):
+ if ospf_params["graceful_restart"].get("set"):
+ commands.append("graceful-restart")
+ else:
+ commands.append(
+ "graceful-restart grace-period "
+ + str(
+ ospf_params["graceful_restart"].get("grace_period")
+ )
+ )
+ if ospf_params.get("graceful_restart_helper"):
+ commands.append("graceful-restart-helper")
+ if ospf_params.get("log_adjacency_changes"):
+ cmd = "log-adjacency-changes"
+ if ospf_params["log_adjacency_changes"].get("detail"):
+ cmd = cmd + " detail"
+ commands.append(cmd)
+ if ospf_params.get("max_lsa"):
+ commands.append(_parse_max_lsa(ospf_params["max_lsa"]))
+ if ospf_params.get("max_metric"):
+ commands.append(_parse_max_metric(ospf_params["max_metric"]))
+ if ospf_params.get("maximum_paths"):
+ commands.append(
+ "maximum-paths " + str(ospf_params["maximum_paths"])
+ )
+ if ospf_params.get("mpls_ldp"):
+ commands.append("mpls ldp sync default")
+ if ospf_params.get("networks"):
+ command_list = _parse_networks(ospf_params["networks"])
+ for c in command_list:
+ commands.append(c)
+ if ospf_params.get("passive_interface"):
+ if "interface_list" in ospf_params["passive_interface"].keys():
+ commands.append(
+ "passive-interface "
+ + ospf_params["passive_interface"]["interface_list"]
+ )
+ else:
+ commands.append("passive-interface default")
+ if ospf_params.get("point_to_point"):
+ commands.append("point-to-point routes")
+ if ospf_params.get("redistribute"):
+ command_list = _parse_redistribute(ospf_params["redistribute"])
+ for c in command_list:
+ commands.append(c)
+ if ospf_params.get("retransmission_threshold"):
+ commands.append(
+ "retransmission-threshold lsa "
+ + str(ospf_params["retransmission_threshold"])
+ )
+ if ospf_params.get("rfc1583compatibility"):
+ commands.append("compatible rfc1583")
+ if ospf_params.get("router_id"):
+ commands.append("router-id " + ospf_params.get("router_id"))
+ if ospf_params.get("summary_address"):
+ commands.append(
+ _parse_summary_address(ospf_params["summary_address"])
+ )
+ if ospf_params.get("timers"):
+ command_list = _parse_timers(ospf_params["timers"])
+ for c in command_list:
+ commands.append(c)
+ commands.append("exit")
+ commandset = []
+ for command in commands:
+ commandset.append(command.strip())
+ return commandset
+
+ def del_commands(self, want, have):
+ commands = []
+ other_commands = 0
+ want = remove_empties(want)
+ if want.get("processes"):
+ for w_inst in want["processes"]:
+ router_context = 0
+ d_cmds = []
+ instance_list = []
+ if have.get("processes"):
+ for h_inst in have["processes"]:
+ if h_inst["process_id"] == w_inst["process_id"]:
+ if w_inst.get("vrf") and w_inst.get(
+ "vrf"
+ ) == h_inst.get("vrf"):
+ if list(w_inst.keys()) == [
+ "process_id",
+ "vrf",
+ ]:
+ commands.append(
+ "no router ospf "
+ + str(w_inst["process_id"])
+ + " vrf "
+ + w_inst["vrf"]
+ )
+ router_context = 1
+ if len(w_inst.keys()) == 1 and list(
+ w_inst.keys()
+ ) == ["process_id"]:
+ commands.append(
+ "no router ospf "
+ + str(w_inst["process_id"])
+ )
+ router_context = 1
+ if not router_context:
+ instance_list = self.compare_dicts(
+ w_inst, h_inst
+ )
+ if not instance_list:
+ del_want = {"processes": [w_inst]}
+ d_cmds = self.add_commands(del_want, have)
+ for cmd in d_cmds:
+ if "router ospf" in cmd:
+ other_commands = 0
+ if cmd not in commands:
+ commands.append(cmd)
+ else:
+ cmd = "no " + cmd
+ if cmd not in commands:
+ commands.append(cmd)
+ other_commands += 1
+ if (
+ not other_commands
+ and len(commands) == 1
+ and not router_context
+ ):
+ if (
+ "no" not in commands[0]
+ and "router ospf" in commands[0]
+ ):
+ commands[0] = "no " + commands[0]
+ return commands
+
+
+def _get_router_command(inst):
+ command = ""
+ if inst.get("vrf") and inst.get("vrf") != "default":
+ command = (
+ "router ospf " + str(inst["process_id"]) + " vrf " + inst["vrf"]
+ )
+ else:
+ command = "router ospf " + str(inst["process_id"])
+ return command
+
+
+def _get_vrf_list(want):
+ vrf_list = []
+ if not want:
+ return vrf_list
+ for w in want["processes"]:
+ if w.get("vrf"):
+ vrf_list.append(w["vrf"])
+ return vrf_list
+
+
+def _parse_areas(areas):
+ command = []
+ for area in areas:
+ area_cmd = "area " + area["area_id"]
+ if area.get("default_cost"):
+ command.append(
+ area_cmd + " default-cost " + str(area.get("default_cost"))
+ )
+ elif area.get("filter"):
+ command.append(
+ area_cmd + " " + _parse_areas_filter(area["filter"])
+ )
+ elif area.get("not_so_stubby"):
+ command.append(
+ area_cmd
+ + " "
+ + _parse_areas_filter_notsostubby(area["not_so_stubby"])
+ )
+ elif area.get("nssa"):
+ command.append(
+ area_cmd + " " + _parse_areas_filter_nssa(area["nssa"])
+ )
+ elif area.get("range"):
+ command.append(area_cmd + " " + _parse_areas_range(area["range"]))
+ return command
+
+
+def _parse_areas_filter(filter_dict):
+ filter_cmd = "filter "
+ if filter_dict.get("prefix_list"):
+ filter_cmd = filter_cmd + filter_dict.get("filter")
+ elif filter_dict.get("address"):
+ filter_cmd = filter_cmd + filter_dict.get("address")
+ else:
+ filter_cmd = (
+ filter_cmd
+ + filter_dict.get("subnet_address")
+ + " "
+ + filter_dict.get("subnet_mask")
+ )
+ return filter_cmd
+
+
+def _parse_areas_filter_notsostubby(nss_dict):
+ nss_cmd = "not-so-stubby "
+ if nss_dict.get("default_information_originate"):
+ nss_cmd = nss_cmd + "default-information-originate "
+ for def_keys in nss_dict["default_information_originate"].keys():
+ if (
+ def_keys == "nssa_only"
+ and nss_dict["default_information_originate"]["nssa_only"]
+ ):
+ nss_cmd = nss_cmd + " nssa-only "
+ elif nss_dict["default_information_originate"].get(def_keys):
+ nss_cmd = (
+ nss_cmd
+ + def_keys
+ + " "
+ + nss_dict["default_information_originate"][def_keys]
+ )
+ elif "lsa" in nss_dict.keys() and nss_dict.get("lsa"):
+ nss_cmd = nss_cmd + " lsa type-7 convert type-5"
+ elif "no_summary" in nss_dict.keys() and nss_dict.get("no_summary"):
+ nss_cmd = nss_cmd + " no-summary"
+ elif "nssa_only" in nss_dict.keys() and nss_dict.get("nssa_only"):
+ nss_cmd = nss_cmd + " nssa-only"
+ return nss_cmd
+
+
+def _parse_areas_filter_nssa(nss_dict):
+ nss_cmd = "nssa "
+ if nss_dict.get("default_information_originate"):
+ nss_cmd = nss_cmd + "default-information-originate "
+ for def_keys in nss_dict["default_information_originate"].keys():
+ if (
+ def_keys == "nssa_only"
+ and nss_dict["default_information_originate"]["nssa_only"]
+ ):
+ nss_cmd = nss_cmd + " nssa-only "
+ elif nss_dict["default_information_originate"].get(def_keys):
+ nss_cmd = (
+ nss_cmd
+ + def_keys
+ + " "
+ + nss_dict["default_information_originate"][def_keys]
+ )
+ elif "no_summary" in nss_dict.keys() and nss_dict.get("no_summary"):
+ nss_cmd = nss_cmd + " no-summary"
+ elif "nssa_only" in nss_dict.keys() and nss_dict.get("nssa_only"):
+ nss_cmd = nss_cmd + " nssa-only"
+ return nss_cmd
+
+
+def _parse_areas_range(range_dict):
+ range_cmd = " range "
+ if range_dict.get("address"):
+ range_cmd = range_cmd + range_dict["address"]
+ if range_dict.get("subnet_address"):
+ range_cmd = (
+ range_cmd
+ + range_dict["subnet_address"]
+ + " "
+ + range_dict["subnet_mask"]
+ )
+ if range_dict.get("advertise") is not None:
+ if range_dict["advertise"]:
+ range_cmd = range_cmd + " advertise "
+ else:
+ range_cmd = range_cmd + " not-advertise "
+ if range_dict.get("cost"):
+ range_cmd = range_cmd + " cost " + str(range_dict["cost"])
+ return range_cmd
+
+
+def _parse_default_information(default_dict):
+ def_cmd = "default-information originate"
+ for def_key in sorted(default_dict.keys()):
+ if def_key == "always":
+ if default_dict.get(def_key):
+ def_cmd = def_cmd + " " + def_key
+ elif def_key in ["metric", "metric_type", "route_map"]:
+ if default_dict.get(def_key):
+ k = re.sub(r"_", "-", def_key)
+ def_cmd = def_cmd + " " + k + " " + str(default_dict[def_key])
+ return def_cmd
+
+
+def _parse_max_lsa(max_lsa_dict):
+ max_lsa_cmd = "max-lsa "
+ if max_lsa_dict.get("count"):
+ max_lsa_cmd = max_lsa_cmd + " " + str(max_lsa_dict["count"])
+ if max_lsa_dict.get("threshold"):
+ max_lsa_cmd = max_lsa_cmd + " " + str(max_lsa_dict["threshold"])
+ for lsa_key, lsa_val in sorted(max_lsa_dict.items()):
+ if lsa_key == "warning" and lsa_val:
+ max_lsa_cmd = max_lsa_cmd + " warning-only"
+ elif lsa_key in ["ignore_count", "reset_time", "ignore_time"]:
+ if lsa_val:
+ k = re.sub(r"_", "-", lsa_key)
+ max_lsa_cmd = max_lsa_cmd + " " + k + " " + str(lsa_val) + " "
+ return max_lsa_cmd
+
+
+def _parse_max_metric(max_metric_dict):
+ metric_cmd = "max-metric router-lsa "
+ for k, v in max_metric_dict["router_lsa"].items():
+ if not v:
+ continue
+ if k == "include_stub" and v:
+ metric_cmd = metric_cmd + " include-stub"
+ elif k == "on_startup":
+ metric_cmd = metric_cmd + " on-startup " + str(v["wait_period"])
+ elif k in ["summary_lsa", "external_lsa"]:
+ k = re.sub(r"_", "-", k)
+ if v.get("set"):
+ metric_cmd = metric_cmd + " " + k
+ else:
+ metric_cmd = (
+ metric_cmd + " " + k + " " + str(v.get("max_metric_value"))
+ )
+ return metric_cmd
+
+
+def _parse_networks(net_list):
+ network_cmd = []
+ for net_dict in net_list:
+ net_cmd = "network "
+ if net_dict.get("prefix"):
+ net_cmd = net_cmd + net_dict.get("prefix")
+ else:
+ net_cmd = (
+ net_cmd
+ + net_dict.get("network_address")
+ + " "
+ + net_dict.get("mask")
+ )
+ if net_dict.get("area"):
+ net_cmd = net_cmd + " area " + net_dict.get("area")
+ network_cmd.append(net_cmd)
+ return network_cmd
+
+
+def _parse_redistribute(r_list):
+ rcmd_list = []
+ for r_dict in r_list:
+ r_cmd = "redistribute "
+ r_cmd = r_cmd + r_dict["routes"]
+ if r_dict.get("isis_level"):
+ k = re.sub(r"_", "-", r_dict["isis_level"])
+ r_cmd = r_cmd + " " + k
+ if r_dict.get("route_map"):
+ r_cmd = r_cmd + " route-map " + r_dict["route_map"]
+ rcmd_list.append(r_cmd)
+ return rcmd_list
+
+
+def _parse_summary_address(addr_dict):
+ sum_cmd = "summary-address "
+ if addr_dict.get("prefix"):
+ sum_cmd = sum_cmd + addr_dict.get("prefix")
+ else:
+ sum_cmd = (
+ sum_cmd + addr_dict.get("address") + " " + addr_dict.get("mask")
+ )
+ if "attribute_map" in addr_dict.keys():
+ sum_cmd = sum_cmd + " attribute-map " + addr_dict["attribute_map"]
+ elif addr_dict.get("not_advertise"):
+ sum_cmd = sum_cmd + " not-advertise "
+ elif "tag" in addr_dict.keys():
+ sum_cmd = sum_cmd + " tag " + addr_dict["tag"]
+ return sum_cmd
+
+
+def _parse_timers(timers_list):
+ timers_cmd = []
+ for t_dict in timers_list:
+ t_cmd = "timers "
+ for t_key in t_dict.keys():
+ if not t_dict.get(t_key):
+ break
+ if t_key == "lsa":
+ if t_dict["lsa"].get("rx"):
+ t_cmd = (
+ t_cmd
+ + "lsa rx min interval "
+ + str(t_dict["lsa"]["rx"]["min_interval"])
+ )
+ else:
+ t_cmd = (
+ t_cmd
+ + "lsa tx delay initial "
+ + str(t_dict["lsa"]["tx"]["delay"]["initial"])
+ + " "
+ + str(t_dict["lsa"]["tx"]["delay"]["min"])
+ + " "
+ + str(t_dict["lsa"]["tx"]["delay"]["max"])
+ )
+ elif t_key == "out_delay":
+ t_cmd = t_cmd + " out-delay " + str(t_dict["out_delay"])
+ elif t_key == "pacing":
+ t_cmd = t_cmd + " pacing flood " + str(t_dict["pacing"])
+ elif t_key == "spf":
+ if "seconds" in t_dict["spf"].keys():
+ t_cmd = t_cmd + " spf " + str(t_dict["spf"]["seconds"])
+ else:
+ t_cmd = (
+ t_cmd
+ + " spf delay initial "
+ + str(t_dict["spf"]["initial"])
+ + " "
+ + str(t_dict["spf"]["max"])
+ + " "
+ + str(t_dict["spf"]["min"])
+ )
+ timers_cmd.append(t_cmd)
+ return timers_cmd
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py
new file mode 100644
index 00000000..a0430737
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py
@@ -0,0 +1,285 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos_ospfv3 config file.
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to its desired end-state is
+created.
+"""
+
+import re
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ dict_merge,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import (
+ ResourceModule,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospfv3 import (
+ Ospfv3Template,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ get_from_dict,
+)
+
+
+class Ospfv3(ResourceModule):
+ """
+ The eos_ospfv3 config class
+ """
+
+ def __init__(self, module):
+ super(Ospfv3, self).__init__(
+ empty_fact_val={},
+ facts_module=Facts(module),
+ module=module,
+ resource="ospfv3",
+ tmplt=Ospfv3Template(),
+ )
+ self.parsers = [
+ "vrf",
+ "address_family",
+ "adjacency",
+ "auto_cost",
+ "area.default_cost",
+ "area.authentication",
+ "area.encryption",
+ "area.nssa",
+ "area.ranges",
+ "area.stub",
+ "bfd",
+ "default_information",
+ "default_metric",
+ "distance",
+ "fips_restrictions",
+ "graceful_restart",
+ "graceful_restart_period",
+ "graceful_restart_helper",
+ "log_adjacency_changes",
+ "max_metric",
+ "maximum_paths",
+ "passive_interface",
+ "redistribute",
+ "router_id",
+ "shutdown",
+ "timers.lsa",
+ "timers.out_delay",
+ "timers.pacing",
+ "timers.throttle.lsa",
+ "timers.throttle.spf",
+ ]
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ if self.state not in ["parsed", "gathered"]:
+ self.generate_commands()
+ self.run_commands()
+ return self.result
+
+ def generate_commands(self):
+ """ Generate configuration commands to send based on
+ want, have and desired state.
+ """
+
+ wantd = {
+ entry["vrf"]: entry for entry in self.want.get("processes", [])
+ }
+ haved = {
+ entry["vrf"]: entry for entry in self.have.get("processes", [])
+ }
+
+ # turn all lists of dicts into dicts prior to merge
+ for entry in wantd, haved:
+ self._ospf_list_to_dict(entry)
+ # if state is merged, merge want onto have and then compare
+ if self.state == "merged":
+ wantd = dict_merge(haved, wantd)
+
+ # if state is deleted, empty out wantd and set haved to wantd
+ if self.state == "deleted":
+ haved = {
+ k: v for k, v in iteritems(haved) if k in wantd or not wantd
+ }
+ wantd = {}
+
+ # remove superfluous config for overridden and deleted
+ if self.state in ["overridden", "deleted"]:
+ for k, have in iteritems(haved):
+ if k not in wantd and have.get("vrf") == k:
+ self.commands.append(self._tmplt.render(have, "vrf", True))
+
+ for k, want in iteritems(wantd):
+ self._compare(want=want, have=haved.pop(k, {}))
+
+ def _compare(self, want, have):
+ """Leverages the base class `compare()` method and
+ populates the list of commands to be run by comparing
+ the `want` and `have` data with the `parsers` defined
+ for the Ospfv3 network resource.
+ """
+ begin = len(self.commands)
+ self._af_compare(want=want, have=have)
+ self._global_compare(want=want, have=have)
+
+ if len(self.commands) != begin or (not have and want):
+ self.commands.insert(
+ begin, self._tmplt.render(want or have, "vrf", False)
+ )
+ self.commands.append("exit")
+
+ def _global_compare(self, want, have):
+ for name, entry in iteritems(want):
+ if name in ["vrf", "address_family"]:
+ continue
+ if not isinstance(entry, dict) and name != "areas":
+ self.compare(
+ parsers=self.parsers,
+ want={name: entry},
+ have={name: have.pop(name, None)},
+ )
+ else:
+ if name == "areas" and entry:
+ self._areas_compare(
+ want={name: entry}, have={name: have.get(name, {})}
+ )
+ else:
+ # passing dict without vrf, inorder to avoid no router ospfv3 command
+ h = {i: have[i] for i in have if i != "vrf"}
+ self.compare(
+ parsers=self.parsers,
+ want={name: entry},
+ have={name: h.pop(name, {})},
+ )
+ # remove remaining items in have for replaced
+ for name, entry in iteritems(have):
+ if name in ["vrf", "address_family"]:
+ continue
+ if not isinstance(entry, dict):
+ self.compare(
+ parsers=self.parsers,
+ want={name: want.pop(name, None)},
+ have={name: entry},
+ )
+ else:
+ # passing dict without vrf, inorder to avoid no router ospfv3 command
+ # w = {i: want[i] for i in want if i != "vrf"}
+ self.compare(
+ parsers=self.parsers,
+ want={name: want.pop(name, {})},
+ have={name: entry},
+ )
+
+ def _af_compare(self, want, have):
+ wafs = want.get("address_family", {})
+ hafs = have.get("address_family", {})
+ for name, entry in iteritems(wafs):
+ begin = len(self.commands)
+ self._compare_lists(want=entry, have=hafs.get(name, {}))
+ self._areas_compare(want=entry, have=hafs.get(name, {}))
+ self.compare(
+ parsers=self.parsers, want=entry, have=hafs.pop(name, {})
+ )
+ if (
+ len(self.commands) != begin
+ and "afi" in entry
+ and entry["afi"] != "router"
+ ):
+ self._rotate_commands(begin=begin)
+ self.commands.insert(
+ begin, self._tmplt.render(entry, "address_family", False)
+ )
+ self.commands.append("exit")
+ for name, entry in iteritems(hafs):
+ self.addcmd(entry, "address_family", True)
+
+ def _rotate_commands(self, begin=0):
+ # move negate commands to beginning
+ for cmd in self.commands[begin::]:
+ negate = re.match(r"^no .*", cmd)
+ if negate:
+ self.commands.insert(
+ begin, self.commands.pop(self.commands.index(cmd))
+ )
+ begin += 1
+
+ def _areas_compare(self, want, have):
+ wareas = want.get("areas", {})
+ hareas = have.get("areas", {})
+ for name, entry in iteritems(wareas):
+ self._area_compare(want=entry, have=hareas.pop(name, {}))
+ for name, entry in iteritems(hareas):
+ self._area_compare(want={}, have=entry)
+
+ def _area_compare(self, want, have):
+ parsers = [
+ "area.default_cost",
+ "area.encryption",
+ "area.authentication",
+ "area.nssa",
+ "area.stub",
+ ]
+ self.compare(parsers=parsers, want=want, have=have)
+ self._area_compare_lists(want=want, have=have)
+
+ def _area_compare_lists(self, want, have):
+ for attrib in ["ranges"]:
+ wdict = want.get(attrib, {})
+ hdict = have.get(attrib, {})
+ for key, entry in iteritems(wdict):
+ if entry != hdict.pop(key, {}):
+ entry["area_id"] = want["area_id"]
+ self.addcmd(entry, "area.{0}".format(attrib), False)
+ # remove remaining items in have for replaced
+ for entry in hdict.values():
+ entry["area_id"] = have["area_id"]
+ self.addcmd(entry, "area.{0}".format(attrib), True)
+
+ def _compare_lists(self, want, have):
+ for attrib in ["redistribute"]:
+ wdict = get_from_dict(want, attrib) or {}
+ hdict = get_from_dict(have, attrib) or {}
+ for key, entry in iteritems(wdict):
+ if entry != hdict.pop(key, {}):
+ self.addcmd(entry, attrib, False)
+ # remove remaining items in have for replaced
+ for entry in hdict.values():
+ self.addcmd(entry, attrib, True)
+
+ def _ospf_list_to_dict(self, entry):
+ for name, proc in iteritems(entry):
+ for area in proc.get("areas", []):
+ if "ranges" in area:
+ area["ranges"] = {
+ entry["address"]: entry
+ for entry in area.get("ranges", [])
+ }
+ proc["areas"] = {
+ entry["area_id"]: entry for entry in proc.get("areas", [])
+ }
+ proc["redistribute"] = {
+ entry["routes"]: entry
+ for entry in proc.get("redistribute", [])
+ }
+ if "address_family" in proc:
+ proc["address_family"] = {
+ entry["afi"]: entry
+ for entry in proc.get("address_family", [])
+ }
+ self._ospf_list_to_dict(proc["address_family"])
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py
new file mode 100644
index 00000000..5dd28b11
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py
@@ -0,0 +1,364 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_static_routes class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_empties,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Static_routes(ConfigBase):
+ """
+ The eos_static_routes class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["static_routes"]
+
+ def __init__(self, module):
+ super(Static_routes, self).__init__(module)
+
+ def get_static_routes_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ static_routes_facts = facts["ansible_network_resources"].get(
+ "static_routes"
+ )
+ if not static_routes_facts:
+ return []
+ return static_routes_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+ if self.state in self.ACTION_STATES:
+ existing_static_routes_facts = self.get_static_routes_facts()
+ else:
+ existing_static_routes_facts = []
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_static_routes_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_static_routes_facts = self.get_static_routes_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ if not self._module.params["running_config"]:
+ self._module.fail_json(
+ msg="Value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_static_routes_facts(
+ data=self._module.params["running_config"]
+ )
+ else:
+ changed_static_routes_facts = []
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_static_routes_facts
+ if result["changed"]:
+ result["after"] = changed_static_routes_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_static_routes_facts
+
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_static_routes_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ onbox_configs = []
+ for h in existing_static_routes_facts:
+ return_command = add_commands(h)
+ for command in return_command:
+ onbox_configs.append(command)
+ config = self._module.params.get("config")
+ want = []
+ if config:
+ for w in config:
+ want.append(remove_empties(w))
+ have = existing_static_routes_facts
+ resp = self.set_state(want, have)
+ for want_config in resp:
+ if want_config not in onbox_configs:
+ commands.append(want_config)
+ return commands
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ if self.state in ("merged", "replaced", "overridden") and not want:
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ self.state
+ )
+ )
+ state = self._module.params["state"]
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or self.state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ haveconfigs = []
+ vrf = get_vrf(want)
+ dest = get_dest(want)
+ for h in have:
+ return_command = add_commands(h)
+ for command in return_command:
+ for d in dest:
+ if d in command:
+ if vrf is None:
+ if "vrf" not in command:
+ haveconfigs.append(command)
+ else:
+ if vrf in command:
+ haveconfigs.append(command)
+ wantconfigs = set_commands(want, have)
+
+ removeconfigs = list(set(haveconfigs) - set(wantconfigs))
+ for command in removeconfigs:
+ commands.append("no " + command)
+ for wantcmd in wantconfigs:
+ commands.append(wantcmd)
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ haveconfigs = []
+ for h in have:
+ return_command = add_commands(h)
+ for command in return_command:
+ haveconfigs.append(command)
+ wantconfigs = set_commands(want, have)
+ idempotentconfigs = list(set(haveconfigs) - set(wantconfigs))
+ if not idempotentconfigs:
+ return idempotentconfigs
+ removeconfigs = list(set(haveconfigs) - set(wantconfigs))
+ for command in removeconfigs:
+ commands.append("no " + command)
+ for wantcmd in wantconfigs:
+ commands.append(wantcmd)
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ return set_commands(want, have)
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ if not want:
+ for h in have:
+ return_command = add_commands(h)
+ for command in return_command:
+ command = "no " + command
+ commands.append(command)
+ else:
+ for w in want:
+ return_command = del_commands(w, have)
+ for command in return_command:
+ commands.append(command)
+ return commands
+
+
+def set_commands(want, have):
+ commands = []
+ for w in want:
+ return_command = add_commands(w)
+ for command in return_command:
+ commands.append(command)
+ return commands
+
+
+def add_commands(want):
+ commandset = []
+ if not want:
+ return commandset
+ vrf = (
+ want["vrf"]
+ if "vrf" in want.keys() and want["vrf"] is not None
+ else None
+ )
+ for address_family in want["address_families"]:
+ for route in address_family["routes"]:
+ for next_hop in route["next_hops"]:
+ commands = []
+ if address_family["afi"] == "ipv4":
+ commands.append("ip route")
+ else:
+ commands.append("ipv6 route")
+ if vrf:
+ commands.append(" vrf " + vrf)
+ if not re.search(r"/", route["dest"]):
+ mask = route["dest"].split()[1]
+ cidr = get_net_size(mask)
+ commands.append(
+ " " + route["dest"].split()[0] + "/" + cidr
+ )
+ else:
+ commands.append(" " + route["dest"])
+ if "interface" in next_hop.keys():
+ commands.append(" " + next_hop["interface"])
+ if "nexthop_grp" in next_hop.keys():
+ commands.append(
+ " Nexthop-Group" + " " + next_hop["nexthop_grp"]
+ )
+ if "forward_router_address" in next_hop.keys():
+ commands.append(" " + next_hop["forward_router_address"])
+ if "mpls_label" in next_hop.keys():
+ commands.append(" label " + str(next_hop["mpls_label"]))
+ if "track" in next_hop.keys():
+ commands.append(" track " + next_hop["track"])
+ if "admin_distance" in next_hop.keys():
+ commands.append(" " + str(next_hop["admin_distance"]))
+ if "description" in next_hop.keys():
+ commands.append(" name " + str(next_hop["description"]))
+ if "tag" in next_hop.keys():
+ commands.append(" tag " + str(next_hop["tag"]))
+
+ config_commands = "".join(commands)
+ commandset.append(config_commands)
+ return commandset
+
+
+def del_commands(want, have):
+ commandset = []
+ haveconfigs = []
+ for h in have:
+ return_command = add_commands(h)
+ for command in return_command:
+ command = "no " + command
+ haveconfigs.append(command)
+ if want is None or "address_families" not in want.keys():
+ commandset = haveconfigs
+ if "address_families" not in want.keys() and "vrf" in want.keys():
+ commandset = []
+ for command in haveconfigs:
+ if want["vrf"] in command:
+ commandset.append(command)
+ elif (
+ want is not None
+ and "vrf" not in want.keys()
+ and "address_families" not in want.keys()
+ ):
+ commandset = []
+ for command in haveconfigs:
+ if "vrf" not in command:
+ commandset.append(command)
+
+ elif want["address_families"]:
+ for address_family in want["address_families"]:
+ for command in haveconfigs:
+ afi = "ip " if address_family["afi"] == "ipv4" else "ipv6"
+ if afi in command:
+ commandset.append(command)
+ return commandset
+
+
+def get_net_size(netmask):
+ binary_str = ""
+ netmask = netmask.split(".")
+ for octet in netmask:
+ binary_str += bin(int(octet))[2:].zfill(8)
+ return str(len(binary_str.rstrip("0")))
+
+
+def get_vrf(config):
+ vrf = ""
+ for c in config:
+ vrf = c["vrf"] if "vrf" in c.keys() and c["vrf"] else None
+ return vrf
+
+
+def get_dest(config):
+ dest = []
+ for c in config:
+ for address_family in c["address_families"]:
+ for route in address_family["routes"]:
+ dest.append(route["dest"])
+ return dest
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py
new file mode 100644
index 00000000..9e0fedf5
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py
@@ -0,0 +1,258 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos_vlans class
+It is in this file where the current configuration (as dict)
+is compared to the provided configuration (as dict) and the command set
+necessary to bring the current configuration to it's desired end-state is
+created
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import (
+ ConfigBase,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ dict_diff,
+ param_list_to_dict,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+
+
+class Vlans(ConfigBase):
+ """
+ The eos_vlans class
+ """
+
+ gather_subset = ["!all", "!min"]
+
+ gather_network_resources = ["vlans"]
+
+ def get_vlans_facts(self, data=None):
+ """ Get the 'facts' (the current configuration)
+
+ :rtype: A dictionary
+ :returns: The current configuration as a dictionary
+ """
+ facts, _warnings = Facts(self._module).get_facts(
+ self.gather_subset, self.gather_network_resources, data=data
+ )
+ vlans_facts = facts["ansible_network_resources"].get("vlans")
+ if not vlans_facts:
+ return []
+ return vlans_facts
+
+ def execute_module(self):
+ """ Execute the module
+
+ :rtype: A dictionary
+ :returns: The result from module execution
+ """
+ result = {"changed": False}
+ warnings = list()
+ commands = list()
+
+ if self.state in self.ACTION_STATES:
+ existing_vlans_facts = self.get_vlans_facts()
+ else:
+ existing_vlans_facts = {}
+
+ if self.state in self.ACTION_STATES or self.state == "rendered":
+ commands.extend(self.set_config(existing_vlans_facts))
+
+ if commands and self.state in self.ACTION_STATES:
+ if not self._module.check_mode:
+ self._connection.edit_config(commands)
+ result["changed"] = True
+ if self.state in self.ACTION_STATES:
+ result["commands"] = commands
+
+ if self.state in self.ACTION_STATES or self.state == "gathered":
+ changed_vlans_facts = self.get_vlans_facts()
+ elif self.state == "rendered":
+ result["rendered"] = commands
+ elif self.state == "parsed":
+ running_config = self._module.params["running_config"]
+ if not running_config:
+ self._module.fail_json(
+ msg="value of running_config parameter must not be empty for state parsed"
+ )
+ result["parsed"] = self.get_vlans_facts(data=running_config)
+
+ if self.state in self.ACTION_STATES:
+ result["before"] = existing_vlans_facts
+ if result["changed"]:
+ result["after"] = changed_vlans_facts
+ elif self.state == "gathered":
+ result["gathered"] = changed_vlans_facts
+ result["warnings"] = warnings
+ return result
+
+ def set_config(self, existing_vlans_facts):
+ """ Collect the configuration from the args passed to the module,
+ collect the current configuration (as a dict from facts)
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ want = self._module.params["config"]
+ have = existing_vlans_facts
+ resp = self.set_state(want, have)
+ return to_list(resp)
+
+ def set_state(self, want, have):
+ """ Select the appropriate function based on the state provided
+
+ :param want: the desired configuration as a dictionary
+ :param have: the current configuration as a dictionary
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ state = self._module.params["state"]
+ if (
+ state in ("merged", "replaced", "overridden", "rendered")
+ and not want
+ ):
+ self._module.fail_json(
+ msg="value of config parameter must not be empty for state {0}".format(
+ state
+ )
+ )
+ want = param_list_to_dict(want, "vlan_id", remove_key=False)
+ have = param_list_to_dict(have, "vlan_id", remove_key=False)
+ if state == "overridden":
+ commands = self._state_overridden(want, have)
+ elif state == "deleted":
+ commands = self._state_deleted(want, have)
+ elif state == "merged" or state == "rendered":
+ commands = self._state_merged(want, have)
+ elif state == "replaced":
+ commands = self._state_replaced(want, have)
+ return commands
+
+ @staticmethod
+ def _state_replaced(want, have):
+ """ The command generator when state is replaced
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for vlan_id, desired in want.items():
+ if vlan_id in have:
+ extant = have[vlan_id]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(vlan_id, add_config, del_config))
+
+ return commands
+
+ @staticmethod
+ def _state_overridden(want, have):
+ """ The command generator when state is overridden
+
+ :rtype: A list
+ :returns: the commands necessary to migrate the current configuration
+ to the desired configuration
+ """
+ commands = []
+ for vlan_id, extant in have.items():
+ if vlan_id in want:
+ desired = want[vlan_id]
+ else:
+ desired = dict()
+
+ add_config = dict_diff(extant, desired)
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(vlan_id, add_config, del_config))
+
+ # Handle vlans not already in config
+ new_vlans = [vlan_id for vlan_id in want if vlan_id not in have]
+ for vlan_id in new_vlans:
+ desired = want[vlan_id]
+ extant = dict(vlan_id=vlan_id)
+ add_config = dict_diff(extant, desired)
+
+ commands.extend(generate_commands(vlan_id, add_config, {}))
+
+ return commands
+
+ @staticmethod
+ def _state_merged(want, have):
+ """ The command generator when state is merged
+
+ :rtype: A list
+ :returns: the commands necessary to merge the provided into
+ the current configuration
+ """
+ commands = []
+ for vlan_id, desired in want.items():
+ if vlan_id in have:
+ extant = have[vlan_id]
+ else:
+ extant = dict()
+
+ add_config = dict_diff(extant, desired)
+
+ commands.extend(generate_commands(vlan_id, add_config, {}))
+
+ return commands
+
+ @staticmethod
+ def _state_deleted(want, have):
+ """ The command generator when state is deleted
+
+ :rtype: A list
+ :returns: the commands necessary to remove the current configuration
+ of the provided objects
+ """
+ commands = []
+ for vlan_id in want:
+ desired = dict()
+ if vlan_id in have:
+ extant = have[vlan_id]
+ else:
+ continue
+
+ del_config = dict_diff(desired, extant)
+
+ commands.extend(generate_commands(vlan_id, {}, del_config))
+
+ return commands
+
+
+def generate_commands(vlan_id, to_set, to_remove):
+ commands = []
+ if "vlan_id" in to_remove:
+ return ["no vlan {0}".format(vlan_id)]
+
+ for key in to_remove:
+ if key in to_set.keys():
+ continue
+ commands.append("no {0}".format(key))
+
+ for key, value in to_set.items():
+ if key == "vlan_id" or value is None:
+ continue
+
+ commands.append("{0} {1}".format(key, value))
+
+ if commands:
+ commands.insert(0, "vlan {0}".format(vlan_id))
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py
new file mode 100644
index 00000000..609cec5f
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py
@@ -0,0 +1,751 @@
+#
+# This code is part of Ansible, but is an independent component.
+#
+# This particular file snippet, and this file snippet only, is BSD licensed.
+# Modules you write using this snippet, which is embedded dynamically by Ansible
+# still belong to the author of the module, and may assign their own license
+# to the complete work.
+#
+# (c) 2017 Red Hat, Inc.
+#
+# Redistribution and use in source and binary forms, with or without modification,
+# are permitted provided that the following conditions are met:
+#
+# * Redistributions of source code must retain the above copyright
+# notice, this list of conditions and the following disclaimer.
+# * Redistributions in binary form must reproduce the above copyright notice,
+# this list of conditions and the following disclaimer in the documentation
+# and/or other materials provided with the distribution.
+#
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
+# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+# IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
+# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+import json
+import os
+import time
+
+from ansible.module_utils._text import to_text
+from ansible.module_utils.basic import env_fallback
+from ansible.module_utils.connection import Connection, ConnectionError
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+ dumps,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+ ComplexList,
+)
+from ansible.module_utils.urls import fetch_url
+
+_DEVICE_CONNECTION = None
+
+eos_provider_spec = {
+ "host": dict(),
+ "port": dict(type="int"),
+ "username": dict(fallback=(env_fallback, ["ANSIBLE_NET_USERNAME"])),
+ "password": dict(
+ fallback=(env_fallback, ["ANSIBLE_NET_PASSWORD"]), no_log=True
+ ),
+ "ssh_keyfile": dict(
+ fallback=(env_fallback, ["ANSIBLE_NET_SSH_KEYFILE"]), type="path"
+ ),
+ "authorize": dict(
+ fallback=(env_fallback, ["ANSIBLE_NET_AUTHORIZE"]),
+ type="bool",
+ default=False,
+ ),
+ "auth_pass": dict(
+ no_log=True, fallback=(env_fallback, ["ANSIBLE_NET_AUTH_PASS"])
+ ),
+ "use_ssl": dict(default=True, type="bool"),
+ "use_proxy": dict(default=True, type="bool"),
+ "validate_certs": dict(default=True, type="bool"),
+ "timeout": dict(type="int"),
+ "transport": dict(default="cli", choices=["cli", "eapi"]),
+}
+eos_argument_spec = {
+ "provider": dict(
+ type="dict",
+ options=eos_provider_spec,
+ removed_at_date="2022-06-01",
+ removed_from_collection="arista.eos",
+ )
+}
+
+
+def get_provider_argspec():
+ return eos_provider_spec
+
+
+def get_connection(module):
+ global _DEVICE_CONNECTION
+ if not _DEVICE_CONNECTION:
+ if is_local_eapi(module):
+ conn = LocalEapi(module)
+ else:
+ connection_proxy = Connection(module._socket_path)
+ cap = json.loads(connection_proxy.get_capabilities())
+ if cap["network_api"] == "cliconf":
+ conn = Cli(module)
+ elif cap["network_api"] == "eapi":
+ conn = HttpApi(module)
+ _DEVICE_CONNECTION = conn
+ return _DEVICE_CONNECTION
+
+
+class Cli:
+ def __init__(self, module):
+ self._module = module
+ self._device_configs = {}
+ self._session_support = None
+ self._connection = None
+
+ @property
+ def supports_sessions(self):
+ if self._session_support is None:
+ self._session_support = self._get_connection().supports_sessions()
+ return self._session_support
+
+ def _get_connection(self):
+ if self._connection:
+ return self._connection
+ self._connection = Connection(self._module._socket_path)
+
+ return self._connection
+
+ def get_config(self, flags=None):
+ """Retrieves the current config from the device or cache
+ """
+ flags = [] if flags is None else flags
+
+ cmd = "show running-config "
+ cmd += " ".join(flags)
+ cmd = cmd.strip()
+
+ try:
+ return self._device_configs[cmd]
+ except KeyError:
+ conn = self._get_connection()
+ try:
+ out = conn.get_config(flags=flags)
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+
+ cfg = to_text(out, errors="surrogate_then_replace").strip()
+ self._device_configs[cmd] = cfg
+ return cfg
+
+ def run_commands(self, commands, check_rc=True):
+ """Run list of commands on remote device and return results
+ """
+ connection = self._get_connection()
+ try:
+ response = connection.run_commands(
+ commands=commands, check_rc=check_rc
+ )
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+ return response
+
+ def load_config(self, commands, commit=False, replace=False):
+ """Loads the config commands onto the remote device
+ """
+ conn = self._get_connection()
+ try:
+ response = conn.edit_config(commands, commit, replace)
+ except ConnectionError as exc:
+ message = getattr(exc, "err", to_text(exc))
+ if (
+ "check mode is not supported without configuration session"
+ in message
+ ):
+ self._module.warn(
+ "EOS can not check config without config session"
+ )
+ response = {"changed": True}
+ else:
+ self._module.fail_json(
+ msg="%s" % message,
+ data=to_text(message, errors="surrogate_then_replace"),
+ )
+
+ return response
+
+ def get_diff(
+ self,
+ candidate=None,
+ running=None,
+ diff_match="line",
+ diff_ignore_lines=None,
+ path=None,
+ diff_replace="line",
+ ):
+ conn = self._get_connection()
+ try:
+ diff = conn.get_diff(
+ candidate=candidate,
+ running=running,
+ diff_match=diff_match,
+ diff_ignore_lines=diff_ignore_lines,
+ path=path,
+ diff_replace=diff_replace,
+ )
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+ return diff
+
+ def get_capabilities(self):
+ """Returns platform info of the remove device
+ """
+ if hasattr(self._module, "_capabilities"):
+ return self._module._capabilities
+
+ connection = self._get_connection()
+ try:
+ capabilities = connection.get_capabilities()
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+ self._module._capabilities = json.loads(capabilities)
+ return self._module._capabilities
+
+
+class LocalEapi:
+ def __init__(self, module):
+ self._module = module
+ self._enable = None
+ self._session_support = None
+ self._device_configs = {}
+
+ provider = module.params.get("provider") or {}
+ host = provider.get("host")
+ port = provider.get("port")
+
+ self._module.params["url_username"] = provider.get("username")
+ self._module.params["url_password"] = provider.get("password")
+
+ if provider.get("use_ssl"):
+ proto = "https"
+ else:
+ proto = "http"
+
+ module.params["validate_certs"] = provider.get("validate_certs")
+
+ self._url = "%s://%s:%s/command-api" % (proto, host, port)
+
+ if provider.get("auth_pass"):
+ self._enable = {
+ "cmd": "enable",
+ "input": provider.get("auth_pass"),
+ }
+ else:
+ self._enable = "enable"
+
+ @property
+ def supports_sessions(self):
+ if self._session_support is None:
+ response = self.send_request(["show configuration sessions"])
+ self._session_support = "error" not in response
+ return self._session_support
+
+ def _request_builder(self, commands, output, reqid=None):
+ params = dict(version=1, cmds=commands, format=output)
+ return dict(jsonrpc="2.0", id=reqid, method="runCmds", params=params)
+
+ def send_request(self, commands, output="text"):
+ commands = to_list(commands)
+
+ if self._enable:
+ commands.insert(0, self._enable)
+
+ body = self._request_builder(commands, output)
+ data = self._module.jsonify(body)
+
+ headers = {"Content-Type": "application/json-rpc"}
+ timeout = self._module.params["provider"]["timeout"]
+ use_proxy = self._module.params["provider"]["use_proxy"]
+
+ response, headers = fetch_url(
+ self._module,
+ self._url,
+ data=data,
+ headers=headers,
+ method="POST",
+ timeout=timeout,
+ use_proxy=use_proxy,
+ )
+
+ if headers["status"] != 200:
+ self._module.fail_json(**headers)
+
+ try:
+ data = response.read()
+ response = self._module.from_json(
+ to_text(data, errors="surrogate_then_replace")
+ )
+ except ValueError:
+ self._module.fail_json(
+ msg="unable to load response from device", data=data
+ )
+
+ if self._enable and "result" in response:
+ response["result"].pop(0)
+
+ return response
+
+ def run_commands(self, commands, check_rc=True):
+ """Runs list of commands on remote device and returns results
+ """
+ output = None
+ queue = list()
+ responses = list()
+
+ def _send(commands, output):
+ response = self.send_request(commands, output=output)
+ if "error" in response:
+ err = response["error"]
+ self._module.fail_json(msg=err["message"], code=err["code"])
+ return response["result"]
+
+ for item in to_list(commands):
+ if is_json(item["command"]):
+ item["command"] = str(item["command"]).replace("| json", "")
+ item["output"] = "json"
+
+ if output and output != item["output"]:
+ responses.extend(_send(queue, output))
+ queue = list()
+
+ output = item["output"] or "json"
+ queue.append(item["command"])
+
+ if queue:
+ responses.extend(_send(queue, output))
+
+ for index, item in enumerate(commands):
+ try:
+ responses[index] = responses[index]["output"].strip()
+ except KeyError:
+ pass
+
+ return responses
+
+ def get_config(self, flags=None):
+ """Retrieves the current config from the device or cache
+ """
+ flags = [] if flags is None else flags
+
+ cmd = "show running-config "
+ cmd += " ".join(flags)
+ cmd = cmd.strip()
+
+ try:
+ return self._device_configs[cmd]
+ except KeyError:
+ out = self.send_request(cmd)
+ cfg = str(out["result"][0]["output"]).strip()
+ self._device_configs[cmd] = cfg
+ return cfg
+
+ def configure(self, commands):
+ """Sends the ordered set of commands to the device
+ """
+ cmds = ["configure terminal"]
+ cmds.extend(commands)
+
+ responses = self.send_request(commands)
+ if "error" in responses:
+ err = responses["error"]
+ self._module.fail_json(msg=err["message"], code=err["code"])
+
+ return responses[1:]
+
+ def load_config(self, config, commit=False, replace=False):
+ """Loads the configuration onto the remote devices
+
+ If the device doesn't support configuration sessions, this will
+ fallback to using configure() to load the commands. If that happens,
+ there will be no returned diff or session values
+ """
+ use_session = os.getenv("ANSIBLE_EOS_USE_SESSIONS", True)
+ try:
+ use_session = int(use_session)
+ except ValueError:
+ pass
+
+ if not all((bool(use_session), self.supports_sessions)):
+ if commit:
+ return self.configure(config)
+ else:
+ self._module.warn(
+ "EOS can not check config without config session"
+ )
+ result = {"changed": True}
+ return result
+
+ session = "ansible_%s" % int(time.time())
+ result = {"session": session}
+ commands = ["configure session %s" % session]
+
+ if replace:
+ commands.append("rollback clean-config")
+
+ commands.extend(config)
+
+ response = self.send_request(commands)
+ if "error" in response:
+ commands = ["configure session %s" % session, "abort"]
+ self.send_request(commands)
+ err = response["error"]
+ error_text = []
+ for data in err["data"]:
+ error_text.extend(data.get("errors", []))
+ error_text = "\n".join(error_text) or err["message"]
+ self._module.fail_json(msg=error_text, code=err["code"])
+
+ commands = [
+ "configure session %s" % session,
+ "show session-config diffs",
+ ]
+ if commit:
+ commands.append("commit")
+ else:
+ commands.append("abort")
+
+ response = self.send_request(commands, output="text")
+ diff = response["result"][1]["output"]
+ if len(diff) > 0:
+ result["diff"] = diff
+
+ return result
+
+ # get_diff added here to support connection=local and transport=eapi scenario
+ def get_diff(
+ self,
+ candidate,
+ running=None,
+ diff_match="line",
+ diff_ignore_lines=None,
+ path=None,
+ diff_replace="line",
+ ):
+ diff = {}
+
+ # prepare candidate configuration
+ candidate_obj = NetworkConfig(indent=3)
+ candidate_obj.load(candidate)
+
+ if running and diff_match != "none" and diff_replace != "config":
+ # running configuration
+ running_obj = NetworkConfig(
+ indent=3, contents=running, ignore_lines=diff_ignore_lines
+ )
+ configdiffobjs = candidate_obj.difference(
+ running_obj, path=path, match=diff_match, replace=diff_replace
+ )
+
+ else:
+ configdiffobjs = candidate_obj.items
+
+ configdiff = (
+ dumps(configdiffobjs, "commands") if configdiffobjs else ""
+ )
+ diff["config_diff"] = configdiff if configdiffobjs else {}
+ return diff
+
+ def get_capabilities(self):
+ # Implement the bare minimum to support eos_facts
+ return dict(device_info=dict(network_os="eos"), network_api="eapi")
+
+
+class HttpApi:
+ def __init__(self, module):
+ self._module = module
+ self._device_configs = {}
+ self._session_support = None
+ self._connection_obj = None
+
+ @property
+ def _connection(self):
+ if not self._connection_obj:
+ self._connection_obj = Connection(self._module._socket_path)
+
+ return self._connection_obj
+
+ @property
+ def supports_sessions(self):
+ if self._session_support is None:
+ self._session_support = self._connection.supports_sessions()
+ return self._session_support
+
+ def run_commands(self, commands, check_rc=True):
+ """Runs list of commands on remote device and returns results
+ """
+ output = None
+ queue = list()
+ responses = list()
+
+ def run_queue(queue, output):
+ try:
+ response = to_list(
+ self._connection.send_request(queue, output=output)
+ )
+ except ConnectionError as exc:
+ if check_rc:
+ raise
+ return to_list(to_text(exc))
+
+ if output == "json":
+ response = [json.loads(item) for item in response]
+ return response
+
+ for item in to_list(commands):
+ cmd_output = "text"
+ if isinstance(item, dict):
+ command = item["command"]
+ if "output" in item:
+ cmd_output = item["output"]
+ else:
+ command = item
+
+ # Emulate '| json' from CLI
+ if is_json(command):
+ command = command.rsplit("|", 1)[0]
+ cmd_output = "json"
+
+ if output and output != cmd_output:
+ responses.extend(run_queue(queue, output))
+ queue = list()
+
+ output = cmd_output
+ queue.append(command)
+
+ if queue:
+ responses.extend(run_queue(queue, output))
+
+ return responses
+
+ def get_config(self, flags=None):
+ """Retrieves the current config from the device or cache
+ """
+ flags = [] if flags is None else flags
+
+ cmd = "show running-config "
+ cmd += " ".join(flags)
+ cmd = cmd.strip()
+
+ try:
+ return self._device_configs[cmd]
+ except KeyError:
+ try:
+ out = self._connection.send_request(cmd)
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+
+ cfg = to_text(out).strip()
+ self._device_configs[cmd] = cfg
+ return cfg
+
+ def get_diff(
+ self,
+ candidate=None,
+ running=None,
+ diff_match="line",
+ diff_ignore_lines=None,
+ path=None,
+ diff_replace="line",
+ ):
+ diff = {}
+
+ # prepare candidate configuration
+ candidate_obj = NetworkConfig(indent=3)
+ candidate_obj.load(candidate)
+
+ if running and diff_match != "none" and diff_replace != "config":
+ # running configuration
+ running_obj = NetworkConfig(
+ indent=3, contents=running, ignore_lines=diff_ignore_lines
+ )
+ configdiffobjs = candidate_obj.difference(
+ running_obj, path=path, match=diff_match, replace=diff_replace
+ )
+
+ else:
+ configdiffobjs = candidate_obj.items
+
+ diff["config_diff"] = (
+ dumps(configdiffobjs, "commands") if configdiffobjs else {}
+ )
+ return diff
+
+ def load_config(self, config, commit=False, replace=False):
+ """Loads the configuration onto the remote devices
+
+ If the device doesn't support configuration sessions, this will
+ fallback to using configure() to load the commands. If that happens,
+ there will be no returned diff or session values
+ """
+ return self.edit_config(config, commit, replace)
+
+ def edit_config(self, config, commit=False, replace=False):
+ """Loads the configuration onto the remote devices
+
+ If the device doesn't support configuration sessions, this will
+ fallback to using configure() to load the commands. If that happens,
+ there will be no returned diff or session values
+ """
+ session = "ansible_%s" % int(time.time())
+ result = {"session": session}
+ banner_cmd = None
+ banner_input = []
+
+ commands = ["configure session %s" % session]
+ if replace:
+ commands.append("rollback clean-config")
+
+ for command in config:
+ if command.startswith("banner"):
+ banner_cmd = command
+ banner_input = []
+ elif banner_cmd:
+ if command == "EOF":
+ command = {
+ "cmd": banner_cmd,
+ "input": "\n".join(banner_input),
+ }
+ banner_cmd = None
+ commands.append(command)
+ else:
+ banner_input.append(command)
+ continue
+ else:
+ commands.append(command)
+
+ try:
+ response = self._connection.send_request(commands)
+ except Exception:
+ commands = ["configure session %s" % session, "abort"]
+ response = self._connection.send_request(commands, output="text")
+ raise
+
+ commands = [
+ "configure session %s" % session,
+ "show session-config diffs",
+ ]
+ if commit:
+ commands.append("commit")
+ else:
+ commands.append("abort")
+
+ response = self._connection.send_request(commands, output="text")
+ diff = response[1].strip()
+ if diff:
+ result["diff"] = diff
+
+ return result
+
+ def get_capabilities(self):
+ """Returns platform info of the remove device
+ """
+ try:
+ capabilities = self._connection.get_capabilities()
+ except ConnectionError as exc:
+ self._module.fail_json(
+ msg=to_text(exc, errors="surrogate_then_replace")
+ )
+
+ return json.loads(capabilities)
+
+
+def is_json(cmd):
+ return to_text(cmd, errors="surrogate_then_replace").endswith("| json")
+
+
+def is_local_eapi(module):
+ provider = module.params.get("provider")
+ if provider:
+ return provider.get("transport") == "eapi"
+ return False
+
+
+def to_command(module, commands):
+ if is_local_eapi(module):
+ default_output = "json"
+ else:
+ default_output = "text"
+
+ transform = ComplexList(
+ dict(
+ command=dict(key=True),
+ output=dict(default=default_output),
+ prompt=dict(type="list"),
+ answer=dict(type="list"),
+ newline=dict(type="bool", default=True),
+ sendonly=dict(type="bool", default=False),
+ check_all=dict(type="bool", default=False),
+ ),
+ module,
+ )
+
+ return transform(to_list(commands))
+
+
+def get_config(module, flags=None):
+ flags = None if flags is None else flags
+
+ conn = get_connection(module)
+ return conn.get_config(flags)
+
+
+def run_commands(module, commands, check_rc=True):
+ conn = get_connection(module)
+ return conn.run_commands(to_command(module, commands), check_rc=check_rc)
+
+
+def load_config(module, config, commit=False, replace=False):
+ conn = get_connection(module)
+ return conn.load_config(config, commit, replace)
+
+
+def get_diff(
+ self,
+ candidate=None,
+ running=None,
+ diff_match="line",
+ diff_ignore_lines=None,
+ path=None,
+ diff_replace="line",
+):
+ conn = self.get_connection()
+ return conn.get_diff(
+ candidate=candidate,
+ running=running,
+ diff_match=diff_match,
+ diff_ignore_lines=diff_ignore_lines,
+ path=path,
+ diff_replace=diff_replace,
+ )
+
+
+def get_capabilities(module):
+ conn = get_connection(module)
+ return conn.get_capabilities()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py
new file mode 100644
index 00000000..4157bf61
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py
@@ -0,0 +1,146 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos acl_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acl_interfaces.acl_interfaces import (
+ Acl_interfacesArgs,
+)
+
+
+class Acl_interfacesFacts(object):
+ """ The eos acl_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Acl_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get(
+ "show running-config | include interface | access-group | traffic-filter"
+ )
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for acl_interfaces
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+
+ if not data:
+ data = self.get_device_data(connection)
+ # split the config into instances of the resource
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+ objs = []
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+
+ ansible_facts["ansible_network_resources"].pop("acl_interfaces", None)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["acl_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ access_group_list = []
+ access_group_v6_list = []
+ acls_list = []
+ group_list = []
+ group_dict = {}
+ config["name"] = utils.parse_conf_arg(conf, "interface")
+ conf_lines = conf.split("\n")
+ for line in conf_lines:
+ if config["name"] in line:
+ continue
+ access_group = utils.parse_conf_arg(line, "ip access-group")
+ # This module was verified on an ios device since vEOS doesnot support
+ # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as
+ # traffic-filter and in eos it is access-group
+
+ # access_group_v6 = utils.parse_conf_arg(line, 'ipv6 traffic-filter')
+ access_group_v6 = utils.parse_conf_arg(line, "ipv6 access-group")
+ if access_group:
+ access_group_list.append(access_group)
+ if access_group_v6:
+ access_group_v6_list.append(access_group_v6)
+ if access_group_list:
+ for acl in access_group_list:
+ a_name = acl.split()[0]
+ a_dir = acl.split()[1]
+ acls_dict = {"name": a_name, "direction": a_dir}
+ acls_list.append(acls_dict)
+ group_dict = {"afi": "ipv4", "acls": acls_list}
+ group_list.append(group_dict)
+ acls_list = []
+ if group_list:
+ config["access_groups"] = group_list
+ if access_group_v6_list:
+ for acl in access_group_v6_list:
+ a_name = acl.split()[0]
+ a_dir = acl.split()[1]
+ acls_dict = {"name": a_name, "direction": a_dir}
+ acls_list.append(acls_dict)
+ group_dict = {"acls": acls_list, "afi": "ipv6"}
+ group_list.append(group_dict)
+ acls_list = []
+ if group_list:
+ config["access_groups"] = group_list
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py
new file mode 100644
index 00000000..e6282303
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py
@@ -0,0 +1,387 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos acls fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acls.acls import (
+ AclsArgs,
+)
+
+
+class AclsFacts(object):
+ """ The eos acls fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = AclsArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section access-list")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for acls
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ find_pattern = r"(?:^|\n)(?:ip|ipv6) access\-list.*?(?=(?:^|\n)(?:ip|ipv6) access\-list|$)"
+ resources = []
+ for p in re.findall(find_pattern, data, re.DOTALL):
+ resources.append(p)
+
+ objs = []
+ ipv4list = []
+ ipv6list = []
+ for resource in resources:
+ if "ipv6" in resource:
+ ipv6list.append(resource)
+ else:
+ ipv4list.append(resource)
+ ipv4list = ["\n".join(ipv4list)]
+ ipv6list = ["\n".join(ipv6list)]
+ for resource in ipv4list:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+ for resource in ipv6list:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+
+ ansible_facts["ansible_network_resources"].pop("acls", None)
+ facts = {}
+ if objs:
+ facts["acls"] = []
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ for cfg in params["config"]:
+ facts["acls"].append(utils.remove_empties(cfg))
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ afi_list = []
+ acls_list = []
+ name_dict = {}
+ standard = 0
+ operator = ["eq", "lt", "neq", "range", "gt"]
+ flags = ["ack", "established", "fin", "psh", "rst", "syn", "urg"]
+ others = ["hop_limit", "log", "ttl", "fragments", "tracked"]
+ for dev_config in conf.split("\n"):
+ ace_dict = {}
+ if not dev_config:
+ continue
+ if dev_config == "!":
+ continue
+ dev_config = dev_config.strip()
+ matches = re.findall(r"(ip.*?) access-list (.*)", dev_config)
+ if matches:
+ afi = "ipv4" if matches[0][0] == "ip" else "ipv6"
+ ace_list = []
+ if bool(name_dict):
+ acls_list.append(name_dict.copy())
+ name_dict = {}
+ if afi not in afi_list:
+ afi_list.append(afi)
+ config.update({"afi": afi})
+ if "standard" in matches[0][1]:
+ standard = 1
+ name = matches[0][1].split()
+ name_dict.update({"name": name[1]})
+ name_dict.update({"standard": True})
+ else:
+ name_dict.update({"name": matches[0][1]})
+ else:
+ source_dict = {}
+ dest_dict = {}
+ dev_config = re.sub("-", "_", dev_config)
+ dev_config_remainder = dev_config.split()
+ if "fragment_rules" in dev_config:
+ ace_dict.update({"sequence": dev_config_remainder.pop(0)})
+ ace_dict.update({"fragment_rules": True})
+ if "remark" in dev_config:
+ ace_dict.update({"sequence": dev_config_remainder.pop(0)})
+ ace_dict.update(
+ {"remark": " ".join(dev_config_remainder[1:])}
+ )
+ seq = re.search(r"\d+ (permit|deny) .*", dev_config)
+ if seq:
+ ace_dict.update({"sequence": dev_config_remainder.pop(0)})
+ ace_dict.update({"grant": dev_config_remainder.pop(0)})
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "vlan"
+ ):
+ vlan_str = ""
+ dev_config_remainder.pop(0)
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "inner"
+ ):
+ vlan_str = dev_config_remainder.pop(0) + " "
+ vlan_str = (
+ dev_config_remainder.pop(0)
+ + " "
+ + dev_config_remainder.pop(0)
+ )
+ ace_dict.update({"vlan": vlan_str})
+ if not standard:
+ protocol = dev_config_remainder[0]
+ ace_dict.update(
+ {"protocol": dev_config_remainder.pop(0)}
+ )
+ src_prefix = re.search(r"/", dev_config_remainder[0])
+ src_address = re.search(
+ r"[a-z\d:\.]+", dev_config_remainder[0]
+ )
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "host"
+ ):
+ source_dict.update(
+ {"host": dev_config_remainder.pop(1)}
+ )
+ dev_config_remainder.pop(0)
+ elif (
+ dev_config_remainder
+ and dev_config_remainder[0] == "any"
+ ):
+ source_dict.update({"any": True})
+ dev_config_remainder.pop(0)
+ elif src_prefix:
+ source_dict.update(
+ {"subnet_address": dev_config_remainder.pop(0)}
+ )
+ elif src_address:
+ source_dict.update(
+ {"address": dev_config_remainder.pop(0)}
+ )
+ source_dict.update(
+ {"wildcard_bits": dev_config_remainder.pop(0)}
+ )
+ if dev_config_remainder:
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] in operator
+ ):
+ port_dict = {}
+ src_port = ""
+ src_opr = dev_config_remainder.pop(0)
+ portlist = dev_config_remainder[:]
+ for config_remainder in portlist:
+ addr = re.search(r"[\.\:]", config_remainder)
+ if (
+ config_remainder == "any"
+ or config_remainder == "host"
+ or addr
+ ):
+ break
+ src_port = src_port + " " + config_remainder
+ dev_config_remainder.pop(0)
+ src_port = src_port.strip()
+ port_dict.update({src_opr: src_port})
+ source_dict.update({"port_protocol": port_dict})
+ ace_dict.update({"source": source_dict})
+ if not dev_config_remainder or standard:
+ if (
+ dev_config_remainder
+ and "log" in dev_config_remainder
+ ):
+ ace_dict.update({"log": True})
+ if bool(ace_dict):
+ ace_list.append(ace_dict.copy())
+ if len(ace_list):
+ name_dict = name_dict.copy()
+ name_dict.update({"aces": ace_list[:]})
+ # acls_list.append(name_dict)
+ continue
+ dest_prefix = re.search(r"/", dev_config_remainder[0])
+ dest_address = re.search(
+ r"[a-z\d:\.]+", dev_config_remainder[0]
+ )
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "host"
+ ):
+ dest_dict.update({"host": dev_config_remainder.pop(1)})
+ dev_config_remainder.pop(0)
+ elif (
+ dev_config_remainder
+ and dev_config_remainder[0] == "any"
+ ):
+ dest_dict.update({"any": True})
+ dev_config_remainder.pop(0)
+ elif dest_prefix:
+ dest_dict.update(
+ {"subnet_address": dev_config_remainder.pop(0)}
+ )
+ elif dest_address:
+ dest_dict.update(
+ {"address": dev_config_remainder.pop(0)}
+ )
+ dest_dict.update(
+ {"wildcard_bits": dev_config_remainder.pop(0)}
+ )
+ if dev_config_remainder:
+ if dev_config_remainder[0] in operator:
+ port_dict = {}
+ dest_port = ""
+ dest_opr = dev_config_remainder.pop(0)
+ portlist = dev_config_remainder[:]
+ for config_remainder in portlist:
+ if (
+ config_remainder in operator
+ or config_remainder in others
+ ):
+ break
+ dest_port = dest_port + " " + config_remainder
+ dev_config_remainder.pop(0)
+ dest_port = dest_port.strip()
+ port_dict.update({dest_opr: dest_port})
+ dest_dict.update({"port_protocol": port_dict})
+ ace_dict.update({"destination": dest_dict})
+ protocol_option_dict = {}
+ tcp_dict = {}
+ icmp_dict = {}
+ ip_dict = {}
+ if not dev_config_remainder:
+ if bool(ace_dict):
+ ace_list.append(ace_dict.copy())
+ if len(ace_list):
+ name_dict = name_dict.copy()
+ name_dict.update({"aces": ace_list[:]})
+ # acls_list.append(name_dict)
+ continue
+ if protocol == "tcp" or "6":
+ protocol = "tcp"
+ flags_dict = {}
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] in flags
+ ):
+ flaglist = dev_config_remainder[:]
+ for config_remainder in flaglist:
+ if config_remainder not in flags:
+ break
+ flags_dict.update({config_remainder: True})
+ dev_config_remainder.pop(0)
+ if bool(flags_dict):
+ tcp_dict.update({"flags": flags_dict})
+ if bool(tcp_dict):
+ protocol_option_dict.update({"tcp": tcp_dict})
+ if (
+ protocol == "icmp"
+ or protocol == "icmpv6"
+ or protocol == "1"
+ or protocol == "58"
+ ):
+ if protocol == "1":
+ protocol = "icmp"
+ elif protocol == "58":
+ protocol = "icmpv6"
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] not in others
+ ):
+ icmp_dict.update({dev_config_remainder[0]: True})
+ dev_config_remainder.pop(0)
+ if bool(icmp_dict):
+ protocol_option_dict.update({protocol: icmp_dict})
+ if protocol in ["ip", "ipv6"]:
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "nexthop_group"
+ ):
+ dev_config_remainder.pop(0)
+ ip_dict.update(
+ {"nexthop_group": dev_config_remainder.pop(0)}
+ )
+ if bool(ip_dict):
+ protocol_option_dict.update({protocol: ip_dict})
+ if bool(protocol_option_dict):
+ ace_dict.update(
+ {"protocol_options": protocol_option_dict}
+ )
+ if (
+ dev_config_remainder
+ and dev_config_remainder[0] == "ttl"
+ ):
+ dev_config_remainder.pop(0)
+ op = dev_config_remainder.pop(0)
+ ttl_dict = {op: dev_config_remainder.pop(0)}
+ ace_dict.update({"ttl": ttl_dict})
+ for config_remainder in dev_config_remainder:
+ if config_remainder in others:
+ if config_remainder == "hop_limit":
+ hop_index = dev_config_remainder.index(
+ config_remainder
+ )
+ hoplimit_dict = {
+ dev_config_remainder[
+ hop_index + 1
+ ]: dev_config_remainder[hop_index + 2]
+ }
+ ace_dict.update({"hop_limit": hoplimit_dict})
+ dev_config_remainder.pop(0)
+ continue
+ ace_dict.update({config_remainder: True})
+ dev_config_remainder.pop(0)
+ if dev_config_remainder:
+ config.update({"line": dev_config})
+ return utils.remove_empties(config)
+ if bool(ace_dict):
+ ace_list.append(ace_dict.copy())
+ if len(ace_list):
+ name_dict = name_dict.copy()
+ name_dict.update({"aces": ace_list[:]})
+ acls_list.append(name_dict.copy())
+ config.update({"acls": acls_list})
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py
new file mode 100644
index 00000000..497ccfd7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py
@@ -0,0 +1,116 @@
+# -*- coding: utf-8 -*-
+# Copyright 2021 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos bgp_address_family fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_address_family import (
+ Bgp_afTemplate,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_address_family.bgp_address_family import (
+ Bgp_afArgs,
+)
+import re
+
+
+class Bgp_afFacts(object):
+ """ The eos bgp_address_family facts class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Bgp_afArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_config(self, connection):
+ """Wrapper method for `connection.get()`
+ This method exists solely to allow the unit test framework to mock device connection calls.
+ """
+ return connection.get("show running-config | section bgp ")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for Bgp_af network resource
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+
+ :rtype: dictionary
+ :returns: facts
+ """
+ facts = {}
+ objs = []
+
+ if not data:
+ data = self.get_config(connection)
+
+ # remove global configs from bgp_address_family
+ bgp_af_config = []
+ vrf_set = ""
+ start = False
+ for bgp_line in data.splitlines():
+ if "router bgp" in bgp_line:
+ bgp_af_config.append(bgp_line)
+ vrf_present = re.search(r"vrf\s\S+", bgp_line)
+ if vrf_present:
+ vrf_set = vrf_present.group(0)
+ if start:
+ bgp_af_config.append(bgp_line)
+ if "address-family" in bgp_line:
+ af_line = vrf_set + bgp_line
+ bgp_af_config.append(af_line)
+ start = True
+ if start and "!" in bgp_line:
+ start = False
+
+ # parse native config using the Bgp_af template
+ bgp_af_parser = Bgp_afTemplate(lines=bgp_af_config)
+ objs = bgp_af_parser.parse()
+ if objs:
+ if "address_family" in objs:
+ objs["address_family"] = list(objs["address_family"].values())
+ for af in objs["address_family"]:
+ if "neighbor" in af:
+ af["neighbor"] = list(af["neighbor"].values())
+ if "network" in af:
+ af["network"] = list(af["network"].values())
+ af["network"] = sorted(
+ af["network"], key=lambda k: k["address"]
+ )
+
+ ansible_facts["ansible_network_resources"].pop(
+ "bgp_address_family", None
+ )
+
+ params = utils.remove_empties(
+ utils.validate_config(self.argument_spec, {"config": objs})
+ )
+
+ facts["bgp_address_family"] = params.get("config", [])
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py
new file mode 100644
index 00000000..3603ea6d
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py
@@ -0,0 +1,119 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos bgp_global fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_global import (
+ Bgp_globalTemplate,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_global.bgp_global import (
+ Bgp_globalArgs,
+)
+
+
+class Bgp_globalFacts(object):
+ """ The eos bgp_global facts class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Bgp_globalArgs.argument_spec
+
+ def get_config(self, connection):
+ """Wrapper method for `connection.get()`
+ This method exists solely to allow the unit test framework to mock device connection calls.
+ """
+ return connection.get("show running-config | section bgp ")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for Bgp_global network resource
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+
+ :rtype: dictionary
+ :returns: facts
+ """
+ facts = {}
+ objs = []
+
+ if not data:
+ data = self.get_config(connection)
+
+ # remove address_family configs from bgp_global
+ bgp_global_config = []
+ start = False
+ self._af = False
+ for bgp_line in data.splitlines():
+ if not start:
+ bgp_global_config.append(bgp_line)
+ if "address-family" in bgp_line:
+ start = True
+ self._af = True
+ if start and "!" in bgp_line:
+ start = False
+
+ # parse native config using the Bgp_global template
+ bgp_global_parser = Bgp_globalTemplate(lines=bgp_global_config)
+ objs = bgp_global_parser.parse()
+
+ if objs:
+ global_vals = objs.get("vrfs", {}).pop("vrf_", {})
+ for key, value in iteritems(global_vals):
+ objs[key] = value
+
+ if "vrfs" in objs:
+ objs["vrfs"] = list(objs["vrfs"].values())
+ for vrf in objs["vrfs"]:
+ if "neighbor" in vrf:
+ vrf["neighbor"] = list(vrf["neighbor"].values())
+ if "network" in vrf:
+ vrf["network"] = list(vrf["network"].values())
+ vrf["network"] = sorted(
+ vrf["network"], key=lambda k: k["address"]
+ )
+ if "aggregate_address" in vrf:
+ vrf["aggregate_address"] = sorted(
+ vrf["aggregate_address"],
+ key=lambda k: k["address"],
+ )
+
+ if "neighbor" in objs:
+ objs["neighbor"] = list(objs["neighbor"].values())
+
+ if "network" in objs:
+ objs["network"] = list(objs["network"].values())
+ objs["network"] = sorted(
+ objs["network"], key=lambda k: k["address"]
+ )
+ if "aggregate_address" in objs:
+ objs["aggregate_address"] = sorted(
+ objs["aggregate_address"], key=lambda k: k["address"]
+ )
+
+ ansible_facts["ansible_network_resources"].pop("bgp_global", None)
+
+ params = utils.remove_empties(
+ utils.validate_config(self.argument_spec, {"config": objs})
+ )
+
+ facts["bgp_global"] = params.get("config", [])
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py
new file mode 100644
index 00000000..c1f4fbc0
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py
@@ -0,0 +1,127 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The facts class for eos
+this file validates each subset of facts and selectively
+calls the appropriate facts gathering function
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.facts.facts import (
+ FactsBase,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.interfaces.interfaces import (
+ InterfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.l2_interfaces.l2_interfaces import (
+ L2_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.l3_interfaces.l3_interfaces import (
+ L3_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lacp.lacp import (
+ LacpFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lacp_interfaces.lacp_interfaces import (
+ Lacp_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lag_interfaces.lag_interfaces import (
+ Lag_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lldp_global.lldp_global import (
+ Lldp_globalFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lldp_interfaces.lldp_interfaces import (
+ Lldp_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.vlans.vlans import (
+ VlansFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.legacy.base import (
+ Default,
+ Hardware,
+ Config,
+ Interfaces,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.acl_interfaces.acl_interfaces import (
+ Acl_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.acls.acls import (
+ AclsFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.static_routes.static_routes import (
+ Static_routesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospfv2.ospfv2 import (
+ Ospfv2Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospfv3.ospfv3 import (
+ Ospfv3Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospf_interfaces.ospf_interfaces import (
+ Ospf_interfacesFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.bgp_address_family.bgp_address_family import (
+ Bgp_afFacts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.bgp_global.bgp_global import (
+ Bgp_globalFacts,
+)
+
+
+FACT_LEGACY_SUBSETS = dict(
+ default=Default, hardware=Hardware, interfaces=Interfaces, config=Config
+)
+FACT_RESOURCE_SUBSETS = dict(
+ interfaces=InterfacesFacts,
+ l2_interfaces=L2_interfacesFacts,
+ l3_interfaces=L3_interfacesFacts,
+ lacp=LacpFacts,
+ lacp_interfaces=Lacp_interfacesFacts,
+ lag_interfaces=Lag_interfacesFacts,
+ lldp_global=Lldp_globalFacts,
+ lldp_interfaces=Lldp_interfacesFacts,
+ vlans=VlansFacts,
+ acl_interfaces=Acl_interfacesFacts,
+ acls=AclsFacts,
+ static_routes=Static_routesFacts,
+ ospfv2=Ospfv2Facts,
+ ospfv3=Ospfv3Facts,
+ ospf_interfaces=Ospf_interfacesFacts,
+ bgp_address_family=Bgp_afFacts,
+ bgp_global=Bgp_globalFacts,
+)
+
+
+class Facts(FactsBase):
+ """ The fact class for eos
+ """
+
+ VALID_LEGACY_GATHER_SUBSETS = frozenset(FACT_LEGACY_SUBSETS.keys())
+ VALID_RESOURCE_SUBSETS = frozenset(FACT_RESOURCE_SUBSETS.keys())
+
+ def get_facts(
+ self, legacy_facts_type=None, resource_facts_type=None, data=None
+ ):
+ """ Collect the facts for eos
+ :param legacy_facts_type: List of legacy facts types
+ :param resource_facts_type: List of resource fact types
+ :param data: previously collected conf
+ :rtype: dict
+ :return: the facts gathered
+ """
+ if self.VALID_RESOURCE_SUBSETS:
+ self.get_network_resources_facts(
+ FACT_RESOURCE_SUBSETS, resource_facts_type, data
+ )
+
+ if self.VALID_LEGACY_GATHER_SUBSETS:
+ self.get_network_legacy_facts(
+ FACT_LEGACY_SUBSETS, legacy_facts_type
+ )
+
+ return self.ansible_facts, self._warnings
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py
new file mode 100644
index 00000000..c52535a7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py
@@ -0,0 +1,110 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from copy import deepcopy
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.interfaces.interfaces import (
+ InterfacesArgs,
+)
+
+
+class InterfacesFacts(object):
+ """ The eos interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = InterfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ^interface")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for interfaces
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # operate on a collection of resource x
+ config = ("\n" + data).split("\ninterface ")
+ objs = []
+ for conf in config:
+ if conf:
+ obj = self.render_config(self.generated_spec, conf)
+ if obj:
+ objs.append(obj)
+ facts = {"interfaces": []}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ for cfg in params["config"]:
+ facts["interfaces"].append(utils.remove_empties(cfg))
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+
+ # populate the facts from the configuration
+ config["name"] = re.match(r"(\S+)", conf).group(1)
+ description = utils.parse_conf_arg(conf, "description")
+ if description is not None:
+ config["description"] = description.replace('"', "")
+ shutdown = utils.parse_conf_cmd_arg(conf, "shutdown", False)
+ config["enabled"] = shutdown if shutdown is False else True
+ config["mtu"] = utils.parse_conf_arg(conf, "mtu")
+ config["mode"] = utils.parse_conf_cmd_arg(
+ conf, "switchport", "layer2", "layer3"
+ )
+
+ state = utils.parse_conf_arg(conf, "speed")
+ if state:
+ if state == "auto":
+ config["duplex"] = state
+ else:
+ # remaining options are all e.g., 10half or 40gfull
+ config["speed"] = state[:-4]
+ config["duplex"] = state[-4:]
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py
new file mode 100644
index 00000000..4ef5f445
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py
@@ -0,0 +1,117 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos l2_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from copy import deepcopy
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l2_interfaces.l2_interfaces import (
+ L2_interfacesArgs,
+)
+
+
+class L2_interfacesFacts(object):
+ """ The eos l2_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = L2_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ^interface")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for l2_interfaces
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # operate on a collection of resource x
+ config = ("\n" + data).split("\ninterface ")
+ objs = []
+ for conf in config:
+ if conf:
+ obj = self.render_config(self.generated_spec, conf)
+ if obj:
+ objs.append(obj)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["l2_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+
+ # populate the facts from the configuration
+ config["name"] = re.match(r"(\S+)", conf).group(1).replace('"', "")
+ has_mode = re.search(r"switchport mode (\S+)", conf)
+ if has_mode:
+ config["mode"] = has_mode.group(1)
+
+ has_access = re.search(r"switchport access vlan (\d+)", conf)
+ if has_access:
+ config["access"] = {"vlan": int(has_access.group(1))}
+
+ has_trunk = re.findall(r"switchport trunk (.+)", conf)
+ if has_trunk:
+ trunk = {}
+ for match in has_trunk:
+ has_native = re.match(r"native vlan (\d+)", match)
+ if has_native:
+ trunk["native_vlan"] = int(has_native.group(1))
+ continue
+
+ has_allowed = re.match(r"allowed vlan (\S+)", match)
+ if has_allowed:
+ # TODO: listify?
+ trunk["trunk_allowed_vlans"] = has_allowed.group(1)
+ continue
+ config["trunk"] = trunk
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py
new file mode 100644
index 00000000..afce1445
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py
@@ -0,0 +1,118 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos l3_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from copy import deepcopy
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l3_interfaces.l3_interfaces import (
+ L3_interfacesArgs,
+)
+
+
+class L3_interfacesFacts(object):
+ """ The eos l3_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = L3_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ^interface")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for l3_interfaces
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = []
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["l3_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+
+ config["name"] = utils.parse_conf_arg(conf, "interface")
+
+ matches = re.findall(r".*ip address (.+)$", conf, re.MULTILINE)
+ if matches:
+ config["ipv4"] = []
+ for match in matches:
+ address, dummy, remainder = match.partition(" ")
+ ipv4 = {"address": address}
+ if remainder == "secondary":
+ ipv4["secondary"] = True
+ config["ipv4"].append(ipv4)
+
+ matches = re.findall(r".*ipv6 address (.+)$", conf, re.MULTILINE)
+ if matches:
+ config["ipv6"] = []
+ for match in matches:
+ address, dummy, remainder = match.partition(" ")
+ ipv6 = {"address": address}
+ config["ipv6"].append(ipv6)
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py
new file mode 100644
index 00000000..2c70188e
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py
@@ -0,0 +1,102 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos lacp fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp.lacp import (
+ LacpArgs,
+)
+
+
+class LacpFacts(object):
+ """ The eos lacp fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = LacpArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ^lacp")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for lacp
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "lacp"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = {}
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.update(obj)
+
+ ansible_facts["ansible_network_resources"].pop("lacp", None)
+ facts = {"lacp": {}}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["lacp"] = utils.remove_empties(params["config"])
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ config["system"]["priority"] = utils.parse_conf_arg(
+ conf, "system-priority"
+ )
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py
new file mode 100644
index 00000000..82c42114
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py
@@ -0,0 +1,104 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos lacp_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp_interfaces.lacp_interfaces import (
+ Lacp_interfacesArgs,
+)
+
+
+class Lacp_interfacesFacts(object):
+ """ The eos lacp_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Lacp_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section lacp")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for lacp_interfaces
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = []
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+
+ ansible_facts["ansible_network_resources"].pop("lacp_interfaces", None)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["lacp_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ config["name"] = utils.parse_conf_arg(conf, "interface")
+ config["port_priority"] = utils.parse_conf_arg(conf, "port-priority")
+ config["rate"] = utils.parse_conf_arg(conf, "rate")
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py
new file mode 100644
index 00000000..fe96b65d
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py
@@ -0,0 +1,119 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos lag_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from copy import deepcopy
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lag_interfaces.lag_interfaces import (
+ Lag_interfacesArgs,
+)
+
+
+class Lag_interfacesFacts(object):
+ """ The eos lag_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Lag_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ^interface")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for lag_interfaces
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = {}
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ group_name = obj["name"]
+ if group_name in objs and "members" in obj:
+ config = objs[group_name]
+ if "members" not in config:
+ config["members"] = []
+ objs[group_name]["members"].extend(obj["members"])
+ else:
+ objs[group_name] = obj
+ objs = list(objs.values())
+ facts = {"lag_interfaces": []}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["lag_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ interface_name = utils.parse_conf_arg(conf, "interface")
+ if interface_name.startswith("Port-Channel"):
+ config["name"] = interface_name
+ return utils.remove_empties(config)
+
+ interface = {"member": interface_name}
+ match = re.match(
+ r".*channel-group (\d+) mode (\S+)", conf, re.MULTILINE | re.DOTALL
+ )
+ if match:
+ config["name"], interface["mode"] = match.groups()
+ config["name"] = "Port-Channel" + config["name"]
+ config["members"] = [interface]
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py
new file mode 100644
index 00000000..8dc886f3
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py
@@ -0,0 +1,181 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import platform
+import re
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ run_commands,
+ get_capabilities,
+)
+
+
+class FactsBase(object):
+
+ COMMANDS = frozenset()
+
+ def __init__(self, module):
+ self.module = module
+ self.warnings = list()
+ self.facts = dict()
+ self.responses = None
+
+ def populate(self):
+ self.responses = run_commands(
+ self.module, list(self.COMMANDS), check_rc=False
+ )
+
+
+class Default(FactsBase):
+
+ SYSTEM_MAP = {"serialNumber": "serialnum"}
+
+ COMMANDS = ["show version | json", "show hostname | json"]
+
+ def populate(self):
+ super(Default, self).populate()
+ data = self.responses[0]
+ for key, value in iteritems(self.SYSTEM_MAP):
+ if key in data:
+ self.facts[value] = data[key]
+
+ self.facts.update(self.responses[1])
+ self.facts.update(self.platform_facts())
+
+ def platform_facts(self):
+ platform_facts = {}
+
+ resp = get_capabilities(self.module)
+ device_info = resp["device_info"]
+
+ platform_facts["system"] = device_info["network_os"]
+
+ for item in ("model", "image", "version", "platform", "hostname"):
+ val = device_info.get("network_os_%s" % item)
+ if val:
+ platform_facts[item] = val
+
+ platform_facts["api"] = resp["network_api"]
+ platform_facts["python_version"] = platform.python_version()
+
+ return platform_facts
+
+
+class Hardware(FactsBase):
+
+ COMMANDS = ["dir all-filesystems", "show version | json"]
+
+ def populate(self):
+ super(Hardware, self).populate()
+ self.facts.update(self.populate_filesystems())
+ self.facts.update(self.populate_memory())
+
+ def populate_filesystems(self):
+ data = self.responses[0]
+
+ if isinstance(data, dict):
+ data = data["messages"][0]
+
+ fs = re.findall(r"^Directory of (.+)/", data, re.M)
+ return dict(filesystems=fs)
+
+ def populate_memory(self):
+ values = self.responses[1]
+ return dict(
+ memfree_mb=int(values["memFree"]) / 1024,
+ memtotal_mb=int(values["memTotal"]) / 1024,
+ )
+
+
+class Config(FactsBase):
+
+ COMMANDS = ["show running-config"]
+
+ def populate(self):
+ super(Config, self).populate()
+ self.facts["config"] = self.responses[0]
+
+
+class Interfaces(FactsBase):
+
+ INTERFACE_MAP = {
+ "description": "description",
+ "physicalAddress": "macaddress",
+ "mtu": "mtu",
+ "bandwidth": "bandwidth",
+ "duplex": "duplex",
+ "lineProtocolStatus": "lineprotocol",
+ "interfaceStatus": "operstatus",
+ "forwardingModel": "type",
+ }
+
+ COMMANDS = ["show interfaces | json", "show lldp neighbors | json"]
+
+ def populate(self):
+ super(Interfaces, self).populate()
+
+ self.facts["all_ipv4_addresses"] = list()
+ self.facts["all_ipv6_addresses"] = list()
+
+ data = self.responses[0]
+ if data and "LLDP is not enabled" not in data:
+ self.facts["interfaces"] = self.populate_interfaces(data)
+
+ if len(self.responses) > 1:
+ data = self.responses[1]
+ if data:
+ self.facts["neighbors"] = self.populate_neighbors(
+ data["lldpNeighbors"]
+ )
+
+ def populate_interfaces(self, data):
+ facts = dict()
+ for key, value in iteritems(data["interfaces"]):
+ intf = dict()
+
+ for remote, local in iteritems(self.INTERFACE_MAP):
+ if remote in value:
+ intf[local] = value[remote]
+
+ if "interfaceAddress" in value:
+ intf["ipv4"] = dict()
+ for entry in value["interfaceAddress"]:
+ intf["ipv4"]["address"] = entry["primaryIp"]["address"]
+ intf["ipv4"]["masklen"] = entry["primaryIp"]["maskLen"]
+ self.add_ip_address(entry["primaryIp"]["address"], "ipv4")
+
+ if "interfaceAddressIp6" in value:
+ intf["ipv6"] = dict()
+ for entry in value["interfaceAddressIp6"]["globalUnicastIp6s"]:
+ intf["ipv6"]["address"] = entry["address"]
+ intf["ipv6"]["subnet"] = entry["subnet"]
+ self.add_ip_address(entry["address"], "ipv6")
+
+ facts[key] = intf
+
+ return facts
+
+ def add_ip_address(self, address, family):
+ if family == "ipv4":
+ self.facts["all_ipv4_addresses"].append(address)
+ else:
+ self.facts["all_ipv6_addresses"].append(address)
+
+ def populate_neighbors(self, neighbors):
+ facts = dict()
+ for value in neighbors:
+ port = value["port"]
+ if port not in facts:
+ facts[port] = list()
+ lldp = dict()
+ lldp["host"] = value["neighborDevice"]
+ lldp["port"] = value["neighborPort"]
+ facts[port].append(lldp)
+ return facts
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py
new file mode 100644
index 00000000..0a9774f5
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py
@@ -0,0 +1,96 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos lldp_global fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_global.lldp_global import (
+ Lldp_globalArgs,
+)
+
+
+class Lldp_globalFacts(object):
+ """ The eos lldp_global fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Lldp_globalArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section lldp")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for lldp_global
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ obj = {}
+ if data:
+ obj.update(self.render_config(self.generated_spec, data))
+
+ ansible_facts["ansible_network_resources"].pop("lldp_global", None)
+ facts = {}
+ if obj:
+ params = utils.validate_config(self.argument_spec, {"config": obj})
+ facts["lldp_global"] = utils.remove_empties(params["config"])
+ else:
+ facts["lldp_global"] = {}
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ config["holdtime"] = utils.parse_conf_arg(conf, "holdtime")
+ config["reinit"] = utils.parse_conf_arg(conf, "reinit")
+ config["timer"] = utils.parse_conf_arg(conf, "timer")
+
+ for match in re.findall(
+ r"^(no)? lldp tlv-select (\S+)", conf, re.MULTILINE
+ ):
+ tlv_option = match[1].replace("-", "_")
+ config["tlv_select"][tlv_option] = bool(match[0] != "no")
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py
new file mode 100644
index 00000000..fb1b938e
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py
@@ -0,0 +1,103 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos lldp_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_interfaces.lldp_interfaces import (
+ Lldp_interfacesArgs,
+)
+
+
+class Lldp_interfacesFacts(object):
+ """ The eos lldp_interfaces fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Lldp_interfacesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for lldp_interfaces
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected configuration
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = connection.get("show running-config | section lldp")
+
+ # split the config into instances of the resource
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = []
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+
+ ansible_facts["ansible_network_resources"].pop("lldp_interfaces", None)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["lldp_interfaces"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ config["name"] = utils.parse_conf_arg(conf, "interface")
+
+ matches = re.findall(r"(no )?lldp (\S+)", conf)
+ for match in matches:
+ config[match[1]] = not bool(match[0])
+
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py
new file mode 100644
index 00000000..16d99fd0
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py
@@ -0,0 +1,103 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos ospf_interfaces fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospf_interfaces import (
+ Ospf_interfacesTemplate,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospf_interfaces.ospf_interfaces import (
+ Ospf_interfacesArgs,
+)
+
+
+class Ospf_interfacesFacts(object):
+ """ The eos ospf_interfaces facts class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Ospf_interfacesArgs.argument_spec
+
+ def get_config(self, connection):
+ """Wrapper method for `connection.get()`
+ This method exists solely to allow the unit test framework to mock device connection calls.
+ """
+ return connection.get("show running-config | section interface ")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for Ospf_interfaces network resource
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+
+ :rtype: dictionary
+ :returns: facts
+ """
+ facts = {}
+
+ if not data:
+ data = self.get_config(connection)
+
+ resource_delim = "interface"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+ # parse native config using the Ospf_interfaces template
+ ospf_interfaces_facts = []
+ for resource in resources:
+ ospf_interfaces_parser = Ospf_interfacesTemplate(
+ lines=resource.splitlines()
+ )
+ entry = ospf_interfaces_parser.parse()
+ if entry:
+ if "address_family" in entry and entry["address_family"]:
+ entry["address_family"] = sorted(
+ list(entry["address_family"].values()),
+ key=lambda k, sk="afi": k[sk],
+ )
+ if entry:
+ if entry.get("address_family"):
+ for addr in entry["address_family"]:
+ if "ip_params" in addr:
+ addr["ip_params"] = sorted(
+ list(addr["ip_params"].values()),
+ key=lambda k, sk="afi": k[sk],
+ )
+ ospf_interfaces_facts.append(entry)
+
+ ansible_facts["ansible_network_resources"].pop("ospf_interfaces", None)
+ facts = {"ospf_interfaces": []}
+ params = utils.remove_empties(
+ utils.validate_config(
+ self.argument_spec, {"config": ospf_interfaces_facts}
+ )
+ )
+
+ if params.get("config"):
+ for cfg in params["config"]:
+ facts["ospf_interfaces"].append(utils.remove_empties(cfg))
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py
new file mode 100644
index 00000000..a7419eaf
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py
@@ -0,0 +1,496 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos ospfv2 fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv2.ospfv2 import (
+ Ospfv2Args,
+)
+
+
+class Ospfv2Facts(object):
+ """ The eos ospfv2 fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Ospfv2Args.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | section ospf")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for ospfv2
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "router ospf"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+ objs_list = []
+ objs = {}
+ for resource in resources:
+ if resource and "router ospfv3" not in resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs_list.append(obj)
+ objs = {"processes": objs_list}
+ ansible_facts["ansible_network_resources"].pop("ospfv2", None)
+
+ facts = {}
+ if objs:
+ facts["ospfv2"] = {}
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["ospfv2"].update(utils.remove_empties(params["config"]))
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ instance_list = []
+ ospf_params_dict = {}
+ areas_list = []
+ distance_dict = {}
+ network_list = []
+ redistribute_list = []
+ timers_list = []
+ areas_list = []
+ for dev_config in conf.split("\n"):
+ if not dev_config:
+ continue
+ network_dict = {}
+ redistribute_dict = {}
+ dev_config = dev_config.strip()
+ dev_config = re.sub(r"-", "_", dev_config).strip()
+ matches = re.findall(r"router (ospf) (.*)", dev_config)
+ if matches:
+ if ospf_params_dict:
+ instance_list.append(ospf_params_dict)
+ ospf_params_dict = {}
+ instance = matches[0][1].split()
+ ospf_params_dict.update({"process_id": str(instance[0])})
+ if "vrf" in dev_config:
+ vrf_name = instance[-1]
+ else:
+ vrf_name = None
+ ospf_params_dict.update({"vrf": vrf_name})
+ if "traffic_engineering" in dev_config:
+ ospf_params_dict.update({"traffic_engineering": True})
+ config_params = dev_config.split()
+ if config_params[0] == "adjacency":
+ threshold = config_params[-1]
+ adjacency_dict = {"exchange_start": {"threshold": threshold}}
+ ospf_params_dict.update({"adjacency": adjacency_dict})
+ elif "auto_cost" in dev_config:
+ bw = config_params[-1]
+ ospf_params_dict.update(
+ {"auto_cost": {"reference_bandwidth": bw}}
+ )
+ elif "bfd" in dev_config:
+ ospf_params_dict.update({"bfd": {"all_interfaces": True}})
+ elif config_params[0] == "default_information":
+ def_dict = {"originate": True}
+ for i, val in enumerate(config_params[2::]):
+ if val == "always":
+ def_dict.update({"always": True})
+ elif val in ["route_map", "metric", "metric_type"]:
+ def_dict.update({val: config_params[i + 3]})
+ ospf_params_dict.update({"default_information": def_dict})
+ elif "default_metric" in dev_config:
+ ospf_params_dict.update({"default_metric": config_params[-1]})
+ elif "distance" in dev_config:
+ distance_dict.update({config_params[-2]: config_params[-1]})
+ ospf_params_dict.update({"distance": distance_dict})
+ elif "distribute_list" in dev_config:
+ ospf_params_dict.update(
+ {"distribute_list": {config_params[1]: config_params[2]}}
+ )
+ elif "dn_bit_ignore" in dev_config:
+ ospf_params_dict.update({"dn_bit_ignore": True})
+ elif "fips_restrictions" in dev_config:
+ ospf_params_dict.update({"fips_restrictions": True})
+ elif "graceful_restart" in dev_config:
+ if "grace_period" in dev_config:
+ ospf_params_dict.update(
+ {
+ "graceful_restart": {
+ "grace_period": config_params[-1]
+ }
+ }
+ )
+ else:
+ ospf_params_dict.update(
+ {"graceful_restart": {"set": True}}
+ )
+ elif "graceful_restart_helper" in dev_config:
+ ospf_params_dict.update({"graceful_restart_helper": True})
+ elif "log_adjacency_changes" in dev_config:
+ detail = True if "detail" in dev_config else False
+ ospf_params_dict.update(
+ {"log_adjacency_changes": {"detail": detail}}
+ )
+ elif "max_lsa" in dev_config:
+ max_lsa_dict = {}
+ config_params.pop(0)
+ max_lsa_dict.update({"count": config_params.pop(0)})
+ if config_params:
+ if config_params[0].isdigit():
+ max_lsa_dict.update(
+ {"threshold": config_params.pop(0)}
+ )
+ for i, el in enumerate(config_params):
+ if el == "warning_only":
+ max_lsa_dict.update({"warning": True})
+ if el in ["ignore_count", "ignore_time", "reset_time"]:
+ max_lsa_dict.update({el: config_params[i + 1]})
+ ospf_params_dict.update({"max_lsa": max_lsa_dict})
+ elif "maximum_paths" in dev_config:
+ ospf_params_dict.update({"maximum_paths": config_params[1]})
+ elif "mpls ldp sync default" in dev_config:
+ ospf_params_dict.update({"mpls_ldp": True})
+ elif config_params[0] == "network":
+ config_params.pop(0)
+ prefix = re.search(r"\/", config_params[0])
+ if prefix:
+ network_dict.update({"prefix": config_params.pop(0)})
+ else:
+ network_dict.update(
+ {"network_address": config_params.pop(0)}
+ )
+ network_dict.update({"mask": config_params.pop(0)})
+ network_dict.update({"area": config_params[-1]})
+ network_list.append(network_dict)
+ ospf_params_dict.update({"networks": network_list})
+ elif "passive_interface" in dev_config:
+ if config_params[1] == "default":
+ ospf_params_dict.update(
+ {"passive_interface": {"default": True}}
+ )
+ else:
+ ospf_params_dict.update(
+ {
+ "passive_interface": {
+ "interface_list": config_params[1]
+ }
+ }
+ )
+ elif "point_to_point" in dev_config:
+ ospf_params_dict.update({"point_to_point": True})
+ elif "redistribute" in dev_config:
+ redistribute_dict.update({"routes": config_params[1]})
+ if config_params[1] == "isis":
+ if "level" in config_params[2]:
+ k = re.sub(r"_", "-", config_params[2])
+ redistribute_dict.update({"isis_level": k})
+ if "route_map" in dev_config:
+ redistribute_dict.update({"route_map": config_params[-1]})
+ redistribute_list.append(redistribute_dict)
+ ospf_params_dict.update({"redistribute": redistribute_list})
+ elif "router_id" in dev_config:
+ ospf_params_dict.update({"router_id": config_params[-1]})
+ elif "retransmission_threshold" in dev_config:
+ ospf_params_dict.update(
+ {"retransmission_threshold": config_params[-1]}
+ )
+ elif config_params[0] == "compatible":
+ ospf_params_dict.update({"rfc1583compatibility": True})
+ elif "shutdown" in dev_config:
+ ospf_params_dict.update({"shutdown": True})
+ elif "summary_address" in dev_config:
+ summary_address_dict = {}
+ config_params.pop(0)
+ prefix = re.search(r"\/", config_params[0])
+ if prefix:
+ summary_address_dict.update(
+ {"prefix": config_params.pop(0)}
+ )
+ else:
+ summary_address_dict.update(
+ {"address": config_params.pop(0)}
+ )
+ summary_address_dict.update({"mask": config_params.pop(0)})
+ if "not_advertise" in dev_config:
+ summary_address_dict.update({"not_advertise": True})
+ config_params.pop(0)
+ else:
+ if config_params:
+ summary_address_dict.update(
+ {config_params[0]: config_params[1]}
+ )
+ ospf_params_dict.update(
+ {"summary_address": summary_address_dict}
+ )
+ elif "timers" in dev_config:
+ timers_dict = {}
+ if config_params[1] == "lsa":
+ if config_params[2] == "rx":
+ timers_dict.update(
+ {
+ "lsa": {
+ "rx": {"min_interval": config_params[-1]}
+ }
+ }
+ )
+ else:
+ timers_dict.update(
+ {
+ "lsa": {
+ "tx": {
+ "delay": {
+ "initial": config_params[-3],
+ "min": config_params[-2],
+ "max": config_params[-1],
+ }
+ }
+ }
+ }
+ )
+ elif config_params[1] == "out_delay":
+ timers_dict.update({"out_delay": config_params[-1]})
+ elif config_params[1] == "pacing":
+ timers_dict.update({"pacing": config_params[-1]})
+ elif config_params[1] == "spf":
+ if config_params[2] == "delay":
+ timers_dict.update(
+ {
+ "spf": {
+ "tx": {
+ "delay": {
+ "initial": config_params[-3],
+ "min": config_params[-2],
+ "max": config_params[-1],
+ }
+ }
+ }
+ }
+ )
+ else:
+ timers_dict.update(
+ {"spf": {"seconds": config_params[-1]}}
+ )
+ timers_list.append(timers_dict)
+ ospf_params_dict.update({"timers": timers_list})
+ elif config_params[0] == "area":
+ areas_dict = {}
+ areas_dict.update({"area_id": config_params[1]})
+ if config_params[2] == "default_cost":
+ areas_dict.update({"default_cost": config_params[-1]})
+ elif config_params[2] == "filter":
+ prefix = re.search(r"\/", config_params[3])
+ if prefix:
+ areas_dict.update(
+ {"filter": {"address": config_params[3]}}
+ )
+ elif config_params[3] == "prefix_list":
+ areas_dict.update(
+ {"filter": {"prefix_list": config_params[-1]}}
+ )
+ else:
+ areas_dict.update(
+ {"filter": {"subnet_address": config_params[3]}}
+ )
+ areas_dict.update(
+ {"filter": {"subnet_mask": config_params[4]}}
+ )
+ elif config_params[2] == "not_so_stubby":
+ if len(config_params) == 3:
+ areas_dict.update({"not_so_stubby": {"set": True}})
+ continue
+ if config_params[3] == "lsa":
+ areas_dict.update({"not_so_stubby": {"lsa": True}})
+ elif config_params[3] == "default_information_originate":
+ default_dict = {}
+ for i, val in enumerate(config_params):
+ if val == "nssa_only":
+ default_dict.update({"nssa_only": True})
+ if val == "metric_type":
+ default_dict.update(
+ {"metric_type": config_params[i + 1]}
+ )
+ if val == "metric":
+ default_dict.update(
+ {"metric": config_params[i + 1]}
+ )
+ areas_dict.update(
+ {
+ "not_so_stubby": {
+ "default_information_originate": default_dict
+ }
+ }
+ )
+ elif config_params[3] == "no_summary":
+ areas_dict.update(
+ {"not_so_stubby": {"no_summary": True}}
+ )
+ elif config_params[3] == "nssa_only":
+ areas_dict.update(
+ {"not_so_stubby": {"nssa_only": True}}
+ )
+ elif config_params[2] == "nssa":
+ if len(config_params) == 3:
+ areas_dict.update({"nssa": {"set": True}})
+ continue
+ if config_params[3] == "default_information_originate":
+ default_dict = {}
+ for i, val in enumerate(config_params):
+ if val == "nssa_only":
+ default_dict.update({"nssa_only": True})
+ if val == "metric_type":
+ default_dict.update(
+ {"metric_type": config_params[i + 1]}
+ )
+ if val == "metric":
+ default_dict.update(
+ {"metric": config_params[i + 1]}
+ )
+ areas_dict.update(
+ {
+ "nssa": {
+ "default_information_originate": default_dict
+ }
+ }
+ )
+ elif config_params[3] == "no_summary":
+ areas_dict.update({"nssa": {"no_summary": True}})
+ elif config_params[3] == "nssa_only":
+ areas_dict.update({"nssa": {"nssa_only": True}})
+ elif config_params[2] == "range":
+ prefix = re.search(r"\/", config_params[3])
+ range_dict = {}
+ if prefix:
+ range_dict.update({"address": config_params[3]})
+ else:
+ range_dict.update({"subnet_address": config_params[3]})
+ range_dict.update({"subnet_mask": config_params[4]})
+ if "advertise" in dev_config:
+ range_dict.update({"advertise": True})
+ if "not_advertise" in dev_config:
+ range_dict.update({"advertise": False})
+ if "cost" in dev_config:
+ range_dict.update({"cost": config_params[-1]})
+ areas_dict.update({"range": range_dict})
+ elif config_params[2] == "stub":
+ if "no_summary" in dev_config:
+ areas_dict.update({"stub": {"no_summary": True}})
+ else:
+ areas_dict.update({"stub": {"set": True}})
+ areas_list.append(areas_dict)
+ ospf_params_dict.update({"areas": areas_list})
+ elif config_params[0] == "max_metric":
+ config_params.pop(0)
+ router_lsa_dict = {}
+ config_params.pop(0)
+ if not config_params:
+ ospf_params_dict.update(
+ {"max_metric": {"router_lsa": {"set": True}}}
+ )
+ else:
+ for i, val in enumerate(config_params):
+ if val == "include_stub":
+ router_lsa_dict.update({"include_stub": True})
+ elif val == "on_startup":
+ if config_params[i + 1] == "wait_for_bgp":
+ router_lsa_dict.update(
+ {"on_startup": {"wait_for_bgp": True}}
+ )
+ else:
+ router_lsa_dict.update(
+ {
+ "on_startup": {
+ "time": config_params[i + 1]
+ }
+ }
+ )
+ elif val == "external_lsa":
+ if (
+ i < len(config_params)
+ and config_params[i + 1].isdigit()
+ ):
+ router_lsa_dict.update(
+ {
+ "external_lsa": {
+ "max_metric_value": config_params[
+ i + 1
+ ]
+ }
+ }
+ )
+ else:
+ router_lsa_dict.update(
+ {"external_lsa": {"set": True}}
+ )
+ elif val == "summary_lsa":
+ if (
+ i < len(config_params) - 1
+ and config_params[i + 1].isdigit()
+ ):
+ router_lsa_dict.update(
+ {
+ "summary_lsa": {
+ "max_metric_value": config_params[
+ i + 1
+ ]
+ }
+ }
+ )
+ else:
+ router_lsa_dict.update(
+ {"summary_lsa": {"set": True}}
+ )
+ ospf_params_dict.update(
+ {"max_metric": {"router_lsa": router_lsa_dict}}
+ )
+ # instance_list.append(ospf_params_dict)
+ # config.update({"ospf_version": "v2", "ospf_processes": instance_list})
+ return utils.remove_empties(ospf_params_dict)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py
new file mode 100644
index 00000000..adef556f
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py
@@ -0,0 +1,115 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The eos ospfv3 fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospfv3 import (
+ Ospfv3Template,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv3.ospfv3 import (
+ Ospfv3Args,
+)
+
+
+class Ospfv3Facts(object):
+ """ The eos ospfv3 facts class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Ospfv3Args.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_config(self, connection):
+ """Wrapper method for `connection.get()`
+ This method exists solely to allow the unit test framework to mock device connection calls.
+ """
+ return connection.get("show running-config | section ospfv3")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for Ospfv3 network resource
+
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+
+ :rtype: dictionary
+ :returns: facts
+ """
+ facts = {}
+ objs = []
+
+ if not data:
+ data = self.get_config(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "router ospfv3"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ # parse native config using the Ospfv3 template
+ ospfv3_facts = {"processes": []}
+
+ for resource in resources:
+ ospfv3_parser = Ospfv3Template(lines=resource.splitlines())
+ objs = ospfv3_parser.parse()
+
+ for key, sortv in [("address_family", "afi")]:
+ if key in objs["processes"] and objs["processes"][key]:
+ objs["processes"][key] = list(
+ objs["processes"][key].values()
+ )
+
+ for addr_family in objs["processes"]["address_family"]:
+ if "areas" in addr_family:
+ addr_family["areas"] = list(addr_family["areas"].values())
+
+ for addr_family in objs["processes"]["address_family"]:
+ if not addr_family.get("afi"):
+ # global vars
+ objs["processes"].update(addr_family)
+ objs["processes"]["address_family"].remove(addr_family)
+
+ ospfv3_facts["processes"].append(objs["processes"])
+
+ ansible_facts["ansible_network_resources"].pop("ospfv3", None)
+ params = utils.validate_config(
+ self.argument_spec, {"config": ospfv3_facts}
+ )
+ params = utils.remove_empties(params)
+
+ facts["ospfv3"] = params.get("config", [])
+ ansible_facts["ansible_network_resources"].update(facts)
+
+ return ansible_facts
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py
new file mode 100644
index 00000000..6ae9cc2e
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py
@@ -0,0 +1,237 @@
+#
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos static_routes fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+from copy import deepcopy
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.static_routes.static_routes import (
+ Static_routesArgs,
+)
+
+
+class Static_routesFacts(object):
+ """ The eos static_routes fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = Static_routesArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def get_device_data(self, connection):
+ return connection.get("show running-config | grep route")
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for static_routes
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = self.get_device_data(connection)
+
+ # split the config into instances of the resource
+ resource_delim = "ip.* route"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [p.strip() for p in re.findall(find_pattern, data)]
+ resources_without_vrf = []
+ resource_vrf = {}
+ for resource in resources:
+ if resource and "vrf" not in resource:
+ resources_without_vrf.append(resource)
+ else:
+ vrf = re.search(r"ip(v6)* route vrf (.*?) .*", resource)
+ if vrf.group(2) in resource_vrf.keys():
+ vrf_val = resource_vrf[vrf.group(2)]
+ vrf_val.append(resource)
+ resource_vrf.update({vrf.group(2): vrf_val})
+ else:
+ resource_vrf.update({vrf.group(2): [resource]})
+ resources_without_vrf = ["\n".join(resources_without_vrf)]
+ for vrf in resource_vrf.keys():
+ vrflist = ["\n".join(resource_vrf[vrf])]
+ resource_vrf.update({vrf: vrflist})
+ objs = []
+ for resource in resources_without_vrf:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.append(obj)
+ for resource in resource_vrf.keys():
+ if resource:
+ obj = self.render_config(
+ self.generated_spec, resource_vrf[resource][0]
+ )
+ if obj:
+ objs.append(obj)
+ ansible_facts["ansible_network_resources"].pop("static_routes", None)
+ facts = {}
+ if objs:
+ facts["static_routes"] = []
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ for cfg in params["config"]:
+ facts["static_routes"].append(utils.remove_empties(cfg))
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ address_family_dict = {}
+ route_dict = {}
+ dest_list = []
+ afi_list = []
+ vrf_list = []
+ routes = []
+ config["address_families"] = []
+ next_hops = {}
+ interface_list = [
+ "Ethernet",
+ "Loopback",
+ "Management",
+ "Port-Channel",
+ "Tunnel",
+ "Vlan",
+ "Vxlan",
+ "vtep",
+ ]
+ conf_list = conf.split("\n")
+ for conf_elem in conf_list:
+ matches = re.findall(
+ r"(ip|ipv6) route ([\d\.\/:]+|vrf) (.+)$", conf_elem
+ )
+ if matches:
+ remainder = matches[0][2].split()
+ route_update = False
+ if matches[0][1] == "vrf":
+ vrf = remainder.pop(0)
+ # new vrf
+ if vrf not in vrf_list and vrf_list:
+ route_dict.update({"next_hops": next_hops})
+ routes.append(route_dict)
+ address_family_dict.update({"routes": routes})
+ config["address_families"].append(address_family_dict)
+ route_update = True
+ config.update({"vrf": vrf})
+ vrf_list.append(vrf)
+ dest = remainder.pop(0)
+ else:
+ config["vrf"] = None
+ dest = matches[0][1]
+ afi = "ipv4" if matches[0][0] == "ip" else "ipv6"
+ if afi not in afi_list:
+ if afi_list and not route_update:
+ # new afi and not the first updating all prev configs
+ route_dict.update({"next_hops": next_hops})
+ routes.append(route_dict)
+ address_family_dict.update({"routes": routes})
+ config["address_families"].append(address_family_dict)
+ route_update = True
+ address_family_dict = {}
+ address_family_dict.update({"afi": afi})
+ routes = []
+ afi_list.append(afi)
+ # To check the format of the dest
+ prefix = re.search(r"/", dest)
+ if not prefix:
+ dest = dest + " " + remainder.pop(0)
+ if dest not in dest_list:
+ # For new dest and not the first dest
+ if dest_list and not route_update:
+ route_dict.update({"next_hops": next_hops})
+ routes.append(route_dict)
+ dest_list.append(dest)
+ next_hops = []
+ route_dict = {}
+ route_dict.update({"dest": dest})
+ nexthops = {}
+ nxthop_addr = re.search(r"[\.\:]", remainder[0])
+ if nxthop_addr:
+ nexthops.update({"interface": remainder.pop(0)})
+ if remainder and remainder[0] == "label":
+ nexthops.update({"mpls_label": remainder.pop(1)})
+ remainder.pop(0)
+ elif re.search(r"Nexthop-Group", remainder[0]):
+ nexthops.update({"nexthop_grp": remainder.pop(1)})
+ remainder.pop(0)
+ else:
+ interface = remainder.pop(0)
+ if interface in interface_list:
+ interface = interface + " " + remainder.pop(0)
+ nexthops.update({"interface": interface})
+ for attribute in remainder:
+ forward_addr = re.search(
+ r"([\dA-Fa-f]+[:\.]+)+[\dA-Fa-f]+", attribute
+ )
+ if forward_addr:
+ nexthops.update(
+ {
+ "forward_router_address": remainder.pop(
+ remainder.index(attribute)
+ )
+ }
+ )
+ for attribute in remainder:
+ for params in ["tag", "name", "track"]:
+ if attribute == params:
+ keyname = params
+ if attribute == "name":
+ keyname = "description"
+ nexthops.update(
+ {
+ keyname: remainder.pop(
+ remainder.index(attribute) + 1
+ )
+ }
+ )
+ remainder.pop(remainder.index(attribute))
+ if remainder:
+ metric = re.search(r"\d+", remainder[0])
+ if metric:
+ nexthops.update({"admin_distance": remainder.pop(0)})
+ next_hops.append(nexthops)
+ route_dict.update({"next_hops": next_hops})
+ routes.append(route_dict)
+ address_family_dict.update({"routes": routes})
+ config["address_families"].append(address_family_dict)
+ return utils.remove_empties(config)
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py
new file mode 100644
index 00000000..0605fdb6
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py
@@ -0,0 +1,120 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+"""
+The eos vlans fact class
+It is in this file the configuration is collected from the device
+for a given resource, parsed, and the facts tree is populated
+based on the configuration.
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+from copy import deepcopy
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import (
+ utils,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.vlans.vlans import (
+ VlansArgs,
+)
+
+
+class VlansFacts(object):
+ """ The eos vlans fact class
+ """
+
+ def __init__(self, module, subspec="config", options="options"):
+ self._module = module
+ self.argument_spec = VlansArgs.argument_spec
+ spec = deepcopy(self.argument_spec)
+ if subspec:
+ if options:
+ facts_argument_spec = spec[subspec][options]
+ else:
+ facts_argument_spec = spec[subspec]
+ else:
+ facts_argument_spec = spec
+
+ self.generated_spec = utils.generate_dict(facts_argument_spec)
+
+ def populate_facts(self, connection, ansible_facts, data=None):
+ """ Populate the facts for vlans
+ :param connection: the device connection
+ :param ansible_facts: Facts dictionary
+ :param data: previously collected conf
+ :rtype: dictionary
+ :returns: facts
+ """
+ if not data:
+ data = connection.get("show running-config | section ^vlan")
+
+ # split the config into instances of the resource
+ resource_delim = "vlan"
+ find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % (
+ resource_delim,
+ resource_delim,
+ )
+ resources = [
+ p.strip() for p in re.findall(find_pattern, data, re.DOTALL)
+ ]
+
+ objs = []
+ for resource in resources:
+ if resource:
+ obj = self.render_config(self.generated_spec, resource)
+ if obj:
+ objs.extend(obj)
+
+ ansible_facts["ansible_network_resources"].pop("vlans", None)
+ facts = {}
+ if objs:
+ params = utils.validate_config(
+ self.argument_spec, {"config": objs}
+ )
+ facts["vlans"] = [
+ utils.remove_empties(cfg) for cfg in params["config"]
+ ]
+
+ ansible_facts["ansible_network_resources"].update(facts)
+ return ansible_facts
+
+ def render_config(self, spec, conf):
+ """
+ Render config as dictionary structure and delete keys
+ from spec for null values
+
+ :param spec: The facts tree, generated from the argspec
+ :param conf: The configuration
+ :rtype: dictionary
+ :returns: The generated config
+ """
+ config = deepcopy(spec)
+ vlans = []
+
+ vlan_list = vlan_to_list(utils.parse_conf_arg(conf, "vlan"))
+ for vlan in vlan_list:
+ config["vlan_id"] = vlan
+ config["name"] = utils.parse_conf_arg(conf, "name")
+ config["state"] = utils.parse_conf_arg(conf, "state")
+ if config["state"] is None:
+ config["state"] = "active"
+ vlans.append(utils.remove_empties(config))
+
+ return vlans
+
+
+def vlan_to_list(vlan_str):
+ vlans = []
+ for vlan in vlan_str.split(","):
+ if "-" in vlan:
+ start, stop = vlan.split("-")
+ vlans.extend(range(int(start), int(stop) + 1))
+ else:
+ vlans.append(int(vlan))
+
+ return vlans
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py
new file mode 100644
index 00000000..76524b0a
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py
@@ -0,0 +1,143 @@
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+import re
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import (
+ CliProvider,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.neighbors import (
+ AFNeighbors,
+)
+
+
+class AddressFamily(CliProvider):
+ def render(self, config=None):
+ commands = list()
+ safe_list = list()
+
+ router_context = "router bgp %s" % self.get_value("config.bgp_as")
+ context_config = None
+
+ for item in self.get_value("config.address_family"):
+ context = "address-family %s" % item["afi"]
+ context_commands = list()
+
+ if config:
+ context_path = [router_context, context]
+ context_config = self.get_config_context(
+ config, context_path, indent=2
+ )
+
+ for key, value in iteritems(item):
+ if value is not None:
+ meth = getattr(self, "_render_%s" % key, None)
+ if meth:
+ resp = meth(item, context_config)
+ if resp:
+ context_commands.extend(to_list(resp))
+
+ if context_commands:
+ commands.append(context)
+ commands.extend(context_commands)
+ commands.append("exit")
+
+ safe_list.append(context)
+
+ if self.params["operation"] == "replace":
+ if config:
+ resp = self._negate_config(config, safe_list)
+ commands.extend(resp)
+
+ return commands
+
+ def _negate_config(self, config, safe_list=None):
+ commands = list()
+ matches = re.findall(r"(address-family .+)$", config, re.M)
+ for item in set(matches).difference(safe_list):
+ commands.append("no %s" % item)
+ return commands
+
+ def _render_auto_summary(self, item, config=None):
+ cmd = "auto-summary"
+ if item["auto_summary"] is False:
+ cmd = "no %s" % cmd
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_synchronization(self, item, config=None):
+ cmd = "synchronization"
+ if item["synchronization"] is False:
+ cmd = "no %s" % cmd
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_networks(self, item, config=None):
+ commands = list()
+ safe_list = list()
+
+ for entry in item["networks"]:
+ network = entry["prefix"]
+ if entry["masklen"]:
+ network = "%s/%s" % (entry["prefix"], entry["masklen"])
+ safe_list.append(network)
+
+ cmd = "network %s" % network
+
+ if entry["route_map"]:
+ cmd += " route-map %s" % entry["route_map"]
+
+ if not config or cmd not in config:
+ commands.append(cmd)
+
+ if self.params["operation"] == "replace":
+ if config:
+ matches = re.findall(r"network (\S+)", config, re.M)
+ for entry in set(matches).difference(safe_list):
+ commands.append("no network %s" % entry)
+
+ return commands
+
+ def _render_redistribute(self, item, config=None):
+ commands = list()
+ safe_list = list()
+
+ for entry in item["redistribute"]:
+ option = entry["protocol"]
+
+ cmd = "redistribute %s" % entry["protocol"]
+
+ if entry["route_map"]:
+ cmd += " route-map %s" % entry["route_map"]
+
+ if not config or cmd not in config:
+ commands.append(cmd)
+
+ safe_list.append(option)
+
+ if self.params["operation"] == "replace":
+ if config:
+ matches = re.findall(
+ r"redistribute (\S+)(?:\s*)(\d*)", config, re.M
+ )
+ for i in range(0, len(matches)):
+ matches[i] = " ".join(matches[i]).strip()
+ for entry in set(matches).difference(safe_list):
+ commands.append("no redistribute %s" % entry)
+
+ return commands
+
+ def _render_neighbors(self, item, config):
+ """ generate bgp neighbor configuration
+ """
+ return AFNeighbors(self.params).render(
+ config, nbr_list=item["neighbors"]
+ )
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py
new file mode 100644
index 00000000..2dbcb81a
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py
@@ -0,0 +1,193 @@
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+import re
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import (
+ CliProvider,
+)
+
+
+class Neighbors(CliProvider):
+ def render(self, config=None, nbr_list=None):
+ commands = list()
+ safe_list = list()
+ if not nbr_list:
+ nbr_list = self.get_value("config.neighbors")
+
+ for item in nbr_list:
+ neighbor_commands = list()
+ context = "neighbor %s" % item["neighbor"]
+ cmd = "%s remote-as %s" % (context, item["remote_as"])
+
+ if not config or cmd not in config:
+ neighbor_commands.append(cmd)
+
+ for key, value in iteritems(item):
+ if value is not None:
+ meth = getattr(self, "_render_%s" % key, None)
+ if meth:
+ resp = meth(item, config)
+ if resp:
+ neighbor_commands.extend(to_list(resp))
+ commands.extend(neighbor_commands)
+ safe_list.append(context)
+
+ if self.params["operation"] == "replace":
+ if config and safe_list:
+ commands.extend(self._negate_config(config, safe_list))
+
+ return commands
+
+ def _negate_config(self, config, safe_list=None):
+ commands = list()
+ matches = re.findall(r"(neighbor \S+)", config, re.M)
+ for item in set(matches).difference(safe_list):
+ commands.append("no %s" % item)
+ return commands
+
+ def _render_description(self, item, config=None):
+ cmd = "neighbor %s description %s" % (
+ item["neighbor"],
+ item["description"],
+ )
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_enabled(self, item, config=None):
+ cmd = "neighbor %s shutdown" % item["neighbor"]
+ if item["enabled"] is True:
+ if not config or cmd in config:
+ cmd = "no %s" % cmd
+ return cmd
+ elif not config or cmd not in config:
+ return cmd
+
+ def _render_update_source(self, item, config=None):
+ cmd = "neighbor %s update-source %s" % (
+ item["neighbor"],
+ item["update_source"],
+ )
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_password(self, item, config=None):
+ cmd = "neighbor %s password %s" % (item["neighbor"], item["password"])
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_ebgp_multihop(self, item, config=None):
+ cmd = "neighbor %s ebgp-multihop %s" % (
+ item["neighbor"],
+ item["ebgp_multihop"],
+ )
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_peer_group(self, item, config=None):
+ cmd = "neighbor %s peer-group %s" % (
+ item["neighbor"],
+ item["peer_group"],
+ )
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_route_reflector_client(self, item, config=None):
+ cmd = "neighbor %s route-reflector-client" % item["neighbor"]
+ if item["route_reflector_client"] is False:
+ if not config or cmd in config:
+ cmd = "no %s" % cmd
+ return cmd
+ elif not config or cmd not in config:
+ return cmd
+
+ def _render_maximum_prefix(self, item, config=None):
+ cmd = "neighbor %s maximum-routes %s" % (
+ item["neighbor"],
+ item["maximum_prefix"],
+ )
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_remove_private_as(self, item, config=None):
+ cmd = "neighbor %s remove-private-AS" % item["neighbor"]
+ if item["remove_private_as"] is False:
+ if not config or cmd in config:
+ cmd = "no %s" % cmd
+ return cmd
+ elif not config or cmd not in config:
+ return cmd
+
+ def _render_timers(self, item, config):
+ """generate bgp timer related configuration
+ """
+ keepalive = item["timers"]["keepalive"]
+ holdtime = item["timers"]["holdtime"]
+ neighbor = item["neighbor"]
+
+ if keepalive and holdtime:
+ cmd = "neighbor %s timers %s %s" % (neighbor, keepalive, holdtime)
+ if not config or cmd not in config:
+ return cmd
+
+
+class AFNeighbors(CliProvider):
+ def render(self, config=None, nbr_list=None):
+ commands = list()
+ if not nbr_list:
+ return
+
+ for item in nbr_list:
+ neighbor_commands = list()
+ for key, value in iteritems(item):
+ if value is not None:
+ meth = getattr(self, "_render_%s" % key, None)
+ if meth:
+ resp = meth(item, config)
+ if resp:
+ neighbor_commands.extend(to_list(resp))
+
+ commands.extend(neighbor_commands)
+
+ return commands
+
+ def _render_activate(self, item, config=None):
+ cmd = "neighbor %s activate" % item["neighbor"]
+ if item["activate"] is False:
+ if not config or cmd in config:
+ cmd = "no %s" % cmd
+ return cmd
+ elif not config or cmd not in config:
+ return cmd
+
+ def _render_default_originate(self, item, config=None):
+ cmd = "neighbor %s default-originate" % item["neighbor"]
+ if item["default_originate"] is False:
+ if not config or cmd in config:
+ cmd = "no %s" % cmd
+ return cmd
+ elif not config or cmd not in config:
+ return cmd
+
+ def _render_graceful_restart(self, item, config=None):
+ cmd = "neighbor %s graceful-restart" % item["neighbor"]
+ if item["graceful_restart"] is False:
+ cmd = "no " + cmd
+ if config:
+ config_el = [x.strip() for x in config.split("\n")]
+ if cmd in config_el:
+ return
+ return cmd
+
+ def _render_weight(self, item, config=None):
+ cmd = "neighbor %s weight %s" % (item["neighbor"], item["weight"])
+ if not config or cmd not in config:
+ return cmd
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py
new file mode 100644
index 00000000..bc75c0a2
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py
@@ -0,0 +1,185 @@
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+import re
+
+from ansible.module_utils.six import iteritems
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import (
+ register_provider,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import (
+ CliProvider,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.neighbors import (
+ Neighbors,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.address_family import (
+ AddressFamily,
+)
+
+REDISTRIBUTE_PROTOCOLS = frozenset(
+ ["ospf", "ospf3", "rip", "isis", "static", "connected"]
+)
+
+
+@register_provider("eos", "eos_bgp")
+class Provider(CliProvider):
+ def render(self, config=None):
+ commands = list()
+
+ existing_as = None
+ if config:
+ match = re.search(r"router bgp (\d+)", config, re.M)
+ if match:
+ existing_as = match.group(1)
+
+ operation = self.params["operation"]
+
+ context = None
+ if self.params["config"]:
+ context = "router bgp %s" % self.get_value("config.bgp_as")
+
+ if operation == "delete":
+ if existing_as:
+ commands.append("no router bgp %s" % existing_as)
+ elif context:
+ commands.append("no %s" % context)
+
+ else:
+ self._validate_input(config)
+ if operation == "replace":
+ if existing_as and int(existing_as) != self.get_value(
+ "config.bgp_as"
+ ):
+ commands.append("no router bgp %s" % existing_as)
+ config = None
+
+ elif operation == "override":
+ if existing_as:
+ commands.append("no router bgp %s" % existing_as)
+ config = None
+
+ context_commands = list()
+
+ for key, value in iteritems(self.get_value("config")):
+ if value is not None:
+ meth = getattr(self, "_render_%s" % key, None)
+ if meth:
+ resp = meth(config)
+ if resp:
+ context_commands.extend(to_list(resp))
+
+ if context and context_commands:
+ commands.append(context)
+ commands.extend(context_commands)
+ commands.append("exit")
+ return commands
+
+ def _render_router_id(self, config=None):
+ cmd = "router-id %s" % self.get_value("config.router_id")
+ if not config or cmd not in config:
+ return cmd
+
+ def _render_log_neighbor_changes(self, config=None):
+ cmd = "bgp log-neighbor-changes"
+ log_neighbor_changes = self.get_value("config.log_neighbor_changes")
+ if log_neighbor_changes is True:
+ if not config or cmd not in config:
+ return cmd
+ elif log_neighbor_changes is False:
+ if config and cmd in config:
+ return "no %s" % cmd
+
+ def _render_networks(self, config=None):
+ commands = list()
+ safe_list = list()
+
+ for entry in self.get_value("config.networks"):
+ network = entry["prefix"]
+ if entry["masklen"]:
+ network = "%s/%s" % (entry["prefix"], entry["masklen"])
+ safe_list.append(network)
+
+ cmd = "network %s" % network
+
+ if entry["route_map"]:
+ cmd += " route-map %s" % entry["route_map"]
+
+ if not config or cmd not in config:
+ commands.append(cmd)
+
+ if self.params["operation"] == "replace":
+ if config:
+ matches = re.findall(r"network (\S+)", config, re.M)
+ for entry in set(matches).difference(safe_list):
+ commands.append("no network %s" % entry)
+
+ return commands
+
+ def _render_redistribute(self, config=None):
+ commands = list()
+ safe_list = list()
+
+ for entry in self.get_value("config.redistribute"):
+ option = entry["protocol"]
+
+ cmd = "redistribute %s" % entry["protocol"]
+
+ if entry["route_map"]:
+ cmd += " route-map %s" % entry["route_map"]
+
+ if not config or cmd not in config:
+ commands.append(cmd)
+
+ safe_list.append(option)
+
+ if self.params["operation"] == "replace":
+ if config:
+ matches = re.findall(
+ r"redistribute (\S+)(?:\s*)(\d*)", config, re.M
+ )
+ for i in range(0, len(matches)):
+ matches[i] = " ".join(matches[i]).strip()
+ for entry in set(matches).difference(safe_list):
+ commands.append("no redistribute %s" % entry)
+
+ return commands
+
+ def _render_neighbors(self, config):
+ """ generate bgp neighbor configuration
+ """
+ return Neighbors(self.params).render(config)
+
+ def _render_address_family(self, config):
+ """ generate address-family configuration
+ """
+ return AddressFamily(self.params).render(config)
+
+ def _validate_input(self, config):
+ def device_has_AF(config):
+ return re.search(r"address-family (?:.*)", config)
+
+ address_family = self.get_value("config.address_family")
+ root_networks = self.get_value("config.networks")
+ operation = self.params["operation"]
+
+ if operation == "replace" and root_networks:
+ if address_family:
+ for item in address_family:
+ if item["networks"]:
+ raise ValueError(
+ "operation is replace but provided both root level networks and networks under %s address family"
+ % item["afi"]
+ )
+
+ if config and device_has_AF(config):
+ raise ValueError(
+ "operation is replace and device has one or more address family activated but root level network(s) provided"
+ )
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py
new file mode 100644
index 00000000..2325fcfd
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py
@@ -0,0 +1,72 @@
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+from ansible.module_utils.basic import AnsibleModule
+from ansible.module_utils.connection import Connection
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers import (
+ providers,
+)
+from ansible.module_utils._text import to_text
+
+
+class NetworkModule(AnsibleModule):
+
+ fail_on_missing_provider = True
+
+ def __init__(self, connection=None, *args, **kwargs):
+ super(NetworkModule, self).__init__(*args, **kwargs)
+
+ if connection is None:
+ connection = Connection(self._socket_path)
+
+ self.connection = connection
+
+ @property
+ def provider(self):
+ if not hasattr(self, "_provider"):
+ capabilities = self.from_json(self.connection.get_capabilities())
+
+ network_os = capabilities["device_info"]["network_os"]
+ network_api = capabilities["network_api"]
+
+ if network_api == "cliconf":
+ connection_type = "network_cli"
+
+ cls = providers.get(
+ network_os, self._name.split(".")[-1], connection_type
+ )
+
+ if not cls:
+ msg = (
+ "unable to find suitable provider for network os %s"
+ % network_os
+ )
+ if self.fail_on_missing_provider:
+ self.fail_json(msg=msg)
+ else:
+ self.warn(msg)
+
+ obj = cls(self.params, self.connection, self.check_mode)
+
+ setattr(self, "_provider", obj)
+
+ return getattr(self, "_provider")
+
+ def get_facts(self, subset=None):
+ try:
+ self.provider.get_facts(subset)
+ except Exception as exc:
+ self.fail_json(msg=to_text(exc))
+
+ def edit_config(self, config_filter=None):
+ current_config = self.connection.get_config(flags=config_filter)
+ try:
+ commands = self.provider.edit_config(current_config)
+ changed = bool(commands)
+ return {"commands": commands, "changed": changed}
+ except Exception as exc:
+ self.fail_json(msg=to_text(exc))
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py
new file mode 100644
index 00000000..a7dbdaa5
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py
@@ -0,0 +1,128 @@
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+import json
+
+from threading import RLock
+
+from ansible.module_utils.six import itervalues
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ to_list,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+)
+
+
+_registered_providers = {}
+_provider_lock = RLock()
+
+
+def register_provider(network_os, module_name):
+ def wrapper(cls):
+ _provider_lock.acquire()
+ try:
+ if network_os not in _registered_providers:
+ _registered_providers[network_os] = {}
+ for ct in cls.supported_connections:
+ if ct not in _registered_providers[network_os]:
+ _registered_providers[network_os][ct] = {}
+ for item in to_list(module_name):
+ for entry in itervalues(_registered_providers[network_os]):
+ entry[item] = cls
+ finally:
+ _provider_lock.release()
+ return cls
+
+ return wrapper
+
+
+def get(network_os, module_name, connection_type):
+ network_os_providers = _registered_providers.get(network_os)
+ if network_os_providers is None:
+ raise ValueError("unable to find a suitable provider for this module")
+ if connection_type not in network_os_providers:
+ raise ValueError("provider does not support this connection type")
+ elif module_name not in network_os_providers[connection_type]:
+ raise ValueError("could not find a suitable provider for this module")
+ return network_os_providers[connection_type][module_name]
+
+
+class ProviderBase(object):
+
+ supported_connections = ()
+
+ def __init__(self, params, connection=None, check_mode=False):
+ self.params = params
+ self.connection = connection
+ self.check_mode = check_mode
+
+ @property
+ def capabilities(self):
+ if not hasattr(self, "_capabilities"):
+ resp = self.from_json(self.connection.get_capabilities())
+ setattr(self, "_capabilities", resp)
+ return getattr(self, "_capabilities")
+
+ def get_value(self, path):
+ params = self.params.copy()
+ for key in path.split("."):
+ params = params[key]
+ return params
+
+ def get_facts(self, subset=None):
+ raise NotImplementedError(self.__class__.__name__)
+
+ def edit_config(self):
+ raise NotImplementedError(self.__class__.__name__)
+
+
+class CliProvider(ProviderBase):
+
+ supported_connections = ("network_cli",)
+
+ @property
+ def capabilities(self):
+ if not hasattr(self, "_capabilities"):
+ resp = self.from_json(self.connection.get_capabilities())
+ setattr(self, "_capabilities", resp)
+ return getattr(self, "_capabilities")
+
+ def get_config_context(self, config, path, indent=2):
+ if config is not None:
+ netcfg = NetworkConfig(indent=indent, contents=config)
+ try:
+ config = netcfg.get_block_config(to_list(path))
+ except ValueError:
+ config = None
+ return config
+
+ def render(self, config=None):
+ raise NotImplementedError(self.__class__.__name__)
+
+ def cli(self, command):
+ try:
+ if not hasattr(self, "_command_output"):
+ setattr(self, "_command_output", {})
+ return self._command_output[command]
+ except KeyError:
+ out = self.connection.get(command)
+ try:
+ out = json.loads(out)
+ except ValueError:
+ pass
+ self._command_output[command] = out
+ return out
+
+ def get_facts(self, subset=None):
+ return self.populate()
+
+ def edit_config(self, config=None):
+ commands = self.render(config)
+ if commands and self.check_mode is False:
+ self.connection.edit_config(commands)
+ return commands
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py
new file mode 100644
index 00000000..d54d6bbc
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py
@@ -0,0 +1,661 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The Bgp_global parser templates file. This contains
+a list of parser definitions and associated functions that
+facilitates both facts gathering and native command generation for
+the given network resource.
+"""
+
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import (
+ NetworkTemplate,
+)
+
+
+def _tmplt_router_bgp_cmd(config_data):
+ command = "router bgp {as_number}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_address_family(config_data):
+ command = ""
+ if config_data.get("vrf"):
+ command = "vrf {vrf}\n".format(**config_data)
+ command += "address-family {afi}".format(**config_data)
+ if config_data.get("safi"):
+ command += " {safi}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_params(config_data):
+ command = "bgp"
+ if config_data["bgp_params"].get("additional_paths"):
+ command += " additional-paths {additional_paths}".format(
+ **config_data["bgp_params"]
+ )
+ if config_data["bgp_params"]["additional_paths"] == "send":
+ command += " any"
+ elif config_data["bgp_params"].get("next_hop_address_family"):
+ command += " next-hop address-family ipv6"
+ elif config_data["bgp_params"].get("next_hop_unchanged"):
+ command += " next-hop-unchanged"
+ elif config_data["bgp_params"].get("redistribute_internal"):
+ command += " redistribute-internal"
+ elif config_data["bgp_params"].get("route"):
+ command += " route install-map {route}".format(
+ **config_data["bgp_params"]
+ )
+ return command
+
+
+def _tmplt_bgp_graceful_restart(config_data):
+ command = "graceful-restart"
+ return command
+
+
+def _tmplt_bgp_neighbor(config_data):
+ command = "neighbor {peer}".format(**config_data["neighbor"])
+ if config_data["neighbor"].get("additional_paths"):
+ command += " additional-paths {additional_paths}".format(
+ **config_data["neighbor"]
+ )
+ if config_data["neighbor"]["additional_paths"] == "send":
+ command += "any"
+ elif config_data["neighbor"].get("activate"):
+ command += " activate"
+ elif config_data["neighbor"].get("default_originate"):
+ command += " default-originate"
+ if config_data["neighbor"]["default_originate"].get("route_map"):
+ command += " route-map {route_map}".format(
+ **config_data["neighbor"]["default_originate"]
+ )
+ if config_data["neighbor"]["default_originate"].get("always"):
+ command += " always"
+ elif config_data["neighbor"].get("graceful_restart"):
+ command += " graceful-restart"
+ elif config_data["neighbor"].get("next_hop_unchanged"):
+ command += " next-hop-unchanged"
+ elif config_data["neighbor"].get("next_hop_address_family"):
+ command += " next-hop addres-family ipv6"
+ elif config_data["neighbor"].get("prefix_list"):
+ command += " prefix-list {name} {direction}".format(
+ **config_data["neighbor"]["prefix_list"]
+ )
+ elif config_data["neighbor"].get("route_map"):
+ command += " route-map {name} {direction}".format(
+ **config_data["neighbor"]["route_map"]
+ )
+ elif config_data["neighbor"].get("weight"):
+ command += " weight {weight}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("encapsulation"):
+ command += " encapsulation {transport}".format(
+ **config_data["neighbor"]
+ )
+ if config_data["neighbor"]["encapsulation"].get("source_interface"):
+ command += " next-hop-self source-interface {source_interface}".format(
+ **config_data["neighbor"]
+ )
+ return command
+
+
+def _tmplt_bgp_network(config_data):
+ command = "network {address}".format(**config_data)
+ if config_data.get("route_map"):
+ command += " route-map {route_map}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_redistribute(config_data):
+ command = "redistribute {protocol}".format(**config_data)
+ if config_data.get("isis_level"):
+ command += " {isis_level}".format(**config_data)
+ if config_data.get("ospf_route"):
+ command += " match {ospf_route}".format(**config_data)
+ if config_data.get("route_map"):
+ command += " route-map {route_map}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_route_target(config_data):
+ command = "route-target {mode} {target}".format(
+ **config_data["route_target"]
+ )
+ return command
+
+
+class Bgp_afTemplate(NetworkTemplate):
+ def __init__(self, lines=None):
+ super(Bgp_afTemplate, self).__init__(lines=lines, tmplt=self)
+
+ # fmt: off
+ PARSERS = [
+ {
+ "name": "router",
+ "getval": re.compile(
+ r"""
+ ^router\s
+ bgp
+ \s(?P<as_num>\S+)
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_router_bgp_cmd,
+ "compval": "as_number",
+ "result": {"as_number": "{{ as_num }}"},
+ "shared": True
+ },
+ {
+ "name": "address_family",
+ "getval": re.compile(
+ r"""
+ \s*(?P<vrf>vrf\s\S+)*
+ \s*address-family
+ \s(?P<afi>ipv4|ipv6|evpn)
+ \s*(?P<type>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_address_family,
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "afi": "{{ afi }}",
+ "safi": "{{ type }}",
+ "vrf": "{{ vrf.split(" ")[1] }}"
+ }
+ }
+ },
+ "shared": True,
+ },
+ {
+ "name": "bgp_params_additional_paths",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+additional-paths
+ \s+(?P<action>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.additional_paths",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "bgp_params": {
+ "additional_paths": "{{ action }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.nexthop_address_family",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+next-hop
+ \s+address-family
+ \s+ipv6
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.next_hop_address_family",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "bgp_params": {
+ "next_hop_unchanged": "{{ 'ipv6' }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.nexthop_unchanged",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+next-hop-unchanged
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.next_hop_unchanged",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "bgp_params": {
+ "next_hop_unchanged": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.redistribute_internal",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+redistribute-internal
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.redistribute_internal",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "bgp_params": {
+ "redistribute_internal": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.route",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+route
+ \s+install-map
+ \s+(?P<route>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.route",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "bgp_params": {
+ "route": "{{ route }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart",
+ "getval": re.compile(
+ r"""
+ \s*graceful-restart
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_graceful_restart,
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "graceful_restart": "{{ True }}"
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.activate",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+activate
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.activate",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "activate": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.additional_paths",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+additional-paths
+ \s+(?P<action>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.additional_paths",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "additional_paths": "{{ action }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.default_originate",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+default-originate
+ \s*(?P<route_map>route-map\s\S+)*
+ \s*(?P<always>always)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.default_originate",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "default_originate": {
+ "route_map": "{{ route_map.split(" ")[1] }}",
+ "always": "{{ True if always is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.graceful_restart",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+graceful-restart
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.graceful_restart",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "graceful_restart": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.next_hop_unchanged",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+next-hop-unchanged
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.next_hop_unchanged",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "next_hop_unchanged": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.next_hop_address_family",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+next-hop
+ \s+address-family
+ \s+ipv6
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.next_hop_address_family",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "next_hop_address_family": "{{ 'ipv6' }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.prefix_list",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+prefix-list
+ \s+(?P<name>\S+)
+ \s+(?P<dir>in|out)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.prefix_list",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "prefix_list": {
+ "name": "{{ name }}",
+ "direction": "{{ dir }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.route_map",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+route-map
+ \s+(?P<name>\S+)
+ \s+(?P<dir>in|out)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.route_map",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "route_map": {
+ "name": "{{ name }}",
+ "direction": "{{ dir }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.weight",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+weight
+ \s+(?P<weight>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.weight",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "weight": "{{ weight }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.encapsulation",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+default
+ \s+encapsulation
+ \s+(?P<type>mpls|vxlan)
+ \s*(next-hop-self)*
+ \s*(source-interface)*
+ \s*(?P<interface>\S+\s\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.encapsulation",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "encapsulation": {
+ "transport": "{{ type }}",
+ "source_interface": "{{ interface }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "network",
+ "getval": re.compile(
+ r"""
+ \s*network
+ \s+(?P<address>\S+)
+ \s*(route-map)*
+ \s*(?P<route_map>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_network,
+ "compval": "network",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "network": {
+ "{{ address }}": {
+ "address": "{{ address }}",
+ "route_map": "{{ route_map }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "redistribute",
+ "getval": re.compile(
+ r"""
+ \s*redistribute
+ \s+(?P<route>\S+)
+ \s*(?P<level>level-1|level-2|level-1-2)*
+ \s*(?P<match>match\s\S+)*
+ \s*(?P<route_map>route-map\s\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_redistribute,
+ "compval": "redistribute",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "redistribute": [
+ {
+ "protocol": "{{ route }}",
+ "route_map": "{{ route_map.split(" ")[1] }}",
+ "isis_level": "{{ level }}",
+ "ospf_route": "{{ match.split(" ")[1] }}"
+ }
+ ]
+ }
+ }
+ },
+ },
+ {
+ "name": "route_target",
+ "getval": re.compile(
+ r"""
+ \s*route-target
+ \s+(?P<mode>both|import|export)
+ \s+(?P<target>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_route_target,
+ "compval": "route_target",
+ "result": {
+ "address_family": {
+ '{{ afi + "_" + vrf|d() }}': {
+ "route_target": {
+ "mode": "{{ mode }}",
+ "target": "{{ target }}",
+ }
+ }
+ }
+ },
+ },
+ ]
+ # fmt: on
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py
new file mode 100644
index 00000000..9225574c
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py
@@ -0,0 +1,2881 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The Bgp_global parser templates file. This contains
+a list of parser definitions and associated functions that
+facilitates both facts gathering and native command generation for
+the given network resource.
+"""
+
+import re
+
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import (
+ NetworkTemplate,
+)
+
+
+def _tmplt_router_bgp_cmd(config_data):
+ command = "router bgp {as_number}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_vrf(config_data):
+ command = "vrf {vrf}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_aggregate_address(config_data):
+ command = "aggregate-address {address}".format(**config_data)
+ if config_data.get("as_set"):
+ command += " as-set"
+ if config_data.get("summary_only"):
+ command += " summary-only"
+ if config_data.get("attribute_map"):
+ command += " attribute-map {attribute_map}".format(**config_data)
+ if config_data.get("match_map"):
+ command += " match-map {match_map}".format(**config_data)
+ if config_data.get("advertise_only"):
+ command += " advertise-only"
+ return command
+
+
+def _tmplt_bgp_params(config_data):
+ command = "bgp"
+ if config_data["bgp_params"].get("additional_paths"):
+ command += (
+ " additional-paths "
+ + config_data["bgp_params"]["additional_paths"]
+ )
+ if config_data["bgp_params"]["additional_paths"] == "send":
+ command += " any"
+ elif config_data["bgp_params"].get("advertise_inactive"):
+ command += " advertise-inactive"
+ elif config_data["bgp_params"].get("allowas_in"):
+ command += " allowas-in"
+ if config_data["bgp_params"]["allowas_in"].get("count"):
+ command += " {count}".format(
+ **config_data["bgp_params"]["allowas_in"]
+ )
+ elif config_data["bgp_params"].get("always_compare_med"):
+ command += " always-comapre-med"
+ elif config_data["bgp_params"].get("asn"):
+ command += " asn notaion {asn}".format(**config_data["bgp_params"])
+ elif config_data["bgp_params"].get("auto_local_addr"):
+ command += " auto-local-addr"
+ elif config_data["bgp_params"].get("bestpath"):
+ command += " bestpath"
+ if config_data["bgp_params"]["bestpath"].get("as_path"):
+ command += " as-path {as_path}".format(
+ **config_data["bgp_params"]["as_path"]
+ )
+ elif config_data["bgp_params"]["bestpath"].get("ecmp_fast"):
+ command += " ecmp-fast"
+ elif config_data["bgp_params"].get("med"):
+ command += " med"
+ if config_data["bgp_params"]["med"].get("confed"):
+ command += " confed"
+ else:
+ command += " missing-as-worst"
+ elif config_data["bgp_params"].get("skip"):
+ command += " skip next-hop igp-cost"
+ elif config_data["bgp_params"].get("tie_break"):
+ tie = re.sub(r"_", r"-", config_data["bgp_params"]["tie_break"])
+ command += " tie-break " + tie
+ elif config_data["bgp_params"].get("client_to_client"):
+ command += " client-to-client"
+ elif config_data["bgp_params"].get("cluster_id"):
+ command += " cluster-id {cluster_id}".format(
+ **config_data["bgp_params"]
+ )
+ elif config_data["bgp_params"].get("confederation"):
+ command += " confederation"
+ if config_data["bgp_params"]["confederation"].get("identifier"):
+ command += " identifier {identifier}".format(
+ **config_data["bgp_params"]["confederation"]
+ )
+ else:
+ command += " peers {peers}".format(
+ **config_data["bgp_params"]["confederation"]
+ )
+ elif config_data["bgp_params"].get("control_plane_filter"):
+ command += " control-plane-filter default-allow"
+ elif config_data["bgp_params"].get("convergence"):
+ command += " convergence"
+ if config_data["bgp_params"]["convergence"].get("slow_peer"):
+ command += " slow-peer"
+ command += " time {time}".format(
+ **config_data["bgp_params"]["convergence"]
+ )
+ elif config_data["bgp_params"].get("default"):
+ command += " default {default}".format(**config_data["bgp_params"])
+ elif config_data["bgp_params"].get("enforce_first-as"):
+ command += " enforce-first-as"
+ elif config_data["bgp_params"].get("host_routes"):
+ command += " host-routes fib direct-install"
+ elif config_data["bgp_params"].get("labeled_unicast"):
+ command += " labeled-unicast rib {labeled_unicast}".format(
+ **config_data["bgp_params"]
+ )
+ elif config_data["bgp_params"].get("listen"):
+ command += " listen"
+ if config_data["bgp_params"]["listen"].get("limit"):
+ command += " limit {limit}".format(
+ **config_data["bgp_params"]["listen"]
+ )
+ else:
+ command += " range {address} peer-group".format(
+ **config_data["bgp_params"]["listen"]["range"]
+ )
+ if config_data["bgp_params"]["listen"]["range"]["peer_group"].get(
+ "peer_filter"
+ ):
+ command += " {name} peer-filter {peer_filter}".format(
+ **config_data["bgp_params"]["listen"]["range"][
+ "peer_group"
+ ]
+ )
+ else:
+ command += " {name} remote-as {remote_as}".format(
+ **config_data["bgp_params"]["listen"]["range"][
+ "peer_group"
+ ]
+ )
+ elif config_data["bgp_params"].get("log_neighbor_changes"):
+ command += " log-neighbor-changes"
+ elif config_data["bgp_params"].get("missing_policy"):
+ command += " missing-policy direction {direction} action {action}".format(
+ **config_data["bgp_params"]["missing_policy"]
+ )
+ elif config_data["bgp_params"].get("monitoring"):
+ command += " monitoring"
+ elif config_data["bgp_params"].get("next_hop_unchanged"):
+ command += " next-hop-unchanged"
+ elif config_data["bgp_params"].get("redistribute_internal"):
+ command += " redistribute-internal"
+ elif config_data["bgp_params"].get("route"):
+ command += " route install-map {route}".format(
+ **config_data["bgp_params"]
+ )
+ elif config_data["bgp_params"].get("route_reflector"):
+ command += " route-reflector preserve-attributes"
+ if config_data["bgp_params"]["reoute_reflector"].get("always"):
+ command += " always"
+ elif config_data["bgp_params"].get("transport"):
+ command += " transport listen-port {transport}".format(
+ **config_data["bgp_params"]
+ )
+ return command
+
+
+def _tmplt_bgp_redistribute(config_data):
+ command = "redistribute {protocol}".format(**config_data)
+ if config_data.get("isis_level"):
+ command += " {isis_level}".format(**config_data)
+ if config_data.get("ospf_route"):
+ if config_data["ospf_route"] == "nssa_external_2":
+ route = "nssa-external 2"
+ elif config_data["ospf_route"] == "nssa_external_1":
+ route = "nssa-external 1"
+ else:
+ route = config_data["ospf_route"]
+ command += " match " + route
+ if config_data.get("route_map"):
+ command += " route-map {route_map}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_default_metric(config_data):
+ command = "default-metric {default_metric}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_distance(config_data):
+ command = "distance bgp"
+ if config_data["distance"].get("external"):
+ command += " {external}".format(**config_data["distance"])
+ if config_data["distance"].get("internal"):
+ command += " {internal}".format(**config_data["distance"])
+ if config_data["distance"].get("local"):
+ command += " {local}".format(**config_data["distance"])
+ return command
+
+
+def _tmplt_bgp_graceful_restart(config_data):
+ command = "graceful-restart"
+ if config_data.get("restart_time"):
+ command += " restart-time {restart_time}".format(**config_data)
+ if config_data.get("stalepath_time"):
+ command += " stalepath-time {stalepath_time}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_graceful_restart_helper(config_data):
+ command = "graceful-restart-helper"
+ return command
+
+
+def _tmplt_bgp_access_group(config_data):
+ if config_data["accesS_group"].get("afi") == "ipv4":
+ afi = "ip"
+ else:
+ afi = "ipv6"
+ command = afi + " access-group {acl_name}".format(
+ **config_data["access_group"]
+ )
+ if config_data["access_group"].get("direction"):
+ command += " {direction}".format(**config_data["access_group"])
+ return command
+
+
+def _tmplt_bgp_maximum_paths(config_data):
+ command = "maximum-paths {max_equal_cost_paths}".format(
+ **config_data["maximum_paths"]
+ )
+ if config_data.get("max_installed_ecmp_paths"):
+ command += " ecmp {max_installed_ecmp_paths}".format(
+ **config_data["maximum_paths"]
+ )
+ return command
+
+
+def _tmplt_bgp_monitoring(config_data):
+ cmd = "monitoring"
+ command = ""
+ if config_data.get("timestamp"):
+ command = cmd + " timestamp {timestamp}".format(**config_data)
+ if config_data.get("port"):
+ command = cmd + " port {port}".format(**config_data)
+ if config_data.get("received"):
+ command = cmd + " received routes {received}".format(**config_data)
+ if config_data.get("station"):
+ command = cmd + " station {station}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_neighbor(config_data):
+ command = "neighbor {peer}".format(**config_data["neighbor"])
+ if config_data["neighbor"].get("additional_paths"):
+ command += " additional-paths {additional_paths}".format(
+ **config_data["neighbor"]
+ )
+ if config_data["neighbor"]["additional_paths"] == "send":
+ command += "any"
+ elif config_data["neighbor"].get("peer_group"):
+ command += " peer-group"
+ if (
+ config_data["neighbor"]["peer_group"]
+ != config_data["neighbor"]["peer_group"]
+ ):
+ command += config_data["neighbor"]["peer_group"]
+ elif config_data["neighbor"].get("allowas_in"):
+ command += " allowas-in"
+ if config_data["neighbor"]["allowas_in"].get("count"):
+ command += " {count}".format(
+ **config_data["neighbor"]["allowas_in"]
+ )
+ elif config_data["neighbor"].get("auto_local_addr"):
+ command += " auto-local-addr"
+ elif config_data["neighbor"].get("default_originate"):
+ command += " default-originate"
+ if config_data["neighbor"]["default_originate"].get("route_map"):
+ command += " route-map {route_map}".format(
+ **config_data["neighbor"]["default_originate"]
+ )
+ if config_data["neighbor"]["default_originate"].get("always"):
+ command += " always"
+ elif config_data["neighbor"].get("description"):
+ command += " description {description}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("dont_capability_negotiate"):
+ command += " dont-capability-negotiate"
+ elif config_data["neighbor"].get("ebgp_multihop"):
+ command += " ebgp-multiphop"
+ if config_data["neighbor"]["ebgp_multihop"].get("ttl"):
+ command += " {ttl}".format(
+ **config_data["neighbor"]["ebgp_multihop"]
+ )
+ elif config_data["neighbor"].get("enforce_first_as"):
+ command += " enforce-first-as"
+ elif config_data["neighbor"].get("export_localpref"):
+ command += " export-localpref {export_localpref}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("fall_over"):
+ command += " fall-over bfd"
+ elif config_data["neighbor"].get("graceful_restart"):
+ command += " graceful-restart"
+ elif config_data["neighbor"].get("graceful_restart_helper"):
+ command += " graceful-restart-helper"
+ elif config_data["neighbor"].get("idle_restart_timer"):
+ command += " idle-restart-timer {idle_restart_timer}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("import_localpref"):
+ command += " import-localpref {import_localpref}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("link_bandwidth"):
+ command += " link-bandwidth"
+ if config_data["neighbor"]["link_bandwidth"].get("auto"):
+ command += " auto"
+ if config_data["neighbor"]["link_bandwidth"].get("default"):
+ command += " default {default}".format(
+ **config_data["neighbor"]["link_bandwidth"]
+ )
+ if config_data["neighbor"]["link_bandwidth"].get("update_delay"):
+ command += " update-delay {update_delay}".format(
+ **config_data["neighbor"]["link_bandwidth"]
+ )
+ elif config_data["neighbor"].get("local_as"):
+ command += " local-as {as_number} no-prepend replace-as".format(
+ **config_data["neighbor"]
+ )
+ if config_data["neighbor"]["local_as"].get("fallback"):
+ command += " fallback"
+ elif config_data["neighbor"].get("local_v6_addr"):
+ command += " local-v6-addr {local_v6_addr}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("maximum_accepted_routes"):
+ command += " maximum-accepted-routes {count}".format(
+ **config_data["neighbor"]["maximum_accepted_routes"]
+ )
+ if config_data["neighbor"]["maximum_accepted_routes"].get(
+ "warning_limit"
+ ):
+ command += " warning-limit {warning_limit}".format(
+ **config_data["neighbor"]["maximum_accepted_routes"]
+ )
+ elif config_data["neighbor"].get("maximum_received_routes"):
+ command += " maximum-routes {count}".format(
+ **config_data["neighbor"]["maximum_received_routes"]
+ )
+ if config_data["neighbor"]["maximum_received_routes"].get(
+ "warning_limit"
+ ):
+ if config_data["neighbor"]["maximum_received_routes"][
+ "warning_limit"
+ ].get("limit_count"):
+ command += " warning-limit {limit_count}".format(
+ **config_data["neighbor"]["maximum_received_routes"][
+ "warning_limit"
+ ]
+ )
+ if config_data["neighbor"]["maximum_received_routes"][
+ "warning_limit"
+ ].get("limit_percent"):
+ command += (
+ " warning-limit "
+ + str(
+ config_data["neighbor"]["maximum_received_routes"][
+ "warning_limit"
+ ]["limit_percent"]
+ )
+ + " percent"
+ )
+ if config_data["neighbor"]["maximum_received_routes"].get(
+ "warning_only"
+ ):
+ command += " warning-only"
+ elif config_data["neighbor"].get("metric_out"):
+ command += " metric-out {metric_out}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("monitoring"):
+ command += " monitoring"
+ elif config_data["neighbor"].get("next_hop_self"):
+ command += " next-hop-self"
+ elif config_data["neighbor"].get("next_hop_unchanged"):
+ command += " next-hop-unchanged"
+ elif config_data["neighbor"].get("next_hop_v6_addr"):
+ command += " next-hop-v6-addt {next_hop_v6_addr} in".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("out_delay"):
+ command += " out-delay {out_delay}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("remote_as"):
+ command += " remote-as {remote_as}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("remove_private_as"):
+ command += " remove-private-as"
+ if config_data["neighbor"]["remove_private_as"].get("all"):
+ command += " all"
+ if config_data["neighbor"]["remove_private_as"].get("replace_as"):
+ command += " replace-as"
+ elif config_data["neighbor"].get("peer_as"):
+ command += " peer-as {peer_as}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("prefix_list"):
+ command += " prefix-list {name} {direction}".format(
+ **config_data["neighbor"]["prefix_list"]
+ )
+ elif config_data["neighbor"].get("route_map"):
+ command += " route-map {name} {direction}".format(
+ **config_data["neighbor"]["route_map"]
+ )
+ elif config_data["neighbor"].get("route_reflector_client"):
+ command += " route-reflector-client"
+ elif config_data["neighbor"].get("route_to_peer"):
+ command += " route-to-peer"
+ elif config_data["neighbor"].get("send_community"):
+ command += " send-community"
+ if config_data["neighbor"]["send_community"].get(
+ "community_attribute"
+ ):
+ command += (
+ " "
+ + config_data["neighbor"]["send_community"][
+ "community_attribute"
+ ]
+ )
+ if config_data["neighbor"]["send_community"].get("sub_attribute"):
+ command += (
+ " "
+ + config_data["neighbor"]["send_community"]["sub_attribute"]
+ )
+ if config_data["neighbor"]["send_community"].get(
+ "link_bandwidth_attribute"
+ ):
+ command += (
+ " "
+ + config_data["neighbor"]["send_community"][
+ "link_bandwidth_attribute"
+ ]
+ )
+ if config_data["neighbor"]["send_community"].get("speed"):
+ command += " " + config_data["neighbor"]["send_community"]["speed"]
+ if config_data["neighbor"]["send_community"].get("divide"):
+ command += (
+ " " + config_data["neighbor"]["send_community"]["divide"]
+ )
+ elif config_data["neighbor"].get("shutdown"):
+ command += " shutdown"
+ elif config_data["neighbor"].get("soft_reconfiguration"):
+ command += " soft-reconfiguration inbound"
+ if config_data["neighbor"]["soft_reconfiguration"] == "all":
+ command += " all"
+ elif config_data["neighbor"].get("transport"):
+ command += " transport"
+ if config_data["neighbor"]["transport"].get("connection_mode"):
+ command += " connection-mode passive"
+ else:
+ command += " remote-port {remote_port}".format(
+ **config_data["neighbor"]["transport"]
+ )
+ elif config_data["neighbor"].get("timers"):
+ command += " timers {keepalive} {holdtime}".format(
+ **config_data["neighbor"]["timers"]
+ )
+ elif config_data["neighbor"].get("ttl"):
+ command += " ttl maximum-hops {ttl}".format(**config_data["neighbor"])
+ elif config_data["neighbor"].get("update_source"):
+ command += " update-source {update_source}".format(
+ **config_data["neighbor"]
+ )
+ elif config_data["neighbor"].get("weight"):
+ command += " weight {weight}".format(**config_data["neighbor"])
+ return command
+
+
+def _tmplt_bgp_network(config_data):
+ command = "network {address}".format(**config_data)
+ if config_data.get("route_map"):
+ command += " route-map {route_map}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_route_target(config_data):
+ command = "route-target {action} {target}".format(
+ **config_data["route_target"]
+ )
+ return command
+
+
+def _tmplt_bgp_router_id(config_data):
+ command = "router-id {router_id}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_shutdown(config_data):
+ return "shutdown"
+
+
+def _tmplt_bgp_timers(config_data):
+ command = "timers bgp {keepalive} {holdtime}".format(
+ **config_data["timers"]
+ )
+ return command
+
+
+def _tmplt_bgp_ucmp(config_data):
+ command = "ucmp"
+ if "fec" in config_data["ucmp"]:
+ command += " fec threshold trigger"
+ command += " {trigger} clear {clear} warning-only".format(
+ **config_data["ucmp"]["fec"]
+ )
+ if "link_bandwidth" in config_data["ucmp"]:
+ command += " link-bandwidth {mode}".format(
+ **config_data["ucmp"]["link_bandwidth"]
+ )
+ if config_data["ucmp"]["link_bandwidth"].get("mode") == "update_delay":
+ command += " {update_delay}".format(
+ **config_data["ucmp"]["link_bandwidth"]
+ )
+ if "mode" in config_data["ucmp"]:
+ command += " mode 1"
+ if config_data["ucmp"]["mode"].get("nexthops"):
+ command += " {nexthops}".format(**config_data["ucmp"]["mode"])
+ return command
+
+
+def _tmplt_bgp_update(config_data):
+ command = "update {wait_for}".format(**config_data["update"])
+ if config_data["update"].get("batch_size"):
+ command += " {batch_size}".format(**config_data["update"])
+ return command
+
+
+def _tmplt_bgp_vlan(config_data):
+ command = "vlan {vlan}".format(**config_data)
+ return command
+
+
+def _tmplt_bgp_vlan_aware_bundle(config_data):
+ command = "vlan-aware-bundle " + config_data["vlan_aware_bundle"]
+ return command
+
+
+class Bgp_globalTemplate(NetworkTemplate):
+ def __init__(self, lines=None):
+ super(Bgp_globalTemplate, self).__init__(lines=lines, tmplt=self)
+
+ # fmt: off
+ PARSERS = [
+ {
+ "name": "router",
+ "getval": re.compile(
+ r"""
+ ^router\s
+ bgp
+ \s(?P<as_num>\S+)
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_router_bgp_cmd,
+ "compval": "as_number",
+ "result": {"as_number": "{{ as_num }}"},
+ },
+ {
+ "name": "vrf",
+ "getval": re.compile(
+ r"""
+ \s*vrf
+ \s(?P<vrf>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_vrf,
+ "compval": "vrfs.vrf",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "vrf": "{{ vrf }}"
+ }
+ }
+ },
+ "shared": True,
+ },
+ {
+ "name": "aggregate_address",
+ "getval": re.compile(
+ r"""
+ \s*aggregate-address
+ \s+(?P<address>\S+)
+ \s*(?P<as_set>as-set)*
+ \s*(?P<summary_only>summary-only)*
+ \s*(?P<attribute_map>attribute-map\s\S+)*
+ \s*(?P<match_map>match-map\s\S+)*
+ \s*(?P<advertise_only>advertise-only)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_aggregate_address,
+ "compval": "aggregate_address",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "aggregate_address": [
+ {
+ "address": "{{ address }}",
+ "advertise_only": "{{ True if advertise_only is defined }}",
+ "as_set": "{{ True if as_set is defined }}",
+ "summary_only": "{{ True if summary_only is defined }}",
+ "attribute_map": "{{ attribute_map.split(" ")[1] }}",
+ "match_map": "{{ match_map.split(" ")[1] }}"
+ }
+ ]
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_additional_paths",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+additional-paths
+ \s+(?P<action>\S+)
+ \s*(any)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.additional_paths",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "additional_paths": "{{ action }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_advertise_inactive",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+advertise-inactive
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.advertise_inactive",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "advertise_inactive": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_allowas_in",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+allowas-in
+ \s*(?P<count>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.allowas_in",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "allowas_in": {
+ "set": "{{ True if count is undefined }}",
+ "count": "{{ count }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_always_compare_med",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+always-compare-med
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.always_compare_med",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "always_compare_med": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_asn",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+asn
+ \s+notation
+ \s+(?P<notation>asdot|asplain)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.asn",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "asn": "{{ notation }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_auto_local_addr",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+auto-local-addr
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.auto_local_addr",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "auto_local_addr": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_bestpath_as_path",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+bestpath
+ \s*(?P<as_path>as-path\s\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.bestpath",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "bestpath": {
+ "as_path": "{{ as_path.split(" ")[1] }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_bestpath_ecmp_fast",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+bestpath
+ \s+ecmp-fast
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.bestpath",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "bestpath": {
+ "ecmp_fast": "{{ True }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_bestpath_med",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+bestpath
+ \s+med
+ \s*(?P<confed>confed)*
+ \s*(?P<missing>missing-as-worst)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.bestpath",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "med": {
+ "confed": "{{ True if confed is defined }}",
+ "missing_as_worst": "{{ True if missing is defined }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_bestpath_skip",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+bestpath
+ \s+(?P<skip>skip)
+ \s+next-hop
+ \s+igp-cost
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.bestpath",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "bestpath": {
+ "skip": "{{ True }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_tie_break",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+tie-break
+ \s+(?P<tie>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.bestpath",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "tie_break": "{{ tie }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_client_to_client",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+client-to-client
+ \s+reflection
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.client_to_client",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "client_to_client": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_cluster_id",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+cluster-id
+ \s+(?P<address>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.cluster_id",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "cluster_id": "{{ address }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_confederation",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+confederation
+ \s*(?P<identifier>identifier\s.+)*
+ \s*(?P<peers>peers\s.+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.confederation",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "confederation": {
+ "identifier": "{{ identifier.split(" ")[1] }}",
+ "peers": "{{ peers.split(" ")[1] }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_control_plane_filter",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+control-plane-filter
+ \s+default-allow
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.control_plane_filter",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "control_plane_filter": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_convergence",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+convergence
+ \s*(?P<slow>slow-peer)*
+ \s+(?P<time>time\s\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.convergence",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "convergence": {
+ "slow_peer": "{{ True if slow is defined else False}}",
+ "time": "{{ time.split(" ")[1] }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params_default",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+default
+ \s(?P<param>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.default",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "default": "{{ param }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.enforce_first_as",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+enforce-first-as
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.enforce_first_as",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "enforce_first_as": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.host_routes",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+host-routes
+ \s+fib
+ \s+direct-install
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.host_routes",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "host_routes": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.labelled_unicast",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+labeled-unicast
+ \s+rib
+ \s+(?P<rib>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.labelled_unicast",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "labeled_unicast": "{{ rib }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.listen_limit",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+listen
+ \s+limit
+ \s+(?P<limit>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.listen.limit",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "listen": {
+ "limit": "{{ limit }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.listen_range",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+listen
+ \s+range
+ \s+(?P<address>\S+)
+ \s+peer-group
+ \s+(?P<group>\S+)
+ \s*(?P<filter>peer-filter \S+)*
+ \s*(?P<remote_as>remote-as \S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.listen.range",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "listen": {
+ "range": {
+ "address": "{{ address }}",
+ "peer_group": {
+ "name": "{{ group }}",
+ "peer_filter": "{{ filter }}",
+ "remote_as": "{{ remote_as }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.log_neighbor_changes",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+log-neighbor-changes
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.log_neighbor_changes",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "log_neighbor_changes": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.missing_policy",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+missing-policy
+ \s+direction
+ \s+(?P<dir>in|out)
+ \s+action
+ \s+(?P<action>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.missing_policy",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "missing_policy": {
+ "direction": "{{ dir }}",
+ "action": "{{ action }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.monitoring",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+monitoring
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.monitoring",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "monitoring": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.nexthop_unchanged",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+next-hop-unchanged
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.next_hop_unchanged",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "next_hop_unchanged": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.redistribute_internal",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+redistribute-internal
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.redistribute_internal",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "redistribute_internal": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.route",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+route
+ \s+install-map
+ \s+(?P<route>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.route",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "route": "{{ route }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.route_reflector",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+route-reflector
+ \s+preserve-attributes
+ \s*(?P<preserve>always)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.route_reflector",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "route_reflector": {
+ "set": "{{ True if presever is undefined }}",
+ "preserve": "{{ True if preserve is defined }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bgp_params.transport",
+ "getval": re.compile(
+ r"""
+ \s*bgp
+ \s+transport
+ \s+listen-port
+ \s+(?P<port>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_params,
+ "compval": "bgp_params.transport",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "bgp_params": {
+ "transport": "{{ port }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "default_metric",
+ "getval": re.compile(
+ r"""
+ \s*default-metric
+ \s(?P<metric>\d+)
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_default_metric,
+ "compval": "default_metric",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "default_metric": "{{ metric }}"
+ }
+ }
+ },
+ },
+ {
+ "name": "redistribute",
+ "getval": re.compile(
+ r"""
+ \s*redistribute
+ \s+(?P<route>\S+)
+ \s*(?P<level>level-1|level-2|level-1-2)*
+ \s*(?P<match>match\s.+)*
+ \s*(?P<route_map>route-map\s\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_redistribute,
+ "compval": "redistribute",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "redistribute": [
+ {
+ "protocol": "{{ route }}",
+ "route_map": "{{ route_map.split(" ")[1] }}",
+ "isis_level": "{{ level }}",
+ "ospf_route": "{{ 'nssa_external_' + match.split(" ")[2] if match.split(" ")[1] == 'nssa-external' else match.split(" ")[1]}}"
+ }
+ ]
+ }
+ }
+ },
+ },
+ {
+ "name": "distance",
+ "getval": re.compile(
+ r"""
+ \s*distance
+ \s+bgp
+ \s(?P<external>\d+)
+ \s*(?P<internal>\d+)*
+ \s*(?P<local>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_distance,
+ "compval": "distance",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "distance": {
+ "external": "{{ external }}",
+ "internal": "{{ internal }}",
+ "local": "{{ local }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart",
+ "getval": re.compile(
+ r"""
+ \s*graceful-restart
+ \s*(?P<restart_time>restart-time\s\d+)*
+ \s*(?P<stalepath_time>stalepath-time\s\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_graceful_restart,
+ "remval": "graceful-restart",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "graceful_restart": {
+ "set": "{{ True if restart_time and stalepath_time is not defined }}",
+ "restart_time": "{{ restart_time.split(" ")[1]|int }}",
+ "stalepath_time": "{{ stalepath_time.split(" ")[1]|int }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart_helper",
+ "getval": re.compile(
+ r"""
+ \s*graceful-restart-helper
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_graceful_restart_helper,
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "graceful_restart": {
+ "set": "{{ True }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "acccess_group",
+ "getval": re.compile(
+ r"""
+ \s*(?P<afi>ip|ipv6)
+ \s+access-group
+ \s+(?P<acl_name>\S+)
+ \s*(?P<direction>in)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_access_group,
+ "compval": "access_group",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "access_group": {
+ "afi": "{{ ipv4 if afi == 'ip' else afi }}",
+ "acl_name": "{{ acl_name }}",
+ "direction": "{{ direction }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "maximum_paths",
+ "getval": re.compile(
+ r"""
+ \s*maximum-paths
+ \s+(?P<max_equal_cost_paths>\d+)
+ \s*(ecmp)*
+ \s*(?P<max_installed_ecmp_paths>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_maximum_paths,
+ "compval": "maximum_paths",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "maximum_paths": {
+ "max_equal_cost_paths": "{{ max_equal_cost_paths }}",
+ "max_installed_ecmp_paths": "{{ max_installed_ecmp_paths }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "monitoring",
+ "getval": re.compile(
+ r"""
+ \s*monitoring
+ \s+(?P<port>\d+)
+ \s*(?P<received>received\sroutes\s\S+)*
+ \s*(?P<time>timestamp\s\S+)*
+ \s*(?P<station>station\s\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_monitoring,
+ "compval": "monitoring",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "monitoring": {
+ "port": "{{ port }}",
+ "received": "{{ received.split(" ")[2] }}",
+ "timestamp": "{{ timestamp.split(" ")[1] }}",
+ "station": "{{ station.split(" ")[1] }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.additional_paths",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+additional-paths
+ \s+(?P<action>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.additional_paths",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "additional_paths": "{{ action }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.allowas_in",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+allowas-in
+ \s*(?P<count>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.allowas_in",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "allowas_in": {
+ "set": "{{ True if count is undefined }}",
+ "count": "{{ count }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.auto_local_addr",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+auto-local-addr
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.auto_local_addr",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "auto_local_addr": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.default_originate",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+default-originate
+ \s*(?P<route_map>route-map\s\S+)*
+ \s*(?P<always>always)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.default_originate",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "default_originate": {
+ "route_map": "{{ route_map.split(" ")[1] }}",
+ "always": "{{ True if always is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.description",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+description
+ \s+(?P<desc>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.description",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "description": "{{ desc }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.dont_capability_negotiate",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+dont-capability-negotiate
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.dont_capability_negotiate",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "dont_capability_negotiate": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.ebgp_multihop",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+ebgp-multihop
+ \s*(?P<ttl>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.ebgp_multihop",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "ebgp_multihop": {
+ "set": "{{ True if ttl is not set }}",
+ "ttl": "{{ ttl }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.enforce_first_as",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+enforce-first-as
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.enforce_first_as",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "enforce_first_as": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.export_localpref",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+export-localpref
+ \s+(?P<pref>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.export_localpref",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "export_localpref": "{{ pref }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.fall_over",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+fall-over
+ \s+bfd
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.fall_over",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "fall_over": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.graceful_restart",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+graceful-restart
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "remval": "neighbor {{ peer }} graceful-restart",
+ "compval": "neighbor.graceful_restart",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "graceful_restart": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.graceful_restart_helper",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+graceful-restart-helper
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.graceful_restart_helper",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "graceful_restart_helper": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.idle_restart_timer",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+idle-restart-timer
+ \s+(?P<time>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.idle_restart_timer",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "idle_restart_timer": "{{ time }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.import_localpref",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+import-localpref
+ \s+(?P<pref>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.import_localpref",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "import_localpref": "{{ pref }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.link_bandwidth",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+link-bandwidth
+ \s*(?P<auto>auto)*
+ \s*(?P<default>default\s\S+)*
+ \s*(?P<update>update-delay\s\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.link_bandwidth",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "link_bandwidth": {
+ "set": "{{ True }}",
+ "auto": "{{ True if auto is defined }}",
+ "default": "{{ default.split(" ")[1] if default is defined }}",
+ "update_delay": "{{ update.split(" ")[1] if update is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.local_as",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+local-as
+ \s+(?P<num>\S+)
+ \s+no-prepend
+ \s+replace-as
+ \s*(?P<fallback>fallback)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.local_as",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "local_as": {
+ "as_number": "{{ num }}",
+ "fallback": "{{ True if fallback is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.local_v6_addr",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+local-v6-addr
+ \s+(?P<addr>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.local_v6_addr",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "local_v6_addr": "{{ addr }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.maximum_accepted_routes",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+maximum-accepted-routes
+ \s+(?P<count>\d+)
+ \s*warning-limit*
+ \s*(?P<limit>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.maximum_accepted_routes",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "maximum_accepted_routes": {
+ "count": "{{ count }}",
+ "warning_limit": "{{ limit }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.maximum_received_routes",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+maximum-routes
+ \s+(?P<count>\d+)*
+ \s*(warning-limit)*
+ \s*(?P<limit>\d+)*
+ \s*(?P<percent>percent)*
+ \s*(?P<warning_only>warning-only)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.maximum_received_routes",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "maximum_received_routes": {
+ "count": "{{ count }}",
+ "warning_limit": {
+ "limit_count": "{{ limit if percent is undefined }}",
+ "limit_percent": "{{ limit if percent is defined }}"
+ },
+ "warning_only": "{{ True if warning_only is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.metric_out",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+metric-out
+ \s+(?P<metric>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.metric_out",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "metric_out": "{{ metric }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.monitoring",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+monitoring
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.monitoring",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "monitoring": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.next_hop_self",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+next-hop-self
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.next_hop_self",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "next_hop_self": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.next_hop_unchanged",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+next-hop-unchanged
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.next_hop_unchanged",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "next_hop_unchanged": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.next_hop_v6_addr",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+next-hop-v6-addr
+ \s+(?P<addr>\S+)
+ \s+in
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.next_hop_v6_addr",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "next_hop_v6_addr": "{{ addr }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.out_delay",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+out-delay
+ \s+(?P<delay>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.out_delay",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "out_delay": "{{ delay }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.remote_as",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+remote-as
+ \s+(?P<num>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.remote_as",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "remote_as": "{{ num }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.remove_private_as",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+remove-private-as
+ \s*(?P<all>all)*
+ \s*(?P<replace>replace-as)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.remove_private_as",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "remove_private_as": {
+ "set": "{{ True if all is undefined and replace is undefined }}",
+ "all": "{{ True if all is defined }}",
+ "replace_as": "{{ True if replace is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.peer_group",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+peer-group
+ \s*(?P<name>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.peer_group",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "peer_group": "{{ name if name is defined else peer}}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.prefix_list",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+prefix-list
+ \s+(?P<name>\S+)
+ \s+(?P<dir>in|out)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.prefix_list",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "prefix_list": {
+ "name": "{{ name }}",
+ "direction": "{{ dir }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.route_map",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+route-map
+ \s+(?P<name>\S+)
+ \s+(?P<dir>in|out)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.route_map",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "route_map": {
+ "name": "{{ name }}",
+ "direction": "{{ dir }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.route_reflector_client",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+route-reflector-client
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.route_reflector_client",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "route_reflector_client": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.route_to_peer",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+route-to-peer
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.route_to_peer",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "route_to_peer": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.send_community",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+send-community
+ \s+(?P<comm>add|extended|link-bandwidth|remove|standard)*
+ \s*(?P<attr>extended|link-bandwidth|standard)*
+ \s*(?P<link>aggregate|divide)*
+ \s*(?P<div>equal|ratio)*
+ \s*(?P<speed>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.send_community",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "send_community": {
+ "set": "{{ True if comm is not defined }}",
+ "community_attribute": "{{ comm }}",
+ "sub_attribute": "{{ attr }}",
+ "link_bandwidth_attribute": "{{ link }}",
+ "speed": "{{ speed }}",
+ "divide": "{{ div }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.shutdown",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+shutdown
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.shutdown",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "shutdown": "{{ True }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.soft_reconfiguration",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+soft-reconfiguration
+ \s+inbound
+ \s*(?P<all>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.soft_reconfiguration",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "soft_reconfiguration": "{{ 'all' if all is defined else 'None' }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.transport",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+transport
+ \s+(?P<mode>\S+)
+ \s*(passive)*
+ \s*(?P<port>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.transport`",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "transport": {
+ "connection_mode": "{{ mode }}",
+ "remote_port": "{{ port if port is defined }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.timers",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+timers
+ \s+(?P<keepalive>\d+)
+ \s+(?P<hold>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.timers",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "timers": {
+ "keepalive": "{{ keepalive }}",
+ "holdtime": "{{ holdtime }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.ttl",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+ttl
+ \s+maximum-hops
+ \s+(?P<hop>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.ttl",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "ttl": "{{ hop }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.update_source",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+update-source
+ \s+(?P<src>.*)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.update_source",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "update_source": "{{ src }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "neighbor.weight",
+ "getval": re.compile(
+ r"""
+ \s*neighbor
+ \s+(?P<peer>\S+)
+ \s+weight
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_neighbor,
+ "compval": "neighbor.weight",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "neighbor": {
+ "{{ peer }}": {
+ "peer": "{{ peer }}",
+ "weight": "{{ val }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "network",
+ "getval": re.compile(
+ r"""
+ \s*network
+ \s+(?P<address>\S+)
+ \s*(route-map)*
+ \s*(?P<route_map>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_network,
+ "compval": "network",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "network": {
+ "{{ address }}": {
+ "address": "{{ address }}",
+ "route_map": "{{ route_map }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "route_target",
+ "getval": re.compile(
+ r"""
+ \s*route-target
+ \s+(?P<action>\S+)
+ \s+(?P<target>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_route_target,
+ "compval": "route_target",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "route_target": {
+ "action": "{{ action }}",
+ "target": "{{ target }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "router_id",
+ "getval": re.compile(
+ r"""
+ \s*router-id
+ \s+(?P<id>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_router_id,
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "router_id": "{{ id }}"
+ }
+ }
+ },
+ },
+ {
+ "name": "shutdown",
+ "getval": re.compile(
+ r"""
+ \s*shutdown
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_shutdown,
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "shutdown": "{{ True }}"
+ }
+ }
+ },
+ },
+ {
+ "name": "timers",
+ "getval": re.compile(
+ r"""
+ \s*timers
+ \s+bgp
+ \s+(?P<keepalive>\d+)
+ \s+(?P<holdtime>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_timers,
+ "compval": "timers",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "timers": {
+ "keepalive": "{{ keepalive }}",
+ "holdtime": "{{ holdtime }}",
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "ucmp_fec",
+ "getval": re.compile(
+ r"""
+ \s*ucmp
+ \s+fec
+ \s+threshold
+ \s+(?P<trigger>trigger \d+)
+ \s+(?P<clear>clear \d+)
+ \s+warning-only
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_ucmp,
+ "compval": "ucmp.fec",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "ucmp": {
+ "fec": {
+ "trigger": "{{ trigger }}",
+ "clear": "{{ clear }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "ucmp_link_bandwidth",
+ "getval": re.compile(
+ r"""
+ \s*ucmp
+ \s+link-bandwidth
+ \s*(?P<ucmp_mode>recursive|encoding-weighted)
+ \s*(?P<update_delay>update-delay \d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_ucmp,
+ "compval": "ucmp.link_bandwidth",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "ucmp": {
+ "link_bandwidth": {
+ "mode": "{{ ucmp_mode }}",
+ "update_delay": "{{ update_delay.split(" ")[1] }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "ucmp_mode",
+ "getval": re.compile(
+ r"""
+ \s*ucmp
+ \s+mode
+ \s+(?P<ucmp_set>\d+)
+ \s*(?P<nexthop>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_ucmp,
+ "compval": "ucmp.mode",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "ucmp": {
+ "mode": {
+ "set": "{{ True if ucmp_set == '1'}}",
+ "nexthops": "{{ nexthop }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "update",
+ "getval": re.compile(
+ r"""
+ \s*update
+ \s+(?P<wait>\S+)
+ \s*(?P<size>batch-size\s\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_update,
+ "compval": "update",
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "update": {
+ "wait_for": "{{ wait }}",
+ "batch_size": "{{ size.split(" ")[1] }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "vlan",
+ "getval": re.compile(
+ r"""
+ \s*vlan
+ \s+(?P<id>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_vlan,
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "vlan": "{{ id }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "vlan_aware_bundle",
+ "getval": re.compile(
+ r"""
+ \s*vlan-aware-bundle
+ \s+(?P<bundle>.+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_bgp_vlan_aware_bundle,
+ "result": {
+ "vrfs": {
+ '{{ "vrf_" + vrf|d() }}': {
+ "vlan_aware_bundle": "{{ bundle }}",
+ }
+ }
+ },
+ },
+ ]
+ # fmt: on
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py
new file mode 100644
index 00000000..bd3e5774
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py
@@ -0,0 +1,1082 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The Ospf_interfaces parser templates file. This contains
+a list of parser definitions and associated functions that
+facilitates both facts gathering and native command generation for
+the given network resource.
+"""
+
+import re
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import (
+ NetworkTemplate,
+)
+
+
+def _tmplt_ospf_int_authentication(config_data):
+ if "authentication_v2" in config_data:
+ command = "ip ospf authentication"
+ if "message_digest" in config_data["authentication_v2"]:
+ command += " message-digest"
+ return command
+ if "authentication_v3" in config_data:
+ command = "ospfv3 authentication ipsec spi "
+ command += "{spi} {algorithm}".format(
+ **config_data["authentication_v3"]
+ )
+ if "passphrase" in config_data["authentication_v3"]:
+ command += " passphrase"
+ if "keytype" in config_data["authentication_v3"]:
+ command += " {keytype}".format(**config_data["authentication_v3"])
+ if "passphrase" not in config_data["authentication_v3"]:
+ command += " {key}".format(**config_data["authentication_v3"])
+ else:
+ command += " {passphrase}".format(
+ **config_data["authentication_v3"]
+ )
+ return command
+
+
+def _tmplt_ospf_int_encryption_v3(config_data):
+ if "encryption" in config_data:
+ command = "ospfv3 encryption ipsec spi ".format(**config_data)
+ command += "{spi} esp {encryption} {algorithm}".format(
+ **config_data["encryption"]
+ )
+ if "passphrase" in config_data["encryption"]:
+ command += " passphrase"
+ if "keytype" in config_data["encryption"]:
+ command += " {keytype}".format(**config_data["encryption"])
+ if "passphrase" not in config_data["encryption"]:
+ command += " {key}".format(**config_data["encryption"])
+ else:
+ command += " {passphrase}".format(**config_data["encryption"])
+ return command
+
+
+def _tmplt_ospf_int_authentication_key(config_data):
+ if "authentication_key" in config_data:
+ command = "ip ospf authentication-key"
+ if "encryption" in config_data["authentication_key"]:
+ command += " {encryption} {key}".format(
+ **config_data["authentication_key"]
+ )
+ else:
+ command += " {key}".format(**config_data["authentication_key"])
+ return command
+
+
+def _tmplt_ospf_int_cost(config_data):
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf cost {cost}".format(**config_data)
+ else:
+ command = "ospfv3 cost {cost}".format(**config_data)
+ return command
+
+
+def _tmplt_ospf_int_bfd(config_data):
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf bfd"
+ else:
+ command = "ospfv3 bfd"
+ return command
+
+
+def _tmplt_ospf_int_hello_interval(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} hello-interval {hello_interval}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf hello-interval {hello_interval}".format(
+ **config_data
+ )
+ else:
+ command = "ospfv3 hello-interval {hello_interval}".format(
+ **config_data
+ )
+ return command
+
+
+def _tmplt_ospf_int_mtu_ignore(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} mtu-ignore".format(**config_data["ip_params"])
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf mtu-ignore"
+ else:
+ command = "ospfv3 mtu-ignore"
+ return command
+
+
+def _tmplt_ospf_int_network(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} network {network}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf network {network}".format(**config_data)
+ else:
+ command = "ospfv3 network {network}".format(**config_data)
+ return command
+
+
+def _tmplt_ospf_int_priority(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} priority {priority}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf priority {priority}".format(**config_data)
+ else:
+ command = "ospfv3 priority {priority}".format(**config_data)
+ return command
+
+
+def _tmplt_ospf_int_retransmit_interval(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} retransmit-interval {retransmit_interval}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf retransmit-interval {retransmit_interval}".format(
+ **config_data
+ )
+ else:
+ command = "ospfv3 retransmit-interval {retransmit_interval}".format(
+ **config_data
+ )
+ return command
+
+
+def _tmplt_ospf_int_transmit_delay(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} transmit-delay {transmit_delay}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf transmit-delay {transmit_delay}".format(
+ **config_data
+ )
+ else:
+ command = "ospfv3 transmit-delay {transmit_delay}".format(
+ **config_data
+ )
+ return command
+
+
+def _tmplt_ospf_int_dead_interval(config_data):
+ if "ip_params" in config_data:
+ command = "ospfv3 {afi} dead-interval {dead_interval}".format(
+ **config_data["ip_params"]
+ )
+ else:
+ if config_data["afi"] == "ipv4":
+ command = "ip ospf dead-interval {dead_interval}".format(
+ **config_data
+ )
+ else:
+ command = "ospfv3 dead-interval {dead_interval}".format(
+ **config_data
+ )
+ return command
+
+
+class Ospf_interfacesTemplate(NetworkTemplate):
+ def __init__(self, lines=None):
+ super(Ospf_interfacesTemplate, self).__init__(lines=lines, tmplt=self)
+
+ PARSERS = [
+ {
+ "name": "interfaces",
+ "getval": re.compile(
+ r"""
+ ^interface
+ \s+(?P<name>\S+)
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": "interface {{ name }}",
+ "result": {"name": "{{ name }}"},
+ "shared": True,
+ },
+ {
+ "name": "area",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+area
+ \s+(?P<area_id>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": "ip ospf area {{ area.area_id }}",
+ "compval": "area",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "area": {"area_id": "{{ area_id }}"},
+ }
+ }
+ },
+ },
+ {
+ "name": "authentication_v2",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+authentication
+ \s*(?P<message_digest>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_authentication,
+ "compval": "authentication_v2",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "authentication_v2": {
+ "set": "{{ True if message_digest is undefined }}",
+ "message_digest": "{{ True if message_digest is defined }}",
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "authentication_v3",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+authentication
+ \s+ipsec
+ \s+spi
+ \s+(?P<val>\d+)
+ \s+(?P<algorithm>md5|sha1)
+ \s*(?P<passphrase>passphrase)*
+ \s*(?P<type>0|7)*
+ \s*(?P<line>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_authentication,
+ "compval": "authentication_v3",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "authentication_v3": {
+ "spi": "{{ val }}",
+ "algorithm": "{{ algorithm }}",
+ "keytype": "{{ type }}",
+ "passphrase": "{{ line if passphrase is defined }}",
+ "key": "{{ str(line) if passphrase is undefined }}",
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "authentication_key",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+authentication-key
+ \s*(?P<encryption>\d+)*
+ \s*(?P<line>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_authentication_key,
+ "compval": "authentication_key",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "authentication_key": {
+ "encryption": "{{ encryption }}",
+ "key": "{{ line }}",
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "bfd",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+bfd
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_bfd,
+ "compval": "bfd",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "bfd": "{{ True }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "deadinterval",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+dead-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_dead_interval,
+ "compval": "dead_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "dead_interval": "{{ interval }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "encryption",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+encryption
+ \s+ipsec
+ \s+spi
+ \s+(?P<val>\d+)
+ \s+esp
+ \s+(?P<encryption>\S+)
+ \s*(?P<algorithm>md5|sha1)
+ \s*(?P<passphrase>passphrase)
+ \s*(?P<type>0|7)
+ \s*(?P<line>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_encryption_v3,
+ "compval": "encryption_v3",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "encryption_v3": {
+ "spi": "{{ val }}",
+ "encryption": "{{ encryption }}",
+ "algorithm": "{{ algorithm }}",
+ "keytype": "{{ type }}",
+ "passphrase": "{{ line if passphrase is defined }}",
+ "key": "{{ str(line) if passphrase is undefined }}",
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "hellointerval",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+hello-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_hello_interval,
+ "compval": "hello_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "hello_interval": "{{ interval }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "bfd",
+ "getval": re.compile(
+ r"""
+ \s+ospfv3
+ \s+bfd
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_bfd,
+ "compval": "bfd",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "bfd": "{{ True }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "cost",
+ "getval": re.compile(
+ r"""
+ \s+ip
+ \s+ospf
+ \s+cost
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_cost,
+ "compval": "cost",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "cost": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "cost",
+ "getval": re.compile(
+ r"""
+ \s+ospfv3
+ \s+cost
+ \s+(?P<cost>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_cost,
+ "compval": "cost",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "cost": "{{ cost }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "deadinterval",
+ "getval": re.compile(
+ r"""
+ \s+ospfv3
+ \s+dead-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_dead_interval,
+ "compval": "dead_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "dead_interval": "{{ interval }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "hellointerval",
+ "getval": re.compile(
+ r"""
+ \s+ospfv3
+ \s+hello-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_hello_interval,
+ "compval": "hello_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "hello_interval": "{{ interval }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_area",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+area
+ \s+(?P<area_id>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 {{ ip_params.afi }} area {{ ip_params.area.area_id }}",
+ "compval": "ip_params.area",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "area": {"area_id": "{{ area_id }}"},
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_bfd",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+bfd
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 {{ afi }} bfd",
+ "compval": "ip_params.bfd",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "bfd": "{{ True }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_cost",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+cost
+ \s+(?P<cost>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 {{ afi }} cost {{ cost }}",
+ "compval": "ip_params.cost",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "cost": "{{ cost }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_dead_interval",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+dead-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_dead_interval,
+ "compval": "ip_params.dead_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "dead_interval": "{{ interval }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_hello_interval",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+hello-interval
+ \s+(?P<interval>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 {{ ip_params.afi }} hello-interval {{ ip_params.hello_interval }}",
+ "compval": "ip_params.hello_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "hello_interval": "{{ interval }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_mtu_ignore",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+(?P<mtu>mtu-ignore)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_mtu_ignore,
+ "compval": "ip_params.mtu_ignore",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "mtu_ignore": "{{ True if mtu is defined}}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_network",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+network
+ \s+(?P<val>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_network,
+ "compval": "ip_params.network",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "network": "{{ val }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_priority",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+priority
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_priority,
+ "compval": "ip_params.priority",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "priority": "{{ val }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_passive_interface",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+passive-interface
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 {{ ip_params.afi }} passive-interface",
+ "compval": "ip_params.passive_interface",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "passive_interface": "{{ True }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_retransmit_interval",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+retransmit-interval
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_retransmit_interval,
+ "compval": "ip_params.retransmit_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "retransmit_interval": "{{ val }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "ip_params_transmit_delay",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+(?P<afi>ipv4|ipv6)
+ \s+transmit-delay
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_transmit_delay,
+ "compval": "ip_params.transmit_delay",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "ip_params": {
+ "{{ afi }}": {
+ "afi": "{{ afi }}",
+ "transmit_delay": "{{ val }}",
+ }
+ },
+ }
+ }
+ },
+ },
+ {
+ "name": "mtu_ignore",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+mtu-ignore
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_mtu_ignore,
+ "compval": "mtu_ignore",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "mtu_ignore": "{{ True }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "network",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+network
+ \s+(?P<interface>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_network,
+ "compval": "network",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "network": "{{ interface }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "priority",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+priority
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_priority,
+ "compval": "priority",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "priority": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "passive_interface",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+passive-interface
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ospfv3 passive-interface",
+ "compval": "passive_interface",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "passive_interface": "{{ True }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "retransmit_interval",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+retransmit-interval
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_retransmit_interval,
+ "compval": "retransmit_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "retransmit_interval": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "transmit_delay",
+ "getval": re.compile(
+ r"""
+ \s*ospfv3
+ \s+transmit-delay
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_transmit_delay,
+ "compval": "transmit_delay",
+ "result": {
+ "address_family": {
+ "{{ 'ipv6' }}": {
+ "afi": '{{ "ipv6" }}',
+ "transmit_delay": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "mtu_ignore",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+mtu-ignore
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_mtu_ignore,
+ "compval": "mtu_ignore",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "mtu_ignore": "{{ True }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "network",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+network
+ \s+(?P<interface>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_network,
+ "compval": "network",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "network": "{{ interface }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "priority",
+ "getval": re.compile(
+ r"""
+ \s*ip ospf
+ \s+priority
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_priority,
+ "compval": "priority",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "priority": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "retransmit_interval",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+retransmit-interval
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_retransmit_interval,
+ "compval": "retransmit_interval",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "retransmit_interval": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "transmit_delay",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+transmit-delay
+ \s+(?P<val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_int_transmit_delay,
+ "compval": "transmit_delay",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "transmit_delay": "{{ val }}",
+ }
+ }
+ },
+ },
+ {
+ "name": "message_digest_key",
+ "getval": re.compile(
+ r"""
+ \s*ip
+ \s+ospf
+ \s+message-digest-key
+ \s+(?P<id>\d+)
+ \s+md5
+ \s*(?P<type>0|7)*
+ \s+(?P<line>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "ip ospf message-digest-key {{ message_digest_key.key_id }} md5 {{ message_digest_key.encryption }} {{ message_digest_key.key }}",
+ "compval": "message_digest_key",
+ "result": {
+ "address_family": {
+ "{{ 'ipv4' }}": {
+ "afi": '{{ "ipv4" }}',
+ "message_digest_key": {
+ "key_id": "{{ id }}",
+ "encryption": "{{ type }}",
+ "key": "{{ line }}",
+ },
+ }
+ }
+ },
+ },
+ ]
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py
new file mode 100644
index 00000000..857521b8
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py
@@ -0,0 +1,1055 @@
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+"""
+The Ospfv3 parser templates file. This contains
+a list of parser definitions and associated functions that
+facilitates both facts gathering and native command generation for
+the given network resource.
+"""
+
+import re
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import (
+ NetworkTemplate,
+)
+
+
+def _tmplt_ospf_vrf_cmd(process):
+ command = "router ospfv3"
+ vrf = "{vrf}".format(**process)
+ if "vrf" in process and vrf != "default":
+ command += " vrf " + vrf
+ return command
+
+
+def _tmplt_ospf_address_family_cmd(config_data):
+ afi = "{afi}".format(**config_data)
+ if afi == "router":
+ command = ""
+ else:
+ command = "address-family " + afi
+ return command
+
+
+def _tmplt_ospf_adjacency_cmd(config_data):
+ command = "adjacency exchange-start threshold"
+ if "adjacency" in config_data:
+ command += " {threshold}".format(
+ **config_data["adjacency"]["exchange_start"]
+ )
+ return command
+
+
+def _tmplt_ospf_auto_cost(config_data):
+ if "auto_cost" in config_data:
+ command = "auto-cost"
+ if "reference_bandwidth" in config_data["auto_cost"]:
+ command += " reference-bandwidth {reference_bandwidth}".format(
+ **config_data["auto_cost"]
+ )
+ return command
+
+
+def _tmplt_ospf_area_authentication(config_data):
+ if "area_id" in config_data:
+ command = "area {area_id} authentication ipsec spi ".format(
+ **config_data
+ )
+ command += "{spi} {algorithm}".format(**config_data["authentication"])
+ if "passphrase" in config_data["authentication"]:
+ command += " passphrase"
+ if (
+ "encrypt_key" in config_data["authentication"]
+ and config_data["authentication"]["encrypt_key"] is False
+ ):
+ command += " 0"
+ if (
+ "hidden_key" in config_data["authentication"]
+ and config_data["authentication"]["hidden_key"] is True
+ ):
+ command += " 7"
+ if "passphrase" not in config_data["authentication"]:
+ command += " {key}".format(**config_data["authentication"])
+ else:
+ command += " {passphrase}".format(**config_data["authentication"])
+ return command
+
+
+def _tmplt_ospf_area_encryption(config_data):
+ if "area_id" in config_data:
+ command = "area {area_id} encryption ipsec spi ".format(**config_data)
+ command += "{spi} esp {encryption} {algorithm}".format(
+ **config_data["encryption"]
+ )
+ if "passphrase" in config_data["encryption"]:
+ command += " passphrase"
+ if (
+ "encrypt_key" in config_data["encryption"]
+ and config_data["encryption"]["encrypt_key"] is False
+ ):
+ command += " 0"
+ if (
+ "hidden_key" in config_data["encryption"]
+ and config_data["encryption"]["hidden_key"] is True
+ ):
+ command += " 7"
+ if "passphrase" not in config_data["encryption"]:
+ command += " {key}".format(**config_data["encryption"])
+ else:
+ command += " {passphrase}".format(**config_data["encryption"])
+ return command
+
+
+def _tmplt_ospf_area_nssa(config_data):
+ if "nssa" in config_data:
+ command = "area {area_id} nssa".format(**config_data)
+ if "default_information_originate" in config_data["nssa"]:
+ command += " default-information-originate"
+ if (
+ "metric"
+ in config_data["nssa"]["default_information_originate"]
+ ):
+ command += " metric {metric}".format(
+ **config_data["nssa"]["default_information_originate"]
+ )
+ if (
+ "metric_type"
+ in config_data["nssa"]["default_information_originate"]
+ ):
+ command += " metric-type {metric_type}".format(
+ **config_data["nssa"]["default_information_originate"]
+ )
+ if (
+ "nssa_only"
+ in config_data["nssa"]["default_information_originate"]
+ ):
+ command += " nssa-only"
+ if config_data["nssa"].get("nssa_only"):
+ command += " nssa-only"
+ if config_data["nssa"].get("translate"):
+ command += " translate type7 always"
+ if config_data["nssa"].get("no_summary"):
+ command += " no-summary"
+ return command
+
+
+def _tmplt_ospf_area_range(config_data):
+ if "area_id" in config_data:
+ command = "area {area_id} range".format(**config_data)
+ if "address" in config_data:
+ command += " {address}".format(**config_data)
+ if "subnet_address" in config_data:
+ command += " {subnet_address}".format(**config_data)
+ if "subnet_mask" in config_data:
+ command += " {subnet_mask}".format(**config_data)
+ if "advertise" in config_data:
+ if config_data.get("advertise"):
+ command += " advertise"
+ else:
+ command += " not-advertise"
+ if "cost" in config_data:
+ command += " cost {cost}".format(**config_data)
+ return command
+
+
+def _tmplt_ospf_area_stub(config_data):
+ if "stub" in config_data:
+ command = "area {area_id} stub".format(**config_data)
+ if "summary_lsa" in config_data["stub"]:
+ if not config_data["stub"]["summary_lsa"]:
+ command += " no-summary"
+ return command
+
+
+def _tmplt_ospf_default_information(config_data):
+ if "default_information" in config_data:
+ command = "default-information"
+ if "originate" in config_data["default_information"]:
+ command += " originate"
+ if "always" in config_data["default_information"]:
+ command += " always"
+ if "metric" in config_data["default_information"]:
+ command += " metric {metric}".format(
+ **config_data["default_information"]
+ )
+ if "metric_type" in config_data["default_information"]:
+ command += " metric-type {metric_type}".format(
+ **config_data["default_information"]
+ )
+ if "route_map" in config_data["default_information"]:
+ command += " route-map {route_map}".format(
+ **config_data["default_information"]
+ )
+ return command
+
+
+def _tmplt_ospf_log_adjacency_changes(config_data):
+ if "log_adjacency_changes" in config_data:
+ command = "log-adjacency-changes"
+ if "detail" in config_data["log_adjacency_changes"]:
+ if config_data["log_adjacency_changes"].get("detail"):
+ command += " detail"
+ return command
+
+
+def _tmplt_ospf_max_metric(config_data):
+ if "max_metric" in config_data:
+ command = "max-metric"
+ if "router_lsa" in config_data["max_metric"]:
+ command += " router-lsa"
+ if "external_lsa" in config_data["max_metric"]["router_lsa"]:
+ command += " external-lsa"
+ if (
+ "max_metric_value"
+ in config_data["max_metric"]["router_lsa"]["external_lsa"]
+ ):
+ command += " {max_metric_value}".format(
+ **config_data["max_metric"]["router_lsa"]["external_lsa"]
+ )
+ if "include_stub" in config_data["max_metric"]["router_lsa"]:
+ if config_data["max_metric"]["router_lsa"].get("include_stub"):
+ command += " include-stub"
+ if "on_startup" in config_data["max_metric"]["router_lsa"]:
+ command += " on-startup {wait_period}".format(
+ **config_data["max_metric"]["router_lsa"]["on_startup"]
+ )
+ if "summary_lsa" in config_data["max_metric"]["router_lsa"]:
+ command += " summary-lsa"
+ if (
+ "max_metric_value"
+ in config_data["max_metric"]["router_lsa"]["summary_lsa"]
+ ):
+ command += " {max_metric_value}".format(
+ **config_data["max_metric"]["router_lsa"]["summary_lsa"]
+ )
+ return command
+
+
+def _tmplt_ospf_redistribute(config_data):
+ command = "redistribute {routes}".format(**config_data)
+ if "route_map" in config_data:
+ command += " route-map {route_map}".format(**config_data)
+ return command
+
+
+def _tmplt_ospf_timers_throttle(config_data):
+ if "throttle" in config_data["timers"]:
+ command = "timers throttle"
+ if "lsa" in config_data["timers"]["throtle"]:
+ if config_data["timers"]["throtle"].get("lsa"):
+ command += " lsa all"
+ if "spf" in config_data["timers"]["throtle"]:
+ if config_data["timers"]["throtle"].get("spf"):
+ command += " spf"
+ if "initial" in config_data["timers"]["throttle"]:
+ command += " {initial}".format(**config_data["timers"]["throttle"])
+ if "min" in config_data["timers"]["throttle"]:
+ command += " {min}".format(**config_data["timers"]["throttle"])
+ if "max" in config_data["timers"]["throttle"]:
+ command += " {max}".format(**config_data["timers"]["max"])
+
+ return command
+
+
+class Ospfv3Template(NetworkTemplate):
+ def __init__(self, lines=None):
+ super(Ospfv3Template, self).__init__(lines=lines, tmplt=self)
+
+ PARSERS = [
+ {
+ "name": "vrf",
+ "getval": re.compile(
+ r"""
+ ^router\s
+ ospfv3
+ \svrf
+ \s(?P<vrf>\S+)
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_vrf_cmd,
+ "result": {"processes": {"vrf": "{{ vrf }}"}},
+ "shared": True,
+ },
+ {
+ "name": "vrf",
+ "getval": re.compile(
+ r"""
+ ^router\s
+ ospfv3
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_vrf_cmd,
+ "result": {"processes": {"vrf": '{{ "default" }}'}},
+ "shared": True,
+ },
+ {
+ "name": "address_family",
+ "getval": re.compile(
+ r"""
+ \s*address-family
+ \s(?P<afi>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_address_family_cmd,
+ "compval": "address_family",
+ "result": {
+ "processes": {
+ "address_family": {"{{ afi }}": {"afi": "{{ afi }}"}}
+ }
+ },
+ "shared": True,
+ },
+ {
+ "name": "adjacency",
+ "getval": re.compile(
+ r"""
+ \s*adjacency
+ \s+exchange-start
+ \s+threshold
+ \s+(?P<threshold>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_adjacency_cmd,
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "adjacency": {
+ "exchange_start": {
+ "threshold": "{{ threshold|int }}"
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "auto_cost",
+ "getval": re.compile(
+ r"""\s+(?P<auto_cost>auto-cost)*
+ \s*(?P<ref_band>reference-bandwidth\s\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_auto_cost,
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "auto_cost": {
+ "reference_bandwidth": '{{ ref_band.split(" ")[1] }}'
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.default_cost",
+ "getval": re.compile(
+ r"""\s+area
+ \s(?P<area_id>\S+)*
+ \sdefault-cost*
+ \s*(?P<default_cost>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "area {{ area_id }} default-cost {{ default_cost }}",
+ "compval": "default_cost",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "default_cost": "{{ default_cost|int }}",
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.authentication",
+ "getval": re.compile(
+ r"""
+ \s*area
+ \s+(?P<area_id>\S+)
+ \s+authentication
+ \s+ipsec
+ \s+spi
+ \s+(?P<val>\d+)
+ \s+(?P<algorithm>md5|sha1)
+ \s*(?P<passphrase>passphrase)*
+ \s*(?P<type>0|7)*
+ \s*(?P<line>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_area_authentication,
+ "compval": "authentication",
+ "remval": "area {{ area_id }} authentication",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "authentication": {
+ "spi": "{{ val }}",
+ "algorithm": "{{ algorithm }}",
+ "encrypt_key": '{{ False if type is defined and type == "0" }}',
+ "hidden_key": '{{ True if type is defined and type == "7" }}',
+ "passphrase": "{{ line if passphrase is defined }}",
+ "key": "{{ str(line) if passphrase is undefined }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.encryption",
+ "getval": re.compile(
+ r"""
+ \s*area
+ \s+(?P<area_id>\S+)*
+ \s+encryption
+ \s+ipsec
+ \s+spi
+ \s+(?P<val>\d+)
+ \s+esp
+ \s*(?P<encryption>\S+)
+ \s*(?P<algorithm>md5|sha1)*
+ \s*(?P<passphrase>passphrase)*
+ \s*(?P<type>0|7)*
+ \s*(?P<line>\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_area_encryption,
+ "compval": "encryption",
+ "remval": "area {{ area_id }} encryption",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "encryption": {
+ "spi": "{{ val }}",
+ "encryption": "{{ encryption }}",
+ "algorithm": "{{ algorithm }}",
+ "encrypt_key": "{{ False if type is defined and type == '0'}}",
+ "passphrase": "{{line if passphrase is defined }}",
+ "hidden_key": "{{ True if type is defined and type == '7'}}",
+ "key": "{{ line if passphrase is not defined }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.nssa",
+ "getval": re.compile(
+ r"""
+ \s+area\s(?P<area_id>\S+)
+ \s(?P<nssa>nssa)
+ \s*(?P<def_origin>default-information-originate)*
+ \s*(metric)*
+ \s*(?P<metric>\d+)*
+ \s*(metric-type)*
+ \s*(?P<metric_type>\d+)*
+ \s*(?P<no_summary>no-summary)*
+ \s*(?P<translate>translate.*)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_area_nssa,
+ "compval": "nssa",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "nssa": {
+ "set": "{{ True if nssa is defined and def_origin is undefined and "
+ "no_summary is undefined and translate is undefined }}",
+ "default_information_originate": {
+ "set": "{{ True if def_origin is defined and metric is undefined and "
+ "metric_type is undefined and nssa_only is undefined }}",
+ "metric": "{{ metric.split("
+ ")[1]|int }}",
+ "metric_type": "{{ metric_type.split("
+ ")[1]|int }}",
+ "nssa_only": "{{ True if nssa_only is defined }}",
+ },
+ "translate": "{{ True if translate is defined }}",
+ "no_summary": "{{ True if no_summary is defined }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.ranges",
+ "getval": re.compile(
+ r"""
+ \s*area
+ \s+(?P<area_id>\S+)*
+ \s+range
+ \s+(?P<address>\S+)*
+ \s*(?P<not_advertise>not-advertise)*
+ \s*(?P<subnet_mask>\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})*
+ \s*(?P<subnet_address>\S+)*
+ \s*(?P<cost>cost)*
+ \s*(?P<cost_val>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_area_range,
+ "compval": "ranges",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "ranges": [
+ {
+ "address": "{{ address }}",
+ "subnet_mask": "{{ subnet_mask }}",
+ "advertise": "{{ not not_advertise }}",
+ "cost": "{{ cost_val }}",
+ }
+ ],
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "area.stub",
+ "getval": re.compile(
+ r"""\s+area\s(?P<area_id>\S+)
+ \s(?P<stub>stub)*
+ \s*(?P<no_sum>no-summary)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_area_stub,
+ "compval": "stub",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "areas": {
+ "{{ area_id }}": {
+ "area_id": "{{ area_id }}",
+ "stub": {
+ "set": "{{ True if stub is defined and no_sum is undefined }}",
+ "summary_lsa": "{{ True if no_sum is defined }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "bfd",
+ "getval": re.compile(
+ r"""\s+bfd*
+ \s*(?P<bfd>all-interfaces)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "bfd all-interfaces",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "bfd": {
+ "all_interfaces": "{{ True if bfd is defined }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "default_information",
+ "getval": re.compile(
+ r"""\s+default-information*
+ \s*(?P<originate>originate)*
+ \s*(?P<always>always)*
+ \s*(?P<metric>metric\s\d+)*
+ \s*(?P<metric_type>metric-type\s\d+)*
+ \s*(?P<route_map>route-map\s\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_default_information,
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "default_information": {
+ "originate": "{{ True if originate is defined }}",
+ "always": "{{ True if always is defined }}",
+ "metric": "{{ metric.split(" ")[1]|int }}",
+ "metric_type": "{{ metric_type.split("
+ ")[1]|int }}",
+ "route_map": "{{ route_map.split(" ")[1] }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "default_metric",
+ "getval": re.compile(
+ r"""\s+default-metric(?P<default_metric>\s\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "default-metric {{ default_metric }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "default_metric": "{{ default_metric| int}}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "distance",
+ "getval": re.compile(
+ r"""\s+distance
+ \s+ospf
+ \s+intra-area
+ \s+(?P<distance>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "distance ospf intra-area {{ distance }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "distance": "{{ distance| int}}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "fips_restrictions",
+ "getval": re.compile(
+ r"""
+ \s+(?P<fips>fips\s*\S+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "fips restrictions",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "fips_restrictions": "{{ True if fips is defined }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart_period",
+ "getval": re.compile(
+ r"""
+ \s+graceful-restart
+ \s*grace-period*
+ \s*(?P<period>\d+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": "graceful-restart grace-period {{ graceful_restart.grace_period }}",
+ "remval": "graceful-restart",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "graceful_restart": {
+ "grace_period": "{{ period|int }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart",
+ "getval": re.compile(
+ r"""
+ \s+graceful-restart
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": "graceful-restart",
+ "remval": "graceful-restart",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "graceful_restart": {"set": "{{ True }}"}
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "graceful_restart_helper",
+ "getval": re.compile(
+ r"""\s+(?P<grace>graceful-restart-helper)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "{{ grace }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "graceful_restart_helper": {
+ "set": "{{ True if grace is defined }}"
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "log_adjacency_changes",
+ "getval": re.compile(
+ r"""\s+(?P<log>log-adjacency-changes)*
+ \s*(?P<detail>detail)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_log_adjacency_changes,
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "log_adjacency_changes": {
+ "set": "{{ True if log is defined and detail is undefined }}",
+ "detail": "{{ True if detail is defined }}",
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "max_metric",
+ "getval": re.compile(
+ r"""\s+max-metric
+ \s+(?P<router_lsa>router-lsa)
+ \s*(?P<external_lsa>external-lsa)*
+ \s*(?P<external_lsa_metric>\d+)*
+ \s*(?P<on_startup>on-startup)*
+ \s*(?P<wait_for_bgp>wait-for-bgp)*
+ \s*(?P<startup_time>\d+)*
+ \s*(?P<summary_lsa>summary-lsa)*
+ \s*(?P<summary_lsa_metric>\d+)*
+ \s*(?P<include_stub>include-stub)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_max_metric,
+ "remval": "max-metric router-lsa",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "max_metric": {
+ "router_lsa": {
+ "set": "{{ True if router_lsa is defined and external_lsa is undefined and "
+ "include_stub is undefined and on_startup is undefined and "
+ "summary_lsa is undefined }}",
+ "external_lsa": {
+ "set": "{{ True if external_lsa is defined and external_lsa_metric is undefined }}",
+ "max_metric_value": "{{ external_lsa_metric }}",
+ },
+ "include_stub": "{{ True if include_stub is defined }}",
+ "on_startup": {
+ "wait_period": "{{ startup_time }}",
+ "wait_for_bgp": "{{ True if wait_for_bgp is defined }}",
+ },
+ "summary_lsa": {
+ "set": "{{ True if summary_lsa is defined and summary_lsa_metric is undefined }}",
+ "max_metric_value": "{{ summary_lsa_metric }}",
+ },
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "maximum_paths",
+ "getval": re.compile(
+ r"""\s+maximum-paths*
+ \s+(?P<paths>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "maximum-paths {{ maximum_paths }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "maximum_paths": "{{ paths }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "passive_interface",
+ "getval": re.compile(
+ r"""
+ \s*(?P<passive>passive-interface.*)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "passive-interface default",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "passive_interface": "{{ True if passive is defined }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "redistribute",
+ "getval": re.compile(
+ r"""
+ \s+redistribute
+ \s(?P<route>\S+)
+ \s*(?P<rmpa>route-map)*
+ \s*(?P<map>\S+)*
+ $""",
+ re.VERBOSE,
+ ),
+ "setval": _tmplt_ospf_redistribute,
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "redistribute": [
+ {
+ "routes": "{{ route }}",
+ "route_map": "{{ map }}",
+ }
+ ]
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "router_id",
+ "getval": re.compile(
+ r"""
+ \s+router-id
+ \s(?P<id>\S+)$""",
+ re.VERBOSE,
+ ),
+ "setval": ("router-id" " {{ router_id }}"),
+ "remval": "router-id",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "router_id": "{{ id }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "shutdown",
+ "getval": re.compile(
+ r"""\s+(?P<shutdown>shutdown)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "shutdown",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "shutdown": "{{ True if shutdown is defined }}"
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "timers.lsa",
+ "getval": re.compile(
+ r"""\s+timers
+ \slsa
+ \sarrival
+ \s(?P<lsa>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "timers lsa arrival {{ timers.lsa }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "timers": {"lsa": "{{ lsa }}"}
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "timers.out_delay",
+ "getval": re.compile(
+ r"""\s+timers
+ \sout-delay
+ \s(?P<out_delay>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "timers out-delay {{ timers.out_delay }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "timers": {"out_delay": "{{ out_delay }}"}
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "timers.pacing",
+ "getval": re.compile(
+ r"""\s+timers
+ \spacing
+ \sflood
+ \s(?P<pacing>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "timers pacing flood {{ timers.pacing }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "timers": {"pacing": "{{ pacing }}"}
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "timers.throttle.lsa",
+ "getval": re.compile(
+ r"""\s+timers
+ \sthrottle
+ \s*(?P<lsa>lsa all)*
+ \s*(?P<initial>\d+)*
+ \s*(?P<min>\d+)*
+ \s*(?P<max>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "timers throttle lsa all {{ timers.throttle.initial }} {{ timers.throttle.min }} {{ timers.throttle.max }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "timers": {
+ "throttle": {
+ "lsa": "{{ True if lsa is defined }}",
+ "initial": "{{ initial }}",
+ "min": "{{ min_delay }}",
+ "max": "{{ max_delay }}",
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ {
+ "name": "timers.throttle.spf",
+ "getval": re.compile(
+ r"""\s+timers
+ \sthrottle
+ \s*(?P<spf>spf)
+ \s*(?P<initial>\d+)
+ \s*(?P<min>\d+)
+ \s*(?P<max>\d+)
+ *$""",
+ re.VERBOSE,
+ ),
+ "setval": "timers throttle spf {{ timers.throttle.initial }} {{ timers.throttle.min }} {{ timers.throttle.max }}",
+ "result": {
+ "processes": {
+ "address_family": {
+ '{{ afi|default("router", true) }}': {
+ "timers": {
+ "throttle": {
+ "spf": "{{ True if spf is defined }}",
+ "initial": "{{ initial }}",
+ "min": "{{ min }}",
+ "max": "{{ max }}",
+ }
+ }
+ }
+ }
+ }
+ },
+ },
+ ]
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py
new file mode 100644
index 00000000..f28f17c3
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py
@@ -0,0 +1,85 @@
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+# utils
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+def get_interface_number(name):
+ digits = ""
+ for char in name:
+ if char.isdigit() or char in "/.":
+ digits += char
+ return digits
+
+
+def normalize_interface(name):
+ """Return the normalized interface name
+ """
+ if not name:
+ return None
+
+ if name.lower().startswith("et"):
+ if_type = "Ethernet"
+ elif name.lower().startswith("lo"):
+ if_type = "Loopback"
+ elif name.lower().startswith("ma"):
+ if_type = "Management"
+ elif name.lower().startswith("po"):
+ if_type = "Port-Channel"
+ elif name.lower().startswith("tu"):
+ if_type = "Tunnel"
+ elif name.lower().startswith("vl"):
+ if_type = "Vlan"
+ elif name.lower().startswith("vx"):
+ if_type = "Vxlan"
+ else:
+ if_type = None
+
+ number_list = name.split(" ")
+ if len(number_list) == 2:
+ number = number_list[-1].strip()
+ else:
+ number = get_interface_number(name)
+
+ if if_type:
+ proper_interface = if_type + number
+ else:
+ proper_interface = name
+
+ return proper_interface
+
+
+def vlan_range_to_list(vlans):
+ result = []
+ if vlans:
+ if isinstance(vlans, str):
+ vlans = vlans.split(",")
+ for part in vlans:
+ if part == "none":
+ break
+ if "-" in part:
+ a, b = part.split("-")
+ a, b = int(a), int(b)
+ result.extend(range(a, b + 1))
+ else:
+ a = int(part)
+ result.append(a)
+ return numerical_sort(result)
+ return result
+
+
+def numerical_sort(string_int_list):
+ """Sorts list of integers that are digits in numerical order.
+ """
+ as_int_list = []
+
+ for vlan in string_int_list:
+ as_int_list.append(int(vlan))
+ as_int_list.sort()
+ return list(set(as_int_list))
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py
new file mode 100644
index 00000000..fbec02f1
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py
@@ -0,0 +1,421 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_acl_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_acl_interfaces
+short_description: ACL interfaces resource module
+description:
+- This module manages adding and removing Access Control Lists (ACLs) from interfaces
+ on devices running EOS software.
+version_added: 1.0.0
+author: GomathiSelvi S (@GomathiselviS)
+options:
+ config:
+ description: A dictionary of ACL options for interfaces.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name/Identifier for the interface.
+ type: str
+ required: true
+ access_groups:
+ type: list
+ elements: dict
+ description:
+ - Specifies ACLs attached to the interfaces.
+ suboptions:
+ afi:
+ description:
+ - Specifies the AFI for the ACL(s) to be configured on this interface.
+ type: str
+ choices:
+ - ipv4
+ - ipv6
+ required: true
+ acls:
+ type: list
+ description:
+ - Specifies the ACLs for the provided AFI.
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Specifies the name of the IPv4/IPv4 ACL for the interface.
+ type: str
+ required: true
+ direction:
+ description:
+ - Specifies the direction of packets that the ACL will be applied
+ on.
+ type: str
+ choices:
+ - in
+ - out
+ required: true
+ running_config:
+ description:
+ - The module, by default, will connect to the remote device and retrieve the current
+ running-config to use as a base for comparing against the contents of source.
+ There are times when it is not desirable to have the task get the current running-config
+ for every task in a playbook. The I(running_config) argument allows the implementer
+ to pass in the configuration to use as the base config for comparison. This
+ value of this option should be the output received from device by executing
+ command
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - gathered
+ - parsed
+ - rendered
+ default: merged
+
+"""
+EXAMPLES = """
+# Using Merged
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface|access-group
+# interface Ethernet1
+# interface Ethernet2
+# interface Ethernet3
+
+- name: Merge module attributes of given access-groups
+ arista.eos.eos_acl_interfaces:
+ config:
+ - name: Ethernet2
+ access_groups:
+ - afi: ipv4
+ acls:
+ name: acl01
+ direction: in
+ - afi: ipv6
+ acls:
+ name: acl03
+ direction: out
+ state: merged
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+
+
+# Using Replaced
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface|access-group
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+# ip access-group acl01 in
+
+- name: Replace module attributes of given access-groups
+ arista.eos.eos_acl_interfaces:
+ config:
+ - name: Ethernet2
+ access_groups:
+ - afi: ipv4
+ acls:
+ name: acl01
+ direction: out
+ state: replaced
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# no ip access-group acl01 in
+# no ipv6 access-group acl03 out
+# ip access-group acl01 out
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 out
+# interface Ethernet3
+# ip access-group acl01 in
+
+
+# Using Overridden
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface|access-group
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+# ip access-group acl01 in
+
+- name: Override module attributes of given access-groups
+ arista.eos.eos_acl_interfaces:
+ config:
+ - name: Ethernet2
+ access_groups:
+ - afi: ipv4
+ acls:
+ name: acl01
+ direction: out
+ state: overridden
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# no ip access-group acl01 in
+# no ipv6 access-group acl03 out
+# ip access-group acl01 out
+# interface Ethernet3
+# no ip access-group acl01 in
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 out
+# interface Ethernet3
+
+
+# Using Deleted
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface|access-group
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+# ip access-group acl01 out
+
+- name: Delete module attributes of given access-groups
+ arista.eos.eos_acl_interfaces:
+ config:
+ - name: Ethernet2
+ access_groups:
+ - afi: ipv4
+ acls:
+ name: acl01
+ direction: in
+ - afi: ipv6
+ acls:
+ name: acl03
+ direction: out
+ state: deleted
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# no ip access-group acl01 in
+# no ipv6 access-group acl03 out
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# interface Ethernet3
+# ip access-group acl01 out
+
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+# ip access-group acl01 out
+
+- name: Delete module attributes of given access-groups from ALL Interfaces
+ arista.eos.eos_acl_interfaces:
+ config:
+ state: deleted
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# no ip access-group acl01 in
+# no ipv6 access-group acl03 out
+# interface Ethernet3
+# no ip access-group acl01 out
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# interface Ethernet3
+
+# Before state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# ipv6 access-group acl03 out
+# interface Ethernet3
+# ip access-group acl01 out
+
+- name: Delete acls under afi
+ arista.eos.eos_acl_interfaces:
+ config:
+ - name: Ethernet3
+ access_groups:
+ - afi: ipv4
+ - name: Ethernet2
+ access_groups:
+ - afi: ipv6
+ state: deleted
+
+# Commands Fired:
+# ---------------
+#
+# interface Ethernet2
+# no ipv6 access-group acl03 out
+# interface Ethernet3
+# no ip access-group acl01 out
+
+# After state:
+# -------------
+#
+# eos#sh running-config | include interface| access-group
+# interface Loopback888
+# interface Ethernet1
+# interface Ethernet2
+# ip access-group acl01 in
+# interface Ethernet3
+
+
+"""
+RETURN = """
+before:
+ description: The configuration prior to the model invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The resulting configuration model invocation.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample:
+ - interface Ethernet2
+ - ip access-group acl01 in
+ - ipv6 access-group acl03 out
+ - interface Ethernet3
+ - ip access-group acl01 out
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acl_interfaces.acl_interfaces import (
+ Acl_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.acl_interfaces.acl_interfaces import (
+ Acl_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ module = AnsibleModule(
+ argument_spec=Acl_interfacesArgs.argument_spec,
+ supports_check_mode=True,
+ )
+
+ result = Acl_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py
new file mode 100644
index 00000000..c37c4244
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py
@@ -0,0 +1,931 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_acls
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_acls
+short_description: ACLs resource module
+description: This module manages the IP access-list attributes of Arista EOS interfaces.
+version_added: 1.0.0
+author: Gomathiselvi S (@GomathiselviS)
+notes:
+- Tested against Arista vEOS v4.20.10M
+options:
+ config:
+ description: A dictionary of IP access-list options
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description:
+ - The Address Family Indicator (AFI) for the Access Control Lists (ACL).
+ type: str
+ required: true
+ choices:
+ - ipv4
+ - ipv6
+ acls:
+ description:
+ - A list of Access Control Lists (ACL).
+ type: list
+ elements: dict
+ suboptions:
+ standard:
+ description: standard access-list or not
+ type: bool
+ name:
+ description: Name of the acl-list
+ type: str
+ required: true
+ aces:
+ description: Filtering data
+ type: list
+ elements: dict
+ suboptions:
+ sequence:
+ description: sequence number for the ordered list of rules
+ type: int
+ remark:
+ description: Specify a comment
+ type: str
+ fragment_rules:
+ description: Add fragment rules
+ type: bool
+ grant:
+ description: Action to be applied on the rule
+ type: str
+ choices:
+ - permit
+ - deny
+ line:
+ description: For fact gathering, any ACE that is not fully parsed,
+ while show up as a value of this attribute.
+ type: str
+ aliases:
+ - ace
+ protocol:
+ description:
+ - Specify the protocol to match.
+ - Refer to vendor documentation for valid values.
+ type: str
+ vlan:
+ description: Vlan options
+ type: str
+ protocol_options:
+ description: All the possible sub options for the protocol chosen.
+ type: dict
+ suboptions:
+ tcp:
+ description: Options for tcp protocol.
+ type: dict
+ suboptions:
+ flags:
+ description: Match TCP packet flags
+ type: dict
+ suboptions:
+ ack:
+ description: Match on the ACK bit
+ type: bool
+ established:
+ description: Match established connections
+ type: bool
+ fin:
+ description: Match on the FIN bit
+ type: bool
+ psh:
+ description: Match on the PSH bit
+ type: bool
+ rst:
+ description: Match on the RST bit
+ type: bool
+ syn:
+ description: Match on the SYN bit
+ type: bool
+ urg:
+ description: Match on the URG bit
+ type: bool
+ icmp:
+ description:
+ - Internet Control Message Protocol settings.
+ type: dict
+ suboptions:
+ administratively_prohibited:
+ description: Administratively prohibited
+ type: bool
+ alternate_address:
+ description: Alternate address
+ type: bool
+ conversion_error:
+ description: Datagram conversion
+ type: bool
+ dod_host_prohibited:
+ description: Host prohibited
+ type: bool
+ dod_net_prohibited:
+ description: Net prohibited
+ type: bool
+ echo:
+ description: Echo (ping)
+ type: bool
+ echo_reply:
+ description: Echo reply
+ type: bool
+ general_parameter_problem:
+ description: Parameter problem
+ type: bool
+ host_isolated:
+ description: Host isolated
+ type: bool
+ host_precedence_unreachable:
+ description: Host unreachable for precedence
+ type: bool
+ host_redirect:
+ description: Host redirect
+ type: bool
+ host_tos_redirect:
+ description: Host redirect for TOS
+ type: bool
+ host_tos_unreachable:
+ description: Host unreachable for TOS
+ type: bool
+ host_unknown:
+ description: Host unknown
+ type: bool
+ host_unreachable:
+ description: Host unreachable
+ type: bool
+ information_reply:
+ description: Information replies
+ type: bool
+ information_request:
+ description: Information requests
+ type: bool
+ mask_reply:
+ description: Mask replies
+ type: bool
+ mask_request:
+ description: Mask requests
+ type: bool
+ message_code:
+ description: ICMP message code
+ type: int
+ message_type:
+ description: ICMP message type
+ type: int
+ mobile_redirect:
+ description: Mobile host redirect
+ type: bool
+ net_redirect:
+ description: Network redirect
+ type: bool
+ net_tos_redirect:
+ description: Net redirect for TOS
+ type: bool
+ net_tos_unreachable:
+ description: Network unreachable for TOS
+ type: bool
+ net_unreachable:
+ description: Net unreachable
+ type: bool
+ network_unknown:
+ description: Network unknown
+ type: bool
+ no_room_for_option:
+ description: Parameter required but no room
+ type: bool
+ option_missing:
+ description: Parameter required but not present
+ type: bool
+ packet_too_big:
+ description: Fragmentation needed and DF set
+ type: bool
+ parameter_problem:
+ description: All parameter problems
+ type: bool
+ port_unreachable:
+ description: Port unreachable
+ type: bool
+ precedence_unreachable:
+ description: Precedence cutoff
+ type: bool
+ protocol_unreachable:
+ description: Protocol unreachable
+ type: bool
+ reassembly_timeout:
+ description: Reassembly timeout
+ type: bool
+ redirect:
+ description: All redirects
+ type: bool
+ router_advertisement:
+ description: Router discovery advertisements
+ type: bool
+ router_solicitation:
+ description: Router discovery solicitations
+ type: bool
+ source_quench:
+ description: Source quenches
+ type: bool
+ source_route_failed:
+ description: Source route failed
+ type: bool
+ time_exceeded:
+ description: All time exceededs
+ type: bool
+ timestamp_reply:
+ description: Timestamp replies
+ type: bool
+ timestamp_request:
+ description: Timestamp requests
+ type: bool
+ traceroute:
+ description: Traceroute
+ type: bool
+ ttl_exceeded:
+ description: TTL exceeded
+ type: bool
+ unreachable:
+ description: All unreachables
+ type: bool
+ message_num:
+ description: icmp msg type number.
+ type: int
+ icmpv6:
+ description: Options for icmpv6.
+ type: dict
+ suboptions:
+ address_unreachable:
+ description: address unreachable
+ type: bool
+ beyond_scope:
+ description: beyond_scope
+ type: bool
+ echo_reply:
+ description: echo_reply
+ type: bool
+ echo_request:
+ description: echo reques
+ type: bool
+ erroneous_header:
+ description: erroneous header
+ type: bool
+ fragment_reassembly_exceeded:
+ description: fragment_reassembly_exceeded
+ type: bool
+ hop_limit_exceeded:
+ description: hop limit exceeded
+ type: bool
+ neighbor_advertisement:
+ description: neighbor advertisement
+ type: bool
+ neighbor_solicitation:
+ description: neighbor_solicitation
+ type: bool
+ no_admin:
+ description: no admin
+ type: bool
+ no_route:
+ description: no route
+ type: bool
+ packet_too_big:
+ description: packet too big
+ type: bool
+ parameter_problem:
+ description: parameter problem
+ type: bool
+ port_unreachable:
+ description: port unreachable
+ type: bool
+ redirect_message:
+ description: redirect message
+ type: bool
+ reject_route:
+ description: reject route
+ type: bool
+ router_advertisement:
+ description: router_advertisement
+ type: bool
+ router_solicitation:
+ description: router_solicitation
+ type: bool
+ source_address_failed:
+ description: source_address_failed
+ type: bool
+ source_routing_error:
+ description: source_routing_error
+ type: bool
+ time_exceeded:
+ description: time_exceeded
+ type: bool
+ unreachable:
+ description: unreachable
+ type: bool
+ unrecognized_ipv6_option:
+ description: unrecognized_ipv6_option
+ type: bool
+ unrecognized_next_header:
+ description: unrecognized_next_header
+ type: bool
+ ip:
+ description: Internet Protocol.
+ type: dict
+ suboptions:
+ nexthop_group:
+ description: Nexthop-group name.
+ type: str
+ ipv6:
+ description: Internet V6 Protocol.
+ type: dict
+ suboptions:
+ nexthop_group:
+ description: Nexthop-group name.
+ type: str
+ source:
+ description: The packet's source address
+ type: dict
+ suboptions:
+ address:
+ description: dotted decimal notation of IP address
+ type: str
+ wildcard_bits:
+ description: Source wildcard bits
+ type: str
+ subnet_address:
+ description: A subnet address
+ type: str
+ host:
+ description: Host IP address
+ type: str
+ any:
+ description: Rule matches all source addresses
+ type: bool
+ port_protocol:
+ description: Specify source port/protocoli, along with operator.
+ (comes with tcp/udp).
+ type: dict
+ destination:
+ description: The packet's destination address
+ type: dict
+ suboptions:
+ address:
+ description: dotted decimal notation of IP address
+ type: str
+ wildcard_bits:
+ description: Source wildcard bits
+ type: str
+ subnet_address:
+ description: A subnet address
+ type: str
+ host:
+ description: Host IP address
+ type: str
+ any:
+ description: Rule matches all source addresses
+ type: bool
+ port_protocol:
+ description: Specify dest port/protocol, along with operator .
+ (comes with tcp/udp).
+ type: dict
+ ttl:
+ description: Compares the TTL (time-to-live) value in the packet to
+ a specified value
+ type: dict
+ suboptions:
+ eq:
+ description: Match a single TTL value
+ type: int
+ lt:
+ description: Match TTL lesser than this number
+ type: int
+ gt:
+ description: Match TTL greater than this number
+ type: int
+ neq:
+ description: Match TTL not equal to this value
+ type: int
+ fragments:
+ description: Match non-head fragment packets
+ type: bool
+ log:
+ description: Log matches against this rule
+ type: bool
+ tracked:
+ description: Match packets in existing ICMP/UDP/TCP connections
+ type: bool
+ hop_limit:
+ description: Hop limit value.
+ type: dict
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section access-list).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices:
+ - deleted
+ - merged
+ - overridden
+ - replaced
+ - gathered
+ - rendered
+ - parsed
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+- name: Merge provided configuration with device configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ aces:
+ - sequence: 35
+ grant: deny
+ protocol: ospf
+ source:
+ subnet_address: 20.0.0.0/8
+ destnation:
+ any: true
+ state: merged
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 35 deny ospf 20.0.0.0/8 any
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+# Using merged
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+- name: Merge to update the given configuration with an existing ace
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ aces:
+ - sequence: 35
+ log: true
+ ttl:
+ eq: 33
+ state: merged
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 35 deny ospf 20.0.0.0/8 any ttl eq 33 log
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+# Using replaced
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# !
+# ip access-list test3
+# 10 permit ip 35.33.0.0/16 any log
+# !
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+
+
+- name: Replace device configuration with provided configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ aces:
+ - sequence: 35
+ grant: permit
+ protocol: ospf
+ source:
+ subnet_address: 20.0.0.0/8
+ destination:
+ any: true
+ state: replaced
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+# ip access-list test1
+# 35 permit ospf 20.0.0.0/8 any
+# !
+# ip access-list test3
+# 10 permit ip 35.33.0.0/16 any log
+# !
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+
+# Using overridden
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# !
+# ip access-list test3
+# 10 permit ip 35.33.0.0/16 any log
+# !
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+
+
+- name: override device configuration with provided configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ aces:
+ - sequence: 35
+ action: permit
+ protocol: ospf
+ source:
+ subnet_address: 20.0.0.0/8
+ destination:
+ any: true
+ state: overridden
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+# ip access-list test1
+# 35 permit ospf 20.0.0.0/8 any
+# !
+
+# Using deleted:
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+# !
+
+- name: Delete provided configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ state: deleted
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+
+
+# Before state:
+# -------------
+# show running-config | section access-list
+# ip access-list test1
+# 10 permit ip 10.10.10.0/24 any ttl eq 200
+# 20 permit ip 10.30.10.0/24 host 10.20.10.1
+# 30 deny tcp host 10.10.20.1 eq finger www any syn log
+# 40 permit ip any any
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+# !
+
+- name: Delete provided configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ state: deleted
+
+# After state:
+# ------------
+#
+# show running-config | section access-list
+
+# ipv6 access-list test2
+# 10 deny icmpv6 any any reject-route hop-limit eq 20
+
+
+# using gathered
+
+# ip access-list test1
+# 35 deny ospf 20.0.0.0/8 any
+# ip access-list test2
+# 40 permit vlan 55 0xE2 icmpv6 any any log
+
+- name: Gather the exisitng condiguration
+ arista.eos.eos_acls:
+ state: gathered
+
+# returns:
+
+
+# arista.eos.eos_acls:
+# config:
+# - afi: "ipv4"
+# acls:
+# - name: test1
+# aces:
+# - sequence: 35
+# grant: "deny"
+# protocol: "ospf"
+# source:
+# subnet_address: 20.0.0.0/8
+# destination:
+# any: true
+# - afi: "ipv6"
+# acls:
+# - name: test2
+# aces:
+# - sequence: 40
+# grant: "permit"
+# vlan: "55 0xE2"
+# protocol: "icmpv6"
+# log: true
+# source:
+# any: true
+# destination:
+# any: true
+
+
+# using rendered
+
+- name: Delete provided configuration
+ arista.eos.eos_acls:
+ config:
+ - afi: ipv4
+ acls:
+ - name: test1
+ aces:
+ - sequence: 35
+ grant: deny
+ protocol: ospf
+ source:
+ subnet_address: 20.0.0.0/8
+ destination:
+ any: true
+ - afi: ipv6
+ acls:
+ - name: test2
+ aces:
+ - sequence: 40
+ grant: permit
+ vlan: 55 0xE2
+ protocol: icmpv6
+ log: true
+ source:
+ any: true
+ destination:
+ any: true
+ state: rendered
+
+# returns:
+
+# ip access-list test1
+# 35 deny ospf 20.0.0.0/8 any
+# ip access-list test2
+# 40 permit vlan 55 0xE2 icmpv6 any any log
+
+
+# Using Parsed
+
+# parsed_acls.cfg
+
+# ipv6 access-list standard test2
+# 10 permit any log
+# !
+# ip access-list test1
+# 35 deny ospf 20.0.0.0/8 any
+# 45 remark Run by ansible
+# 55 permit tcp any any
+# !
+
+- name: parse configs
+ arista.eos.eos_acls:
+ running_config: "{{ lookup('file', './parsed_acls.cfg') }}"
+ state: parsed
+
+# returns
+# "parsed": [
+# {
+# "acls": [
+# {
+# "aces": [
+# {
+# "destination": {
+# "any": true
+# },
+# "grant": "deny",
+# "protocol": "ospf",
+# "sequence": 35,
+# "source": {
+# "subnet_address": "20.0.0.0/8"
+# }
+# },
+# {
+# "remark": "Run by ansible",
+# "sequence": 45
+# },
+# {
+# "destination": {
+# "any": true
+# },
+# "grant": "permit",
+# "protocol": "tcp",
+# "sequence": 55,
+# "source": {
+# "any": true
+# }
+# }
+# ],
+# "name": "test1"
+# }
+# ],
+# "afi": "ipv4"
+# },
+# {
+# "acls": [
+# {
+# "aces": [
+# {
+# "grant": "permit",
+# "log": true,
+# "sequence": 10,
+# "source": {
+# "any": true
+# }
+# }
+# ],
+# "name": "test2",
+# "standard": true
+# }
+# ],
+# "afi": "ipv6"
+# }
+# ]
+
+"""
+RETURN = """
+before:
+ description: The configuration prior to the model invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The resulting configuration model invocation.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample:
+ - ipv6 access-list standard test2
+ - 10 permit any log
+ - ip access-list test1
+ - 35 deny ospf 20.0.0.0/8 any
+ - 45 remark Run by ansible
+ - 55 permit tcp any any
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acls.acls import (
+ AclsArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.acls.acls import (
+ Acls,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+
+ module = AnsibleModule(
+ argument_spec=AclsArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Acls(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py
new file mode 100644
index 00000000..2a8077ef
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py
@@ -0,0 +1,219 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_banner
+author: Peter Sprygada (@privateip)
+short_description: Manage multiline banners on Arista EOS devices
+description:
+- This will configure both login and motd banners on remote devices running Arista
+ EOS. It allows playbooks to add or remote banner text from the active running configuration.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+notes:
+- Tested against EOS 4.15
+options:
+ banner:
+ description:
+ - Specifies which banner that should be configured on the remote device.
+ required: true
+ choices:
+ - login
+ - motd
+ type: str
+ text:
+ description:
+ - The banner text that should be present in the remote device running configuration. This
+ argument accepts a multiline string. Requires I(state=present).
+ type: str
+ state:
+ description:
+ - Specifies whether or not the configuration is present in the current devices
+ active running configuration.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+"""
+
+EXAMPLES = """
+- name: configure the login banner
+ arista.eos.eos_banner:
+ banner: login
+ text: |
+ this is my login banner
+ that contains a multiline
+ string
+ state: present
+
+- name: remove the motd banner
+ arista.eos.eos_banner:
+ banner: motd
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - banner login
+ - this is my login banner
+ - that contains a multiline
+ - string
+ - EOF
+session_name:
+ description: The EOS config session name used to load the configuration
+ returned: if changes
+ type: str
+ sample: ansible_1479315771
+"""
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ load_config,
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+ is_local_eapi,
+)
+from ansible.module_utils.six import string_types
+from ansible.module_utils._text import to_text
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+ state = module.params["state"]
+
+ if state == "absent" and have.get("text"):
+ if isinstance(have["text"], string_types):
+ commands.append("no banner %s" % module.params["banner"])
+ elif have["text"].get("loginBanner") or have["text"].get("motd"):
+ commands.append({"cmd": "no banner %s" % module.params["banner"]})
+
+ elif state == "present":
+ if isinstance(have["text"], string_types):
+ if want["text"] != have["text"]:
+ commands.append("banner %s" % module.params["banner"])
+ commands.extend(want["text"].strip().split("\n"))
+ commands.append("EOF")
+ else:
+ have_text = have["text"].get("loginBanner") or have["text"].get(
+ "motd"
+ )
+ if have_text:
+ have_text = have_text.strip()
+
+ if to_text(want["text"]) != have_text or not have_text:
+ # For EAPI we need to construct a dict with cmd/input
+ # key/values for the banner
+ commands.append(
+ {
+ "cmd": "banner %s" % module.params["banner"],
+ "input": want["text"].strip("\n"),
+ }
+ )
+
+ return commands
+
+
+def map_config_to_obj(module):
+ output = run_commands(module, ["show banner %s" % module.params["banner"]])
+ obj = {"banner": module.params["banner"], "state": "absent"}
+ if output:
+ if is_local_eapi(module):
+ # On EAPI we need to extract the banner text from dict key
+ # 'loginBanner'
+ if module.params["banner"] == "login":
+ banner_response_key = "loginBanner"
+ else:
+ banner_response_key = "motd"
+ if (
+ isinstance(output[0], dict)
+ and banner_response_key in output[0].keys()
+ ):
+ obj["text"] = output[0]
+ else:
+ obj["text"] = output[0]
+ obj["state"] = "present"
+ return obj
+
+
+def map_params_to_obj(module):
+ text = module.params["text"]
+ if text:
+ text = to_text(text).strip()
+
+ return {
+ "banner": module.params["banner"],
+ "text": text,
+ "state": module.params["state"],
+ }
+
+
+def main():
+ """ main entry point for module execution
+ """
+ argument_spec = dict(
+ banner=dict(required=True, choices=["login", "motd"]),
+ text=dict(),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ required_if = [("state", "present", ("text",))]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py
new file mode 100644
index 00000000..be30dfd4
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py
@@ -0,0 +1,465 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+#
+# (c) 2019, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_bgp
+author: Nilashish Chakraborty (@NilashishC)
+short_description: (deprecated, removed after 2023-01-29) Configure global BGP protocol settings on Arista EOS.
+description:
+- This module provides configuration management of global BGP parameters on Arista
+ EOS devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_bgp_global
+ why: Updated module released with more functionality.
+ removed_at_date: '2023-01-29'
+notes:
+- Tested against Arista vEOS Version 4.15.9M.
+options:
+ config:
+ description:
+ - Specifies the BGP related configuration.
+ type: dict
+ suboptions:
+ bgp_as:
+ description:
+ - Specifies the BGP Autonomous System (AS) number to configure on the device.
+ type: int
+ required: true
+ router_id:
+ description:
+ - Configures the BGP routing process router-id value.
+ type: str
+ log_neighbor_changes:
+ description:
+ - Enable/disable logging neighbor up/down and reset reason.
+ type: bool
+ neighbors:
+ description:
+ - Specifies BGP neighbor related configurations.
+ type: list
+ elements: dict
+ suboptions:
+ neighbor:
+ description:
+ - Neighbor router address.
+ required: true
+ type: str
+ remote_as:
+ description:
+ - Remote AS of the BGP neighbor to configure.
+ type: int
+ required: true
+ update_source:
+ description:
+ - Source of the routing updates.
+ type: str
+ password:
+ description:
+ - Password to authenticate the BGP peer connection.
+ type: str
+ description:
+ description:
+ - Neighbor specific description.
+ type: str
+ ebgp_multihop:
+ description:
+ - Specifies the maximum hop count for EBGP neighbors not on directly connected
+ networks.
+ - The range is from 1 to 255.
+ type: int
+ peer_group:
+ description:
+ - Name of the peer group that the neighbor is a member of.
+ type: str
+ timers:
+ description:
+ - Specifies BGP neighbor timer related configurations.
+ type: dict
+ suboptions:
+ keepalive:
+ description:
+ - Frequency (in seconds) with which the device sends keepalive messages
+ to its peer.
+ - The range is from 0 to 3600.
+ type: int
+ required: true
+ holdtime:
+ description:
+ - Interval (in seconds) after not receiving a keepalive message that
+ device declares a peer dead.
+ - The range is from 3 to 7200.
+ - Setting this value to 0 will not send keep-alives (hold forever).
+ type: int
+ required: true
+ route_reflector_client:
+ description:
+ - Specify a neighbor as a route reflector client.
+ type: int
+ remove_private_as:
+ description:
+ - Remove the private AS number from outbound updates.
+ type: bool
+ enabled:
+ description:
+ - Administratively shutdown or enable a neighbor.
+ type: bool
+ maximum_prefix:
+ description:
+ - Maximum number of prefixes to accept from this peer.
+ - The range is from 0 to 4294967294.
+ type: int
+ redistribute:
+ description:
+ - Specifies the redistribute information from another routing protocol.
+ type: list
+ elements: dict
+ suboptions:
+ protocol:
+ description:
+ - Specifies the protocol for configuring redistribute information.
+ required: true
+ type: str
+ choices: [ospf, ospf3, static, connected, rip, isis]
+ route_map:
+ description:
+ - Specifies the route map reference.
+ type: str
+ networks:
+ description:
+ - Specify Networks to announce via BGP.
+ - For operation replace, this option is mutually exclusive with networks option
+ under address_family.
+ - For operation replace, if the device already has an address family activated,
+ this option is not allowed.
+ type: list
+ elements: dict
+ suboptions:
+ prefix:
+ description:
+ - Network ID to announce via BGP.
+ required: true
+ type: str
+ masklen:
+ description:
+ - Subnet mask length for the Network to announce(e.g, 8, 16, 24, etc.).
+ type: int
+ route_map:
+ description:
+ - Route map to modify the attributes.
+ type: str
+ address_family:
+ description:
+ - Specifies BGP address family related configurations.
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description:
+ - Type of address family to configure.
+ type: str
+ choices:
+ - ipv4
+ - ipv6
+ required: true
+ redistribute:
+ description:
+ - Specifies the redistribute information from another routing protocol.
+ type: list
+ elements: dict
+ suboptions:
+ protocol:
+ description:
+ - Specifies the protocol for configuring redistribute information.
+ required: true
+ type: str
+ choices:
+ - ospf3
+ - ospf
+ - isis
+ - static
+ - connected
+ - rip
+ route_map:
+ description:
+ - Specifies the route map reference.
+ type: str
+ networks:
+ description:
+ - Specify Networks to announce via BGP.
+ - For operation replace, this option is mutually exclusive with root level
+ networks option.
+ type: list
+ elements: dict
+ suboptions:
+ prefix:
+ description:
+ - Network ID to announce via BGP.
+ required: true
+ type: str
+ masklen:
+ description:
+ - Subnet mask length for the Network to announce(e.g, 8, 16, 24, etc.).
+ type: int
+ route_map:
+ description:
+ - Route map to modify the attributes.
+ type: str
+ neighbors:
+ description:
+ - Specifies BGP neighbor related configurations in Address Family configuration
+ mode.
+ type: list
+ elements: dict
+ suboptions:
+ neighbor:
+ description:
+ - Neighbor router address.
+ required: true
+ type: str
+ activate:
+ description:
+ - Enable the Address Family for this Neighbor.
+ type: bool
+ default_originate:
+ description:
+ - Originate default route to this neighbor.
+ type: bool
+ graceful_restart:
+ description:
+ - Enable/disable graceful restart mode for this neighbor.
+ type: bool
+ weight:
+ description:
+ - Assign weight for routes learnt from this neighbor.
+ - The range is from 0 to 65535
+ type: int
+ operation:
+ description:
+ - Specifies the operation to be performed on the BGP process configured on the
+ device.
+ - In case of merge, the input configuration will be merged with the existing BGP
+ configuration on the device.
+ - In case of replace, if there is a diff between the existing configuration and
+ the input configuration, the existing configuration will be replaced by the
+ input configuration for every option that has the diff.
+ - In case of override, all the existing BGP configuration will be removed from
+ the device and replaced with the input configuration.
+ - In case of delete the existing BGP configuration will be removed from the device.
+ type: str
+ default: merge
+ choices:
+ - merge
+ - replace
+ - override
+ - delete
+"""
+
+EXAMPLES = """
+- name: configure global bgp as 64496
+ arista.eos.eos_bgp:
+ config:
+ bgp_as: 64496
+ router_id: 192.0.2.1
+ log_neighbor_changes: true
+ neighbors:
+ - neighbor: 203.0.113.5
+ remote_as: 64511
+ timers:
+ keepalive: 300
+ holdtime: 360
+ - neighbor: 198.51.100.2
+ remote_as: 64498
+ networks:
+ - prefix: 198.51.100.0
+ route_map: RMAP_1
+ - prefix: 192.0.2.0
+ masklen: 23
+ address_family:
+ - afi: ipv4
+ safi: unicast
+ redistribute:
+ - protocol: isis
+ route_map: RMAP_1
+ operation: merge
+
+- name: Configure BGP neighbors
+ arista.eos.eos_bgp:
+ config:
+ bgp_as: 64496
+ neighbors:
+ - neighbor: 192.0.2.10
+ remote_as: 64496
+ description: IBGP_NBR_1
+ ebgp_multihop: 100
+ timers:
+ keepalive: 300
+ holdtime: 360
+
+ - neighbor: 192.0.2.15
+ remote_as: 64496
+ description: IBGP_NBR_2
+ ebgp_multihop: 150
+ operation: merge
+
+- name: Configure root-level networks for BGP
+ arista.eos.eos_bgp:
+ config:
+ bgp_as: 64496
+ networks:
+ - prefix: 203.0.113.0
+ masklen: 27
+ route_map: RMAP_1
+
+ - prefix: 203.0.113.32
+ masklen: 27
+ route_map: RMAP_2
+ operation: merge
+
+- name: Configure BGP neighbors under address family mode
+ arista.eos.eos_bgp:
+ config:
+ bgp_as: 64496
+ address_family:
+ - afi: ipv4
+ neighbors:
+ - neighbor: 203.0.113.10
+ activate: yes
+ default_originate: true
+
+ - neighbor: 192.0.2.15
+ activate: yes
+ graceful_restart: true
+ operation: merge
+
+- name: remove bgp as 64496 from config
+ arista.eos.eos_bgp:
+ config:
+ bgp_as: 64496
+ operation: delete
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - router bgp 64496
+ - bgp router-id 192.0.2.1
+ - bgp log-neighbor-changes
+ - neighbor 203.0.113.5 remote-as 64511
+ - neighbor 203.0.113.5 timers 300 360
+ - neighbor 198.51.100.2 remote-as 64498
+ - network 198.51.100.0 route-map RMAP_1
+ - network 192.0.2.0 mask 255.255.254.0
+ - address-family ipv4
+ - redistribute isis route-map RMAP_1
+ - exit-address-family
+"""
+from ansible.module_utils._text import to_text
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.module import (
+ NetworkModule,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.process import (
+ REDISTRIBUTE_PROTOCOLS,
+)
+
+
+def main():
+ """ main entry point for module execution
+ """
+ network_spec = {
+ "prefix": dict(required=True),
+ "masklen": dict(type="int"),
+ "route_map": dict(),
+ }
+
+ redistribute_spec = {
+ "protocol": dict(choices=list(REDISTRIBUTE_PROTOCOLS), required=True),
+ "route_map": dict(),
+ }
+
+ timer_spec = {
+ "keepalive": dict(type="int", required=True),
+ "holdtime": dict(type="int", required=True),
+ }
+
+ neighbor_spec = {
+ "neighbor": dict(required=True),
+ "remote_as": dict(type="int", required=True),
+ "update_source": dict(),
+ "password": dict(no_log=True),
+ "enabled": dict(type="bool"),
+ "description": dict(),
+ "ebgp_multihop": dict(type="int"),
+ "timers": dict(type="dict", options=timer_spec),
+ "peer_group": dict(),
+ "maximum_prefix": dict(type="int"),
+ "route_reflector_client": dict(type="int"),
+ "remove_private_as": dict(type="bool"),
+ }
+
+ af_neighbor_spec = {
+ "neighbor": dict(required=True),
+ "activate": dict(type="bool"),
+ "default_originate": dict(type="bool"),
+ "graceful_restart": dict(type="bool"),
+ "weight": dict(type="int"),
+ }
+
+ address_family_spec = {
+ "afi": dict(choices=["ipv4", "ipv6"], required=True),
+ "networks": dict(type="list", elements="dict", options=network_spec),
+ "redistribute": dict(
+ type="list", elements="dict", options=redistribute_spec
+ ),
+ "neighbors": dict(
+ type="list", elements="dict", options=af_neighbor_spec
+ ),
+ }
+
+ config_spec = {
+ "bgp_as": dict(type="int", required=True),
+ "router_id": dict(),
+ "log_neighbor_changes": dict(type="bool"),
+ "neighbors": dict(type="list", elements="dict", options=neighbor_spec),
+ "address_family": dict(
+ type="list", elements="dict", options=address_family_spec
+ ),
+ "redistribute": dict(
+ type="list", elements="dict", options=redistribute_spec
+ ),
+ "networks": dict(type="list", elements="dict", options=network_spec),
+ }
+
+ argument_spec = {
+ "config": dict(type="dict", options=config_spec),
+ "operation": dict(
+ default="merge", choices=["merge", "replace", "override", "delete"]
+ ),
+ }
+
+ module = NetworkModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+
+ try:
+ result = module.edit_config(config_filter="| section bgp")
+ except Exception as exc:
+ module.fail_json(msg=to_text(exc))
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py
new file mode 100644
index 00000000..7e64fdef
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py
@@ -0,0 +1,1332 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2021 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_bgp_address_family
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+module: eos_bgp_address_family
+short_description: Manages BGP address family resource module
+description: This module configures and manages the attributes of BGP AF on Arista
+ EOS platforms.
+version_added: 1.4.0
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+notes:
+- Tested against Arista EOS 4.23.0F
+- This module works with connection C(network_cli). See the L(EOS Platform Options,eos_platform_options).
+options:
+ config:
+ description: Configurations for BGP address family.
+ type: dict
+ suboptions:
+ as_number:
+ description: Autonomous system number.
+ type: str
+ address_family: &address_family
+ description: Enable address family and enter its config mode
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description: address family.
+ type: str
+ choices: ['ipv4', 'ipv6', 'evpn']
+ safi:
+ description: Address family type for ipv4.
+ type: str
+ choices: ['labeled-unicast', 'multicast']
+ bgp_params:
+ description: BGP parameters.
+ type: dict
+ suboptions:
+ additional_paths:
+ description: BGP additional-paths commands
+ type: str
+ choices: ['install', 'send', 'receive']
+ next_hop_address_family:
+ description: Next-hop address-family configuration
+ type: str
+ choices: ['ipv6']
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ redistribute_internal:
+ description: Redistribute internal BGP routes.
+ type: bool
+ route:
+ description: Configure route-map for route installation.
+ type: str
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: bool
+ neighbor:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ peer:
+ type: str
+ description: Neighbor address/ peer-group name.
+ activate:
+ description: Activate neighbor in the address family.
+ type: bool
+ additional_paths:
+ description: BGP additional-paths commands.
+ type: str
+ choices: ['send', 'receive']
+ default_originate:
+ description: Originate default route to this neighbor.
+ type: dict
+ suboptions:
+ route_map:
+ description: Route map reference.
+ type: str
+ always:
+ description: Always originate default route to this neighbor.
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: bool
+ next_hop_address_family:
+ description: Next-hop address-family configuration
+ type: str
+ choices: ['ipv6']
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ prefix_list:
+ description: Prefix list reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound prefix-list.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: prefix list name.
+ type: str
+ route_map:
+ description: Route map reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound route-map.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: Route map name.
+ type: str
+ weight:
+ description: Weight to assign.
+ type: int
+ encapsulation:
+ description: Default transport encapsulation for neighbor. Applicable for evpn address-family.
+ type: dict
+ suboptions:
+ transport:
+ description: MPLS/VXLAN transport.
+ type: str
+ choices: ['mpls', 'vxlan']
+ source_interface:
+ description: Source interface to update BGP next hop address. Applicable for mpls transport.
+ type: str
+ network:
+ description: configure routing for network.
+ type: list
+ elements: dict
+ suboptions:
+ route_map:
+ description: Route map reference.
+ type: str
+ address:
+ description: network address.
+ type: str
+ redistribute:
+ description: Redistribute routes in to BGP.
+ type: list
+ elements: dict
+ suboptions:
+ protocol:
+ description: Routes to be redistributed.
+ type: str
+ choices: ['isis', 'ospf3', 'dhcp']
+ route_map:
+ description: Route map reference.
+ type: str
+ isis_level:
+ description: Applicable for isis routes. Specify isis route level.
+ type: str
+ choices: ['level-1', 'level-2', 'level-1-2']
+ ospf_route:
+ description: ospf route options.
+ type: str
+ choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2']
+ route_target:
+ description: Route target
+ type: dict
+ suboptions:
+ mode:
+ description: route import or route export.
+ type: str
+ choices: ['both', 'import', 'export']
+ target:
+ description: route target
+ type: str
+ vrf:
+ description: name of the VRF in which BGP will be configured.
+ type: str
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section bgp).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed]
+ default: merged
+
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state
+
+# veos(config)#show running-config | section bgp
+# veos(config)#
+
+ - name: Merge provided configuration with device configuration
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv4"
+ redistribute:
+ - protocol: "ospf3"
+ ospf_route: "external"
+ network:
+ - address: "1.1.1.0/24"
+ - address: "1.5.1.0/24"
+ route_map: "MAP01"
+ - afi: "ipv6"
+ bgp_params:
+ additional_paths: "receive"
+ neighbor:
+ - peer: "peer2"
+ default_originate:
+ always: True
+ - afi: "ipv6"
+ redistribute:
+ - protocol: "isis"
+ isis_level: "level-2"
+ route_target:
+ mode: "export"
+ target: "33:11"
+ vrf: "vrft"
+ state: merged
+
+# After state:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# bgp additional-paths receive
+# neighbor peer2 activate
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv6
+# route-target export 33:11
+# redistribute isis level-2
+# veos(config-router-bgp)#
+
+# Module Execution:
+
+# "after": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# }
+# ],
+# "route_target": {
+# "mode": "export",
+# "target": "33:11"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "before": {},
+# "changed": true,
+# "commands": [
+# "router bgp 10",
+# "address-family ipv4",
+# "redistribute ospf3 match external",
+# "network 1.1.1.0/24",
+# "network 1.5.1.0/24 route-map MAP01",
+# "exit",
+# "address-family ipv6",
+# "neighbor peer2 default-originate always",
+# "bgp additional-paths receive",
+# "exit",
+# "vrf vrft",
+# "address-family ipv6",
+# "redistribute isis level-2",
+# "route-target export 33:11",
+# "exit",
+# "exit"
+# ],
+
+# Using replaced:
+
+# Before State:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# bgp additional-paths receive
+# neighbor peer2 activate
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv6
+# route-target export 33:11
+# redistribute isis level-2
+# veos(config-router-bgp)#
+#
+
+ - name: Replace
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv6"
+ vrf: "vrft"
+ redistribute:
+ - protocol: "ospf3"
+ ospf_route: "external"
+ - afi: "ipv6"
+ redistribute:
+ - protocol: "isis"
+ isis_level: "level-2"
+ state: replaced
+
+# After State:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# neighbor peer2 default-originate always
+# redistribute isis level-2
+# !
+# vrf vrft
+# address-family ipv6
+# redistribute ospf3 match external
+# veos(config-router-bgp)#
+#
+#
+# # Module Execution:
+#
+# "after": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "neighbor": [
+# {
+# "activate": true,
+# "peer": "1.1.1.1"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ],
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "before": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "neighbor": [
+# {
+# "activate": true,
+# "peer": "1.1.1.1"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "activate": true,
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# }
+# ],
+# "route_target": {
+# "mode": "export",
+# "target": "33:11"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "changed": true,
+# "commands": [
+# "router bgp 10",
+# "vrf vrft",
+# "address-family ipv6",
+# "redistribute ospf3 match external",
+# "no redistribute isis level-2",
+# "no route-target export 33:11",
+# "exit",
+# "exit",
+# "address-family ipv6",
+# "redistribute isis level-2",
+# "no neighbor peer2 activate",
+# "no bgp additional-paths receive",
+# "exit"
+# ],
+
+# Using overridden (overriding af at global context):
+# Before state:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# neighbor peer2 default-originate always
+# redistribute isis level-2
+# !
+# vrf vrft
+# address-family ipv6
+# redistribute ospf3 match external
+# veos(config-router-bgp)#
+
+ - name: Overridden
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv4"
+ bgp_params:
+ additional_paths: "receive"
+ neighbor:
+ - peer: "peer2"
+ default_originate:
+ always: True
+ state: overridden
+
+# After State:
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv6
+# redistribute ospf3 match external
+# veos(config-router-bgp)#
+#
+# Module Execution:
+#
+# "after": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ],
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "before": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "neighbor": [
+# {
+# "activate": true,
+# "peer": "1.1.1.1"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ],
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "changed": true,
+# "commands": [
+# "router bgp 10",
+# "address-family ipv4",
+# "no redistribute ospf3 match external",
+# "no network 1.1.1.0/24",
+# "no network 1.5.1.0/24 route-map MAP01",
+# "neighbor peer2 default-originate always",
+# "no neighbor 1.1.1.1 activate",
+# "bgp additional-paths receive",
+# "exit",
+# "no address-family ipv6"
+# ],
+
+# using Overridden (overridding af in vrf context):
+
+# Before State:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# no neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv6
+# route-target export 33:11
+# redistribute isis level-2
+# redistribute ospf3 match external
+# veos(config-router-bgp)#
+
+
+ - name: Overridden
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv4"
+ bgp_params:
+ additional_paths: "receive"
+ neighbor:
+ - peer: "peer2"
+ default_originate:
+ always: True
+ vrf: vrft
+ state: overridden
+
+# After State:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv4
+# bgp additional-paths receive
+# veos(config-router-bgp)#
+#
+# Module Execution:
+#
+# "after": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "before": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# },
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ],
+# "route_target": {
+# "mode": "export",
+# "target": "33:11"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "changed": true,
+# "commands": [
+# "router bgp 10",
+# "vrf vrft",
+# "address-family ipv4",
+# "neighbor peer2 default-originate always",
+# "bgp additional-paths receive",
+# "exit",
+# "exit",
+# " vrf vrft",
+# "no address-family ipv6"
+# ],
+
+# Using Deleted:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# no neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# !
+# vrf vrft
+# address-family ipv4
+# bgp additional-paths receive
+# veos(config-router-bgp)#
+
+ - name: Delete
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv6"
+ vrf: "vrft"
+ - afi: "ipv6"
+ state: deleted
+
+# After State:
+
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# no neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# vrf vrft
+# address-family ipv4
+# bgp additional-paths receive
+# veos(config-router-bgp)#
+#
+# Module Execution:
+#
+# "after": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+# "before": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ]
+# },
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+
+# Using parsed:
+
+# parsed_bgp_address_family.cfg :
+
+# router bgp 10
+# neighbor n2 peer-group
+# neighbor n2 next-hop-unchanged
+# neighbor n2 maximum-routes 12000
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# redistribute ospf3 match external
+# !
+# address-family ipv6
+# no bgp additional-paths receive
+# neighbor n2 next-hop-unchanged
+# redistribute isis level-2
+# !
+# vrf bgp_10
+# ip access-group acl01
+# ucmp fec threshold trigger 33 clear 22 warning-only
+# !
+# address-family ipv4
+# route-target import 20:11
+# !
+# vrf vrft
+# address-family ipv4
+# bgp additional-paths receive
+# !
+# address-family ipv6
+# redistribute ospf3 match external
+
+ - name: parse configs
+ arista.eos.eos_bgp_address_family:
+ running_config: "{{ lookup('file', './parsed_bgp_address_family.cfg') }}"
+ state: parsed
+
+# Module Execution:
+# "parsed": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "neighbor": [
+# {
+# "next_hop_unchanged": true,
+# "peer": "n2"
+# }
+# ],
+# "redistribute": [
+# {
+# "isis_level": "level-2",
+# "protocol": "isis"
+# }
+# ]
+# },
+# {
+# "afi": "ipv4",
+# "route_target": {
+# "mode": "import",
+# "target": "20:11"
+# },
+# "vrf": "bgp_10"
+# },
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "vrf": "vrft"
+# },
+# {
+# "afi": "ipv6",
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ],
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# }
+# }
+
+# Using gathered:
+
+# Device config:
+# veos(config-router-bgp)#show running-config | section bgp
+# router bgp 10
+# neighbor peer2 peer-group
+# neighbor peer2 maximum-routes 12000
+# neighbor 1.1.1.1 maximum-routes 12000
+# !
+# address-family ipv4
+# bgp additional-paths receive
+# neighbor peer2 default-originate always
+# no neighbor 1.1.1.1 activate
+# network 1.1.1.0/24
+# network 1.5.1.0/24 route-map MAP01
+# redistribute ospf3 match external
+# !
+# vrf vrft
+# address-family ipv4
+# bgp additional-paths receive
+# veos(config-router-bgp)#
+
+ - name: gather configs
+ arista.eos.eos_bgp_address_family:
+ state: gathered
+
+# Module Execution:
+# "gathered": {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "neighbor": [
+# {
+# "default_originate": {
+# "always": true
+# },
+# "peer": "peer2"
+# }
+# ],
+# "network": [
+# {
+# "address": "1.1.1.0/24"
+# },
+# {
+# "address": "1.5.1.0/24",
+# "route_map": "MAP01"
+# }
+# ],
+# "redistribute": [
+# {
+# "ospf_route": "external",
+# "protocol": "ospf3"
+# }
+# ]
+# },
+# {
+# "afi": "ipv4",
+# "bgp_params": {
+# "additional_paths": "receive"
+# },
+# "vrf": "vrft"
+# }
+# ],
+# "as_number": "10"
+# },
+
+# using rendered:
+
+ - name: Render
+ arista.eos.eos_bgp_address_family:
+ config:
+ as_number: "10"
+ address_family:
+ - afi: "ipv4"
+ redistribute:
+ - protocol: "ospf3"
+ ospf_route: "external"
+ network:
+ - address: "1.1.1.0/24"
+ - address: "1.5.1.0/24"
+ route_map: "MAP01"
+ - afi: "ipv6"
+ bgp_params:
+ additional_paths: "receive"
+ neighbor:
+ - peer: "peer2"
+ default_originate:
+ always: True
+ - afi: "ipv6"
+ redistribute:
+ - protocol: "isis"
+ isis_level: "level-2"
+ route_target:
+ mode: "export"
+ target: "33:11"
+ vrf: "vrft"
+
+ state: rendered
+
+# Module Execution:
+
+# "rendered": [
+# "router bgp 10",
+# "address-family ipv4",
+# "redistribute ospf3 match external",
+# "network 1.1.1.0/24",
+# "network 1.5.1.0/24 route-map MAP01",
+# "exit",
+# "address-family ipv6",
+# "neighbor peer2 default-originate always",
+# "bgp additional-paths receive",
+# "exit",
+# "vrf vrft",
+# "address-family ipv6",
+# "redistribute isis level-2",
+# "route-target export 33:11",
+# "exit",
+# "exit"
+# ]
+#
+
+"""
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_address_family.bgp_address_family import (
+ Bgp_afArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.bgp_address_family.bgp_address_family import (
+ Bgp_af,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ module = AnsibleModule(
+ argument_spec=Bgp_afArgs.argument_spec,
+ mutually_exclusive=[],
+ required_if=[],
+ supports_check_mode=False,
+ )
+
+ result = Bgp_af(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py
new file mode 100644
index 00000000..319d435d
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py
@@ -0,0 +1,2312 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_bgp_global
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+module: eos_bgp_global
+short_description: Manages BGP global resource module
+description: This module configures and manages the attributes of BGP global on Arista
+ EOS platforms.
+version_added: 1.4.0
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+notes:
+- Tested against Arista EOS 4.23.0F
+- This module works with connection C(network_cli). See the L(EOS Platform Options,eos_platform_options).
+options:
+ config:
+ description: A list of configurations for BGP global.
+ type: dict
+ suboptions:
+ as_number:
+ description: Autonomous system number.
+ type: str
+ aggregate_address:
+ description: Configure aggregate address.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description: ipv4/ipv6 address prefix.
+ type: str
+ advertise_only:
+ description: Advertise without installing the generated blackhole route in
+ FIB.
+ type: bool
+ as_set:
+ description: Generate autonomous system set path information.
+ type: bool
+ attribute_map:
+ description: Name of the route map used to set the attribute of the
+ aggregate route.
+ type: str
+ match_map:
+ description: Name of the route map used to filter the contributors of the
+ aggregate route.
+ type: str
+ summary_only:
+ description: Filters all more-specific routes from updates.
+ type: bool
+ bgp_params:
+ description: BGP parameters.
+ type: dict
+ suboptions:
+ additional_paths:
+ description: BGP additional-paths commands
+ type: str
+ choices: ['install', 'send', 'receive']
+ advertise_inactive:
+ description: Advertise BGP routes even if they are inactive in RIB.
+ type: bool
+ allowas_in:
+ description: Allow local-as in updates.
+ type: dict
+ suboptions:
+ set:
+ description: When True, it is set.
+ type: bool
+ count:
+ description: Number of local ASNs allowed in a BGP update.
+ type: int
+ always_compare_med:
+ description: BGP Always Compare MED
+ type: bool
+ asn:
+ description: AS Number notation.
+ type: str
+ choices: ['asdot', 'asplain']
+ auto_local_addr:
+ description: Automatically determine the local address to be used
+ for the non-transport AF.
+ type: bool
+ bestpath:
+ description: Select the bestpath selection algorithim for BGP routes.
+ type: dict
+ suboptions:
+ as_path:
+ description: Select the bestpath selection based on as-path.
+ type: str
+ choices: ['ignore', 'multipath_relax']
+ ecmp_fast:
+ description: Tie-break BGP paths in a ECMP group based on the order of arrival.
+ type: bool
+ med:
+ description: MED attribute
+ type: dict
+ suboptions:
+ confed:
+ description: MED Confed.
+ type: bool
+ missing_as_worst:
+ description: MED missing-as-worst.
+ type: bool
+ skip:
+ description: skip one of the tie breaking rules in the bestpath selection.
+ type: bool
+ tie_break:
+ description: Configure the tie-break option for BGP bestpath selection.
+ choices: ['cluster_list_length', 'router_id']
+ type: str
+ client_to_client:
+ description: client to client configuration.
+ type: bool
+ cluster_id:
+ description: Cluster ID of this router acting as a route reflector.
+ type: str
+ confederation:
+ description: confederation.
+ type: dict
+ suboptions:
+ identifier:
+ description: Confederation identifier.
+ type: str
+ peers:
+ description: Confederation peers.
+ type: str
+ control_plan_filter:
+ description: Control plane filter for BGP.
+ type: bool
+ convergence:
+ description: Bgp convergence parameters.
+ type: dict
+ suboptions:
+ slow_peer:
+ description: Maximum amount of time to wait for slow peers to estabilsh session.
+ type: bool
+ time:
+ description: time in secs
+ type: int
+ default:
+ description: Default neighbor configuration commands.
+ type: str
+ choices: ['ipv4_unicast', 'ipv6_unicast']
+ enforce_first_as:
+ description: Enforce the First AS for EBGP routes(default).
+ type: bool
+ host_routes:
+ description: BGP host routes configuration.
+ type: bool
+ labeled_unicast:
+ description: Labeled Unicast.
+ type: str
+ choices: ['ip', 'tunnel']
+ listen:
+ description: BGP listen.
+ type: dict
+ suboptions:
+ limit:
+ description: Set limit on the number of dynamic BGP peers allowed.
+ type: int
+ range:
+ description: Subnet Range to be associated with the peer-group.
+ type: dict
+ suboptions:
+ address:
+ description: Address prefix
+ type: str
+ peer_group:
+ description: Name of peer group.
+ type: dict
+ suboptions:
+ name:
+ description: name.
+ type: str
+ peer_filter:
+ description: Name of peer filter.
+ type: str
+ remote_as:
+ description: Neighbor AS number
+ type: str
+ log_neighbor_changes:
+ description: Log neighbor up/down events.
+ type: bool
+ missing_policy:
+ description: Missing policy override configuration commands.
+ type: dict
+ suboptions:
+ direction:
+ description: Missing policy direction options.
+ type: str
+ choices: ['in', 'out']
+ action:
+ description: Missing policy action options.
+ type: str
+ choices: ['deny', 'permit', 'deny-in-out']
+ monitoring:
+ description: Enable Bgp monitoring for all/specified stations.
+ type: bool
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ redistribute_internal:
+ description: Redistribute internal BGP routes.
+ type: bool
+ route:
+ description: Configure route-map for route installation.
+ type: str
+ route_reflector:
+ description: Configure route reflector options
+ type: dict
+ suboptions:
+ set:
+ description: When True route_reflector is set.
+ type: bool
+ preserve:
+ description: preserve route attributes, overwriting route-map changes
+ type: bool
+ transport:
+ description: Configure transport port for TCP session
+ type: int
+ default_metric:
+ description: Default metric.
+ type: int
+ distance:
+ description: Define an administrative distance.
+ type: dict
+ suboptions:
+ external:
+ description: distance for external routes.
+ type: int
+ internal:
+ description: distance for internal routes.
+ type: int
+ local:
+ description: distance for local routes.
+ type: int
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: dict
+ suboptions:
+ set:
+ description: When True, graceful restart is set.
+ type: bool
+ restart_time:
+ description: Set the max time needed to restart and come back up.
+ type: int
+ stalepath_time:
+ description: Set the max time to hold onto restarting peer stale paths.
+ type: int
+ graceful_restart_helper:
+ description: Enable graceful restart helper mode.
+ type: bool
+ access_group:
+ description: ip/ipv6 access list configuration.
+ type: dict
+ suboptions:
+ afi:
+ description: Specify ip/ipv6.
+ type: str
+ choices: ['ipv4', 'ipv6']
+ acl_name:
+ description: access list name.
+ type: str
+ direction:
+ description: direction of packets.
+ type: str
+ maximum_paths:
+ description: Maximum number of equal cost paths.
+ type: dict
+ suboptions:
+ max_equal_cost_paths:
+ description: Value for maximum number of equal cost paths.
+ type: int
+ max_installed_ecmp_paths:
+ description: Value for maximum number of installed ECMP routes.
+ type: int
+ monitoring:
+ description: BGP monitoring protocol configuration.
+ type: dict
+ suboptions:
+ port:
+ description: Configure the BGP monitoring protocol port number <1024-65535>.
+ type: int
+ received:
+ description: BGP monitoring protocol received route selection.
+ type: str
+ choices: ['post_policy', 'pre_policy']
+ station:
+ description: BGP monitoring station configuration.
+ type: str
+ timestamp:
+ description: BGP monitoring protocol Per-Peer Header timestamp behavior.
+ type: str
+ choices: ['none', 'send_time']
+ neighbor:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ peer:
+ type: str
+ description: Neighbor address or peer-group.
+ additional_paths:
+ description: BGP additional-paths commands.
+ type: str
+ choices: ['send', 'receive']
+ allowas_in:
+ description: Allow local-as in updates.
+ type: dict
+ suboptions:
+ set:
+ description: When True, it is set.
+ type: bool
+ count:
+ description: Number of local ASNs allowed in a BGP update.
+ type: int
+ auto_local_addr:
+ description: Automatically determine the local address to be used
+ for the non-transport AF.
+ type: bool
+ default_originate:
+ description: Originate default route to this neighbor.
+ type: dict
+ suboptions:
+ route_map:
+ description: Route map reference.
+ type: str
+ always:
+ description: Always originate default route to this neighbor.
+ type: bool
+ description:
+ description: Text describing the neighbor.
+ type: str
+ dont_capability_negotiate:
+ description: Donot perform Capability Negotiation with this
+ neighbor.
+ type: bool
+ ebgp_multihop:
+ description: Allow BGP connections to indirectly connected
+ external peers.
+ type: dict
+ suboptions:
+ ttl:
+ description: Time-to-live in the range 1-255 hops.
+ type: int
+ set:
+ description: If True, ttl is not set.
+ type: bool
+ enforce_first_as:
+ description: Enforce the First AS for EBGP routes(default).
+ type: bool
+ export_localpref:
+ description: Override localpref when exporting to an internal
+ peer.
+ type: int
+ fall_over:
+ description: Configure BFD protocol options for this peer.
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: bool
+ graceful_restart_helper:
+ description: Enable graceful restart helper mode.
+ type: bool
+ idle_restart_timer:
+ description: Neighbor idle restart timer.
+ type: int
+ import_localpref:
+ description: Override localpref when importing from an external
+ peer.
+ type: int
+ link_bandwidth:
+ description: Enable link bandwidth community for routes to this
+ peer.
+ type: dict
+ suboptions:
+ set:
+ description: If True, set link bandwidth
+ type: bool
+ auto:
+ description: Enable link bandwidth auto generation for routes from this peer.
+ type: bool
+ default:
+ description: Enable link bandwidth default generation for routes from this
+ peer.
+ type: str
+ update_delay:
+ description: Delay outbound route updates.
+ type: int
+ local_as:
+ description: Configure local AS number advertised to peer.
+ type: dict
+ suboptions:
+ as_number:
+ description: AS number.
+ type: str
+ fallback:
+ description: Prefer router AS Number over local AS Number.
+ type: bool
+ local_v6_addr:
+ description: The local IPv6 address of the neighbor in A:B:C:D:E:F:G:H format.
+ type: str
+ maximum_accepted_routes:
+ description: Maximum number of routes accepted from this peer.
+ type: dict
+ suboptions:
+ count:
+ description: Maximum number of accepted routes (0 means unlimited).
+ type: int
+ warning_limit:
+ description: Maximum number of accepted routes after which a warning is issued.
+ (0 means never warn)
+ type: int
+ maximum_received_routes:
+ description: Maximum number of routes received from this peer.
+ type: dict
+ suboptions:
+ count:
+ description: Maximum number of routes (0 means unlimited).
+ type: int
+ warning_limit:
+ description: Percentage of maximum-routes at which warning is to be issued.
+ type: dict
+ suboptions:
+ limit_count:
+ description: Number of routes at which to warn.
+ type: int
+ limit_percent:
+ description: Percentage of maximum number of routes at which to warn( 1-100).
+ type: int
+ warning_only:
+ description: Only warn, no restart, if max route limit exceeded.
+ type: bool
+ metric_out:
+ description: MED value to advertise to peer.
+ type: int
+ monitoring:
+ description: Enable BGP Monitoring Protocol for this peer.
+ type: bool
+ next_hop_self:
+ description: Always advertise this router address as the BGP
+ next hop
+ type: bool
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ next_hop_v6_address:
+ description: IPv6 next-hop address for the neighbor
+ type: str
+ out_delay:
+ description: Delay outbound route updates.
+ type: int
+ encryption_password:
+ description: Password to use in computation of MD5 hash.
+ type: dict
+ suboptions:
+ type:
+ description: Encryption type.
+ type: int
+ choices: [0, 7]
+ password:
+ description: password (up to 80 chars).
+ type: str
+ remote_as:
+ description: Neighbor Autonomous System.
+ type: str
+ remove_private_as:
+ description: Remove private AS number from updates to this peer.
+ type: dict
+ suboptions:
+ set:
+ description: If True, set remove_private_as.
+ type: bool
+ all:
+ description: Remove private AS number.
+ type: bool
+ replace_as:
+ description: Replace private AS number with local AS number.
+ type: bool
+ peer_group:
+ description: Name of the peer-group.
+ type: str
+
+ prefix_list:
+ description: Prefix list reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound prefix-list.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: prefix list name.
+ type: str
+ route_map:
+ description: Route map reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound route-map.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: Route map name.
+ type: str
+ route_reflector_client:
+ description: Configure peer as a route reflector client.
+ type: bool
+ route_to_peer:
+ description: Use routing table information to reach the peer.
+ type: bool
+ send_community:
+ description: Send community attribute to this neighbor.
+ type: dict
+ suboptions:
+ community_attribute:
+ description: Type of community attributes to send to this neighbor.
+ type: str
+ sub_attribute:
+ description: Attribute to be sent to the neighbor.
+ type: str
+ choices: ['extended', 'link-bandwidth', 'standard']
+ link_bandwidth_attribute:
+ description: cumulative/aggregate attribute to be sent.
+ type: str
+ choices: ['aggregate', 'divide']
+ speed:
+ description: Reference link speed in bits/second
+ type: str
+ divide:
+ description: link-bandwidth divide attribute.
+ type: str
+ choices: ['equal', 'ratio']
+ shut_down:
+ description: Administratively shut down this neighbor.
+ type: bool
+ soft_recognition:
+ description: Configure how to handle routes that fail import.
+ type: str
+ choices: ['all', 'None']
+ timers:
+ description: Timers.
+ type: dict
+ suboptions:
+ keepalive:
+ description: Keep Alive Interval in secs.
+ type: int
+ holdtime:
+ description: Hold time in secs.
+ type: int
+ transport:
+ description: Configure transport options for TCP session.
+ type: dict
+ suboptions:
+ connection_mode:
+ description: Configure connection-mode for TCP session.
+ type: str
+ remote_port:
+ description: Configure BGP peer TCP port to connect to.
+ type: int
+ ttl:
+ description: BGP ttl security check
+ type: int
+ update_source:
+ description: Specify the local source interface for peer BGP
+ sessions.
+ type: str
+ weight:
+ description: Weight to assign.
+ type: int
+ network:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description: address prefix.
+ type: str
+ route_map:
+ description: Name of route map.
+ type: str
+ redistribute:
+ description: Redistribute routes in to BGP.
+ type: list
+ elements: dict
+ suboptions:
+ protocol:
+ description: Routes to be redistributed.
+ type: str
+ choices: ['isis', 'ospf3', 'ospf', 'attached-host', 'connected', 'rip', 'static']
+ route_map:
+ description: Route map reference.
+ type: str
+ isis_level:
+ description: Applicable for isis routes. Specify isis route level.
+ type: str
+ choices: ['level-1', 'level-2', 'level-1-2']
+ ospf_route:
+ description: ospf route options.
+ type: str
+ choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2']
+ router_id:
+ description: Router id.
+ type: str
+ route_target:
+ description: Route target.
+ type: dict
+ suboptions:
+ action:
+ description: Route action.
+ type: str
+ choices: ['both', 'import', 'export']
+ target:
+ description: Route Target.
+ type: str
+ shutdown:
+ description: When True, shut down BGP.
+ type: bool
+ timers:
+ description: Timers.
+ type: dict
+ suboptions:
+ keepalive:
+ description: Keep Alive Interval in secs.
+ type: int
+ holdtime:
+ description: Hold time in secs.
+ type: int
+ ucmp:
+ description: Configure unequal cost multipathing.
+ type: dict
+ suboptions:
+ fec:
+ description: Configure UCMP fec utilization threshold.
+ type: dict
+ suboptions:
+ trigger:
+ description: UCMP fec utilization too high threshold.
+ type: int
+ clear:
+ description: UCMP FEC utilization Clear thresholds.
+ type: int
+ link_bandwidth:
+ description: Configure link-bandwidth propagation delay.
+ type: dict
+ suboptions:
+ mode:
+ description: UCMP link bandwidth mode
+ type: str
+ choices: ['encoding_weighted', 'recursive']
+ update_delay:
+ description: Link Bandwidth Advertisement delay.
+ type: int
+ mode:
+ description: UCMP mode.
+ type: dict
+ suboptions:
+ set:
+ description: If True, ucmp mode is set to 1.
+ type: bool
+ nexthops:
+ description: Value for total number UCMP nexthops.
+ type: int
+ update:
+ description: Configure BGP update generation.
+ type: dict
+ suboptions:
+ wait_for:
+ description: wait for options before converge or synchronize.
+ type: str
+ choices: ['wait_for_convergence', 'wait_install']
+ batch_size:
+ description: batch size for FIB route acknowledgements.
+ type: int
+ vlan:
+ description: Configure MAC VRF BGP for single VLAN support.
+ type: int
+ vlan_aware_bundle:
+ description: Configure MAC VRF BGP for multiple VLAN support.
+ type: str
+ vrfs:
+ description: Configure BGP in a VRF.
+ type: list
+ elements: dict
+ suboptions:
+ vrf:
+ description: VRF name.
+ type: str
+ aggregate_address:
+ description: Configure aggregate address.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description: ipv4/ipv6 address prefix.
+ type: str
+ advertise_only:
+ description: Advertise without installing the generated blackhole route in
+ FIB.
+ type: bool
+ as_set:
+ description: Generate autonomous system set path information.
+ type: bool
+ attribute_map:
+ description: Name of the route map used to set the attribute of the
+ aggregate route.
+ type: str
+ match_map:
+ description: Name of the route map used to filter the contributors of the
+ aggregate route.
+ type: str
+ summary_only:
+ description: Filters all more-specific routes from updates.
+ type: bool
+ bgp_params:
+ description: BGP parameters.
+ type: dict
+ suboptions:
+ additional_paths:
+ description: BGP additional-paths commands
+ type: str
+ choices: ['install', 'send', 'receive']
+ advertise_inactive:
+ description: Advertise BGP routes even if they are inactive in RIB.
+ type: bool
+ allowas_in:
+ description: Allow local-as in updates.
+ type: dict
+ suboptions:
+ set:
+ description: When True, it is set.
+ type: bool
+ count:
+ description: Number of local ASNs allowed in a BGP update.
+ type: int
+ always_compare_med:
+ description: BGP Always Compare MED
+ type: bool
+ asn:
+ description: AS Number notation.
+ type: str
+ choices: ['asdot', 'asplain']
+ auto_local_addr:
+ description: Automatically determine the local address to be used
+ for the non-transport AF.
+ type: bool
+ bestpath:
+ description: Select the bestpath selection algorithim for BGP routes.
+ type: dict
+ suboptions:
+ as_path:
+ description: Select the bestpath selection based on as-path.
+ type: str
+ choices: ['ignore', 'multipath_relax']
+ ecmp_fast:
+ description: Tie-break BGP paths in a ECMP group based on the order of arrival.
+ type: bool
+ med:
+ description: MED attribute
+ type: dict
+ suboptions:
+ confed:
+ description: MED Confed.
+ type: bool
+ missing_as_worst:
+ description: MED missing-as-worst.
+ type: bool
+ skip:
+ description: skip one of the tie breaking rules in the bestpath selection.
+ type: bool
+ tie_break:
+ description: Configure the tie-break option for BGP bestpath selection.
+ choices: ['cluster_list_length', 'router_id']
+ type: str
+ client_to_client:
+ description: client to client configuration.
+ type: bool
+ cluster_id:
+ description: Cluster ID of this router acting as a route reflector.
+ type: str
+ confederation:
+ description: confederation.
+ type: dict
+ suboptions:
+ identifier:
+ description: Confederation identifier.
+ type: str
+ peers:
+ description: Confederation peers.
+ type: str
+ control_plane_filter:
+ description: Control plane filter for BGP.
+ type: bool
+ convergence:
+ description: Bgp convergence parameters.
+ type: dict
+ suboptions:
+ slow_peer:
+ description: Maximum amount of time to wait for slow peers to estabilsh session.
+ type: bool
+ time:
+ description: time in secs
+ type: int
+ default:
+ description: Default neighbor configuration commands.
+ type: str
+ choices: ['ipv4_unicast', 'ipv6_unicast']
+ enforce_first_as:
+ description: Enforce the First AS for EBGP routes(default).
+ type: bool
+ host_routes:
+ description: BGP host routes configuration.
+ type: bool
+ labeled_unicast:
+ description: Labeled Unicast.
+ type: str
+ choices: ['ip', 'tunnel']
+ listen:
+ description: BGP listen.
+ type: dict
+ suboptions:
+ limit:
+ description: Set limit on the number of dynamic BGP peers allowed.
+ type: int
+ range:
+ description: Subnet Range to be associated with the peer-group.
+ type: dict
+ suboptions:
+ address:
+ description: Address prefix
+ type: str
+ peer_group:
+ description: Name of peer group.
+ type: dict
+ suboptions:
+ name:
+ description: name.
+ type: str
+ peer_filter:
+ description: Name of peer filter.
+ type: str
+ remote_as:
+ description: Neighbor AS number
+ type: str
+ log_neighbor_changes:
+ description: Log neighbor up/down events.
+ type: bool
+ missing_policy:
+ description: Missing policy override configuration commands.
+ type: dict
+ suboptions:
+ direction:
+ description: Missing policy direction options.
+ type: str
+ choices: ['in', 'out']
+ action:
+ description: Missing policy action options.
+ type: str
+ choices: ['deny', 'permit', 'deny-in-out']
+ monitoring:
+ description: Enable Bgp monitoring for all/specified stations.
+ type: bool
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ redistribute_internal:
+ description: Redistribute internal BGP routes.
+ type: bool
+ route:
+ description: Configure route-map for route installation.
+ type: str
+ route_reflector:
+ description: Configure route reflector options
+ type: dict
+ suboptions:
+ set:
+ description: When True route_reflector is set.
+ type: bool
+ preserve:
+ description: preserve route attributes, overwriting route-map changes
+ type: bool
+ transport:
+ description: Configure transport port for TCP session
+ type: int
+ default_metric:
+ description: Default metric.
+ type: int
+ distance:
+ description: Define an administrative distance.
+ type: dict
+ suboptions:
+ external:
+ description: distance for external routes.
+ type: int
+ internal:
+ description: distance for internal routes.
+ type: int
+ local:
+ description: distance for local routes.
+ type: int
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: dict
+ suboptions:
+ set:
+ description: When True, graceful restart is set.
+ type: bool
+ restart_time:
+ description: Set the max time needed to restart and come back up.
+ type: int
+ stalepath_time:
+ description: Set the max time to hold onto restarting peer stale paths.
+ type: int
+ graceful_restart_helper:
+ description: Enable graceful restart helper mode.
+ type: bool
+ access_group:
+ description: ip/ipv6 access list configuration.
+ type: dict
+ suboptions:
+ afi:
+ description: Specify ip/ipv6.
+ type: str
+ choices: ['ip', 'ipv6']
+ acl_name:
+ description: access list name.
+ type: str
+ direction:
+ description: direction of packets.
+ type: str
+ maximum_paths:
+ description: Maximum number of equal cost paths.
+ type: dict
+ suboptions:
+ max_equal_cost_paths:
+ description: Value for maximum number of equal cost paths.
+ type: int
+ max_installed_ecmp_paths:
+ description: Value for maximum number of installed ECMP routes.
+ type: int
+ neighbor:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ peer:
+ type: str
+ description: Neighbor address or peer group.
+ additional_paths:
+ description: BGP additional-paths commands.
+ type: str
+ choices: ['send', 'receive']
+ allowas_in:
+ description: Allow local-as in updates.
+ type: dict
+ suboptions:
+ set:
+ description: When True, it is set.
+ type: bool
+ count:
+ description: Number of local ASNs allowed in a BGP update.
+ type: int
+ auto_local_addr:
+ description: Automatically determine the local address to be used
+ for the non-transport AF.
+ type: bool
+ default_originate:
+ description: Originate default route to this neighbor.
+ type: dict
+ suboptions:
+ route_map:
+ description: Route map reference.
+ type: str
+ always:
+ description: Always originate default route to this neighbor.
+ type: bool
+ description:
+ description: Text describing the neighbor.
+ type: str
+ dont_capability_negotiate:
+ description: Donot perform Capability Negotiation with this
+ neighbor.
+ type: bool
+ ebgp_multihop:
+ description: Allow BGP connections to indirectly connected
+ external peers.
+ type: dict
+ suboptions:
+ ttl:
+ description: Time-to-live in the range 1-255 hops.
+ type: int
+ set:
+ description: If True, ttl is not set.
+ type: bool
+ enforce_first_as:
+ description: Enforce the First AS for EBGP routes(default).
+ type: bool
+ export_localpref:
+ description: Override localpref when exporting to an internal
+ peer.
+ type: int
+ fall_over:
+ description: Configure BFD protocol options for this peer.
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: bool
+ graceful_restart_helper:
+ description: Enable graceful restart helper mode.
+ type: bool
+ idle_restart_timer:
+ description: Neighbor idle restart timer.
+ type: int
+ import_localpref:
+ description: Override localpref when importing from an external
+ peer.
+ type: int
+ link_bandwidth:
+ description: Enable link bandwidth community for routes to this
+ peer.
+ type: dict
+ suboptions:
+ set:
+ description: If True, set link bandwidth
+ type: bool
+ auto:
+ description: Enable link bandwidth auto generation for routes from this peer.
+ type: bool
+ default:
+ description: Enable link bandwidth default generation for routes from this
+ peer.
+ type: str
+ update_delay:
+ description: Delay outbound route updates.
+ type: int
+ local_as:
+ description: Configure local AS number advertised to peer.
+ type: dict
+ suboptions:
+ as_number:
+ description: AS number.
+ type: str
+ fallback:
+ description: Prefer router AS Number over local AS Number.
+ type: bool
+ local_v6_addr:
+ description: The local IPv6 address of the neighbor in A:B:C:D:E:F:G:H format.
+ type: str
+ maximum_accepted_routes:
+ description: Maximum number of routes accepted from this peer.
+ type: dict
+ suboptions:
+ count:
+ description: Maximum number of accepted routes (0 means unlimited).
+ type: int
+ warning_limit:
+ description: Maximum number of accepted routes after which a warning is issued.
+ (0 means never warn)
+ type: int
+ maximum_received_routes:
+ description: Maximum number of routes received from this peer.
+ type: dict
+ suboptions:
+ count:
+ description: Maximum number of routes (0 means unlimited).
+ type: int
+ warning_limit:
+ description: Percentage of maximum-routes at which warning is to be issued.
+ type: dict
+ suboptions:
+ limit_count:
+ description: Number of routes at which to warn.
+ type: int
+ limit_percent:
+ description: Percentage of maximum number of routes at which to warn( 1-100).
+ type: int
+ warning_only:
+ description: Only warn, no restart, if max route limit exceeded.
+ type: bool
+ metric_out:
+ description: MED value to advertise to peer.
+ type: int
+ monitoring:
+ description: Enable BGP Monitoring Protocol for this peer.
+ type: bool
+ next_hop_self:
+ description: Always advertise this router address as the BGP
+ next hop
+ type: bool
+ next_hop_unchanged:
+ description: Preserve original nexthop while advertising routes to
+ eBGP peers.
+ type: bool
+ next_hop_v6_address:
+ description: IPv6 next-hop address for the neighbor
+ type: str
+ out_delay:
+ description: Delay outbound route updates.
+ type: int
+ encryption_password:
+ description: Password to use in computation of MD5 hash.
+ type: dict
+ suboptions:
+ type:
+ description: Encryption type.
+ type: int
+ choices: [0, 7]
+ password:
+ description: password (up to 80 chars).
+ type: str
+ remote_as:
+ description: Neighbor Autonomous System.
+ type: str
+ remove_private_as:
+ description: Remove private AS number from updates to this peer.
+ type: dict
+ suboptions:
+ set:
+ description: If True, set remove_private_as.
+ type: bool
+ all:
+ description: Remove private AS number.
+ type: bool
+ replace_as:
+ description: Replace private AS number with local AS number.
+ type: bool
+ peer_group:
+ description: Name of the peer-group.
+ type: str
+
+ prefix_list:
+ description: Prefix list reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound prefix-list.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: prefix list name.
+ type: str
+ route_map:
+ description: Route map reference.
+ type: dict
+ suboptions:
+ direction:
+ description: Configure an inbound/outbound route-map.
+ type: str
+ choices: ['in', 'out']
+ name:
+ description: Route map name.
+ type: str
+ route_reflector_client:
+ description: Configure peer as a route reflector client.
+ type: bool
+ route_to_peer:
+ description: Use routing table information to reach the peer.
+ type: bool
+ send_community:
+ description: Send community attribute to this neighbor.
+ type: dict
+ suboptions:
+ community_attribute:
+ description: Type of community attributes to send to this neighbor.
+ type: str
+ sub_attribute:
+ description: Attribute to be sent to the neighbor.
+ type: str
+ choices: ['extended', 'link-bandwidth', 'standard']
+ link_bandwidth_attribute:
+ description: cumulative/aggregate attribute to be sent.
+ type: str
+ choices: ['aggregate', 'divide']
+ speed:
+ description: Reference link speed in bits/second
+ type: str
+ divide:
+ description: link-bandwidth divide attribute.
+ type: str
+ choices: ['equal', 'ratio']
+ shut_down:
+ description: Administratively shut down this neighbor.
+ type: bool
+ soft_recognition:
+ description: Configure how to handle routes that fail import.
+ type: str
+ choices: ['all', 'None']
+ timers:
+ description: Timers.
+ type: dict
+ suboptions:
+ keepalive:
+ description: Keep Alive Interval in secs.
+ type: int
+ holdtime:
+ description: Hold time in secs.
+ type: int
+ transport:
+ description: Configure transport options for TCP session.
+ type: dict
+ suboptions:
+ connection_mode:
+ description: Configure connection-mode for TCP session.
+ type: str
+ remote_port:
+ description: Configure BGP peer TCP port to connect to.
+ type: int
+ ttl:
+ description: BGP ttl security check
+ type: int
+ update_source:
+ description: Specify the local source interface for peer BGP
+ sessions.
+ type: str
+ weight:
+ description: Weight to assign.
+ type: int
+ network:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description: address prefix.
+ type: str
+ route_map:
+ description: Name of route map.
+ type: str
+ redistribute:
+ description: Redistribute routes in to BGP.
+ type: list
+ elements: dict
+ suboptions:
+ protocol:
+ description: Routes to be redistributed.
+ type: str
+ choices: ['isis', 'ospf3', 'ospf', 'attached-host', 'connected', 'rip', 'static']
+ route_map:
+ description: Route map reference.
+ type: str
+ isis_level:
+ description: Applicable for isis routes. Specify isis route level.
+ type: str
+ choices: ['level-1', 'level-2', 'level-1-2']
+ ospf_route:
+ description: ospf route options.
+ type: str
+ choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2']
+ route_target:
+ description: Route target.
+ type: dict
+ suboptions:
+ action:
+ description: Route action.
+ type: str
+ choices: ['both', 'import', 'export']
+ target:
+ description: Route Target.
+ type: str
+ router_id:
+ description: Router id.
+ type: str
+ shutdown:
+ description: When True, shut down BGP.
+ type: bool
+ timers:
+ description: Timers.
+ type: dict
+ suboptions:
+ keepalive:
+ description: Keep Alive Interval in secs.
+ type: int
+ holdtime:
+ description: Hold time in secs.
+ type: int
+ ucmp:
+ description: Configure unequal cost multipathing.
+ type: dict
+ suboptions:
+ fec:
+ description: Configure UCMP fec utilization threshold.
+ type: dict
+ suboptions:
+ trigger:
+ description: UCMP fec utilization too high threshold.
+ type: int
+ clear:
+ description: UCMP FEC utilization Clear thresholds.
+ type: int
+ link_bandwidth:
+ description: Configure link-bandwidth propagation delay.
+ type: dict
+ suboptions:
+ mode:
+ description: UCMP link bandwidth mode
+ type: str
+ choices: ['encoding_weighted', 'recursive']
+ update_delay:
+ description: Link Bandwidth Advertisement delay.
+ type: int
+ mode:
+ description: UCMP mode.
+ type: dict
+ suboptions:
+ set:
+ description: If True, ucmp mode is set to 1.
+ type: bool
+ nexthops:
+ description: Value for total number UCMP nexthops.
+ type: int
+ update:
+ description: Configure BGP update generation.
+ type: dict
+ suboptions:
+ wait_for:
+ description: wait for options before converge or synchronize.
+ type: str
+ choices: ['wait_for_convergence', 'wait_install']
+ batch_size:
+ description: batch size for FIB route acknowledgements.
+ type: int
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section bgp).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ - State I(purged) removes all the BGP configurations from the
+ target device. Use caution with this state.('no router bgp <x>')
+ - State I(deleted) only removes BGP attributes that this modules
+ manages and does not negate the BGP process completely. Thereby, preserving
+ address-family related configurations under BGP context.
+ - Running states I(deleted) and I(replaced) will result in an error if there
+ are address-family configuration lines present under vrf context that is
+ is to be removed. Please use the M(arista.eos.eos_bgp_address_family)
+ module for prior cleanup.
+ - Refer to examples for more details.
+ type: str
+ choices: [deleted, merged, purged, replaced, gathered, rendered, parsed]
+ default: merged
+"""
+EXAMPLES = """
+# Using merged
+# Before state
+
+# veos(config)#show running-config | section bgp
+# veos(config)#
+
+ - name: Merge provided configuration with device configuration
+ arista.eos.eos_bgp_global:
+ config:
+ as_number: "100"
+ bgp_params:
+ host_routes: True
+ convergence:
+ slow_peer: True
+ time: 6
+ additional_paths: "send"
+ log_neighbor_changes: True
+ maximum_paths:
+ max_equal_cost_paths: 55
+ aggregate_address:
+ - address: "1.2.1.0/24"
+ as_set: true
+ match_map: "match01"
+ - address: "5.2.1.0/24"
+ attribute_map: "attrmatch01"
+ advertise_only: true
+ redistribute:
+ - protocol: "static"
+ route_map: "map_static"
+ - protocol: "attached-host"
+ distance:
+ internal: 50
+ neighbor:
+ - peer: "10.1.3.2"
+ allowas_in:
+ set: true
+ default_originate:
+ always: true
+ dont_capability_negotiate: true
+ export_localpref: 4000
+ maximum_received_routes:
+ count: 500
+ warning_limit:
+ limit_percent: 5
+ next_hop_unchanged: true
+ prefix_list:
+ name: "prefix01"
+ direction: "out"
+ - peer: "peer1"
+ fall_over: true
+ link_bandwidth:
+ update_delay: 5
+ monitoring: True
+ send_community:
+ community_attribute: "extended"
+ sub_attribute: "link-bandwidth"
+ link_bandwidth_attribute: "aggregate"
+ speed: "600"
+ vlan: 5
+ state: merged
+
+# After State:
+# veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# distance bgp 50 50 50
+# maximum-paths 55
+# bgp additional-paths send any
+# neighbor peer1 peer-group
+# neighbor peer1 link-bandwidth update-delay 5
+# neighbor peer1 fall-over bfd
+# neighbor peer1 monitoring
+# neighbor peer1 send-community extended link-bandwidth aggregate 600
+# neighbor peer1 maximum-routes 12000
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+# vlan 5
+# !
+# address-family ipv4
+# neighbor 10.1.3.2 prefix-list prefix01 out
+# veos(config)#
+#
+# Module Execution:
+#
+# "after": {
+# "aggregate_address": [
+# {
+# "address": "1.2.1.0/24",
+# "as_set": true,
+# "match_map": "match01"
+# },
+# {
+# "address": "5.2.1.0/24",
+# "advertise_only": true,
+# "attribute_map": "attrmatch01"
+# }
+# ],
+# "as_number": "100",
+# "bgp_params": {
+# "additional_paths": "send",
+# "convergence": {
+# "slow_peer": true,
+# "time": 6
+# }
+# },
+# "distance": {
+# "external": 50,
+# "internal": 50,
+# "local": 50
+# },
+# "maximum_paths": {
+# "max_equal_cost_paths": 55
+# },
+# "neighbor": [
+# {
+# "fall_over": true,
+# "link_bandwidth": {
+# "set": true,
+# "update_delay": 5
+# },
+# "maximum_received_routes": {
+# "count": 12000
+# },
+# "monitoring": true,
+# "peer": "peer1",
+# "peer_group": "peer1",
+# "send_community": {
+# "community_attribute": "extended",
+# "link_bandwidth_attribute": "aggregate",
+# "speed": "600",
+# "sub_attribute": "link-bandwidth"
+# }
+# },
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "export_localpref": 4000,
+# "maximum_received_routes": {
+# "count": 500,
+# "warning_limit": {
+# "limit_percent": 5
+# }
+# },
+# "next_hop_unchanged": true,
+# "peer": "10.1.3.2"
+# }
+# ],
+# "redistribute": [
+# {
+# "protocol": "static",
+# "route_map": "map_static"
+# },
+# {
+# "protocol": "attached-host"
+# }
+# ],
+# "vlan": 5
+# },
+# "before": {},
+# "changed": true,
+# "commands": [
+# "router bgp 100",
+# "neighbor 10.1.3.2 allowas-in",
+# "neighbor 10.1.3.2 default-originate always",
+# "neighbor 10.1.3.2 dont-capability-negotiate",
+# "neighbor 10.1.3.2 export-localpref 4000",
+# "neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent",
+# "neighbor 10.1.3.2 next-hop-unchanged",
+# "neighbor 10.1.3.2 prefix-list prefix01 out",
+# "neighbor peer1 fall-over bfd",
+# "neighbor peer1 link-bandwidth update-delay 5",
+# "neighbor peer1 monitoring",
+# "neighbor peer1 send-community extended link-bandwidth aggregate 600",
+# "redistribute static route-map map_static",
+# "redistribute attached-host",
+# "aggregate-address 1.2.1.0/24 as-set match-map match01",
+# "aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only",
+# "bgp host-routes fib direct-install",
+# "bgp convergence slow-peer time 6",
+# "bgp additional-paths send any",
+# "bgp log-neighbor-changes",
+# "maximum-paths 55",
+# "distance bgp 50",
+# "vlan 5"
+# ],
+
+# Using replaced:
+
+# Before state:
+# veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# distance bgp 50 50 50
+# maximum-paths 55
+# bgp additional-paths send any
+# neighbor peer1 peer-group
+# neighbor peer1 link-bandwidth update-delay 5
+# neighbor peer1 fall-over bfd
+# neighbor peer1 monitoring
+# neighbor peer1 send-community extended link-bandwidth aggregate 600
+# neighbor peer1 maximum-routes 12000
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+# vlan 5
+# !
+# address-family ipv4
+# neighbor 10.1.3.2 prefix-list prefix01 out
+# !
+# vrf vrf01
+# route-target import 54:11
+# neighbor 12.1.3.2 dont-capability-negotiate
+# neighbor 12.1.3.2 allowas-in 3
+# neighbor 12.1.3.2 default-originate always
+# neighbor 12.1.3.2 maximum-routes 12000
+# veos(config)#
+
+ - name: replace provided configuration with device configuration
+ arista.eos.eos_bgp_global:
+ config:
+ as_number: "100"
+ bgp_params:
+ host_routes: True
+ convergence:
+ slow_peer: True
+ time: 6
+ additional_paths: "send"
+ log_neighbor_changes: True
+ vrfs:
+ - vrf: "vrf01"
+ maximum_paths:
+ max_equal_cost_paths: 55
+ aggregate_address:
+ - address: "1.2.1.0/24"
+ as_set: true
+ match_map: "match01"
+ - address: "5.2.1.0/24"
+ attribute_map: "attrmatch01"
+ advertise_only: true
+ redistribute:
+ - protocol: "static"
+ route_map: "map_static"
+ - protocol: "attached-host"
+ distance:
+ internal: 50
+ neighbor:
+ - peer: "10.1.3.2"
+ allowas_in:
+ set: true
+ default_originate:
+ always: true
+ dont_capability_negotiate: true
+ export_localpref: 4000
+ maximum_received_routes:
+ count: 500
+ warning_limit:
+ limit_percent: 5
+ next_hop_unchanged: true
+ prefix_list:
+ name: "prefix01"
+ direction: "out"
+ - peer: "peer1"
+ fall_over: true
+ link_bandwidth:
+ update_delay: 5
+ monitoring: True
+ send_community:
+ community_attribute: "extended"
+ sub_attribute: "link-bandwidth"
+ link_bandwidth_attribute: "aggregate"
+ speed: "600"
+ state: replaced
+
+# After State:
+
+# veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# bgp additional-paths send any
+# !
+# vrf vrf01
+# distance bgp 50 50 50
+# maximum-paths 55
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+# address-family ipv4
+# neighbor 10.1.3.2 prefix-list prefix01 out
+# veos(config)#
+#
+#
+# Module Execution:
+#
+# "after": {
+# "as_number": "100",
+# "bgp_params": {
+# "additional_paths": "send",
+# "convergence": {
+# "slow_peer": true,
+# "time": 6
+# }
+# },
+# "vrfs": [
+# {
+# "aggregate_address": [
+# {
+# "address": "1.2.1.0/24",
+# "as_set": true,
+# "match_map": "match01"
+# },
+# {
+# "address": "5.2.1.0/24",
+# "advertise_only": true,
+# "attribute_map": "attrmatch01"
+# }
+# ],
+# "distance": {
+# "external": 50,
+# "internal": 50,
+# "local": 50
+# },
+# "maximum_paths": {
+# "max_equal_cost_paths": 55
+# },
+# "neighbor": [
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "export_localpref": 4000,
+# "maximum_received_routes": {
+# "count": 500,
+# "warning_limit": {
+# "limit_percent": 5
+# }
+# },
+# "next_hop_unchanged": true,
+# "peer": "10.1.3.2"
+# }
+# ],
+# "redistribute": [
+# {
+# "protocol": "static",
+# "route_map": "map_static"
+# },
+# {
+# "protocol": "attached-host"
+# }
+# ],
+# "vrf": "vrf01"
+# }
+# ]
+# },
+# "before": {
+# "aggregate_address": [
+# {
+# "address": "1.2.1.0/24",
+# "as_set": true,
+# "match_map": "match01"
+# },
+# {
+# "address": "5.2.1.0/24",
+# "advertise_only": true,
+# "attribute_map": "attrmatch01"
+# }
+# ],
+# "as_number": "100",
+# "bgp_params": {
+# "additional_paths": "send",
+# "convergence": {
+# "slow_peer": true,
+# "time": 6
+# }
+# },
+# "distance": {
+# "external": 50,
+# "internal": 50,
+# "local": 50
+# },
+# "maximum_paths": {
+# "max_equal_cost_paths": 55
+# },
+# "neighbor": [
+# {
+# "fall_over": true,
+# "link_bandwidth": {
+# "set": true,
+# "update_delay": 5
+# },
+# "maximum_received_routes": {
+# "count": 12000
+# },
+# "monitoring": true,
+# "peer": "peer1",
+# "peer_group": "peer1",
+# "send_community": {
+# "community_attribute": "extended",
+# "link_bandwidth_attribute": "aggregate",
+# "speed": "600",
+# "sub_attribute": "link-bandwidth"
+# }
+# },
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "export_localpref": 4000,
+# "maximum_received_routes": {
+# "count": 500,
+# "warning_limit": {
+# "limit_percent": 5
+# }
+# },
+# "next_hop_unchanged": true,
+# "peer": "10.1.3.2"
+# }
+# ],
+# "redistribute": [
+# {
+# "protocol": "static",
+# "route_map": "map_static"
+# },
+# {
+# "protocol": "attached-host"
+# }
+# ],
+# "vlan": 5,
+# "vrfs": [
+# {
+# "neighbor": [
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "maximum_received_routes": {
+# "count": 12000
+# },
+# "peer": "12.1.3.2"
+# }
+# ],
+# "route_target": {
+# "action": "import",
+# "target": "54:11"
+# },
+# "vrf": "vrf01"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "router bgp 100",
+# "vrf vrf01",
+# "no route-target import 54:11",
+# "neighbor 10.1.3.2 allowas-in",
+# "neighbor 10.1.3.2 default-originate always",
+# "neighbor 10.1.3.2 dont-capability-negotiate",
+# "neighbor 10.1.3.2 export-localpref 4000",
+# "neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent",
+# "neighbor 10.1.3.2 next-hop-unchanged",
+# "neighbor 10.1.3.2 prefix-list prefix01 out",
+# "neighbor peer1 fall-over bfd",
+# "neighbor peer1 link-bandwidth update-delay 5",
+# "neighbor peer1 monitoring",
+# "neighbor peer1 send-community extended link-bandwidth aggregate 600",
+# "no neighbor 12.1.3.2",
+# "redistribute static route-map map_static",
+# "redistribute attached-host",
+# "aggregate-address 1.2.1.0/24 as-set match-map match01",
+# "aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only",
+# "maximum-paths 55",
+# "distance bgp 50",
+# "exit",
+# "no neighbor peer1 peer-group",
+# "no neighbor peer1 link-bandwidth update-delay 5",
+# "no neighbor peer1 fall-over bfd",
+# "no neighbor peer1 monitoring",
+# "no neighbor peer1 send-community extended link-bandwidth aggregate 600",
+# "no neighbor peer1 maximum-routes 12000",
+# "no neighbor 10.1.3.2",
+# "no redistribute static route-map map_static",
+# "no redistribute attached-host",
+# "no aggregate-address 1.2.1.0/24 as-set match-map match01",
+# "no aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only",
+# "bgp host-routes fib direct-install",
+# "bgp log-neighbor-changes",
+# "no distance bgp 50 50 50",
+# "no maximum-paths 55",
+# "no vlan 5"
+# ],
+#
+
+# Using replaced (in presence of address_family under vrf):
+# Before State:
+
+#veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# bgp additional-paths send any
+# !
+# vrf vrf01
+# distance bgp 50 50 50
+# maximum-paths 55
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+# address-family ipv4
+# neighbor 10.1.3.2 prefix-list prefix01 out
+# !
+# address-family ipv6
+# redistribute dhcp
+# veos(config)#
+
+ - name: Replace
+ arista.eos.eos_bgp_global:
+ config:
+ as_number: "100"
+ graceful_restart:
+ set: True
+ router_id: "1.1.1.1"
+ timers:
+ keepalive: 2
+ holdtime: 5
+ ucmp:
+ mode:
+ set: True
+ vlan_aware_bundle: "bundle1 bundle2 bundle3"
+ state: replaced
+
+# Module Execution:
+
+# fatal: [192.168.122.113]: FAILED! => {
+# "changed": false,
+# "invocation": {
+# "module_args": {
+# "config": {
+# "access_group": null,
+# "aggregate_address": null,
+# "as_number": "100",
+# "bgp_params": null,
+# "default_metric": null,
+# "distance": null,
+# "graceful_restart": {
+# "restart_time": null,
+# "set": true,
+# "stalepath_time": null
+# },
+# "graceful_restart_helper": null,
+# "maximum_paths": null,
+# "monitoring": null,
+# "neighbor": null,
+# "network": null,
+# "redistribute": null,
+# "route_target": null,
+# "router_id": "1.1.1.1",
+# "shutdown": null,
+# "timers": {
+# "holdtime": 5,
+# "keepalive": 2
+# },
+# "ucmp": {
+# "fec": null,
+# "link_bandwidth": null,
+# "mode": {
+# "nexthops": null,
+# "set": true
+# }
+# },
+# "update": null,
+# "vlan": null,
+# "vlan_aware_bundle": "bundle1 bundle2 bundle3",
+# "vrfs": null
+# },
+# "running_config": null,
+# "state": "replaced"
+# }
+# },
+# "msg": "Use the _bgp_af module to delete the address_family under vrf, before replacing/deleting the vrf."
+# }
+
+# Using deleted:
+
+# Before state:
+
+# veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# bgp additional-paths send any
+# !
+# vrf vrf01
+# distance bgp 50 50 50
+# maximum-paths 55
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+
+ - name: Delete configuration
+ arista.eos.eos_bgp_global:
+ config:
+ as_number: "100"
+ state: deleted
+
+# After State:
+
+# veos(config)#show running-config | section bgp
+# router bgp 100
+#
+#
+# Module Execution:
+#
+# "after": {
+# "as_number": "100"
+# },
+# "before": {
+# "as_number": "100",
+# "bgp_params": {
+# "additional_paths": "send",
+# "convergence": {
+# "slow_peer": true,
+# "time": 6
+# }
+# },
+# "vrfs": [
+# {
+# "aggregate_address": [
+# {
+# "address": "1.2.1.0/24",
+# "as_set": true,
+# "match_map": "match01"
+# },
+# {
+# "address": "5.2.1.0/24",
+# "advertise_only": true,
+# "attribute_map": "attrmatch01"
+# }
+# ],
+# "distance": {
+# "external": 50,
+# "internal": 50,
+# "local": 50
+# },
+# "maximum_paths": {
+# "max_equal_cost_paths": 55
+# },
+# "neighbor": [
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "export_localpref": 4000,
+# "maximum_received_routes": {
+# "count": 500,
+# "warning_limit": {
+# "limit_percent": 5
+# }
+# },
+# "next_hop_unchanged": true,
+# "peer": "10.1.3.2"
+# }
+# ],
+# "redistribute": [
+# {
+# "protocol": "static",
+# "route_map": "map_static"
+# },
+# {
+# "protocol": "attached-host"
+# }
+# ],
+# "vrf": "vrf01"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "router bgp 100",
+# "no vrf vrf01",
+# "no bgp convergence slow-peer time 6",
+# "no bgp additional-paths send any"
+# ],
+#
+
+# Using purged:
+
+# Before state:
+
+# veos(config)#show running-config | section bgp
+# router bgp 100
+# bgp convergence slow-peer time 6
+# distance bgp 50 50 50
+# maximum-paths 55
+# bgp additional-paths send any
+# neighbor peer1 peer-group
+# neighbor peer1 link-bandwidth update-delay 5
+# neighbor peer1 fall-over bfd
+# neighbor peer1 monitoring
+# neighbor peer1 send-community extended link-bandwidth aggregate 600
+# neighbor peer1 maximum-routes 12000
+# neighbor 10.1.3.2 export-localpref 4000
+# neighbor 10.1.3.2 next-hop-unchanged
+# neighbor 10.1.3.2 dont-capability-negotiate
+# neighbor 10.1.3.2 allowas-in 3
+# neighbor 10.1.3.2 default-originate always
+# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent
+# aggregate-address 1.2.1.0/24 as-set match-map match01
+# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only
+# redistribute static route-map map_static
+# redistribute attached-host
+# !
+# vlan 5
+# !
+# address-family ipv4
+# neighbor 10.1.3.2 prefix-list prefix01 out
+# !
+# vrf vrf01
+# route-target import 54:11
+# neighbor 12.1.3.2 dont-capability-negotiate
+# neighbor 12.1.3.2 allowas-in 3
+# neighbor 12.1.3.2 default-originate always
+# neighbor 12.1.3.2 maximum-routes 12000
+# veos(config)#
+
+ - name: Purge configuration
+ arista.eos.eos_bgp_global:
+ config:
+ as_number: "100"
+ state: purged
+
+# After State:
+
+# veos(config)#show running-config | section bgp
+# veos(config)#
+
+# Module Execution:
+
+# "after": {},
+# "before": {
+# "aggregate_address": [
+# {
+# "address": "1.2.1.0/24",
+# "as_set": true,
+# "match_map": "match01"
+# },
+# {
+# "address": "5.2.1.0/24",
+# "advertise_only": true,
+# "attribute_map": "attrmatch01"
+# }
+# ],
+# "as_number": "100",
+# "bgp_params": {
+# "additional_paths": "send",
+# "convergence": {
+# "slow_peer": true,
+# "time": 6
+# }
+# },
+# "distance": {
+# "external": 50,
+# "internal": 50,
+# "local": 50
+# },
+# "maximum_paths": {
+# "max_equal_cost_paths": 55
+# },
+# "neighbor": [
+# {
+# "fall_over": true,
+# "link_bandwidth": {
+# "set": true,
+# "update_delay": 5
+# },
+# "maximum_received_routes": {
+# "count": 12000
+# },
+# "monitoring": true,
+# "peer": "peer1",
+# "peer_group": "peer1",
+# "send_community": {
+# "community_attribute": "extended",
+# "link_bandwidth_attribute": "aggregate",
+# "speed": "600",
+# "sub_attribute": "link-bandwidth"
+# }
+# },
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "export_localpref": 4000,
+# "maximum_received_routes": {
+# "count": 500,
+# "warning_limit": {
+# "limit_percent": 5
+# }
+# },
+# "next_hop_unchanged": true,
+# "peer": "10.1.3.2"
+# }
+# ],
+# "redistribute": [
+# {
+# "protocol": "static",
+# "route_map": "map_static"
+# },
+# {
+# "protocol": "attached-host"
+# }
+# ],
+# "vlan": 5,
+# "vrfs": [
+# {
+# "neighbor": [
+# {
+# "allowas_in": {
+# "count": 3
+# },
+# "default_originate": {
+# "always": true
+# },
+# "dont_capability_negotiate": true,
+# "maximum_received_routes": {
+# "count": 12000
+# },
+# "peer": "12.1.3.2"
+# }
+# ],
+# "route_target": {
+# "action": "import",
+# "target": "54:11"
+# },
+# "vrf": "vrf01"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "no router bgp 100"
+# ],
+
+"""
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_global.bgp_global import (
+ Bgp_globalArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.bgp_global.bgp_global import (
+ Bgp_global,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ module = AnsibleModule(
+ argument_spec=Bgp_globalArgs.argument_spec,
+ mutually_exclusive=[],
+ required_if=[],
+ supports_check_mode=False,
+ )
+
+ result = Bgp_global(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py
new file mode 100644
index 00000000..c4f43c77
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py
@@ -0,0 +1,260 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_command
+author: Peter Sprygada (@privateip)
+short_description: Run arbitrary commands on an Arista EOS device
+description:
+- Sends an arbitrary set of commands to an EOS node and returns the results read from
+ the device. This module includes an argument that will cause the module to wait
+ for a specific condition before returning or timing out if the condition is not
+ met.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+notes:
+- Tested against EOS 4.15
+options:
+ commands:
+ description:
+ - The commands to send to the remote EOS device over the configured provider. The
+ resulting output from the command is returned. If the I(wait_for) argument
+ is provided, the module is not returned until the condition is satisfied or
+ the number of I(retries) has been exceeded.
+ - If a command sent to the device requires answering a prompt, it is possible to pass
+ a dict containing command, answer and prompt. Common answers are 'y' or "\\r"
+ (carriage return, must be double quotes). Refer below examples.
+ required: true
+ type: list
+ elements: raw
+ wait_for:
+ description:
+ - Specifies what to evaluate from the output of the command and what conditionals
+ to apply. This argument will cause the task to wait for a particular conditional
+ to be true before moving forward. If the conditional is not true by the configured
+ retries, the task fails. Note - With I(wait_for) the value in C(result['stdout'])
+ can be accessed using C(result), that is to access C(result['stdout'][0]) use
+ C(result[0]) See examples.
+ type: list
+ elements: str
+ aliases:
+ - waitfor
+ match:
+ description:
+ - The I(match) argument is used in conjunction with the I(wait_for) argument to
+ specify the match policy. Valid values are C(all) or C(any). If the value
+ is set to C(all) then all conditionals in the I(wait_for) must be satisfied. If
+ the value is set to C(any) then only one of the values must be satisfied.
+ type: str
+ default: all
+ choices:
+ - any
+ - all
+ retries:
+ description:
+ - Specifies the number of retries a command should be tried before it is considered
+ failed. The command is run on the target device every retry and evaluated against
+ the I(wait_for) conditionals.
+ default: 10
+ type: int
+ interval:
+ description:
+ - Configures the interval in seconds to wait between retries of the command. If
+ the command does not pass the specified conditional, the interval indicates
+ how to long to wait before trying the command again.
+ default: 1
+ type: int
+"""
+
+EXAMPLES = """
+- name: run show version on remote devices
+ arista.eos.eos_command:
+ commands: show version
+
+- name: run show version and check to see if output contains Arista
+ arista.eos.eos_command:
+ commands: show version
+ wait_for: result[0] contains Arista
+
+- name: run multiple commands on remote nodes
+ arista.eos.eos_command:
+ commands:
+ - show version
+ - show interfaces
+
+- name: run multiple commands and evaluate the output
+ arista.eos.eos_command:
+ commands:
+ - show version
+ - show interfaces
+ wait_for:
+ - result[0] contains Arista
+ - result[1] contains Loopback0
+
+- name: run commands and specify the output format
+ arista.eos.eos_command:
+ commands:
+ - command: show version
+ output: json
+
+- name: using cli transport, check whether the switch is in maintenance mode
+ arista.eos.eos_command:
+ commands: show maintenance
+ wait_for: result[0] contains 'Under Maintenance'
+
+- name: using cli transport, check whether the switch is in maintenance mode using
+ json output
+ arista.eos.eos_command:
+ commands: show maintenance | json
+ wait_for: result[0].units.System.state eq 'underMaintenance'
+
+- name: using eapi transport check whether the switch is in maintenance, with 8 retries
+ and 2 second interval between retries
+ arista.eos.eos_command:
+ commands: show maintenance
+ wait_for: result[0]['units']['System']['state'] eq 'underMaintenance'
+ interval: 2
+ retries: 8
+ provider:
+ transport: eapi
+"""
+
+RETURN = """
+stdout:
+ description: The set of responses from the commands
+ returned: always apart from low level errors (such as action plugin)
+ type: list
+ sample: ['...', '...']
+stdout_lines:
+ description: The value of stdout split into a list
+ returned: always apart from low level errors (such as action plugin)
+ type: list
+ sample: [['...', '...'], ['...'], ['...']]
+failed_conditions:
+ description: The list of conditionals that have failed
+ returned: failed
+ type: list
+ sample: ['...', '...']
+"""
+import time
+
+from ansible.module_utils._text import to_text
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.parsing import (
+ Conditional,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ transform_commands,
+ to_lines,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def parse_commands(module, warnings):
+ commands = transform_commands(module)
+
+ if module.check_mode:
+ for item in list(commands):
+ if not item["command"].startswith("show"):
+ warnings.append(
+ "Only show commands are supported when using check mode, not "
+ "executing %s" % item["command"]
+ )
+ commands.remove(item)
+
+ return commands
+
+
+def to_cli(obj):
+ cmd = obj["command"]
+ if obj.get("output") == "json":
+ cmd += " | json"
+ return cmd
+
+
+def main():
+ """entry point for module execution
+ """
+ argument_spec = dict(
+ commands=dict(type="list", required=True, elements="raw"),
+ wait_for=dict(type="list", aliases=["waitfor"], elements="str"),
+ match=dict(default="all", choices=["all", "any"]),
+ retries=dict(default=10, type="int"),
+ interval=dict(default=1, type="int"),
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ module = AnsibleModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+
+ warnings = list()
+ result = {"changed": False, "warnings": warnings}
+ commands = parse_commands(module, warnings)
+ wait_for = module.params["wait_for"] or list()
+
+ try:
+ conditionals = [Conditional(c) for c in wait_for]
+ except AttributeError as exc:
+ module.fail_json(msg=to_text(exc))
+
+ retries = module.params["retries"]
+ interval = module.params["interval"]
+ match = module.params["match"]
+
+ while retries > 0:
+ responses = run_commands(module, commands)
+
+ for item in list(conditionals):
+ if item(responses):
+ if match == "any":
+ conditionals = list()
+ break
+ conditionals.remove(item)
+
+ if not conditionals:
+ break
+
+ time.sleep(interval)
+ retries -= 1
+
+ if conditionals:
+ failed_conditions = [item.raw for item in conditionals]
+ msg = "One or more conditional statements have not been satisfied"
+ module.fail_json(msg=msg, failed_conditions=failed_conditions)
+
+ result.update(
+ {"stdout": responses, "stdout_lines": list(to_lines(responses))}
+ )
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py
new file mode 100644
index 00000000..fe53b502
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py
@@ -0,0 +1,593 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_config
+author: Peter Sprygada (@privateip)
+short_description: Manage Arista EOS configuration sections
+description:
+- Arista EOS configurations use a simple block indent file syntax for segmenting configuration
+ into sections. This module provides an implementation for working with EOS configuration
+ sections in a deterministic way. This module works with either CLI or eAPI transports.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+notes:
+- Tested against EOS 4.15
+- Abbreviated commands are NOT idempotent, see
+ L(Network FAQ,../network/user_guide/faq.html#why-do-the-config-modules-always-return-changed-true-with-abbreviated-commands).
+- To ensure idempotency and correct diff the configuration lines in the relevant module options should be similar to how they
+ appear if present in the running configuration on device including the indentation.
+options:
+ lines:
+ description:
+ - The ordered set of commands that should be configured in the section. The commands
+ must be the exact same commands as found in the device running-config as found in the
+ device running-config to ensure idempotency and correct diff. Be sure
+ to note the configuration command syntax as some commands are automatically
+ modified by the device config parser.
+ aliases:
+ - commands
+ type: list
+ elements: str
+ parents:
+ description:
+ - The ordered set of parents that uniquely identify the section or hierarchy the
+ commands should be checked against. If the parents argument is omitted, the
+ commands are checked against the set of top level or global commands.
+ type: list
+ elements: str
+ src:
+ description:
+ - The I(src) argument provides a path to the configuration file to load into the
+ remote system. The path can either be a full system path to the configuration
+ file if the value starts with / or relative to the root of the implemented role
+ or playbook. This argument is mutually exclusive with the I(lines) and I(parents)
+ arguments. It can be a Jinja2 template as well. The configuration lines in the source
+ file should be similar to how it will appear if present in the running-configuration
+ (live switch config) of the device i ncluding the indentation to ensure idempotency
+ and correct diff. Arista EOS device config has 3 spaces indentation.
+ type: path
+ before:
+ description:
+ - The ordered set of commands to push on to the command stack if a change needs
+ to be made. This allows the playbook designer the opportunity to perform configuration
+ commands prior to pushing any changes without affecting how the set of commands
+ are matched against the system.
+ type: list
+ elements: str
+ after:
+ description:
+ - The ordered set of commands to append to the end of the command stack if a change
+ needs to be made. Just like with I(before) this allows the playbook designer
+ to append a set of commands to be executed after the command set.
+ type: list
+ elements: str
+ match:
+ description:
+ - Instructs the module on the way to perform the matching of the set of commands
+ against the current device config. If match is set to I(line), commands are
+ matched line by line. If match is set to I(strict), command lines are matched
+ with respect to position. If match is set to I(exact), command lines must be
+ an equal match. Finally, if match is set to I(none), the module will not attempt
+ to compare the source configuration with the running configuration on the remote
+ device.
+ type: str
+ default: line
+ choices:
+ - line
+ - strict
+ - exact
+ - none
+ replace:
+ description:
+ - Instructs the module on the way to perform the configuration on the device. If
+ the replace argument is set to I(line) then the modified lines are pushed to
+ the device in configuration mode. If the replace argument is set to I(block)
+ then the entire command block is pushed to the device in configuration mode
+ if any line is not correct.
+ type: str
+ default: line
+ choices:
+ - line
+ - block
+ - config
+ backup:
+ description:
+ - This argument will cause the module to create a full backup of the current C(running-config)
+ from the remote device before any changes are made. If the C(backup_options)
+ value is not given, the backup file is written to the C(backup) folder in the
+ playbook root directory or role root directory, if playbook is part of an ansible
+ role. If the directory does not exist, it is created.
+ type: bool
+ default: no
+ running_config:
+ description:
+ - The module, by default, will connect to the remote device and retrieve the current
+ running-config to use as a base for comparing against the contents of source. There
+ are times when it is not desirable to have the task get the current running-config
+ for every task in a playbook. The I(running_config) argument allows the implementer
+ to pass in the configuration to use as the base config for this module.
+ The configuration lines for this option should be similar to how it will appear if present
+ in the running-configuration of the device including the indentation to ensure idempotency
+ and correct diff.
+ type: str
+ aliases:
+ - config
+ defaults:
+ description:
+ - The I(defaults) argument will influence how the running-config is collected
+ from the device. When the value is set to true, the command used to collect
+ the running-config is append with the all keyword. When the value is set to
+ false, the command is issued without the all keyword
+ type: bool
+ default: no
+ save_when:
+ description:
+ - When changes are made to the device running-configuration, the changes are not
+ copied to non-volatile storage by default. Using this argument will change
+ that before. If the argument is set to I(always), then the running-config will
+ always be copied to the startup-config and the I(modified) flag will always
+ be set to True. If the argument is set to I(modified), then the running-config
+ will only be copied to the startup-config if it has changed since the last save
+ to startup-config. If the argument is set to I(never), the running-config will
+ never be copied to the startup-config. If the argument is set to I(changed),
+ then the running-config will only be copied to the startup-config if the task
+ has made a change. I(changed) was added in Ansible 2.5.
+ default: never
+ type: str
+ choices:
+ - always
+ - never
+ - modified
+ - changed
+ diff_against:
+ description:
+ - When using the C(ansible-playbook --diff) command line argument the module can
+ generate diffs against different sources.
+ - When this option is configure as I(startup), the module will return the diff
+ of the running-config against the startup-config.
+ - When this option is configured as I(intended), the module will return the diff
+ of the running-config against the configuration provided in the C(intended_config)
+ argument.
+ - When this option is configured as I(running), the module will return the before
+ and after diff of the running-config with respect to any changes made to the
+ device configuration.
+ - When this option is configured as C(session), the diff returned will be based
+ on the configuration session.
+ default: session
+ type: str
+ choices:
+ - startup
+ - running
+ - intended
+ - session
+ diff_ignore_lines:
+ description:
+ - Use this argument to specify one or more lines that should be ignored during
+ the diff. This is used for lines in the configuration that are automatically
+ updated by the system. This argument takes a list of regular expressions or
+ exact line matches.
+ type: list
+ elements: str
+ intended_config:
+ description:
+ - The C(intended_config) provides the master configuration that the node should
+ conform to and is used to check the final running-config against. This argument
+ will not modify any settings on the remote device and is strictly used to check
+ the compliance of the current device's configuration against. When specifying
+ this argument, the task should also modify the C(diff_against) value and set
+ it to I(intended). The configuration lines for this value should be similar to how it
+ will appear if present in the running-configuration of the device including the indentation
+ to ensure correct diff.
+ type: str
+ backup_options:
+ description:
+ - This is a dict object containing configurable options related to backup file
+ path. The value of this option is read only when C(backup) is set to I(yes),
+ if C(backup) is set to I(no) this option will be silently ignored.
+ suboptions:
+ filename:
+ description:
+ - The filename to be used to store the backup configuration. If the filename
+ is not given it will be generated based on the hostname, current time and
+ date in format defined by <hostname>_config.<current-date>@<current-time>
+ type: str
+ dir_path:
+ description:
+ - This option provides the path ending with directory name in which the backup
+ configuration file will be stored. If the directory does not exist it will
+ be first created and the filename is either the value of C(filename) or
+ default filename as described in C(filename) options description. If the
+ path value is not given in that case a I(backup) directory will be created
+ in the current working directory and backup configuration will be copied
+ in C(filename) within I(backup) directory.
+ type: path
+ type: dict
+"""
+# noqa: E501
+
+EXAMPLES = """
+- name: configure top level settings
+ arista.eos.eos_config:
+ lines: hostname {{ inventory_hostname }}
+
+- name: load an acl into the device
+ arista.eos.eos_config:
+ lines:
+ - 10 permit ip host 192.0.2.1 any log
+ - 20 permit ip host 192.0.2.2 any log
+ - 30 permit ip host 192.0.2.3 any log
+ - 40 permit ip host 192.0.2.4 any log
+ parents: ip access-list test
+ before: no ip access-list test
+ replace: block
+
+- name: load configuration from file
+ arista.eos.eos_config:
+ src: eos.cfg
+
+- name: render a Jinja2 template onto an Arista switch
+ arista.eos.eos_config:
+ backup: yes
+ src: eos_template.j2
+
+- name: diff the running config against a master config
+ arista.eos.eos_config:
+ diff_against: intended
+ intended_config: "{{ lookup('file', 'master.cfg') }}"
+
+- name: for idempotency, use full-form commands
+ arista.eos.eos_config:
+ lines:
+ # - shut
+ - shutdown
+ # parents: int eth1
+ parents: interface Ethernet1
+
+- name: configurable backup path
+ arista.eos.eos_config:
+ src: eos_template.j2
+ backup: yes
+ backup_options:
+ filename: backup.cfg
+ dir_path: /home/user
+"""
+
+RETURN = """
+commands:
+ description: The set of commands that will be pushed to the remote device
+ returned: always
+ type: list
+ sample: ['hostname switch01', 'interface Ethernet1', 'no shutdown']
+updates:
+ description: The set of commands that will be pushed to the remote device
+ returned: always
+ type: list
+ sample: ['hostname switch01', 'interface Ethernet1', 'no shutdown']
+backup_path:
+ description: The full path to the backup file
+ returned: when backup is yes
+ type: str
+ sample: /playbooks/ansible/backup/eos_config.2016-07-16@22:28:34
+filename:
+ description: The name of the backup file
+ returned: when backup is yes and filename is not specified in backup options
+ type: str
+ sample: eos_config.2016-07-16@22:28:34
+shortname:
+ description: The full path to the backup file excluding the timestamp
+ returned: when backup is yes and filename is not specified in backup options
+ type: str
+ sample: /playbooks/ansible/backup/eos_config
+date:
+ description: The date extracted from the backup file name
+ returned: when backup is yes
+ type: str
+ sample: "2016-07-16"
+time:
+ description: The time extracted from the backup file name
+ returned: when backup is yes
+ type: str
+ sample: "22:28:34"
+"""
+from ansible.module_utils._text import to_text
+from ansible.module_utils.basic import AnsibleModule
+from ansible.module_utils.connection import ConnectionError
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+ dumps,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+ get_connection,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def get_candidate(module):
+ candidate = ""
+ if module.params["src"]:
+ candidate = module.params["src"]
+ elif module.params["lines"]:
+ candidate_obj = NetworkConfig(indent=3)
+ parents = module.params["parents"] or list()
+ candidate_obj.add(module.params["lines"], parents=parents)
+ candidate = dumps(candidate_obj, "raw")
+ return candidate
+
+
+def get_running_config(module, config=None, flags=None):
+ contents = module.params["running_config"]
+ if not contents:
+ if config:
+ contents = config
+ else:
+ contents = get_config(module, flags=flags)
+ return contents
+
+
+def save_config(module, result):
+ result["changed"] = True
+ if not module.check_mode:
+ cmd = {
+ "command": "copy running-config startup-config",
+ "output": "text",
+ }
+ run_commands(module, [cmd])
+ else:
+ module.warn(
+ "Skipping command `copy running-config startup-config` "
+ "due to check_mode. Configuration not copied to "
+ "non-volatile storage"
+ )
+
+
+def main():
+ """ main entry point for module execution
+ """
+ backup_spec = dict(filename=dict(), dir_path=dict(type="path"))
+ argument_spec = dict(
+ src=dict(type="path"),
+ lines=dict(aliases=["commands"], type="list", elements="str"),
+ parents=dict(type="list", elements="str"),
+ before=dict(type="list", elements="str"),
+ after=dict(type="list", elements="str"),
+ match=dict(
+ default="line", choices=["line", "strict", "exact", "none"]
+ ),
+ replace=dict(default="line", choices=["line", "block", "config"]),
+ defaults=dict(type="bool", default=False),
+ backup=dict(type="bool", default=False),
+ backup_options=dict(type="dict", options=backup_spec),
+ save_when=dict(
+ choices=["always", "never", "modified", "changed"], default="never"
+ ),
+ diff_against=dict(
+ choices=["startup", "session", "intended", "running"],
+ default="session",
+ ),
+ diff_ignore_lines=dict(type="list", elements="str"),
+ running_config=dict(aliases=["config"]),
+ intended_config=dict(),
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ mutually_exclusive = [("lines", "src"), ("parents", "src")]
+
+ required_if = [
+ ("match", "strict", ["lines"]),
+ ("match", "exact", ["lines"]),
+ ("replace", "block", ["lines"]),
+ ("replace", "config", ["src"]),
+ ("diff_against", "intended", ["intended_config"]),
+ ]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ mutually_exclusive=mutually_exclusive,
+ required_if=required_if,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ diff_ignore_lines = module.params["diff_ignore_lines"]
+ config = None
+ contents = None
+ flags = ["all"] if module.params["defaults"] else []
+ connection = get_connection(module)
+
+ # Refuse to diff_against: session if sessions are disabled
+ if (
+ module.params["diff_against"] == "session"
+ and not connection.supports_sessions
+ ):
+ module.fail_json(
+ msg="Cannot diff against sessions when sessions are disabled. Please change diff_against to another value"
+ )
+
+ if module.params["backup"] or (
+ module._diff and module.params["diff_against"] == "running"
+ ):
+ contents = get_config(module, flags=flags)
+ config = NetworkConfig(indent=1, contents=contents)
+ if module.params["backup"]:
+ result["__backup__"] = contents
+
+ if any((module.params["src"], module.params["lines"])):
+ msg = (
+ "To ensure idempotency and correct diff the input configuration lines should be"
+ " similar to how they appear if present in the running configuration on device"
+ )
+ if module.params["src"]:
+ msg += " including the indentation"
+ warnings.append(msg)
+
+ match = module.params["match"]
+ replace = module.params["replace"]
+ path = module.params["parents"]
+
+ candidate = get_candidate(module)
+ running = get_running_config(module, contents, flags=flags)
+
+ try:
+ response = connection.get_diff(
+ candidate=candidate,
+ running=running,
+ diff_match=match,
+ diff_ignore_lines=diff_ignore_lines,
+ path=path,
+ diff_replace=replace,
+ )
+ except ConnectionError as exc:
+ module.fail_json(msg=to_text(exc, errors="surrogate_then_replace"))
+
+ config_diff = response["config_diff"]
+ if config_diff:
+ commands = config_diff.split("\n")
+ if module.params["before"]:
+ commands[:0] = module.params["before"]
+
+ if module.params["after"]:
+ commands.extend(module.params["after"])
+
+ result["commands"] = commands
+ result["updates"] = commands
+
+ replace = module.params["replace"] == "config"
+ commit = not module.check_mode
+
+ response = load_config(
+ module, commands, replace=replace, commit=commit
+ )
+
+ result["changed"] = True
+
+ if module.params["diff_against"] == "session":
+ if "diff" in response:
+ result["diff"] = {"prepared": response["diff"]}
+ else:
+ result["changed"] = False
+
+ if "session" in response:
+ result["session"] = response["session"]
+
+ running_config = module.params["running_config"]
+ startup_config = None
+
+ if module.params["save_when"] == "always":
+ save_config(module, result)
+ elif module.params["save_when"] == "modified":
+ output = run_commands(
+ module,
+ [
+ {"command": "show running-config", "output": "text"},
+ {"command": "show startup-config", "output": "text"},
+ ],
+ )
+
+ running_config = NetworkConfig(
+ indent=3, contents=output[0], ignore_lines=diff_ignore_lines
+ )
+ startup_config = NetworkConfig(
+ indent=3, contents=output[1], ignore_lines=diff_ignore_lines
+ )
+
+ if running_config.sha1 != startup_config.sha1:
+ save_config(module, result)
+
+ elif module.params["save_when"] == "changed" and result["changed"]:
+ save_config(module, result)
+
+ if module._diff:
+ if not running_config:
+ output = run_commands(
+ module, {"command": "show running-config", "output": "text"}
+ )
+ contents = output[0]
+ else:
+ contents = running_config
+
+ # recreate the object in order to process diff_ignore_lines
+ running_config = NetworkConfig(
+ indent=3, contents=contents, ignore_lines=diff_ignore_lines
+ )
+
+ if module.params["diff_against"] == "running":
+ if module.check_mode:
+ module.warn(
+ "unable to perform diff against running-config due to check mode"
+ )
+ contents = None
+ else:
+ contents = config.config_text
+
+ elif module.params["diff_against"] == "startup":
+ if not startup_config:
+ output = run_commands(
+ module,
+ {"command": "show startup-config", "output": "text"},
+ )
+ contents = output[0]
+ else:
+ contents = startup_config.config_text
+
+ elif module.params["diff_against"] == "intended":
+ contents = module.params["intended_config"]
+
+ if contents is not None:
+ base_config = NetworkConfig(
+ indent=3, contents=contents, ignore_lines=diff_ignore_lines
+ )
+
+ if running_config.sha1 != base_config.sha1:
+ if module.params["diff_against"] == "intended":
+ before = running_config
+ after = base_config
+ elif module.params["diff_against"] in ("startup", "running"):
+ before = base_config
+ after = running_config
+
+ result.update(
+ {
+ "changed": True,
+ "diff": {"before": str(before), "after": str(after)},
+ }
+ )
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py
new file mode 100644
index 00000000..24cbd9c8
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py
@@ -0,0 +1,452 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_eapi
+author: Peter Sprygada (@privateip)
+short_description: Manage and configure Arista EOS eAPI.
+requirements:
+- EOS v4.12 or greater
+description:
+- Use to enable or disable eAPI access, and set the port and state of http, https,
+ local_http and unix-socket servers.
+- When enabling eAPI access the default is to enable HTTP on port 80, enable HTTPS
+ on port 443, disable local HTTP, and disable Unix socket server. Use the options
+ listed below to override the default configuration.
+- Requires EOS v4.12 or greater.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+options:
+ http:
+ description:
+ - The C(http) argument controls the operating state of the HTTP transport protocol
+ when eAPI is present in the running-config. When the value is set to True, the
+ HTTP protocol is enabled and when the value is set to False, the HTTP protocol
+ is disabled. By default, when eAPI is first configured, the HTTP protocol is
+ disabled.
+ type: bool
+ aliases:
+ - enable_http
+ http_port:
+ description:
+ - Configures the HTTP port that will listen for connections when the HTTP transport
+ protocol is enabled. This argument accepts integer values in the valid range
+ of 1 to 65535.
+ type: int
+ https:
+ description:
+ - The C(https) argument controls the operating state of the HTTPS transport protocol
+ when eAPI is present in the running-config. When the value is set to True, the
+ HTTPS protocol is enabled and when the value is set to False, the HTTPS protocol
+ is disabled. By default, when eAPI is first configured, the HTTPS protocol is
+ enabled.
+ type: bool
+ aliases:
+ - enable_https
+ https_port:
+ description:
+ - Configures the HTTP port that will listen for connections when the HTTP transport
+ protocol is enabled. This argument accepts integer values in the valid range
+ of 1 to 65535.
+ type: int
+ local_http:
+ description:
+ - The C(local_http) argument controls the operating state of the local HTTP transport
+ protocol when eAPI is present in the running-config. When the value is set
+ to True, the HTTP protocol is enabled and restricted to connections from localhost
+ only. When the value is set to False, the HTTP local protocol is disabled.
+ - Note is value is independent of the C(http) argument
+ type: bool
+ aliases:
+ - enable_local_http
+ local_http_port:
+ description:
+ - Configures the HTTP port that will listen for connections when the HTTP transport
+ protocol is enabled. This argument accepts integer values in the valid range
+ of 1 to 65535.
+ type: int
+ socket:
+ description:
+ - The C(socket) argument controls the operating state of the UNIX Domain Socket
+ used to receive eAPI requests. When the value of this argument is set to True,
+ the UDS will listen for eAPI requests. When the value is set to False, the
+ UDS will not be available to handle requests. By default when eAPI is first
+ configured, the UDS is disabled.
+ type: bool
+ aliases:
+ - enable_socket
+ timeout:
+ description:
+ - The time (in seconds) to wait for the eAPI configuration to be reflected in
+ the running-config.
+ type: int
+ default: 30
+ vrf:
+ description:
+ - The C(vrf) argument will configure eAPI to listen for connections in the specified
+ VRF. By default, eAPI transports will listen for connections in the global
+ table. This value requires the VRF to already be created otherwise the task
+ will fail.
+ default: default
+ type: str
+ config:
+ description:
+ - The module, by default, will connect to the remote device and retrieve the current
+ running-config to use as a base for comparing against the contents of source. There
+ are times when it is not desirable to have the task get the current running-config
+ for every task in a playbook. The I(config) argument allows the implementer
+ to pass in the configuration to use as the base config for comparison.
+ type: str
+ state:
+ description:
+ - The C(state) argument controls the operational state of eAPI on the remote device. When
+ this argument is set to C(started), eAPI is enabled to receive requests and
+ when this argument is C(stopped), eAPI is disabled and will not receive requests.
+ type: str
+ default: started
+ choices:
+ - started
+ - stopped
+"""
+
+EXAMPLES = """
+- name: Enable eAPI access with default configuration
+ arista.eos.eos_eapi:
+ state: started
+
+- name: Enable eAPI with no HTTP, HTTPS at port 9443, local HTTP at port 80, and socket
+ enabled
+ arista.eos.eos_eapi:
+ state: started
+ http: false
+ https_port: 9443
+ local_http: yes
+ local_http_port: 80
+ socket: yes
+
+- name: Shutdown eAPI access
+ arista.eos.eos_eapi:
+ state: stopped
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - management api http-commands
+ - protocol http port 81
+ - no protocol https
+urls:
+ description: Hash of URL endpoints eAPI is listening on per interface
+ returned: when eAPI is started
+ type: dict
+ sample: {'Management1': ['http://172.26.10.1:80']}
+session_name:
+ description: The EOS config session name used to load the configuration
+ returned: when changed is True
+ type: str
+ sample: ansible_1479315771
+"""
+import re
+import time
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ run_commands,
+ load_config,
+)
+from ansible.module_utils.six import iteritems
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def check_transport(module):
+ transport = (module.params["provider"] or {}).get("transport")
+
+ if transport == "eapi":
+ module.fail_json(
+ msg="eos_eapi module is only supported over cli transport"
+ )
+
+
+def validate_http_port(value, module):
+ if not 1 <= value <= 65535:
+ module.fail_json(msg="http_port must be between 1 and 65535")
+
+
+def validate_https_port(value, module):
+ if not 1 <= value <= 65535:
+ module.fail_json(msg="http_port must be between 1 and 65535")
+
+
+def validate_local_http_port(value, module):
+ if not 1 <= value <= 65535:
+ module.fail_json(msg="http_port must be between 1 and 65535")
+
+
+def validate_vrf(value, module):
+ out = run_commands(module, ["show vrf"])
+ configured_vrfs = []
+ lines = out[0].strip().splitlines()[3:]
+ for line in lines:
+ if not line:
+ continue
+ splitted_line = re.split(r"\s{2,}", line.strip())
+ if len(splitted_line) > 2:
+ configured_vrfs.append(splitted_line[0])
+
+ configured_vrfs.append("default")
+ if value not in configured_vrfs:
+ module.fail_json(
+ msg="vrf `%s` is not configured on the system" % value
+ )
+
+
+def map_obj_to_commands(updates, module, warnings):
+ commands = list()
+ want, have = updates
+
+ def needs_update(x):
+ return want.get(x) is not None and (want.get(x) != have.get(x))
+
+ def add(cmd):
+ if "management api http-commands" not in commands:
+ commands.insert(0, "management api http-commands")
+ commands.append(cmd)
+
+ if any((needs_update("http"), needs_update("http_port"))):
+ if want["http"] is False:
+ add("no protocol http")
+ else:
+ if have["http"] is False and want["http"] in (False, None):
+ warnings.append(
+ "protocol http is not enabled, not configuring http port value"
+ )
+ else:
+ port = want["http_port"] or 80
+ add("protocol http port %s" % port)
+
+ if any((needs_update("https"), needs_update("https_port"))):
+ if want["https"] is False:
+ add("no protocol https")
+ else:
+ if have["https"] is False and want["https"] in (False, None):
+ warnings.append(
+ "protocol https is not enabled, not configuring https port value"
+ )
+ else:
+ port = want["https_port"] or 443
+ add("protocol https port %s" % port)
+
+ if any((needs_update("local_http"), needs_update("local_http_port"))):
+ if want["local_http"] is False:
+ add("no protocol http localhost")
+ else:
+ if have["local_http"] is False and want["local_http"] in (
+ False,
+ None,
+ ):
+ warnings.append(
+ "protocol local_http is not enabled, not configuring local_http port value"
+ )
+ else:
+ port = want["local_http_port"] or 8080
+ add("protocol http localhost port %s" % port)
+
+ if any((needs_update("socket"), needs_update("socket"))):
+ if want["socket"] is False:
+ add("no protocol unix-socket")
+ else:
+ add("protocol unix-socket")
+ if needs_update("state"):
+ if want["state"] == "stopped":
+ add("shutdown")
+ elif want["state"] == "started":
+ add("no shutdown")
+
+ if needs_update("vrf"):
+ add("vrf %s" % want["vrf"])
+ # switching operational vrfs here
+ # need to add the desired state as well
+ if want["state"] == "stopped":
+ add("shutdown")
+ elif want["state"] == "started":
+ add("no shutdown")
+
+ return commands
+
+
+def parse_state(data):
+ if data[0]["enabled"]:
+ return "started"
+ else:
+ return "stopped"
+
+
+def map_config_to_obj(module):
+ out = run_commands(module, ["show management api http-commands | json"])
+ return {
+ "http": out[0]["httpServer"]["configured"],
+ "http_port": out[0]["httpServer"]["port"],
+ "https": out[0]["httpsServer"]["configured"],
+ "https_port": out[0]["httpsServer"]["port"],
+ "local_http": out[0]["localHttpServer"]["configured"],
+ "local_http_port": out[0]["localHttpServer"]["port"],
+ "socket": out[0]["unixSocketServer"]["configured"],
+ "vrf": out[0]["vrf"] or "default",
+ "state": parse_state(out),
+ }
+
+
+def map_params_to_obj(module):
+ obj = {
+ "http": module.params["http"],
+ "http_port": module.params["http_port"],
+ "https": module.params["https"],
+ "https_port": module.params["https_port"],
+ "local_http": module.params["local_http"],
+ "local_http_port": module.params["local_http_port"],
+ "socket": module.params["socket"],
+ "vrf": module.params["vrf"],
+ "state": module.params["state"],
+ }
+
+ for key, value in iteritems(obj):
+ if value:
+ validator = globals().get("validate_%s" % key)
+ if validator:
+ validator(value, module)
+
+ return obj
+
+
+def verify_state(updates, module):
+ want, have = updates
+
+ invalid_state = [
+ ("http", "httpServer"),
+ ("https", "httpsServer"),
+ ("local_http", "localHttpServer"),
+ ("socket", "unixSocketServer"),
+ ]
+
+ timeout = module.params["timeout"]
+ state = module.params["state"]
+
+ while invalid_state:
+ out = run_commands(
+ module, ["show management api http-commands | json"]
+ )
+ for index, item in enumerate(invalid_state):
+ want_key, eapi_key = item
+ if want[want_key] is not None:
+ if want[want_key] == out[0][eapi_key]["running"]:
+ del invalid_state[index]
+ elif state == "stopped":
+ if not out[0][eapi_key]["running"]:
+ del invalid_state[index]
+ else:
+ del invalid_state[index]
+ time.sleep(1)
+ timeout -= 1
+ if timeout == 0:
+ module.fail_json(
+ msg="timeout expired before eapi running state changed"
+ )
+
+
+def collect_facts(module, result):
+ out = run_commands(module, ["show management api http-commands | json"])
+ facts = dict(eos_eapi_urls=dict())
+ for each in out[0]["urls"]:
+ intf, url = each.split(" : ")
+ key = str(intf).strip()
+ if key not in facts["eos_eapi_urls"]:
+ facts["eos_eapi_urls"][key] = list()
+ facts["eos_eapi_urls"][key].append(str(url).strip())
+ result["ansible_facts"] = facts
+
+
+def main():
+ """ main entry point for module execution
+ """
+ argument_spec = dict(
+ http=dict(aliases=["enable_http"], type="bool"),
+ http_port=dict(type="int"),
+ https=dict(aliases=["enable_https"], type="bool"),
+ https_port=dict(type="int"),
+ local_http=dict(aliases=["enable_local_http"], type="bool"),
+ local_http_port=dict(type="int"),
+ socket=dict(aliases=["enable_socket"], type="bool"),
+ timeout=dict(type="int", default=30),
+ vrf=dict(default="default"),
+ config=dict(),
+ state=dict(default="started", choices=["stopped", "started"]),
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ module = AnsibleModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+
+ check_transport(module)
+
+ result = {"changed": False}
+
+ warnings = list()
+ if module.params["config"]:
+ warnings.append(
+ "config parameter is no longer necessary and will be ignored"
+ )
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands((want, have), module, warnings)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ if result["changed"]:
+ verify_state((want, have), module)
+
+ collect_facts(module, result)
+
+ if warnings:
+ result["warnings"] = warnings
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py
new file mode 100644
index 00000000..24e61ea7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py
@@ -0,0 +1,207 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_facts
+author:
+- Peter Sprygada (@privateip)
+- Nathaniel Case (@Qalthos)
+short_description: Collect facts from remote devices running Arista EOS
+description:
+- Collects facts from Arista devices running the EOS operating system. This module
+ places the facts gathered in the fact tree keyed by the respective resource name. The
+ facts module will always collect a base set of facts from the device and can enable
+ or disable collection of additional facts.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+options:
+ gather_subset:
+ description:
+ - When supplied, this argument will restrict the facts collected to a given subset. Possible
+ values for this argument include all, hardware, config, and interfaces. Can
+ specify a list of values to include a larger subset. Values can also be used
+ with an initial C(M(!)) to specify that a specific subset should not be collected.
+ required: false
+ type: list
+ elements: str
+ default: '!config'
+ gather_network_resources:
+ description:
+ - When supplied, this argument will restrict the facts collected to a given subset.
+ Possible values for this argument include all and the resources like interfaces,
+ vlans etc. Can specify a list of values to include a larger subset. Values can
+ also be used with an initial C(M(!)) to specify that a specific subset should
+ not be collected. Values can also be used with an initial C(M(!)) to specify
+ that a specific subset should not be collected. Valid subsets are 'all', 'interfaces',
+ 'l2_interfaces', 'l3_interfaces', 'lacp', 'lacp_interfaces', 'lag_interfaces',
+ 'lldp_global', 'lldp_interfaces', 'vlans', 'acls'.
+ required: false
+ type: list
+ elements: str
+"""
+
+EXAMPLES = """
+- name: Gather all legacy facts
+- arista.eos.eos_facts:
+ gather_subset: all
+
+- name: Gather only the config and default facts
+ arista.eos.eos_facts:
+ gather_subset:
+ - config
+
+- name: Do not gather hardware facts
+ arista.eos.eos_facts:
+ gather_subset:
+ - '!hardware'
+
+- name: Gather legacy and resource facts
+ arista.eos.eos_facts:
+ gather_subset: all
+ gather_network_resources: all
+
+- name: Gather only the interfaces resource facts and no legacy facts
+- arista.eos.eos_facts:
+ gather_subset:
+ - '!all'
+ - '!min'
+ gather_network_resources:
+ - interfaces
+
+- name: Gather all resource facts and minimal legacy facts
+ arista.eos.eos_facts:
+ gather_subset: min
+ gather_network_resources: all
+"""
+
+RETURN = """
+ansible_net_gather_subset:
+ description: The list of fact subsets collected from the device
+ returned: always
+ type: list
+
+ansible_net_gather_network_resources:
+ description: The list of fact for network resource subsets collected from the device
+ returned: when the resource is configured
+ type: list
+
+# default
+ansible_net_model:
+ description: The model name returned from the device
+ returned: always
+ type: str
+ansible_net_serialnum:
+ description: The serial number of the remote device
+ returned: always
+ type: str
+ansible_net_version:
+ description: The operating system version running on the remote device
+ returned: always
+ type: str
+ansible_net_hostname:
+ description: The configured hostname of the device
+ returned: always
+ type: str
+ansible_net_image:
+ description: The image file the device is running
+ returned: always
+ type: str
+ansible_net_fqdn:
+ description: The fully qualified domain name of the device
+ returned: always
+ type: str
+ansible_net_api:
+ description: The name of the transport
+ returned: always
+ type: str
+ansible_net_python_version:
+ description: The Python version Ansible controller is using
+ returned: always
+ type: str
+
+# hardware
+ansible_net_filesystems:
+ description: All file system names available on the device
+ returned: when hardware is configured
+ type: list
+ansible_net_memfree_mb:
+ description: The available free memory on the remote device in Mb
+ returned: when hardware is configured
+ type: int
+ansible_net_memtotal_mb:
+ description: The total memory on the remote device in Mb
+ returned: when hardware is configured
+ type: int
+
+# config
+ansible_net_config:
+ description: The current active config from the device
+ returned: when config is configured
+ type: str
+
+# interfaces
+ansible_net_all_ipv4_addresses:
+ description: All IPv4 addresses configured on the device
+ returned: when interfaces is configured
+ type: list
+ansible_net_all_ipv6_addresses:
+ description: All IPv6 addresses configured on the device
+ returned: when interfaces is configured
+ type: list
+ansible_net_interfaces:
+ description: A hash of all interfaces running on the system
+ returned: when interfaces is configured
+ type: dict
+ansible_net_neighbors:
+ description: The list of LLDP neighbors from the remote device
+ returned: when interfaces is configured
+ type: dict
+"""
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.facts.facts import (
+ FactsArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import (
+ Facts,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: ansible_facts
+ """
+ argument_spec = FactsArgs.argument_spec
+ argument_spec.update(eos_argument_spec)
+
+ module = AnsibleModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+ warnings = [
+ "default value for `gather_subset` "
+ "will be changed to `min` from `!config` v2.11 onwards"
+ ]
+
+ result = Facts(module).get_facts()
+
+ ansible_facts, additional_warnings = result
+ warnings.extend(additional_warnings)
+
+ module.exit_json(ansible_facts=ansible_facts, warnings=warnings)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py
new file mode 100644
index 00000000..b90903ee
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py
@@ -0,0 +1,601 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# Copyright: (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_interface
+author: Ganesh Nalawade (@ganeshrn)
+short_description: (deprecated, removed after 2022-06-01) Manage Interface on Arista
+ EOS network devices
+description:
+- This module provides declarative management of Interfaces on Arista EOS network
+ devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_interfaces
+ why: Updated modules released with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ name:
+ description:
+ - Name of the Interface to be configured on remote device. The name of interface
+ should be in expanded format and not abbreviated.
+ type: str
+ description:
+ description:
+ - Description of Interface upto 240 characters.
+ type: str
+ enabled:
+ description:
+ - Interface link status. If the value is I(True) the interface state will be enabled,
+ else if value is I(False) interface will be in disable (shutdown) state.
+ default: true
+ type: bool
+ speed:
+ description:
+ - This option configures autoneg and speed/duplex/flowcontrol for the interface
+ given in C(name) option.
+ type: str
+ mtu:
+ description:
+ - Set maximum transmission unit size in bytes of transmit packet for the interface
+ given in C(name) option.
+ type: str
+ tx_rate:
+ description:
+ - Transmit rate in bits per second (bps) for the interface given in C(name) option.
+ - This is state check parameter only.
+ - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html)
+ type: str
+ rx_rate:
+ description:
+ - Receiver rate in bits per second (bps) for the interface given in C(name) option.
+ - This is state check parameter only.
+ - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html)
+ type: str
+ neighbors:
+ description:
+ - Check the operational state of given interface C(name) for LLDP neighbor.
+ - The following suboptions are available.
+ type: list
+ elements: dict
+ suboptions:
+ host:
+ description:
+ - LLDP neighbor host for given interface C(name).
+ type: str
+ port:
+ description:
+ - LLDP neighbor port to which given interface C(name) is connected.
+ type: str
+ aggregate:
+ description:
+ - List of Interfaces definitions. Each of the entry in aggregate list should define
+ name of interface C(name) and other options as required.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the Interface to be configured on remote device. The name of interface
+ should be in expanded format and not abbreviated.
+ type: str
+ required: true
+ description:
+ description:
+ - Description of Interface upto 240 characters.
+ type: str
+ enabled:
+ description:
+ - Interface link status. If the value is I(True) the interface state will be enabled,
+ else if value is I(False) interface will be in disable (shutdown) state.
+ type: bool
+ speed:
+ description:
+ - This option configures autoneg and speed/duplex/flowcontrol for the interface
+ given in C(name) option.
+ type: str
+ mtu:
+ description:
+ - Set maximum transmission unit size in bytes of transmit packet for the interface
+ given in C(name) option.
+ type: str
+ tx_rate:
+ description:
+ - Transmit rate in bits per second (bps) for the interface given in C(name) option.
+ - This is state check parameter only.
+ - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html)
+ type: str
+ rx_rate:
+ description:
+ - Receiver rate in bits per second (bps) for the interface given in C(name) option.
+ - This is state check parameter only.
+ - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html)
+ type: str
+ neighbors:
+ description:
+ - Check the operational state of given interface C(name) for LLDP neighbor.
+ - The following suboptions are available.
+ type: list
+ elements: dict
+ suboptions:
+ host:
+ description:
+ - LLDP neighbor host for given interface C(name).
+ type: str
+ port:
+ description:
+ - LLDP neighbor port to which given interface C(name) is connected.
+ type: str
+ delay:
+ description:
+ - Time in seconds to wait before checking for the operational state on remote
+ device. This wait is applicable for operational state argument which are I(state)
+ with values C(up)/C(down), I(tx_rate) and I(rx_rate).
+ default: 10
+ type: int
+ state:
+ description:
+ - State of the Interface configuration, C(up) means present and operationally
+ up and C(down) means present and operationally C(down)
+ type: str
+ choices:
+ - present
+ - absent
+ - up
+ - down
+ delay:
+ description:
+ - Time in seconds to wait before checking for the operational state on remote
+ device. This wait is applicable for operational state argument which are I(state)
+ with values C(up)/C(down), I(tx_rate) and I(rx_rate).
+ default: 10
+ type: int
+ state:
+ description:
+ - State of the Interface configuration, C(up) means present and operationally
+ up and C(down) means present and operationally C(down)
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+ - up
+ - down
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: configure interface
+ arista.eos.eos_interface:
+ name: ethernet1
+ description: test-interface
+ speed: 100full
+ mtu: 512
+
+- name: remove interface
+ arista.eos.eos_interface:
+ name: ethernet1
+ state: absent
+
+- name: make interface up
+ arista.eos.eos_interface:
+ name: ethernet1
+ enabled: true
+
+- name: make interface down
+ arista.eos.eos_interface:
+ name: ethernet1
+ enabled: false
+
+- name: Check intent arguments
+ arista.eos.eos_interface:
+ name: ethernet1
+ state: up
+ tx_rate: ge(0)
+ rx_rate: le(0)
+
+- name: Check neighbors intent arguments
+ arista.eos.eos_interface:
+ name: ethernet1
+ neighbors:
+ - port: eth0
+ host: netdev
+
+- name: Configure interface in disabled state and check if the operational state is
+ disabled or not
+ arista.eos.eos_interface:
+ name: ethernet1
+ enabled: false
+ state: down
+
+- name: Add interface using aggregate
+ arista.eos.eos_interface:
+ aggregate:
+ - {name: ethernet1, mtu: 256, description: test-interface-1}
+ - {name: ethernet2, mtu: 516, description: test-interface-2}
+ speed: 100full
+ state: present
+
+- name: Delete interface using aggregate
+ arista.eos.eos_interface:
+ aggregate:
+ - name: loopback9
+ - name: loopback10
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device.
+ returned: always, except for the platforms that use Netconf transport to manage the device.
+ type: list
+ sample:
+ - interface ethernet1
+ - description test-interface
+ - speed 100full
+ - mtu 512
+"""
+import re
+from copy import deepcopy
+from time import sleep
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ conditional,
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def validate_mtu(value, module):
+ if value and not 68 <= int(value) <= 65535:
+ module.fail_json(msg="mtu must be between 68 and 65535")
+
+
+def validate_param_values(module, obj, param=None):
+ if param is None:
+ param = module.params
+ for key in obj:
+ # validate the param value (if validator func exists)
+ validator = globals().get("validate_%s" % key)
+ if callable(validator):
+ validator(param.get(key), module)
+
+
+def parse_shutdown(configobj, name):
+ cfg = configobj["interface %s" % name]
+ cfg = "\n".join(cfg.children)
+ match = re.search(r"shutdown", cfg, re.M)
+ return bool(match)
+
+
+def parse_config_argument(configobj, name, arg=None):
+ cfg = configobj["interface %s" % name]
+ cfg = "\n".join(cfg.children)
+ match = re.search(r"%s (.+)$" % arg, cfg, re.M)
+ if match:
+ return match.group(1)
+
+
+def search_obj_in_list(name, lst):
+ for o in lst:
+ if o["name"] == name:
+ return o
+
+ return None
+
+
+def add_command_to_interface(interface, cmd, commands):
+ if interface not in commands:
+ commands.append(interface)
+ commands.append(cmd)
+
+
+def map_config_to_obj(module):
+ config = get_config(module)
+ configobj = NetworkConfig(indent=3, contents=config)
+
+ match = re.findall(r"^interface (\S+)", config, re.M)
+ if not match:
+ return list()
+
+ instances = list()
+
+ for item in set(match):
+ obj = {
+ "name": item.lower(),
+ "description": parse_config_argument(
+ configobj, item, "description"
+ ),
+ "speed": parse_config_argument(configobj, item, "speed"),
+ "mtu": parse_config_argument(configobj, item, "mtu"),
+ "disable": parse_shutdown(configobj, item),
+ "state": "present",
+ }
+ instances.append(obj)
+ return instances
+
+
+def map_params_to_obj(module):
+ obj = []
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ item["name"] = item["name"].lower()
+ validate_param_values(module, item, item)
+ d = item.copy()
+
+ if d["enabled"]:
+ d["disable"] = False
+ else:
+ d["disable"] = True
+
+ obj.append(d)
+
+ else:
+ params = {
+ "name": module.params["name"].lower(),
+ "description": module.params["description"],
+ "speed": module.params["speed"],
+ "mtu": module.params["mtu"],
+ "state": module.params["state"],
+ "delay": module.params["delay"],
+ "tx_rate": module.params["tx_rate"],
+ "rx_rate": module.params["rx_rate"],
+ "neighbors": module.params["neighbors"],
+ }
+
+ validate_param_values(module, params)
+ if module.params["enabled"]:
+ params.update({"disable": False})
+ else:
+ params.update({"disable": True})
+
+ obj.append(params)
+ return obj
+
+
+def map_obj_to_commands(updates, modules):
+ commands = list()
+ want, have = updates
+
+ args = ("speed", "description", "mtu")
+ for w in want:
+ name = w["name"]
+ disable = w["disable"]
+ state = w["state"]
+
+ obj_in_have = search_obj_in_list(name, have)
+ interface = "interface " + name
+
+ if state == "absent" and obj_in_have:
+ commands.append("no " + interface)
+
+ elif state in ("present", "up", "down"):
+ if obj_in_have:
+ for item in args:
+ candidate = w.get(item)
+ running = obj_in_have.get(item)
+ if candidate != running:
+ if candidate:
+ cmd = "{0} {1}".format(item, candidate)
+ add_command_to_interface(interface, cmd, commands)
+
+ if disable and not obj_in_have.get("disable", False):
+ add_command_to_interface(interface, "shutdown", commands)
+ elif not disable and obj_in_have.get("disable", False):
+ add_command_to_interface(
+ interface, "no shutdown", commands
+ )
+ else:
+ commands.append(interface)
+ for item in args:
+ value = w.get(item)
+ if value:
+ commands.append("{0} {1}".format(item, value))
+
+ if disable:
+ commands.append("no shutdown")
+ return commands
+
+
+def check_declarative_intent_params(module, want, result):
+ failed_conditions = []
+ have_neighbors = None
+ for w in want:
+ want_state = w.get("state")
+ want_tx_rate = w.get("tx_rate")
+ want_rx_rate = w.get("rx_rate")
+ want_neighbors = w.get("neighbors")
+
+ if (
+ want_state not in ("up", "down")
+ and not want_tx_rate
+ and not want_rx_rate
+ and not want_neighbors
+ ):
+ continue
+
+ if result["changed"]:
+ sleep(w["delay"])
+
+ command = {
+ "command": "show interfaces %s" % w["name"],
+ "output": "text",
+ }
+ output = run_commands(module, [command])
+
+ if want_state in ("up", "down"):
+ match = re.search(
+ r"%s (\w+)" % "line protocol is", output[0], re.M
+ )
+ have_state = None
+ if match:
+ have_state = match.group(1)
+ if have_state is None or not conditional(
+ want_state, have_state.strip()
+ ):
+ failed_conditions.append("state " + "eq(%s)" % want_state)
+
+ if want_tx_rate:
+ match = re.search(r"%s (\d+)" % "output rate", output[0], re.M)
+ have_tx_rate = None
+ if match:
+ have_tx_rate = match.group(1)
+
+ if have_tx_rate is None or not conditional(
+ want_tx_rate, have_tx_rate.strip(), cast=int
+ ):
+ failed_conditions.append("tx_rate " + want_tx_rate)
+
+ if want_rx_rate:
+ match = re.search(r"%s (\d+)" % "input rate", output[0], re.M)
+ have_rx_rate = None
+ if match:
+ have_rx_rate = match.group(1)
+
+ if have_rx_rate is None or not conditional(
+ want_rx_rate, have_rx_rate.strip(), cast=int
+ ):
+ failed_conditions.append("rx_rate " + want_rx_rate)
+
+ if want_neighbors:
+ have_host = []
+ have_port = []
+ if have_neighbors is None:
+ command = {
+ "command": "show lldp neighbors {0}".format(w["name"]),
+ "output": "text",
+ }
+ have_neighbors = run_commands(module, [command])
+
+ if have_neighbors[0]:
+ lines = have_neighbors[0].strip().split("\n")
+ col = None
+ for index, line in enumerate(lines):
+ if re.search(
+ r"^Port\s+Neighbor Device ID\s+Neighbor Port ID\s+TTL",
+ line,
+ ):
+ col = index
+ break
+
+ if col and col < len(lines) - 1:
+ idx = col + 1
+ for items in lines[idx:]:
+ value = re.split(r"\s+", items)
+ try:
+ have_port.append(value[2])
+ have_host.append(value[1])
+ except IndexError:
+ pass
+
+ for item in want_neighbors:
+ host = item.get("host")
+ port = item.get("port")
+ if host and host not in have_host:
+ failed_conditions.append("host " + host)
+ if port and port not in have_port:
+ failed_conditions.append("port " + port)
+ return failed_conditions
+
+
+def main():
+ """ main entry point for module execution
+ """
+ neighbors_spec = dict(host=dict(), port=dict())
+
+ element_spec = dict(
+ name=dict(),
+ description=dict(),
+ speed=dict(),
+ mtu=dict(),
+ enabled=dict(default=True, type="bool"),
+ tx_rate=dict(),
+ rx_rate=dict(),
+ neighbors=dict(type="list", elements="dict", options=neighbors_spec),
+ delay=dict(default=10, type="int"),
+ state=dict(
+ default="present", choices=["present", "absent", "up", "down"]
+ ),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["name"] = dict(required=True)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+ aggregate_spec["delay"].update(default=10)
+
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec)
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["name", "aggregate"]]
+ mutually_exclusive = [["name", "aggregate"]]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_one_of=required_one_of,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ failed_conditions = check_declarative_intent_params(module, want, result)
+
+ if failed_conditions:
+ msg = "One or more conditional statements have not been satisfied"
+ module.fail_json(msg=msg, failed_conditions=failed_conditions)
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py
new file mode 100644
index 00000000..b844816a
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py
@@ -0,0 +1,413 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+##############################################
+# WARNING #
+##############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+##############################################
+
+"""
+The module file for eos_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_interfaces
+short_description: Interfaces resource module
+description:
+- This module manages the interface attributes of Arista EOS interfaces.
+version_added: 1.0.0
+author:
+- Nathaniel Case (@qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: The provided configuration
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Full name of the interface, e.g. GigabitEthernet1.
+ type: str
+ required: True
+ description:
+ description:
+ - Interface description
+ type: str
+ duplex:
+ description:
+ - Interface link status. Applicable for Ethernet interfaces only.
+ - Values other than C(auto) must also set I(speed).
+ - Ignored when I(speed) is set above C(1000).
+ type: str
+ enabled:
+ default: true
+ description:
+ - Administrative state of the interface.
+ - Set the value to C(true) to administratively enable the interface or C(false)
+ to disable it.
+ type: bool
+ mtu:
+ description:
+ - MTU for a specific interface. Must be an even number between 576 and 9216.
+ Applicable for Ethernet interfaces only.
+ type: int
+ speed:
+ description:
+ - Interface link speed. Applicable for Ethernet interfaces only.
+ type: str
+ mode:
+ description:
+ - Manage Layer2 or Layer3 state of the interface. Applicable for Ethernet
+ and port channel interfaces only.
+ choices:
+ - layer2
+ - layer3
+ type: str
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^interface).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - parsed
+ - rendered
+ - gathered
+ default: merged
+ description:
+ - The state of the configuration after module completion.
+ type: str
+
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# description "Interface 1"
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+- name: Merge provided configuration with device configuration
+ arista.eos.eos_interfaces:
+ config:
+ - name: Ethernet1
+ enabled: true
+ mode: layer3
+ - name: Ethernet2
+ description: Configured by Ansible
+ enabled: false
+ state: merged
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# description "Interface 1"
+# no switchport
+# !
+# interface Ethernet2
+# description "Configured by Ansible"
+# shutdown
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+# Using replaced
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# description "Interface 1"
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+- name: Replaces device configuration of listed interfaces with provided configuration
+ arista.eos.eos_interfaces:
+ config:
+ - name: Ethernet1
+ enabled: true
+ - name: Ethernet2
+ description: Configured by Ansible
+ enabled: false
+ state: replaced
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# description "Configured by Ansible"
+# shutdown
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+# Using overridden
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# description "Interface 1"
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+- name: Overrides all device configuration with provided configuration
+ arista.eos.eos_interfaces:
+ config:
+ - name: Ethernet1
+ enabled: true
+ - name: Ethernet2
+ description: Configured by Ansible
+ enabled: false
+ state: overridden
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# description "Configured by Ansible"
+# shutdown
+# !
+# interface Management1
+# ip address dhcp
+# !
+
+# Using deleted
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# description "Interface 1"
+# no switchport
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+- name: Delete or return interface parameters to default settings
+ arista.eos.eos_interfaces:
+ config:
+ - name: Ethernet1
+ state: deleted
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# description "Management interface"
+# ip address dhcp
+# !
+
+# Using rendered
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_interfaces:
+ config:
+ - name: Ethernet1
+ enabled: true
+ mode: layer3
+ - name: Ethernet2
+ description: Configured by Ansible
+ enabled: false
+ state: merged
+
+# Output:
+# ------------
+
+# - "interface Ethernet1"
+# - "description "Interface 1""
+# - "no swithcport"
+# - "interface Ethernet2"
+# - "description "Configured by Ansible""
+# - "shutdown"
+# - "interface Management1"
+# - "description "Management interface""
+# - "ip address dhcp"
+
+# Using parsed
+# parsed.cfg
+
+# interface Ethernet1
+# description "Interface 1"
+# !
+# interface Ethernet2
+# description "Configured by Ansible"
+# shutdown
+# !
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output
+# parsed:
+# - name: Ethernet1
+# enabled: True
+# mode: layer2
+# - name: Ethernet2
+# description: 'Configured by Ansible'
+# enabled: False
+# mode: layer2
+
+# Using gathered:
+
+# Existing config on the device
+# -----------------------------
+# interface Ethernet1
+# description "Interface 1"
+# !
+# interface Ethernet2
+# description "Configured by Ansible"
+# shutdown
+# !
+
+- name: Gather interfaces facts from the device
+ arista.eos.interfaces:
+ state: gathered
+
+# output
+# gathered:
+# - name: Ethernet1
+# enabled: True
+# mode: layer2
+# - name: Ethernet2
+# description: 'Configured by Ansible'
+# enabled: False
+# mode: layer2
+"""
+
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: dict
+ sample: The configuration returned will always be in the same format of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: dict
+ sample: The configuration returned will always be in the same format of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['interface Ethernet2', 'shutdown', 'speed 10full']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.interfaces.interfaces import (
+ InterfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.interfaces.interfaces import (
+ Interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+
+ module = AnsibleModule(
+ argument_spec=InterfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py
new file mode 100644
index 00000000..50edb8d8
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py
@@ -0,0 +1,461 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_l2_interface
+author: Ricardo Carrillo Cruz (@rcarrillocruz)
+short_description: (deprecated, removed after 2022-06-01) Manage L2
+ interfaces on Arista EOS network devices.
+description:
+- This module provides declarative management of L2 interfaces on Arista EOS network
+ devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_l2_interfaces
+ why: Updated modules released with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ name:
+ description:
+ - Name of the interface
+ type: str
+ aliases:
+ - interface
+ mode:
+ description:
+ - Mode in which interface needs to be configured.
+ type: str
+ choices:
+ - access
+ - trunk
+ access_vlan:
+ description:
+ - Configure given VLAN in access port. If C(mode=access), used as the access VLAN
+ ID.
+ type: str
+ native_vlan:
+ description:
+ - Native VLAN to be configured in trunk port. If C(mode=trunk), used as the trunk
+ native VLAN ID.
+ type: str
+ trunk_allowed_vlans:
+ description:
+ - List of allowed VLANs in a given trunk port. If C(mode=trunk), these are the
+ ONLY VLANs that will be configured on the trunk, i.e. C(2-10,15).
+ type: str
+ aliases:
+ - trunk_vlans
+ aggregate:
+ description:
+ - List of Layer-2 interface definitions.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the interface
+ type: str
+ required: true
+ access_vlan:
+ description:
+ - Configure given VLAN in access port. If C(mode=access), used as the access VLAN
+ ID.
+ type: str
+ native_vlan:
+ description:
+ - Native VLAN to be configured in trunk port. If C(mode=trunk), used as the trunk
+ native VLAN ID.
+ type: str
+ trunk_allowed_vlans:
+ description:
+ - List of allowed VLANs in a given trunk port. If C(mode=trunk), these are the
+ ONLY VLANs that will be configured on the trunk, i.e. C(2-10,15).
+ type: str
+ aliases:
+ - trunk_vlans
+ mode:
+ description:
+ - Mode in which interface needs to be configured.
+ type: str
+ choices:
+ - access
+ - trunk
+ state:
+ description:
+ - Manage the state of the Layer-2 Interface configuration.
+ type: str
+ choices:
+ - present
+ - absent
+ state:
+ description:
+ - Manage the state of the Layer-2 Interface configuration.
+ type: str
+ default: present
+ choices:
+ - present
+ - absent
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: Ensure Ethernet1 does not have any switchport
+ arista.eos.eos_l2_interface:
+ name: Ethernet1
+ state: absent
+
+- name: Ensure Ethernet1 is configured for access vlan 20
+ arista.eos.eos_l2_interface:
+ name: Ethernet1
+ mode: access
+ access_vlan: 20
+
+- name: Ensure Ethernet1 is a trunk port and ensure 2-50 are being tagged (doesn't
+ mean others aren't also being tagged)
+ arista.eos.eos_l2_interface:
+ name: Ethernet1
+ mode: trunk
+ native_vlan: 10
+ trunk_allowed_vlans: 2-50
+
+- name: Set switchports on aggregate
+ arista.eos.eos_l2_interface:
+ aggregate:
+ - {name: ethernet1, mode: access, access_vlan: 20}
+ - {name: ethernet2, mode: trunk, native_vlan: 10}
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always.
+ type: list
+ sample:
+ - interface ethernet1
+ - switchport access vlan 20
+"""
+import re
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def parse_config_argument(configobj, name, arg=None):
+ cfg = configobj["interface %s" % name]
+ cfg = "\n".join(cfg.children)
+ match = re.search(r"%s (.+)$" % arg, cfg, re.M)
+ if match:
+ return match.group(1).strip()
+
+
+def search_obj_in_list(name, lst):
+ for o in lst:
+ if o["name"] == name:
+ return o
+
+ return None
+
+
+def map_obj_to_commands(want, have, module):
+ commands = list()
+
+ for w in want:
+ name = w["name"]
+ state = w["state"]
+ mode = w["mode"]
+ access_vlan = w["access_vlan"]
+ native_vlan = w["native_vlan"]
+ trunk_allowed_vlans = w["trunk_allowed_vlans"]
+
+ interface = "interface " + name
+ commands.append(interface)
+
+ obj_in_have = search_obj_in_list(name, have)
+ if not obj_in_have:
+ module.fail_json(msg="invalid interface {0}".format(name))
+
+ if state == "absent":
+ if obj_in_have["mode"] == "access":
+ commands.append(
+ "no switchport access vlan {0}".format(
+ obj_in_have["access_vlan"]
+ )
+ )
+
+ if obj_in_have["mode"] == "trunk":
+ commands.append("no switchport mode trunk")
+
+ if obj_in_have["native_vlan"]:
+ commands.append(
+ "no switchport trunk native vlan {0}".format(
+ obj_in_have["native_vlan"]
+ )
+ )
+
+ if obj_in_have["trunk_allowed_vlans"]:
+ commands.append(
+ "no switchport trunk allowed vlan {0}".format(
+ obj_in_have["trunk_allowed_vlans"]
+ )
+ )
+
+ if obj_in_have["state"] == "present":
+ commands.append("no switchport")
+ else:
+ if obj_in_have["state"] == "absent":
+ commands.append("switchport")
+ commands.append("switchport mode {0}".format(mode))
+
+ if access_vlan:
+ commands.append(
+ "switchport access vlan {0}".format(access_vlan)
+ )
+
+ if native_vlan:
+ commands.append(
+ "switchport trunk native vlan {0}".format(native_vlan)
+ )
+
+ if trunk_allowed_vlans:
+ commands.append(
+ "switchport trunk allowed vlan {0}".format(
+ trunk_allowed_vlans
+ )
+ )
+ else:
+ if mode != obj_in_have["mode"]:
+ if obj_in_have["mode"] == "access":
+ commands.append(
+ "no switchport access vlan {0}".format(
+ obj_in_have["access_vlan"]
+ )
+ )
+ commands.append("switchport mode trunk")
+ if native_vlan:
+ commands.append(
+ "switchport trunk native vlan {0}".format(
+ native_vlan
+ )
+ )
+ if trunk_allowed_vlans:
+ commands.append(
+ "switchport trunk allowed vlan {0}".format(
+ trunk_allowed_vlans
+ )
+ )
+ else:
+ if obj_in_have["native_vlan"]:
+ commands.append(
+ "no switchport trunk native vlan {0}".format(
+ obj_in_have["native_vlan"]
+ )
+ )
+ commands.append("no switchport mode trunk")
+ if obj_in_have["trunk_allowed_vlans"]:
+ commands.append(
+ "no switchport trunk allowed vlan {0}".format(
+ obj_in_have["trunk_allowed_vlans"]
+ )
+ )
+ commands.append("no switchport mode trunk")
+ commands.append(
+ "switchport access vlan {0}".format(access_vlan)
+ )
+ else:
+ if mode == "access":
+ if access_vlan != obj_in_have["access_vlan"]:
+ commands.append(
+ "switchport access vlan {0}".format(
+ access_vlan
+ )
+ )
+ else:
+ if (
+ native_vlan != obj_in_have["native_vlan"]
+ and native_vlan
+ ):
+ commands.append(
+ "switchport trunk native vlan {0}".format(
+ native_vlan
+ )
+ )
+ if (
+ trunk_allowed_vlans
+ != obj_in_have["trunk_allowed_vlans"]
+ and trunk_allowed_vlans
+ ):
+ commands.append(
+ "switchport trunk allowed vlan {0}".format(
+ trunk_allowed_vlans
+ )
+ )
+
+ if commands[-1] == interface:
+ commands.pop(-1)
+
+ return commands
+
+
+def map_config_to_obj(module, warnings):
+ config = get_config(module, flags=["| section interface"])
+ configobj = NetworkConfig(indent=3, contents=config)
+
+ match = re.findall(r"^interface (\S+)", config, re.M)
+ if not match:
+ return list()
+
+ instances = list()
+
+ for item in set(match):
+ command = {
+ "command": "show interfaces {0} switchport | include Switchport".format(
+ item
+ ),
+ "output": "text",
+ }
+ command_result = run_commands(module, command, check_rc=False)
+ if "Interface does not exist" in command_result[0]:
+ warnings.append(
+ "Could not gather switchport information for {0}: {1}".format(
+ item, command_result[0]
+ )
+ )
+ continue
+
+ if command_result[0]:
+ switchport_cfg = command_result[0].split(":")[1].strip()
+
+ if switchport_cfg == "Enabled":
+ state = "present"
+ else:
+ state = "absent"
+
+ obj = {
+ "name": item.lower(),
+ "state": state,
+ "access_vlan": parse_config_argument(
+ configobj, item, "switchport access vlan"
+ ),
+ "native_vlan": parse_config_argument(
+ configobj, item, "switchport trunk native vlan"
+ ),
+ "trunk_allowed_vlans": parse_config_argument(
+ configobj, item, "switchport trunk allowed vlan"
+ ),
+ }
+ if obj["access_vlan"]:
+ obj["mode"] = "access"
+ else:
+ obj["mode"] = "trunk"
+
+ instances.append(obj)
+
+ return instances
+
+
+def map_params_to_obj(module):
+ obj = []
+
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ item["name"] = item["name"].lower()
+ obj.append(item.copy())
+ else:
+ obj.append(
+ {
+ "name": module.params["name"].lower(),
+ "mode": module.params["mode"],
+ "access_vlan": module.params["access_vlan"],
+ "native_vlan": module.params["native_vlan"],
+ "trunk_allowed_vlans": module.params["trunk_allowed_vlans"],
+ "state": module.params["state"],
+ }
+ )
+
+ return obj
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ name=dict(type="str", aliases=["interface"]),
+ mode=dict(choices=["access", "trunk"]),
+ access_vlan=dict(type="str"),
+ native_vlan=dict(type="str"),
+ trunk_allowed_vlans=dict(type="str", aliases=["trunk_vlans"]),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["name"] = dict(required=True)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec)
+ )
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ mutually_exclusive=[
+ ["access_vlan", "native_vlan"],
+ ["access_vlan", "trunk_allowed_vlans"],
+ ],
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+ result = {"changed": False, "warnings": warnings}
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module, warnings)
+ commands = map_obj_to_commands(want, have, module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py
new file mode 100644
index 00000000..0250dd16
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py
@@ -0,0 +1,428 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+##############################################
+# WARNING #
+##############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+##############################################
+
+"""
+The module file for eos_l2_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_l2_interfaces
+short_description: L2 interfaces resource module
+description: This module provides declarative management of Layer-2 interface on Arista
+ EOS devices.
+version_added: 1.0.0
+author: Nathaniel Case (@qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A dictionary of Layer-2 interface options
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Full name of interface, e.g. Ethernet1.
+ type: str
+ required: true
+ access:
+ description:
+ - Switchport mode access command to configure the interface as a layer 2 access.
+ type: dict
+ suboptions:
+ vlan:
+ description:
+ - Configure given VLAN in access port. It's used as the access VLAN ID.
+ type: int
+ trunk:
+ description:
+ - Switchport mode trunk command to configure the interface as a Layer 2 trunk.
+ type: dict
+ suboptions:
+ native_vlan:
+ description:
+ - Native VLAN to be configured in trunk port. It is used as the trunk
+ native VLAN ID.
+ type: int
+ trunk_allowed_vlans:
+ description:
+ - List of allowed VLANs in a given trunk port. These are the only VLANs
+ that will be configured on the trunk.
+ type: list
+ elements: str
+ mode:
+ description:
+ - Mode in which interface needs to be configured.
+ - Access mode is not shown in interface facts, so idempotency will not be
+ maintained for switchport mode access and every time the output will come
+ as changed=True.
+ type: str
+ choices:
+ - access
+ - trunk
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^interface).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - parsed
+ - rendered
+ - gathered
+ default: merged
+ description:
+ - The state of the configuration after module completion
+ type: str
+
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport access vlan 20
+# !
+# interface Ethernet2
+# switchport trunk native vlan 20
+# switchport mode trunk
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+- name: Merge provided configuration with device configuration.
+ arista.eos.eos_l2_interfaces:
+ config:
+ - name: Ethernet1
+ mode: trunk
+ trunk:
+ native_vlan: 10
+ - name: Ethernet2
+ mode: access
+ access:
+ vlan: 30
+ state: merged
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport trunk native vlan 10
+# switchport mode trunk
+# !
+# interface Ethernet2
+# switchport access vlan 30
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+# Using replaced
+
+# Before state:
+# -------------
+#
+# veos2#show running-config | s int
+# interface Ethernet1
+# switchport access vlan 20
+# !
+# interface Ethernet2
+# switchport trunk native vlan 20
+# switchport mode trunk
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+- name: Replace device configuration of specified L2 interfaces with provided configuration.
+ arista.eos.eos_l2_interfaces:
+ config:
+ - name: Ethernet1
+ mode: trunk
+ trunk:
+ native_vlan: 20
+ trunk_vlans: 5-10, 15
+ state: replaced
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport trunk native vlan 20
+# switchport trunk allowed vlan 5-10,15
+# switchport mode trunk
+# !
+# interface Ethernet2
+# switchport trunk native vlan 20
+# switchport mode trunk
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+# Using overridden
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport access vlan 20
+# !
+# interface Ethernet2
+# switchport trunk native vlan 20
+# switchport mode trunk
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+- name: Override device configuration of all L2 interfaces on device with provided
+ configuration.
+ arista.eos.eos_l2_interfaces:
+ config:
+ - name: Ethernet2
+ mode: access
+ access:
+ vlan: 30
+ state: overridden
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# switchport access vlan 30
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+# Using deleted
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport access vlan 20
+# !
+# interface Ethernet2
+# switchport trunk native vlan 20
+# switchport mode trunk
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+# !
+
+- name: Delete EOS L2 interfaces as in given arguments.
+ arista.eos.eos_l2_interfaces:
+ config:
+ - name: Ethernet1
+ - name: Ethernet2
+ state: deleted
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+# using rendered
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_l2_interfaces:
+ config:
+ - name: Ethernet1
+ mode: trunk
+ trunk:
+ native_vlan: 10
+ - name: Ethernet2
+ mode: access
+ access:
+ vlan: 30
+ state: merged
+
+# Output :
+# ------------
+#
+# - "interface Ethernet1"
+# - "switchport trunk native vlan 10"
+# - "switchport mode trunk"
+# - "interface Ethernet2"
+# - "switchport access vlan 30"
+# - "interface Management1"
+# - "ip address dhcp"
+# - "ipv6 address auto-config"
+
+
+# using parsed
+
+# parsed.cfg
+
+# interface Ethernet1
+# switchport trunk native vlan 10
+# switchport mode trunk
+# !
+# interface Ethernet2
+# switchport access vlan 30
+# !
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.l2_interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+# parsed:
+# - name: Ethernet1
+# mode: trunk
+# trunk:
+# native_vlan: 10
+# - name: Ethernet2
+# mode: access
+# access:
+# vlan: 30
+
+
+# Using gathered:
+# Existing config on the device:
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# switchport trunk native vlan 10
+# switchport mode trunk
+# !
+# interface Ethernet2
+# switchport access vlan 30
+# !
+
+- name: Gather interfaces facts from the device
+ arista.eos.l2_interfaces:
+ state: gathered
+# output:
+# gathered:
+# - name: Ethernet1
+# mode: trunk
+# trunk:
+# native_vlan: 10
+# - name: Ethernet2
+# mode: access
+# access:
+# vlan: 30
+
+"""
+
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: The configuration returned will always be in the same format of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: The configuration returned will always be in the same format of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['interface Ethernet2', 'switchport access vlan 20']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l2_interfaces.l2_interfaces import (
+ L2_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.l2_interfaces.l2_interfaces import (
+ L2_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+
+ module = AnsibleModule(
+ argument_spec=L2_interfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = L2_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py
new file mode 100644
index 00000000..4fb9c249
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py
@@ -0,0 +1,374 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# Copyright: (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_l3_interface
+author: Ganesh Nalawade (@ganeshrn)
+short_description: (deprecated, removed after 2022-06-01) Manage L3 interfaces on
+ Arista EOS network devices.
+description:
+- This module provides declarative management of L3 interfaces on Arista EOS network
+ devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_l3_interfaces
+ why: Updated modules released with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ name:
+ description:
+ - Name of the L3 interface to be configured eg. ethernet1
+ type: str
+ ipv4:
+ description:
+ - IPv4 address to be set for the L3 interface mentioned in I(name) option. The
+ address format is <ipv4 address>/<mask>, the mask is number in range 0-32 eg.
+ 192.168.0.1/24
+ type: str
+ ipv6:
+ description:
+ - IPv6 address to be set for the L3 interface mentioned in I(name) option. The
+ address format is <ipv6 address>/<mask>, the mask is number in range 0-128 eg.
+ fd5d:12c9:2201:1::1/64
+ type: str
+ aggregate:
+ description:
+ - List of L3 interfaces definitions. Each of the entry in aggregate list should
+ define name of interface C(name) and a optional C(ipv4) or C(ipv6) address.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the L3 interface to be configured eg. ethernet1
+ type: str
+ required: True
+ ipv4:
+ description:
+ - IPv4 address to be set for the L3 interface mentioned in I(name) option. The
+ address format is <ipv4 address>/<mask>, the mask is number in range 0-32 eg.
+ 192.168.0.1/24
+ type: str
+ ipv6:
+ description:
+ - IPv6 address to be set for the L3 interface mentioned in I(name) option. The
+ address format is <ipv6 address>/<mask>, the mask is number in range 0-128 eg.
+ fd5d:12c9:2201:1::1/64
+ type: str
+ state:
+ description:
+ - State of the L3 interface configuration. It indicates if the configuration should
+ be present or absent on remote device.
+ type: str
+ choices:
+ - present
+ - absent
+ state:
+ description:
+ - State of the L3 interface configuration. It indicates if the configuration should
+ be present or absent on remote device.
+ type: str
+ default: present
+ choices:
+ - present
+ - absent
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: Remove ethernet1 IPv4 and IPv6 address
+ arista.eos.eos_l3_interface:
+ name: ethernet1
+ state: absent
+
+- name: Set ethernet1 IPv4 address
+ arista.eos.eos_l3_interface:
+ name: ethernet1
+ ipv4: 192.168.0.1/24
+
+- name: Set ethernet1 IPv6 address
+ arista.eos.eos_l3_interface:
+ name: ethernet1
+ ipv6: fd5d:12c9:2201:1::1/64
+
+- name: Set interface Vlan1 (SVI) IPv4 address
+ arista.eos.eos_l3_interface:
+ name: Vlan1
+ ipv4: 192.168.0.5/24
+
+- name: Set IP addresses on aggregate
+ arista.eos.eos_l3_interface:
+ aggregate:
+ - name: ethernet1
+ ipv4: 192.168.2.10/24
+ - name: ethernet1
+ ipv4: 192.168.3.10/24
+ ipv6: fd5d:12c9:2201:1::1/64
+
+- name: Remove IP addresses on aggregate
+ arista.eos.eos_l3_interface:
+ aggregate:
+ - name: ethernet1
+ ipv4: 192.168.2.10/24
+ - name: ethernet1
+ ipv4: 192.168.3.10/24
+ ipv6: fd5d:12c9:2201:1::1/64
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always, except for the platforms that use Netconf transport to manage the device.
+ type: list
+ sample:
+ - interface ethernet1
+ - ip address 192.168.0.1/24
+ - ipv6 address fd5d:12c9:2201:1::1/64
+"""
+import re
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import (
+ NetworkConfig,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ is_masklen,
+)
+
+
+def validate_ipv4(value, module):
+ if value:
+ address = value.split("/")
+ if len(address) != 2:
+ module.fail_json(
+ msg="address format is <ipv4 address>/<mask>, got invalid format %s"
+ % value
+ )
+
+ if not is_masklen(address[1]):
+ module.fail_json(
+ msg="invalid value for mask: %s, mask should be in range 0-32"
+ % address[1]
+ )
+
+
+def validate_ipv6(value, module):
+ if value:
+ address = value.split("/")
+ if len(address) != 2:
+ module.fail_json(
+ msg="address format is <ipv6 address>/<mask>, got invalid format %s"
+ % value
+ )
+ else:
+ if not 0 <= int(address[1]) <= 128:
+ module.fail_json(
+ msg="invalid value for mask: %s, mask should be in range 0-128"
+ % address[1]
+ )
+
+
+def validate_param_values(module, obj, param=None):
+ if param is None:
+ param = module.params
+ for key in obj:
+ # validate the param value (if validator func exists)
+ validator = globals().get("validate_%s" % key)
+ if callable(validator):
+ validator(param.get(key), module)
+
+
+def parse_config_argument(configobj, name, arg=None):
+ cfg = configobj["interface %s" % name]
+ cfg = "\n".join(cfg.children)
+ match = re.search(r"%s (.+)$" % arg, cfg, re.M)
+ if match:
+ return match.group(1).strip()
+
+
+def search_obj_in_list(name, lst):
+ for o in lst:
+ if o["name"] == name:
+ return o
+
+ return None
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+
+ for w in want:
+ name = w["name"]
+ ipv4 = w["ipv4"]
+ ipv6 = w["ipv6"]
+ state = w["state"]
+
+ interface = "interface " + name
+ commands.append(interface)
+
+ obj_in_have = search_obj_in_list(name, have)
+ if state == "absent" and obj_in_have:
+ if obj_in_have["ipv4"]:
+ if ipv4:
+ commands.append("no ip address {0}".format(ipv4))
+ else:
+ commands.append("no ip address")
+ if obj_in_have["ipv6"]:
+ if ipv6:
+ commands.append("no ipv6 address {0}".format(ipv6))
+ else:
+ commands.append("no ipv6 address")
+
+ elif state == "present":
+ if ipv4:
+ if (
+ obj_in_have is None
+ or obj_in_have["ipv4"] is None
+ or ipv4 != obj_in_have["ipv4"]
+ ):
+ commands.append("ip address {0}".format(ipv4))
+
+ if ipv6:
+ if (
+ obj_in_have is None
+ or obj_in_have["ipv6"] is None
+ or ipv6.lower() != obj_in_have["ipv6"].lower()
+ ):
+ commands.append("ipv6 address {0}".format(ipv6))
+
+ if commands[-1] == interface:
+ commands.pop(-1)
+
+ return commands
+
+
+def map_config_to_obj(module):
+ config = get_config(module, flags=["| section interface"])
+ configobj = NetworkConfig(indent=3, contents=config)
+
+ match = re.findall(r"^interface (\S+)", config, re.M)
+ if not match:
+ return list()
+
+ instances = list()
+
+ for item in set(match):
+ obj = {
+ "name": item.lower(),
+ "ipv4": parse_config_argument(configobj, item, "ip address"),
+ "ipv6": parse_config_argument(configobj, item, "ipv6 address"),
+ "state": "present",
+ }
+ instances.append(obj)
+
+ return instances
+
+
+def map_params_to_obj(module):
+ obj = []
+
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ item["name"] = item["name"].lower()
+ validate_param_values(module, item, item)
+ obj.append(item.copy())
+ else:
+ obj.append(
+ {
+ "name": module.params["name"].lower(),
+ "ipv4": module.params["ipv4"],
+ "ipv6": module.params["ipv6"],
+ "state": module.params["state"],
+ }
+ )
+
+ validate_param_values(module, obj)
+
+ return obj
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ name=dict(),
+ ipv4=dict(),
+ ipv6=dict(),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["name"] = dict(required=True)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec)
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["name", "aggregate"]]
+ mutually_exclusive = [["name", "aggregate"]]
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_one_of=required_one_of,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py
new file mode 100644
index 00000000..38ab683b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py
@@ -0,0 +1,406 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_l3_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_l3_interfaces
+short_description: L3 interfaces resource module
+description: This module provides declarative management of Layer 3 interfaces on
+ Arista EOS devices.
+version_added: 1.0.0
+author: Nathaniel Case (@qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+ 'eos_l2_interfaces/eos_interfaces' should be used for preparing the interfaces , before applying L3 configurations using
+ this module (eos_l3_interfaces).
+options:
+ config:
+ description: A dictionary of Layer 3 interface options
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Full name of the interface, i.e. Ethernet1.
+ type: str
+ required: true
+ ipv4:
+ description:
+ - List of IPv4 addresses to be set for the Layer 3 interface mentioned in
+ I(name) option.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description:
+ - IPv4 address to be set in the format <ipv4 address>/<mask> eg. 192.0.2.1/24,
+ or C(dhcp) to query DHCP for an IP address.
+ type: str
+ secondary:
+ description:
+ - Whether or not this address is a secondary address.
+ type: bool
+ ipv6:
+ description:
+ - List of IPv6 addresses to be set for the Layer 3 interface mentioned in
+ I(name) option.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description:
+ - IPv6 address to be set in the address format is <ipv6 address>/<mask>
+ eg. 2001:db8:2201:1::1/64 or C(auto-config) to use SLAAC to chose an
+ address.
+ type: str
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^interface).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - parsed
+ - gathered
+ - rendered
+ default: merged
+
+"""
+
+EXAMPLES = """
+
+# Using deleted
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 192.0.2.12/24
+# !
+# interface Ethernet2
+# ipv6 address 2001:db8::1/64
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+- name: Delete L3 attributes of given interfaces.
+ arista.eos.eos_l3_interfaces:
+ config:
+ - name: Ethernet1
+ - name: Ethernet2
+ state: deleted
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+
+# Using merged
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 192.0.2.12/24
+# !
+# interface Ethernet2
+# ipv6 address 2001:db8::1/64
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+- name: Merge provided configuration with device configuration.
+ arista.eos.eos_l3_interfaces:
+ config:
+ - name: Ethernet1
+ ipv4:
+ - address: 198.51.100.14/24
+ - name: Ethernet2
+ ipv4:
+ - address: 203.0.113.27/24
+ state: merged
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 198.51.100.14/24
+# !
+# interface Ethernet2
+# ip address 203.0.113.27/24
+# ipv6 address 2001:db8::1/64
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+
+# Using overridden
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 192.0.2.12/24
+# !
+# interface Ethernet2
+# ipv6 address 2001:db8::1/64
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+- name: Override device configuration of all L2 interfaces on device with provided
+ configuration.
+ arista.eos.eos_l3_interfaces:
+ config:
+ - name: Ethernet1
+ ipv6:
+ - address: 2001:db8:feed::1/96
+ - name: Management1
+ ipv4:
+ - address: dhcp
+ ipv6: auto-config
+ state: overridden
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ipv6 address 2001:db8:feed::1/96
+# !
+# interface Ethernet2
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+
+# Using replaced
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 192.0.2.12/24
+# !
+# interface Ethernet2
+# ipv6 address 2001:db8::1/64
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+- name: Replace device configuration of specified L2 interfaces with provided configuration.
+ arista.eos.eos_l3_interfaces:
+ config:
+ - name: Ethernet2
+ ipv4:
+ - address: 203.0.113.27/24
+ state: replaced
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 192.0.2.12/24
+# !
+# interface Ethernet2
+# ip address 203.0.113.27/24
+# !
+# interface Management1
+# ip address dhcp
+# ipv6 address auto-config
+
+# Using parsed:
+
+# parsed.cfg
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 198.51.100.14/24
+# !
+# interface Ethernet2
+# ip address 203.0.113.27/24
+# !
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+
+# parsed:
+# - name: Ethernet1
+# ipv4:
+# - address: 198.51.100.14/24
+# - name: Ethernet2
+# ipv4:
+# - address: 203.0.113.27/24
+
+# Using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_l3_interfaces:
+ config:
+ - name: Ethernet1
+ ipv4:
+ - address: 198.51.100.14/24
+ - name: Ethernet2
+ ipv4:
+ - address: 203.0.113.27/24
+ state: rendered
+
+# Output
+# ------------
+#rendered:
+# - "interface Ethernet1"
+# - "ip address 198.51.100.14/24"
+# - "interface Ethernet2"
+# - "ip address 203.0.113.27/24"
+
+# using gathered:
+
+# Native COnfig:
+# veos#show running-config | section interface
+# interface Ethernet1
+# ip address 198.51.100.14/24
+# !
+# interface Ethernet2
+# ip address 203.0.113.27/24
+# !
+
+- name: Gather l3 interfaces facts from the device
+ arista.eos.l3_interfaces:
+ state: gathered
+
+# gathered:
+# - name: Ethernet1
+# ipv4:
+# - address: 198.51.100.14/24
+# - name: Ethernet2
+# ipv4:
+# - address: 203.0.113.27/24
+
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['interface Ethernet2', 'ip address 192.0.2.12/24']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l3_interfaces.l3_interfaces import (
+ L3_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.l3_interfaces.l3_interfaces import (
+ L3_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=L3_interfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = L3_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py
new file mode 100644
index 00000000..33102756
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py
@@ -0,0 +1,245 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_lacp
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lacp
+short_description: LACP resource module
+description:
+- This module manages Global Link Aggregation Control Protocol (LACP) on Arista EOS
+ devices.
+version_added: 1.0.0
+author: Nathaniel Case (@Qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: LACP global options.
+ type: dict
+ suboptions:
+ system:
+ description: LACP system options.
+ type: dict
+ suboptions:
+ priority:
+ description:
+ - The system priority to use in LACP negotiations.
+ - Lower value is higher priority.
+ - Refer to vendor documentation for valid values.
+ type: int
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^lacp).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - deleted
+ - parsed
+ - rendered
+ - gathered
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+
+# Before state:
+# -------------
+# veos# show running-config | include lacp
+# lacp system-priority 10
+
+- name: Merge provided global LACP attributes with device attributes
+ arista.eos.eos_lacp:
+ config:
+ system:
+ priority: 20
+ state: merged
+
+# After state:
+# ------------
+# veos# show running-config | include lacp
+# lacp system-priority 20
+#
+
+
+# Using replaced
+
+# Before state:
+# -------------
+# veos# show running-config | include lacp
+# lacp system-priority 10
+
+- name: Replace device global LACP attributes with provided attributes
+ arista.eos.eos_lacp:
+ config:
+ system:
+ priority: 20
+ state: replaced
+
+# After state:
+# ------------
+# veos# show running-config | include lacp
+# lacp system-priority 20
+#
+
+
+# Using deleted
+
+# Before state:
+# -------------
+# veos# show running-config | include lacp
+# lacp system-priority 10
+
+- name: Delete global LACP attributes
+ arista.eos.eos_lacp:
+ state: deleted
+
+# After state:
+# ------------
+# veos# show running-config | include lacp
+#
+
+#Using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_lacp:
+ config:
+ system:
+ priority: 20
+ state: rendered
+
+# Output:
+# ------------
+# rendered:
+# - "lacp system-priority 20"
+#
+
+# Using parsed:
+
+# parsed.cfg
+# lacp system-priority 20
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.eos_lacp:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+# parsed:
+# system:
+# priority: 20
+
+# Using gathered:
+# nathive config:
+# -------------
+# lacp system-priority 10
+
+- name: Gather lacp facts from the device
+ arista.eos.eos_lacp:
+ state: gathered
+
+# Output:
+# gathered:
+# system:
+# priority: 10
+#
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: dict
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: dict
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['lacp system-priority 10']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp.lacp import (
+ LacpArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lacp.lacp import (
+ Lacp,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=LacpArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Lacp(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py
new file mode 100644
index 00000000..e841f2c9
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py
@@ -0,0 +1,336 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_lacp_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lacp_interfaces
+short_description: LACP interfaces resource module
+description:
+- This module manages Link Aggregation Control Protocol (LACP) attributes of interfaces
+ on Arista EOS devices.
+version_added: 1.0.0
+author: Nathaniel Case (@Qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A dictionary of LACP interfaces options.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Full name of the interface (i.e. Ethernet1).
+ type: str
+ port_priority:
+ description:
+ - LACP port priority for the interface. Range 1-65535.
+ type: int
+ rate:
+ description:
+ - Rate at which PDUs are sent by LACP. At fast rate LACP is transmitted once
+ every 1 second. At normal rate LACP is transmitted every 30 seconds after
+ the link is bundled.
+ type: str
+ choices:
+ - fast
+ - normal
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^interfaces).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - parsed
+ - rendered
+ - gathered
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# interface Ethernet2
+# lacp rate fast
+
+- name: Merge provided configuration with device configuration
+ arista.eos.eos_lacp_interfaces:
+ config:
+ - name: Ethernet1
+ rate: fast
+ - name: Ethernet2
+ rate: normal
+ state: merged
+
+#
+# -----------
+# After state
+# -----------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# lacp rate fast
+# interface Ethernet2
+
+
+# Using replaced
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# interface Ethernet2
+# lacp rate fast
+
+- name: Replace existing LACP configuration of specified interfaces with provided
+ configuration
+ arista.eos.eos_lacp_interfaces:
+ config:
+ - name: Ethernet1
+ rate: fast
+ state: replaced
+
+#
+# -----------
+# After state
+# -----------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp rate fast
+# interface Ethernet2
+# lacp rate fast
+
+
+# Using overridden
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# interface Ethernet2
+# lacp rate fast
+
+- name: Override the LACP configuration of all the interfaces with provided configuration
+ arista.eos.eos_lacp_interfaces:
+ config:
+ - name: Ethernet1
+ rate: fast
+ state: overridden
+
+#
+# -----------
+# After state
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp rate fast
+# interface Ethernet2
+
+
+# Using deleted
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# interface Ethernet2
+# lacp rate fast
+
+- name: Delete LACP attributes of given interfaces (or all interfaces if none specified).
+ arista.eos.eos_lacp_interfaces:
+ state: deleted
+
+#
+# -----------
+# After state
+# -----------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# interface Ethernet2
+
+# using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_lacp_interfaces:
+ config:
+ - name: Ethernet1
+ rate: fast
+ - name: Ethernet2
+ rate: normal
+ state: rendered
+
+#
+# -----------
+# Output
+# -----------
+# rendered:
+# - "interface Ethernet1"
+# - "lacp rate fast"
+
+# Using parsed:
+
+# parsed.cfg:
+# "interface Ethernet1"
+# "lacp rate fast"
+# "interface Ethernet2"
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.eos_lacp_interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+# parsed:
+# - name: Ethernet1
+# rate: fast
+# - name: Ethernet2
+# rate: normal
+
+# Using gathered:
+# native config:
+# veos#show run | section ^interface
+# interface Ethernet1
+# lacp port-priority 30
+# interface Ethernet2
+# lacp rate fast
+
+- name: Gather LACP facts from the device
+ arista.eos.eos_lacp_interfaces:
+ state: gathered
+
+# Output:
+# gathered:
+# - name: Ethernet1
+# - name: Ethernet2
+# rate: fast
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['interface Ethernet1', 'lacp rate fast']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp_interfaces.lacp_interfaces import (
+ Lacp_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lacp_interfaces.lacp_interfaces import (
+ Lacp_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=Lacp_interfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Lacp_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py
new file mode 100644
index 00000000..0d3bf983
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py
@@ -0,0 +1,340 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_lag_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lag_interfaces
+short_description: LAG interfaces resource module
+description: This module manages attributes of link aggregation groups on Arista EOS
+ devices.
+version_added: 1.0.0
+author: Nathaniel Case (@Qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A list of link aggregation group configurations.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the port-channel interface of the link aggregation group (LAG) e.g.,
+ Port-Channel5.
+ type: str
+ required: true
+ members:
+ description:
+ - Ethernet interfaces that are part of the group.
+ type: list
+ elements: dict
+ suboptions:
+ member:
+ description:
+ - Name of ethernet interface that is a member of the LAG.
+ type: str
+ mode:
+ description:
+ - LAG mode for this interface.
+ type: str
+ choices:
+ - 'active'
+ - 'on'
+ - 'passive'
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section interfaces).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - rendered
+ - gathered
+ - parsed
+ default: merged
+
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+
+- name: Merge provided LAG attributes with existing device configuration
+ arista.eos.eos_lag_interfaces:
+ config:
+ - name: 5
+ members:
+ - member: Ethernet2
+ mode: on
+ state: merged
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+# channel-group 5 mode on
+
+
+# Using replaced
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+
+- name: Replace all device configuration of specified LAGs with provided configuration
+ arista.eos.eos_lag_interfaces:
+ config:
+ - name: 5
+ members:
+ - member: Ethernet2
+ mode: on
+ state: replaced
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# interface Ethernet2
+# channel-group 5 mode on
+
+
+# Using overridden
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+
+- name: Override all device configuration of all LAG attributes with provided configuration
+ arista.eos.eos_lag_interfaces:
+ config:
+ - name: 10
+ members:
+ - member: Ethernet2
+ mode: on
+ state: overridden
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# interface Ethernet2
+# channel-group 10 mode on
+
+
+# Using deleted
+
+# Before state:
+# -------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+# channel-group 5 mode on
+
+- name: Delete LAG attributes of the given interfaces.
+ arista.eos.eos_lag_interfaces:
+ config:
+ - name: 5
+ members:
+ - member: Ethernet1
+ state: deleted
+
+# After state:
+# ------------
+#
+# veos#show running-config | section interface
+# interface Ethernet1
+# interface Ethernet2
+# channel-group 5 mode on
+
+# Using parsed:
+
+# parsed.cfg
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+# channel-group 5 mode on
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.lag_interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+# parsed:
+# - name: 5
+# members:
+# - member: Ethernet2
+# mode: on
+# - member: Ethernet1
+# mode: on
+
+# using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_lag_interfaces:
+ config:
+ - name: 5
+ members:
+ - member: Ethernet2
+ mode: on
+ - member: Ethernet1
+ mode: on
+ state: rendered
+# -----------
+# Output
+# -----------
+#
+# rendered:
+
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+# channel-group 5 mode on
+
+
+# Using gathered:
+
+# native config:
+# interface Ethernet1
+# channel-group 5 mode on
+# interface Ethernet2
+# channel-group 5 mode on
+
+- name: Gather lldp_global facts from the device
+ arista.eos.lldp_global:
+ state: gathered
+
+# Output:
+# gathered:
+# - name: 5
+# members:
+# - member: Ethernet2
+# mode: on
+# - member: Ethernet1
+# mode: on
+
+"""
+
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['command 1', 'command 2', 'command 3']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lag_interfaces.lag_interfaces import (
+ Lag_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lag_interfaces.lag_interfaces import (
+ Lag_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=Lag_interfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Lag_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py
new file mode 100644
index 00000000..32d289a1
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py
@@ -0,0 +1,432 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# Copyright: (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_linkagg
+author: Trishna Guha (@trishnaguha)
+short_description: (deprecated, removed after 2022-06-01) Manage link aggregation
+ groups on Arista EOS network devices
+description:
+- This module provides declarative management of link aggregation groups on Arista
+ EOS network devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_lag_interfaces
+ why: Updated modules released with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ group:
+ description:
+ - Channel-group number for the port-channel Link aggregation group. Range 1-2000.
+ type: int
+ mode:
+ description:
+ - Mode of the link aggregation group.
+ type: str
+ choices:
+ - active
+ - "on"
+ - passive
+ members:
+ description:
+ - List of members of the link aggregation group.
+ type: list
+ elements: str
+ min_links:
+ description:
+ - Minimum number of ports required up before bringing up the link aggregation
+ group.
+ type: int
+ aggregate:
+ description: List of link aggregation definitions.
+ type: list
+ elements: dict
+ suboptions:
+ group:
+ description:
+ - Channel-group number for the port-channel Link aggregation group. Range 1-2000.
+ type: int
+ required: True
+ mode:
+ description:
+ - Mode of the link aggregation group.
+ type: str
+ choices:
+ - active
+ - "on"
+ - passive
+ members:
+ description:
+ - List of members of the link aggregation group.
+ type: list
+ elements: str
+ min_links:
+ description:
+ - Minimum number of ports required up before bringing up the link aggregation
+ group.
+ type: int
+ state:
+ description:
+ - State of the link aggregation group.
+ type: str
+ choices:
+ - present
+ - absent
+ state:
+ description:
+ - State of the link aggregation group.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+ purge:
+ description:
+ - Purge links not defined in the I(aggregate) parameter.
+ default: false
+ type: bool
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: create link aggregation group
+ arista.eos.eos_linkagg:
+ group: 10
+ state: present
+
+- name: delete link aggregation group
+ arista.eos.eos_linkagg:
+ group: 10
+ state: absent
+
+- name: set link aggregation group to members
+ arista.eos.eos_linkagg:
+ group: 200
+ min_links: 3
+ mode: active
+ members:
+ - Ethernet0
+ - Ethernet1
+
+- name: remove link aggregation group from Ethernet0
+ arista.eos.eos_linkagg:
+ group: 200
+ min_links: 3
+ mode: active
+ members:
+ - Ethernet1
+
+- name: Create aggregate of linkagg definitions
+ arista.eos.eos_linkagg:
+ aggregate:
+ - {group: 3, mode: on, members: [Ethernet1]}
+ - {group: 100, mode: passive, min_links: 3, members: [Ethernet2]}
+
+- name: Remove aggregate of linkagg definitions
+ arista.eos.eos_linkagg:
+ aggregate:
+ - {group: 3, mode: on, members: [Ethernet1]}
+ - {group: 100, mode: passive, min_links: 3, members: [Ethernet2]}
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always, except for the platforms that use Netconf transport to manage the device.
+ type: list
+ sample:
+ - interface port-channel 30
+ - port-channel min-links 5
+ - interface Ethernet3
+ - channel-group 30 mode on
+ - no interface port-channel 30
+"""
+
+import re
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def search_obj_in_list(group, lst):
+ for o in lst:
+ if o["group"] == group:
+ return o
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+ purge = module.params["purge"]
+
+ for w in want:
+ group = w["group"]
+ mode = w["mode"]
+ min_links = w["min_links"]
+ members = w.get("members") or []
+ state = w["state"]
+ del w["state"]
+
+ obj_in_have = search_obj_in_list(group, have)
+
+ if state == "absent":
+ if obj_in_have:
+ commands.append("no interface port-channel {0}".format(group))
+
+ elif state == "present":
+ cmd = ["interface port-channel {0}".format(group), "end"]
+ if not obj_in_have:
+ if not group:
+ module.fail_json(msg="group is a required option")
+ commands.extend(cmd)
+
+ if min_links != "None":
+ commands.append(
+ "port-channel min-links {0}".format(min_links)
+ )
+
+ if members:
+ for m in members:
+ commands.append("interface {0}".format(m))
+ commands.append(
+ "channel-group {0} mode {1}".format(group, mode)
+ )
+
+ else:
+ if members:
+ if "members" not in obj_in_have.keys():
+ for m in members:
+ commands.extend(cmd)
+ commands.append("interface {0}".format(m))
+ commands.append(
+ "channel-group {0} mode {1}".format(
+ group, mode
+ )
+ )
+
+ elif set(members) != set(obj_in_have["members"]):
+ missing_members = list(
+ set(members) - set(obj_in_have["members"])
+ )
+ for m in missing_members:
+ commands.extend(cmd)
+ commands.append("interface {0}".format(m))
+ commands.append(
+ "channel-group {0} mode {1}".format(
+ group, mode
+ )
+ )
+
+ superfluous_members = list(
+ set(obj_in_have["members"]) - set(members)
+ )
+ for m in superfluous_members:
+ commands.extend(cmd)
+ commands.append("interface {0}".format(m))
+ commands.append(
+ "no channel-group {0}".format(group)
+ )
+
+ if purge:
+ for h in have:
+ obj_in_want = search_obj_in_list(h["group"], want)
+ if not obj_in_want:
+ commands.append(
+ "no interface port-channel {0}".format(h["group"])
+ )
+
+ return commands
+
+
+def map_params_to_obj(module, required_together=None):
+ obj = []
+
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ module._check_required_together(required_together, item)
+ d = item.copy()
+ d["group"] = str(d["group"])
+ d["min_links"] = str(d["min_links"])
+
+ obj.append(d)
+ else:
+ obj.append(
+ {
+ "group": str(module.params["group"]),
+ "mode": module.params["mode"],
+ "min_links": str(module.params["min_links"]),
+ "members": module.params["members"],
+ "state": module.params["state"],
+ }
+ )
+
+ return obj
+
+
+def parse_mode(group, member, config):
+ mode = None
+
+ for line in config.strip().split("!"):
+ match_int = re.findall(r"interface {0}\\b".format(member), line, re.M)
+ if match_int:
+ match = re.search(
+ r"channel-group {0} mode (\S+)".format(group), line, re.M
+ )
+ if match:
+ mode = match.group(1)
+
+ return mode
+
+
+def parse_members(group, config):
+ members = []
+
+ for line in config.strip().split("!"):
+ match_group = re.findall(
+ r"channel-group {0} mode".format(group), line, re.M
+ )
+ if match_group:
+ match = re.search(r"interface (\S+)", line, re.M)
+ if match:
+ members.append(match.group(1))
+
+ return members
+
+
+def get_channel(group, module):
+ channel = {}
+ config = get_config(module, flags=["| section channel-group"])
+
+ for line in config.split("\n"):
+ stripped = line.strip()
+ match = re.search(r"interface (\S+)", stripped, re.M)
+
+ if match:
+ member = match.group(1)
+ channel["mode"] = parse_mode(group, member, config)
+ channel["members"] = parse_members(group, config)
+
+ return channel
+
+
+def parse_min_links(group, config):
+ min_links = ""
+
+ for line in config.strip().split("!"):
+ match_pc = re.findall(
+ r"interface Port-Channel{0}\\b".format(group), line, re.M
+ )
+ if match_pc:
+ match = re.search(r"port-channel min-links (\S+)", line, re.M)
+ if match:
+ min_links = match.group(1)
+
+ return min_links
+
+
+def map_config_to_obj(module):
+ objs = list()
+ config = get_config(module, flags=["| section port-channel"])
+
+ for line in config.split("\n"):
+ stripped = line.strip()
+ match = re.search(r"interface Port-Channel(\S+)", stripped, re.M)
+ if match:
+ obj = {}
+ group = match.group(1)
+ obj["group"] = group
+ obj["min_links"] = parse_min_links(group, config)
+ obj.update(get_channel(group, module))
+ objs.append(obj)
+
+ return objs
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ group=dict(type="int"),
+ mode=dict(choices=["active", "on", "passive"]),
+ min_links=dict(type="int"),
+ members=dict(type="list", elements="str"),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["group"] = dict(required=True, type="int")
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec),
+ purge=dict(default=False, type="bool"),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["group", "aggregate"]]
+ required_together = [["members", "mode"]]
+ mutually_exclusive = [["group", "aggregate"]]
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_one_of=required_one_of,
+ required_together=required_together,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py
new file mode 100644
index 00000000..d2e306cf
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py
@@ -0,0 +1,122 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lldp
+author: Ganesh Nalawade (@ganeshrn)
+short_description: Manage LLDP configuration on Arista EOS network devices
+description:
+- This module provides declarative management of LLDP service on Arista EOS network
+ devices.
+version_added: 1.0.0
+notes:
+- Tested against EOS 4.15
+options:
+ state:
+ description:
+ - State of the LLDP configuration. If value is I(present) lldp will be enabled
+ else if it is I(absent) it will be disabled.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+ - enabled
+ - disabled
+extends_documentation_fragment:
+- arista.eos.eos
+"""
+
+EXAMPLES = """
+- name: Enable LLDP service
+ arista.eos.eos_lldp:
+ state: present
+
+- name: Disable LLDP service
+ arista.eos.eos_lldp:
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always, except for the platforms that use Netconf transport to manage the device.
+ type: list
+ sample:
+ - lldp run
+"""
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def has_lldp(module):
+ config = get_config(module, flags=["| section lldp"])
+
+ is_lldp_enable = False
+ if "no lldp run" not in config:
+ is_lldp_enable = True
+
+ return is_lldp_enable
+
+
+def main():
+ """ main entry point for module execution
+ """
+ argument_spec = dict(
+ state=dict(
+ default="present",
+ choices=["present", "absent", "enabled", "disabled"],
+ )
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ module = AnsibleModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+
+ if warnings:
+ result["warnings"] = warnings
+
+ HAS_LLDP = has_lldp(module)
+
+ commands = []
+
+ if module.params["state"] == "absent" and HAS_LLDP:
+ commands.append("no lldp run")
+ elif module.params["state"] == "present" and not HAS_LLDP:
+ commands.append("lldp run")
+
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py
new file mode 100644
index 00000000..4ad4bf4b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py
@@ -0,0 +1,345 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_lldp_global
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lldp_global
+short_description: LLDP resource module
+description:
+- This module manages Global Link Layer Discovery Protocol (LLDP) settings on Arista
+ EOS devices.
+version_added: 1.0.0
+author: Nathaniel Case (@Qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: The provided global LLDP configuration.
+ type: dict
+ suboptions:
+ holdtime:
+ description:
+ - Specifies the holdtime (in sec) to be sent in packets.
+ type: int
+ reinit:
+ description:
+ - Specifies the delay (in sec) for LLDP initialization on any interface.
+ type: int
+ timer:
+ description:
+ - Specifies the rate at which LLDP packets are sent (in sec).
+ type: int
+ tlv_select:
+ description:
+ - Specifies the LLDP TLVs to enable or disable.
+ type: dict
+ suboptions:
+ link_aggregation:
+ description:
+ - Enable or disable link aggregation TLV.
+ type: bool
+ management_address:
+ description:
+ - Enable or disable management address TLV.
+ type: bool
+ max_frame_size:
+ description:
+ - Enable or disable maximum frame size TLV.
+ type: bool
+ port_description:
+ description:
+ - Enable or disable port description TLV.
+ type: bool
+ system_capabilities:
+ description:
+ - Enable or disable system capabilities TLV.
+ type: bool
+ system_description:
+ description:
+ - Enable or disable system description TLV.
+ type: bool
+ system_name:
+ description:
+ - Enable or disable system name TLV.
+ type: bool
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section lldp).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - deleted
+ - rendered
+ - gathered
+ - parsed
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+#
+# ------------
+# Before State
+# ------------
+#
+# veos# show run | section lldp
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select system-description
+
+- name: Merge provided LLDP configuration with the existing configuration
+ arista.eos.eos_lldp_global:
+ config:
+ holdtime: 100
+ tlv_select:
+ management_address: false
+ port_description: false
+ system_description: true
+ state: merged
+
+# -----------
+# After state
+# -----------
+#
+# veos# show run | section lldp
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select port-description
+
+
+# Using replaced
+#
+# ------------
+# Before State
+# ------------
+#
+# veos# show run | section lldp
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select system-description
+
+- name: Replace existing LLDP device configuration with provided configuration
+ arista.eos.eos_lldp_global:
+ config:
+ holdtime: 100
+ tlv_select:
+ management_address: false
+ port_description: false
+ system_description: true
+ state: replaced
+
+# -----------
+# After state
+# -----------
+#
+# veos# show run | section lldp
+# lldp holdtime 100
+# no lldp tlv-select management-address
+# no lldp tlv-select port-description
+
+
+# Using deleted
+#
+# ------------
+# Before State
+# ------------
+#
+# veos# show run | section lldp
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select system-description
+
+- name: Delete existing LLDP configurations from the device
+ arista.eos.eos_lldp_global:
+ state: deleted
+
+# -----------
+# After state
+# -----------
+#
+# veos# show run | section ^lldp
+
+# Using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_lldp_global:
+ config:
+ holdtime: 100
+ tlv_select:
+ management_address: false
+ port_description: false
+ system_description: true
+ state: rendered
+
+# -----------
+# Output
+# -----------
+#
+# rendered:
+# - "lldp holdtime 100"
+# - "no lldp tlv-select management-address"
+# - "no lldp tlv-select port-description"
+
+# Using parsed
+
+# parsed.cfg
+
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select system-description
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.lldp_global:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# -----------
+# Output
+# -----------
+
+# parsed:
+# holdtime: 100
+# timer 3000
+# reinit 5
+# tlv_select:
+# management_address: False
+# port_description: False
+# system_description: True
+
+# Using gathered:
+# native config:
+# lldp timer 3000
+# lldp holdtime 100
+# lldp reinit 5
+# no lldp tlv-select management-address
+# no lldp tlv-select system-description
+
+
+- name: Gather lldp_global facts from the device
+ arista.eos.lldp_global:
+ state: gathered
+
+# -----------
+# Output
+# -----------
+
+# gathered:
+# holdtime: 100
+# timer 3000
+# reinit 5
+# tlv_select:
+# management_address: False
+# port_description: False
+# system_description: True
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: dict
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: dict
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['lldp holdtime 100', 'no lldp timer', 'lldp tlv-select system-description']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_global.lldp_global import (
+ Lldp_globalArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lldp_global.lldp_global import (
+ Lldp_global,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=Lldp_globalArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Lldp_global(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py
new file mode 100644
index 00000000..98a24913
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py
@@ -0,0 +1,344 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_lldp_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_lldp_interfaces
+short_description: LLDP interfaces resource module
+description:
+- This module manages Link Layer Discovery Protocol (LLDP) attributes of interfaces
+ on Arista EOS devices.
+version_added: 1.0.0
+author: Nathaniel Case (@Qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A dictionary of LLDP interfaces options.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Full name of the interface (i.e. Ethernet1).
+ type: str
+ receive:
+ description:
+ - Enable/disable LLDP RX on an interface.
+ type: bool
+ transmit:
+ description:
+ - Enable/disable LLDP TX on an interface.
+ type: bool
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ^interface).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - parsed
+ - gathered
+ - rendered
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp receive
+# interface Ethernet2
+# no lldp transmit
+
+- name: Merge provided configuration with running configuration
+ arista.eos.eos_lldp_interfaces:
+ config:
+ - name: Ethernet1
+ transmit: false
+ - name: Ethernet2
+ transmit: false
+ state: merged
+
+#
+# ------------
+# After state
+# ------------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp transmit
+# no lldp receive
+# interface Ethernet2
+# no lldp transmit
+
+
+# Using replaced
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp receive
+# interface Ethernet2
+# no lldp transmit
+
+- name: Replace existing LLDP configuration of specified interfaces with provided
+ configuration
+ arista.eos.eos_lldp_interfaces:
+ config:
+ - name: Ethernet1
+ transmit: false
+ state: replaced
+
+#
+# ------------
+# After state
+# ------------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp transmit
+# interface Ethernet2
+# no lldp transmit
+
+
+# Using overridden
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp receive
+# interface Ethernet2
+# no lldp transmit
+
+- name: Override the LLDP configuration of all the interfaces with provided configuration
+ arista.eos.eos_lldp_interfaces:
+ config:
+ - name: Ethernet1
+ transmit: false
+ state: overridden
+
+#
+# ------------
+# After state
+# ------------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp transmit
+# interface Ethernet2
+
+
+# Using deleted
+#
+#
+# ------------
+# Before state
+# ------------
+#
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# no lldp receive
+# interface Ethernet2
+# no lldp transmit
+
+- name: Delete LLDP configuration of specified interfaces (or all interfaces if none
+ are specified)
+ arista.eos.eos_lldp_interfaces:
+ state: deleted
+
+#
+# ------------
+# After state
+# ------------
+#
+# veos#show run | section ^interface
+# interface Ethernet1
+# interface Ethernet2
+
+# using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_lldp_interfaces:
+ config:
+ - name: Ethernet1
+ transmit: false
+ - name: Ethernet2
+ transmit: false
+ state: rendered
+
+#
+# ------------
+# Output
+# ------------
+#
+# interface Ethernet1
+# no lldp transmit
+# interface Ethernet2
+# no lldp transmit
+
+# Using parsed
+# parsed.cfg
+
+# interface Ethernet1
+# no lldp transmit
+# interface Ethernet2
+# no lldp transmit
+
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.lldp_interfaces:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# ------------
+# Output
+# ------------
+
+# parsed:
+# - name: Ethernet1
+# transmit: False
+# - name: Ethernet2
+# transmit: False
+
+# Using gathered:
+
+# native config:
+# interface Ethernet1
+# no lldp transmit
+# interface Ethernet2
+# no lldp transmit
+
+- name: Gather lldp interfaces facts from the device
+ arista.eos.lldp_interfaces:
+ state: gathered
+
+# ------------
+# Output
+# ------------
+
+# gathered:
+# - name: Ethernet1
+# transmit: False
+# - name: Ethernet2
+# transmit: False
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['interface Ethernet1', 'no lldp transmit']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_interfaces.lldp_interfaces import (
+ Lldp_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lldp_interfaces.lldp_interfaces import (
+ Lldp_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=Lldp_interfacesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Lldp_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py
new file mode 100644
index 00000000..a03cf7ef
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py
@@ -0,0 +1,505 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+# Copyright: (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+
+DOCUMENTATION = """
+module: eos_logging
+author: Trishna Guha (@trishnaguha)
+short_description: Manage logging on network devices
+description:
+- This module provides declarative management of logging on Arista Eos devices.
+version_added: 1.0.0
+notes:
+- Tested against EOS 4.15
+options:
+ dest:
+ description:
+ - Destination of the logs.
+ choices:
+ - "on"
+ - host
+ - console
+ - monitor
+ - buffered
+ type: str
+ name:
+ description:
+ - The hostname or IP address of the destination.
+ - Required when I(dest=host).
+ type: str
+ size:
+ description:
+ - Size of buffer. The acceptable value is in range from 10 to 2147483647 bytes.
+ type: int
+ facility:
+ description:
+ - Set logging facility.
+ type: str
+ level:
+ description:
+ - Set logging severity levels.
+ choices:
+ - emergencies
+ - alerts
+ - critical
+ - errors
+ - warnings
+ - notifications
+ - informational
+ - debugging
+ type: str
+ aggregate:
+ description: List of logging definitions.
+ type: list
+ elements: dict
+ suboptions:
+ dest:
+ description:
+ - Destination of the logs.
+ choices:
+ - "on"
+ - host
+ - console
+ - monitor
+ - buffered
+ type: str
+ name:
+ description:
+ - The hostname or IP address of the destination.
+ - Required when I(dest=host).
+ type: str
+ size:
+ description:
+ - Size of buffer. The acceptable value is in range from 10 to 2147483647 bytes.
+ type: int
+ facility:
+ description:
+ - Set logging facility.
+ type: str
+ level:
+ description:
+ - Set logging severity levels.
+ choices:
+ - emergencies
+ - alerts
+ - critical
+ - errors
+ - warnings
+ - notifications
+ - informational
+ - debugging
+ type: str
+ state:
+ description:
+ - State of the logging configuration.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+ state:
+ description:
+ - State of the logging configuration.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+extends_documentation_fragment:
+- arista.eos.eos
+"""
+
+EXAMPLES = """
+- name: configure host logging
+ arista.eos.eos_logging:
+ dest: host
+ name: 172.16.0.1
+ state: present
+
+- name: remove host logging configuration
+ arista.eos.eos_logging:
+ dest: host
+ name: 172.16.0.1
+ state: absent
+
+- name: configure console logging level and facility
+ arista.eos.eos_logging:
+ dest: console
+ facility: local7
+ level: debugging
+ state: present
+
+- name: enable logging to all
+ arista.eos.eos_logging:
+ dest: on
+
+- name: configure buffer size
+ arista.eos.eos_logging:
+ dest: buffered
+ size: 5000
+
+- name: Configure logging using aggregate
+ arista.eos.eos_logging:
+ aggregate:
+ - {dest: console, level: warnings}
+ - {dest: buffered, size: 480000}
+ state: present
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - logging facility local7
+ - logging host 172.16.0.1
+"""
+
+import re
+
+
+from copy import deepcopy
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+DEST_GROUP = ["on", "host", "console", "monitor", "buffered"]
+LEVEL_GROUP = [
+ "emergencies",
+ "alerts",
+ "critical",
+ "errors",
+ "warnings",
+ "notifications",
+ "informational",
+ "debugging",
+]
+
+
+def validate_size(value, module):
+ if value:
+ if not int(10) <= value <= int(2147483647):
+ module.fail_json(msg="size must be between 10 and 2147483647")
+ else:
+ return value
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+
+ for w in want:
+ dest = w["dest"]
+ name = w["name"]
+ size = w["size"]
+ facility = w["facility"]
+ level = w["level"]
+ state = w["state"]
+ del w["state"]
+
+ if state == "absent" and w in have:
+ if dest:
+ if dest == "host":
+ commands.append("no logging host {0}".format(name))
+
+ elif dest in DEST_GROUP:
+ commands.append("no logging {0}".format(dest))
+
+ else:
+ module.fail_json(
+ msg="dest must be among console, monitor, buffered, host, on"
+ )
+
+ if facility:
+ commands.append("no logging facility {0}".format(facility))
+
+ if state == "present" and w not in have:
+ if facility:
+ present = False
+
+ # Iterate over every dictionary in the 'have' list to check if
+ # similar configuration for facility exists or not
+
+ for entry in have:
+ if not entry["dest"] and entry["facility"] == facility:
+ present = True
+
+ if not present:
+ commands.append("logging facility {0}".format(facility))
+
+ if dest == "host":
+ commands.append("logging host {0}".format(name))
+
+ elif dest == "on":
+ commands.append("logging on")
+
+ elif dest == "buffered" and size:
+
+ present = False
+
+ # Deals with the following two cases:
+ # Case 1: logging buffered <size> <level>
+ # logging buffered <same-size>
+ #
+ # Case 2: Same buffered logging configuration
+ # already exists (i.e., both size &
+ # level are same)
+
+ for entry in have:
+ if entry["dest"] == "buffered" and entry["size"] == size:
+
+ if not level or entry["level"] == level:
+ present = True
+
+ if not present:
+ if size and level:
+ commands.append(
+ "logging buffered {0} {1}".format(size, level)
+ )
+ else:
+ commands.append("logging buffered {0}".format(size))
+
+ else:
+ if dest:
+ dest_cmd = "logging {0}".format(dest)
+ if level:
+ dest_cmd += " {0}".format(level)
+
+ commands.append(dest_cmd)
+ return commands
+
+
+def parse_facility(line):
+ facility = None
+ match = re.search(r"logging facility (\S+)", line, re.M)
+ if match:
+ facility = match.group(1)
+
+ return facility
+
+
+def parse_size(line, dest):
+ size = None
+
+ if dest == "buffered":
+ match = re.search(r"logging buffered (\S+)", line, re.M)
+ if match:
+ try:
+ int_size = int(match.group(1))
+ except ValueError:
+ int_size = None
+
+ if int_size:
+ if isinstance(int_size, int):
+ size = str(match.group(1))
+ else:
+ size = str(10)
+
+ return size
+
+
+def parse_name(line, dest):
+ name = None
+ if dest == "host":
+ match = re.search(r"logging host (\S+)", line, re.M)
+ if match:
+ name = match.group(1)
+
+ return name
+
+
+def parse_level(line, dest):
+ level = None
+
+ if dest != "host":
+
+ # Line for buffer logging entry in running-config is of the form:
+ # logging buffered <size> <level>
+
+ if dest == "buffered":
+ match = re.search(r"logging buffered (?:\d+) (\S+)", line, re.M)
+
+ else:
+ match = re.search(r"logging {0} (\S+)".format(dest), line, re.M)
+
+ if match:
+ if match.group(1) in LEVEL_GROUP:
+ level = match.group(1)
+
+ return level
+
+
+def map_config_to_obj(module):
+ obj = []
+
+ data = get_config(module, flags=["section logging"])
+
+ for line in data.split("\n"):
+
+ match = re.search(r"logging (\S+)", line, re.M)
+
+ if match:
+ if match.group(1) in DEST_GROUP:
+ dest = match.group(1)
+
+ else:
+ dest = None
+
+ obj.append(
+ {
+ "dest": dest,
+ "name": parse_name(line, dest),
+ "size": parse_size(line, dest),
+ "facility": parse_facility(line),
+ "level": parse_level(line, dest),
+ }
+ )
+
+ return obj
+
+
+def parse_obj(obj, module):
+ if module.params["size"] is None:
+ obj.append(
+ {
+ "dest": module.params["dest"],
+ "name": module.params["name"],
+ "size": module.params["size"],
+ "facility": module.params["facility"],
+ "level": module.params["level"],
+ "state": module.params["state"],
+ }
+ )
+
+ else:
+ obj.append(
+ {
+ "dest": module.params["dest"],
+ "name": module.params["name"],
+ "size": str(validate_size(module.params["size"], module)),
+ "facility": module.params["facility"],
+ "level": module.params["level"],
+ "state": module.params["state"],
+ }
+ )
+
+ return obj
+
+
+def map_params_to_obj(module, required_if=None):
+ obj = []
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ module._check_required_if(required_if, item)
+ d = item.copy()
+
+ if d["dest"] != "host":
+ d["name"] = None
+
+ if d["dest"] == "buffered":
+ if "size" in d:
+ d["size"] = str(validate_size(d["size"], module))
+ elif "size" not in d:
+ d["size"] = str(10)
+ else:
+ pass
+
+ if d["dest"] != "buffered":
+ d["size"] = None
+
+ obj.append(d)
+
+ else:
+ if module.params["dest"] != "host":
+ module.params["name"] = None
+
+ if module.params["dest"] == "buffered":
+ if not module.params["size"]:
+ module.params["size"] = str(10)
+ else:
+ module.params["size"] = None
+
+ parse_obj(obj, module)
+
+ return obj
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ dest=dict(choices=DEST_GROUP),
+ name=dict(),
+ size=dict(type="int"),
+ facility=dict(),
+ level=dict(choices=LEVEL_GROUP),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ aggregate_spec["state"].update(default="present")
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec)
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_if = [("dest", "host", ["name"])]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ have = map_config_to_obj(module)
+ want = map_params_to_obj(module, required_if=required_if)
+
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py
new file mode 100644
index 00000000..64226223
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py
@@ -0,0 +1,1228 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_ospf_interfaces
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+module: eos_ospf_interfaces
+version_added: 1.1.0
+short_description: OSPF Interfaces Resource Module.
+description:
+- This module manages OSPF configuration of interfaces on devices running Arista EOS.
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+options:
+ config:
+ description: A list of OSPF configuration for interfaces.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name/Identifier of the interface.
+ type: str
+ address_family:
+ description:
+ - OSPF settings on the interfaces in address-family context.
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description:
+ - Address Family Identifier (AFI) for OSPF settings on the interfaces.
+ type: str
+ choices: ['ipv4', 'ipv6']
+ required: True
+ area:
+ description:
+ - Area associated with interface.
+ - Valid only when afi = ipv4.
+ type: dict
+ suboptions:
+ area_id:
+ description:
+ - Area ID as a decimal or IP address format.
+ type: str
+ required: True
+ authentication_v2:
+ description:
+ - Authentication settings on the interface.
+ - Valid only when afi = ipv4.
+ type: dict
+ suboptions:
+ message_digest:
+ description:
+ - Use message-digest authentication.
+ type: bool
+ set:
+ description:
+ - Enable authentication on the interface.
+ type: bool
+ authentication_v3:
+ description:
+ - Authentication settings on the interface.
+ - Valid only when afi = ipv6.
+ type: dict
+ suboptions:
+ spi:
+ description: IPsec Security Parameter Index.
+ type: int
+ algorithm:
+ description: Encryption alsgorithm.
+ type: str
+ choices: ["md5", "sha1"]
+ keytype:
+ description:
+ - Specifies if an unencrypted/hidden follows.
+ - 0 denotes unencrypted key.
+ - 7 denotes hidden key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ key:
+ description: 128 bit MD5 key or 140 bit SHA1 key.
+ type: str
+ authentication_key:
+ description:
+ - Configure the authentication key for the interface.
+ - Valid only when afi = ipv4.
+ type: dict
+ suboptions:
+ encryption:
+ description:
+ - 0 Specifies an UNENCRYPTED authentication key will follow.
+ - 7 Specifies a proprietry encryption type.`
+ type: str
+ key:
+ description:
+ - password (up to 8 chars).
+ type: str
+ bfd:
+ description: Enable BFD.
+ type: bool
+ cost:
+ description:
+ - metric associated with interface.
+ type: int
+ dead_interval:
+ description:
+ - Time interval to detect a dead router.
+ type: int
+ encryption_v3:
+ description:
+ - Authentication settings on the interface.
+ - Valid only when afi = ipv6.
+ type: dict
+ suboptions:
+ spi:
+ description: IPsec Security Parameter Index.
+ type: int
+ encryption:
+ description: encryption type.
+ choices: ["3des-cbc", "aes-128-cbc", "aes-192-cbc", "aes-256-cbc", "null"]
+ type: str
+ algorithm:
+ description: algorithm.
+ type: str
+ choices: ["md5", "sha1"]
+ keytype:
+ description:
+ - Specifies if an unencrypted/hidden follows.
+ - 0 denotes unencrypted key.
+ - 7 denotes hidden key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ key:
+ description: key
+ type: str
+ hello_interval:
+ description:
+ - Timer interval between transmission of hello packets.
+ type: int
+ ip_params:
+ description:
+ - Specify parameters for IPv4/IPv6.
+ - Valid only when afi = ipv6.
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description:
+ - Address Family Identifier (AFI) for OSPF settings on the interfaces.
+ type: str
+ choices: ['ipv4', 'ipv6']
+ required: True
+ area:
+ description:
+ - Area associated with interface.
+ - Valid only when afi = ipv4.
+ type: dict
+ suboptions:
+ area_id:
+ description:
+ - Area ID as a decimal or IP address format.
+ type: str
+ required: True
+ bfd:
+ description: Enable BFD.
+ type: bool
+ cost:
+ description:
+ - metric associated with interface.
+ type: int
+ dead_interval:
+ description:
+ - Time interval to detect a dead router.
+ type: int
+ hello_interval:
+ description:
+ - Timer interval between transmission of hello packets.
+ type: int
+ mtu_ignore:
+ description:
+ - if True, Disable MTU check for Database Description packets.
+ type: bool
+ network:
+ description:
+ - Interface type.
+ type: str
+ priority:
+ description:
+ - Interface priority.
+ type: int
+ retransmit_interval:
+ description:
+ - LSA retransmission interval.
+ type: int
+ passive_interface:
+ description:
+ - Suppress routing updates in an interface.
+ type: bool
+ transmit_delay:
+ description:
+ - LSA transmission delay.
+ type: int
+ message_digest_key:
+ description:
+ - Message digest authentication password (key) settings.
+ type: dict
+ suboptions:
+ key_id:
+ description:
+ - Key ID.
+ type: int
+ encryption:
+ description:
+ - 0 Specifies an UNENCRYPTED ospf password (key) will follow.
+ - 7 Specifies a proprietry encryption type.
+ type: str
+ key:
+ description:
+ - Authentication key (upto 16 chars).
+ type: str
+ mtu_ignore:
+ description:
+ - if True, Disable MTU check for Database Description packets.
+ type: bool
+ network:
+ description:
+ - Interface type.
+ type: str
+ passive_interface:
+ description:
+ - Suppress routing updates in an interface.
+ - Valid only when afi = ipv6.
+ type: bool
+ priority:
+ description:
+ - Interface priority.
+ type: int
+ retransmit_interval:
+ description:
+ - LSA retransmission interval.
+ type: int
+ shutdown:
+ description:
+ - Shutdown OSPF on this interface.
+ type: bool
+ transmit_delay:
+ description:
+ - LSA transmission delay.
+ type: int
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section interface).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - gathered
+ - parsed
+ - rendered
+ default: merged
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state
+
+# veos(config)#show running-config | section interface | ospf
+# veos(config)#
+
+ - name: Merge provided configuration with device configuration
+ arista.eos.eos_ospf_interfaces:
+ config:
+ - name: "Vlan1"
+ address_family:
+ - afi: "ipv4"
+ area:
+ area_id: "0.0.0.50"
+ cost: 500
+ mtu_ignore: True
+ - afi: "ipv6"
+ dead_interval: 44
+ ip_params:
+ - afi: "ipv6"
+ mtu_ignore: True
+ network: "point-to-point"
+ state: merged
+
+# After State
+
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ip ospf cost 500
+# ip ospf mtu-ignore
+# ip ospf area 0.0.0.50
+# ospfv3 dead-interval 44
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# veos(config)#
+#
+#
+# Module Execution:
+#
+# "after": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.50"
+# },
+# "cost": 500,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "dead_interval": 44,
+# "ip_params": [
+# {
+# "afi": "ipv6",
+# "mtu_ignore": True,
+# "network": "point-to-point"
+# }
+# ]
+# }
+# ],
+# "name": "Vlan1"
+# }
+# ],
+# "before": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# }
+# ],
+# "changed": True,
+# "commands": [
+# "interface Vlan1",
+# "ip ospf area 0.0.0.50",
+# "ip ospf cost 500",
+# "ip ospf mtu-ignore",
+# "ospfv3 dead-interval 44",
+# "ospfv3 ipv6 mtu-ignore",
+# "ospfv3 ipv6 network point-to-point"
+# ],
+#
+
+# Using replaced
+#---------------
+
+# Before State:
+
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ip ospf cost 500
+# ip ospf dead-interval 29
+# ip ospf hello-interval 66
+# ip ospf mtu-ignore
+# ip ospf area 0.0.0.50
+# ospfv3 cost 106
+# ospfv3 hello-interval 77
+# ospfv3 dead-interval 44
+# ospfv3 transmit-delay 100
+# ospfv3 ipv4 priority 45
+# ospfv3 ipv4 area 0.0.0.5
+# ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 retransmit-interval 115
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+# veos(config)#
+
+
+ - name: Replace device configuration with provided configuration
+ arista.eos.eos_ospf_interfaces:
+ config:
+ - name: "Vlan1"
+ address_family:
+ - afi: "ipv6"
+ cost: 44
+ bfd: True
+ ip_params:
+ - afi: "ipv6"
+ mtu_ignore: True
+ network: "point-to-point"
+ dead_interval: 56
+ state: replaced
+
+# After State:
+
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ospfv3 bfd
+# ospfv3 cost 44
+# no ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+# veos(config)#
+#
+# Module Execution:
+#
+# "after": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "bfd": True,
+# "cost": 44,
+# "ip_params": [
+# {
+# "afi": "ipv6",
+# "mtu_ignore": True,
+# "network": "point-to-point"
+# }
+# ]
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+# "before": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.50"
+# },
+# "cost": 500,
+# "dead_interval": 29,
+# "hello_interval": 66,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "cost": 106,
+# "dead_interval": 44,
+# "hello_interval": 77,
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.5"
+# },
+# "priority": 45
+# },
+# {
+# "afi": "ipv6",
+# "mtu_ignore": True,
+# "network": "point-to-point",
+# "passive_interface": True,
+# "retransmit_interval": 115
+# }
+# ],
+# "transmit_delay": 100
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+# "changed": True,
+# "commands": [
+# "interface Vlan1",
+# "no ip ospf cost 500",
+# "no ip ospf dead-interval 29",
+# "no ip ospf hello-interval 66",
+# "no ip ospf mtu-ignore",
+# "no ip ospf area 0.0.0.50",
+# "ospfv3 cost 44",
+# "ospfv3 bfd",
+# "ospfv3 authentication ipsec spi 30 md5 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ==",
+# "no ospfv3 ipv4 priority 45",
+# "no ospfv3 ipv4 area 0.0.0.5",
+# "ospfv3 ipv6 dead-interval 56",
+# "no ospfv3 ipv6 passive-interface",
+# "no ospfv3 ipv6 retransmit-interval 115",
+# "no ospfv3 hello-interval 77",
+# "no ospfv3 dead-interval 44",
+# "no ospfv3 transmit-delay 100"
+# ],
+#
+
+# Using overidden:
+# ----------------
+
+# Before State:
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ip ospf dead-interval 29
+# ip ospf hello-interval 66
+# ip ospf mtu-ignore
+# ospfv3 bfd
+# ospfv3 cost 106
+# ospfv3 hello-interval 77
+# ospfv3 transmit-delay 100
+# ospfv3 ipv4 priority 45
+# ospfv3 ipv4 area 0.0.0.5
+# ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 dead-interval 56
+# ospfv3 ipv6 retransmit-interval 115
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+# veos(config)#
+
+ - name: Override device configuration with provided configuration
+ arista.eos.eos_ospf_interfaces:
+ config:
+ - name: "Vlan1"
+ address_family:
+ - afi: "ipv6"
+ cost: 44
+ bfd: True
+ ip_params:
+ - afi: "ipv6"
+ mtu_ignore: True
+ network: "point-to-point"
+ dead_interval: 56
+ state: overridden
+
+# After State:
+
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ospfv3 bfd
+# ospfv3 cost 44
+# no ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 dead-interval 56
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# veos(config)#
+#
+#
+# Module Execution:
+#
+# "after": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "bfd": True,
+# "cost": 44,
+# "ip_params": [
+# {
+# "afi": "ipv6",
+# "dead_interval": 56,
+# "mtu_ignore": True,
+# "network": "point-to-point"
+# }
+# ]
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "name": "Vlan2"
+# }
+# ],
+# "before": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "dead_interval": 29,
+# "hello_interval": 66,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "bfd": True,
+# "cost": 106,
+# "hello_interval": 77,
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.5"
+# },
+# "priority": 45
+# },
+# {
+# "afi": "ipv6",
+# "dead_interval": 56,
+# "mtu_ignore": True,
+# "network": "point-to-point",
+# "passive_interface": True,
+# "retransmit_interval": 115
+# }
+# ],
+# "transmit_delay": 100
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+# "changed": True,
+# "commands": [
+# "interface Vlan2",
+# "no ospfv3 ipv4 hello-interval 45",
+# "no ospfv3 ipv4 retransmit-interval 100",
+# "no ospfv3 ipv4 area 0.0.0.6",
+# "interface Vlan1",
+# "no ip ospf dead-interval 29",
+# "no ip ospf hello-interval 66",
+# "no ip ospf mtu-ignore",
+# "ospfv3 cost 44",
+# "ospfv3 authentication ipsec spi 30 md5 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ==",
+# "no ospfv3 ipv4 priority 45",
+# "no ospfv3 ipv4 area 0.0.0.5",
+# "no ospfv3 ipv6 passive-interface",
+# "no ospfv3 ipv6 retransmit-interval 115",
+# "no ospfv3 hello-interval 77",
+# "no ospfv3 transmit-delay 100"
+# ],
+#
+
+# Using deleted:
+#--------------
+
+# before State:
+
+# veos(config)#show running-config | section interface | ospf
+# interface Vlan1
+# ip ospf dead-interval 29
+# ip ospf hello-interval 66
+# ip ospf mtu-ignore
+# ospfv3 bfd
+# ospfv3 cost 106
+# ospfv3 hello-interval 77
+# ospfv3 transmit-delay 100
+# ospfv3 ipv4 priority 45
+# ospfv3 ipv4 area 0.0.0.5
+# ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 dead-interval 56
+# ospfv3 ipv6 retransmit-interval 115
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+# veos(config)#
+
+ - name: Delete device configuration
+ arista.eos.eos_ospf_interfaces:
+ config:
+ - name: "Vlan1"
+ state: deleted
+
+# After State:
+
+# veos#show running-config | section interface | ospf
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+#
+# Module Execution:
+#
+# "after": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+# "before": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "dead_interval": 29,
+# "hello_interval": 66,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "bfd": True,
+# "cost": 106,
+# "hello_interval": 77,
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.5"
+# },
+# "priority": 45
+# },
+# {
+# "afi": "ipv6",
+# "dead_interval": 56,
+# "mtu_ignore": True,
+# "network": "point-to-point",
+# "passive_interface": True,
+# "retransmit_interval": 115
+# }
+# ],
+# "transmit_delay": 100
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+# "changed": True,
+# "commands": [
+# "interface Vlan1",
+# "no ip ospf dead-interval 29",
+# "no ip ospf hello-interval 66",
+# "no ip ospf mtu-ignore",
+# "no ospfv3 bfd",
+# "no ospfv3 cost 106",
+# "no ospfv3 hello-interval 77",
+# "no ospfv3 transmit-delay 100",
+# "no ospfv3 ipv4 priority 45",
+# "no ospfv3 ipv4 area 0.0.0.5",
+# "no ospfv3 ipv6 passive-interface",
+# "no ospfv3 ipv6 dead-interval 56",
+# "no ospfv3 ipv6 retransmit-interval 115",
+# "no ospfv3 ipv6 network point-to-point",
+# "no ospfv3 ipv6 mtu-ignore"
+# ],
+#
+
+# Using parsed:
+# ------------
+
+# parsed.cfg:
+# ----------
+
+# interface Vlan1
+# ip ospf dead-interval 29
+# ip ospf hello-interval 66
+# ip ospf mtu-ignore
+# ip ospf cost 500
+# ospfv3 bfd
+# ospfv3 cost 106
+# ospfv3 hello-interval 77
+# ospfv3 transmit-delay 100
+# ospfv3 ipv4 priority 45
+# ospfv3 ipv4 area 0.0.0.5
+# ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 dead-interval 56
+# ospfv3 ipv6 retransmit-interval 115
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+#
+
+ - name: parse configs
+ arista.eos.eos_ospf_interfaces:
+ running_config: "{{ lookup('file', './parsed.cfg') }}"
+ state: parsed
+
+# Module Execution:
+# "parsed": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "cost": 500,
+# "dead_interval": 29,
+# "hello_interval": 66,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "bfd": True,
+# "cost": 106,
+# "hello_interval": 77,
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.5"
+# },
+# "priority": 45
+# },
+# {
+# "afi": "ipv6",
+# "dead_interval": 56,
+# "mtu_ignore": True,
+# "network": "point-to-point",
+# "passive_interface": True,
+# "retransmit_interval": 115
+# }
+# ],
+# "transmit_delay": 100
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ]
+
+# Using gathered:
+
+# Device COnfig:
+
+# veos#show running-config | section interface | ospf
+# interface Vlan1
+# ip ospf cost 500
+# ip ospf dead-interval 29
+# ip ospf hello-interval 66
+# ip ospf mtu-ignore
+# ip ospf area 0.0.0.50
+# ospfv3 cost 106
+# ospfv3 hello-interval 77
+# ospfv3 transmit-delay 100
+# ospfv3 ipv4 priority 45
+# ospfv3 ipv4 area 0.0.0.5
+# ospfv3 ipv6 passive-interface
+# ospfv3 ipv6 dead-interval 56
+# ospfv3 ipv6 retransmit-interval 115
+# ospfv3 ipv6 network point-to-point
+# ospfv3 ipv6 mtu-ignore
+# !
+# interface Vlan2
+# ospfv3 ipv4 hello-interval 45
+# ospfv3 ipv4 retransmit-interval 100
+# ospfv3 ipv4 area 0.0.0.6
+# veos#
+
+ - name: gather configs
+ arista.eos.eos_ospf_interfaces:
+ state: gathered
+
+# Module Execution:
+#
+# "gathered": [
+# {
+# "name": "Ethernet1"
+# },
+# {
+# "name": "Ethernet2"
+# },
+# {
+# "name": "Management1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.50"
+# },
+# "cost": 500,
+# "dead_interval": 29,
+# "hello_interval": 66,
+# "mtu_ignore": True
+# },
+# {
+# "afi": "ipv6",
+# "cost": 106,
+# "hello_interval": 77,
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.5"
+# },
+# "priority": 45
+# },
+# {
+# "afi": "ipv6",
+# "dead_interval": 56,
+# "mtu_ignore": True,
+# "network": "point-to-point",
+# "passive_interface": True,
+# "retransmit_interval": 115
+# }
+# ],
+# "transmit_delay": 100
+# }
+# ],
+# "name": "Vlan1"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "ip_params": [
+# {
+# "afi": "ipv4",
+# "area": {
+# "area_id": "0.0.0.6"
+# },
+# "hello_interval": 45,
+# "retransmit_interval": 100
+# }
+# ]
+# }
+# ],
+# "name": "Vlan2"
+# }
+# ],
+#
+
+
+# Using rendered:
+# --------------
+
+ - name: Render provided configuration
+ arista.eos.eos_ospf_interfaces:
+ config:
+ - name: "Vlan1"
+ address_family:
+ - afi: "ipv4"
+ dead_interval: 29
+ mtu_ignore: True
+ hello_interval: 66
+ - afi: "ipv6"
+ hello_interval: 77
+ cost : 106
+ transmit_delay: 100
+ ip_params:
+ - afi: "ipv6"
+ retransmit_interval: 115
+ dead_interval: 56
+ passive_interface: True
+ - afi: "ipv4"
+ area:
+ area_id: "0.0.0.5"
+ priority: 45
+ - name: "Vlan2"
+ address_family:
+ - afi: "ipv6"
+ ip_params:
+ - afi: "ipv4"
+ area:
+ area_id: "0.0.0.6"
+ hello_interval: 45
+ retransmit_interval: 100
+ - afi: "ipv4"
+ message_digest_key:
+ key_id: 200
+ encryption: 7
+ key: "hkdfhtu=="
+
+ state: rendered
+
+# Module Execution:
+#
+# "rendered": [
+# "interface Vlan1",
+# "ip ospf dead-interval 29",
+# "ip ospf mtu-ignore",
+# "ip ospf hello-interval 66",
+# "ospfv3 hello-interval 77",
+# "ospfv3 cost 106",
+# "ospfv3 transmit-delay 100",
+# "ospfv3 ipv4 area 0.0.0.5",
+# "ospfv3 ipv4 priority 45",
+# "ospfv3 ipv6 retransmit-interval 115",
+# "ospfv3 ipv6 dead-interval 56",
+# "ospfv3 ipv6 passive-interface",
+# "interface Vlan2",
+# "ip ospf message-digest-key 200 md5 7 hkdfhtu==",
+# "ospfv3 ipv4 area 0.0.0.6",
+# "ospfv3 ipv4 hello-interval 45",
+# "ospfv3 ipv4 retransmit-interval 100"
+# ]
+#
+
+"""
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospf_interfaces.ospf_interfaces import (
+ Ospf_interfacesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospf_interfaces.ospf_interfaces import (
+ Ospf_interfaces,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ module = AnsibleModule(
+ argument_spec=Ospf_interfacesArgs.argument_spec,
+ mutually_exclusive=[],
+ required_if=[],
+ supports_check_mode=False,
+ )
+
+ result = Ospf_interfaces(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py
new file mode 100644
index 00000000..ff178d9c
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py
@@ -0,0 +1,1545 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_ospfv2
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_ospfv2
+short_description: OSPFv2 resource module
+description: This module configures and manages the attributes of ospfv2 on Arista
+ EOS platforms.
+version_added: 1.0.0
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+notes:
+- Tested against Arista EOS 4.23.0F
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A list of configurations for ospfv2.
+ type: dict
+ suboptions:
+ processes:
+ description: A list of dictionary specifying the ospfv2 processes.
+ type: list
+ elements: dict
+ suboptions:
+ process_id:
+ description: ID of OSPFV2 process.
+ type: int
+ vrf:
+ description: VRF name .
+ type: str
+ traffic_engineering:
+ description: Enter traffic engineering config mode
+ type: bool
+ adjacency:
+ description: Configure adjacency options for OSPF instance.
+ type: dict
+ suboptions:
+ exchange_start:
+ description: Configure exchange-start options for OSPF instance.
+ type: dict
+ suboptions:
+ threshold:
+ description: Number of peers to bring up simultaneously.
+ type: int
+ router_id:
+ description: 32-bit number assigned to a router running OSPFv2.
+ type: str
+ max_lsa:
+ description: Specifies the switch behavior on reaching max lsa count.
+ type: dict
+ suboptions:
+ count:
+ description: maximum count of lsas.
+ type: int
+ threshold:
+ description: percentage of <count> , when a warning should be raised.
+ type: int
+ ignore_time:
+ description: time in minutes, for which the switch shoud be shutdown
+ on max-lsa warning
+ type: int
+ ignore_count:
+ description: No. of times the switch can shut down temporarily on
+ warning
+ type: int
+ reset_time:
+ description: Time in minutes, after which the shutdown counter resets.
+ type: int
+ warning:
+ description: Only give warning message when limit is exceeded
+ type: bool
+ max_metric:
+ description: Set maximum metric.
+ type: dict
+ suboptions:
+ router_lsa:
+ description: Maximum metric in self-originated router-LSAs.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set router-lsa attribute.
+ type: bool
+ external_lsa:
+ description: Override external-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ include_stub:
+ description: Set maximum metric for stub links in router-LSAs.
+ type: bool
+ on_startup:
+ description: Set maximum metric temporarily after reboot.
+ type: dict
+ suboptions:
+ wait_period:
+ description:
+ - Wait period in seconds after startup.
+ type: int
+ summary_lsa:
+ description: Override summary-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ log_adjacency_changes:
+ description: To configure link-state changes and transitions of OSPFv2
+ neighbors.
+ type: dict
+ suboptions:
+ detail:
+ description: If true , configures the switch to log all link-state
+ changes.
+ type: bool
+ maximum_paths:
+ description: Maximum number of next-hops in an ECMP route.
+ type: int
+ mpls_ldp:
+ description: mpls ldp sync configuration.
+ type: bool
+ networks:
+ description: Configure routing for a network.
+ type: list
+ elements: dict
+ suboptions:
+ network_address:
+ description: Network Address.
+ type: str
+ prefix:
+ description: Prefix.
+ type: str
+ mask:
+ description: Network Wildcard Mask.
+ type: str
+ area:
+ description: Configure OSPF area.
+ type: str
+ passive_interface:
+ description: Include interface but without actively running OSPF.
+ type: dict
+ suboptions:
+ interface_list:
+ description: Interface range.
+ type: str
+ default:
+ description: If True, Set all interfaces to passive by default
+ type: bool
+ point_to_point:
+ description: Configure Point-to-point specific features.
+ type: bool
+ rfc1583compatibility:
+ description: Specifies different methods for calculating summary route
+ metrics.
+ type: bool
+ distance:
+ description: Specifies the administrative distance for routes.
+ type: dict
+ suboptions:
+ external:
+ description: Routes external to the area
+ type: int
+ inter_area:
+ description: Routes from other areas
+ type: int
+ intra_area:
+ description: Routes with in an area
+ type: int
+ redistribute:
+ description: Specifies the routes to be redistributed
+ type: list
+ elements: dict
+ suboptions:
+ routes:
+ description: Route types (BGP,isis,connected etc)
+ type: str
+ route_map:
+ description: Specify which route map to use.
+ type: str
+ isis_level:
+ description: ISIS levels.
+ type: str
+ retransmission_threshold:
+ description: Configure threshold for retransmission.
+ type: int
+ distribute_list:
+ description: Specifies the list of routes to be filtered.
+ type: dict
+ suboptions:
+ route_map:
+ description: route map to be filtered
+ type: str
+ prefix_list:
+ description: prefix list to be filtered
+ type: str
+ areas:
+ description: Specifies the configuration for OSPF areas
+ type: list
+ elements: dict
+ suboptions:
+ area_id:
+ description: Specifies a 32 bit number expressed in decimal or dotted-decimal
+ notation.
+ type: str
+ default_cost:
+ description: Specify the cost for default summary route in stub/NSSA
+ area.
+ type: int
+ filter:
+ description: Specify the filter for incoming summary LSAs.
+ type: dict
+ suboptions:
+ address:
+ description: IP address.
+ type: str
+ subnet_address:
+ description: IP address with mask length
+ type: str
+ subnet_mask:
+ description: IP subnet mask
+ type: str
+ prefix_list:
+ description: Specify list to filter for incoming LSAs.
+ type: str
+ nssa:
+ description: Configures NSSA parameters.
+ type: dict
+ suboptions:
+ default_information_originate:
+ description: Originate default Type 7 LSA.
+ type: dict
+ suboptions:
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ nssa_only:
+ description: Limit default advertisement to this NSSA area.
+ type: bool
+ no_summary:
+ description: Filter all type-3 LSAs in the nssa area.
+ type: bool
+ nssa_only:
+ description: Disable Type-7 LSA p-bit setting
+ type: bool
+ set:
+ description: Set config up to nssa
+ type: bool
+ not_so_stubby:
+ description: Configures NSSA parameters.
+ type: dict
+ suboptions:
+ default_information_originate:
+ description: Originate default Type 7 LSA.
+ type: dict
+ suboptions:
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ nssa_only:
+ description: Limit default advertisement to this NSSA area.
+ type: bool
+ lsa:
+ description: lsa parameters
+ type: bool
+ no_summary:
+ description: Filter all type-3 LSAs in the nssa area.
+ type: bool
+ nssa_only:
+ description: Disable Type-7 LSA p-bit setting
+ type: bool
+ set:
+ description: Set config up to not-so-stubby
+ type: bool
+ range:
+ description: Configure route summarization.
+ type: dict
+ suboptions:
+ address:
+ description: IP address.
+ type: str
+ subnet_address:
+ description: IP address with mask length
+ type: str
+ subnet_mask:
+ description: IP subnet mask
+ type: str
+ advertise:
+ description: Enable Advertisement of the range.
+ type: bool
+ cost:
+ description: Configures the metric.
+ type: int
+ stub:
+ description: Stub area.
+ type: dict
+ suboptions:
+ no_summary:
+ description: If False , Filter all type-3 LSAs in the stub area.
+ type: bool
+ set:
+ description: When true sets the stub config alone.
+ type: bool
+ auto_cost:
+ description: Set auto-cost.
+ type: dict
+ suboptions:
+ reference_bandwidth:
+ description: reference bandwidth in megabits per sec.
+ type: int
+ bfd:
+ description: Enable BFD.
+ type: dict
+ suboptions:
+ all_interfaces:
+ description: Enable BFD on all interfaces.
+ type: bool
+ default_information:
+ description: Control distribution of default information.
+ type: dict
+ suboptions:
+ originate:
+ description: Distribute a default route.
+ type: bool
+ always:
+ description: Always advertise default route.
+ type: bool
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ route_map:
+ description: Specify which route-map to use.
+ type: str
+ default_metric:
+ description: Configure the default metric for redistributed routes
+ type: int
+ dn_bit_ignore:
+ description: If True, Disable dn-bit check for Type-3 LSAs in non-default
+ VRFs.
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: dict
+ suboptions:
+ grace_period:
+ description: Specify maximum time to wait for graceful-restart to
+ complete.
+ type: int
+ set:
+ description: When true sets the grace_fulrestart config alone.
+ type: bool
+ graceful_restart_helper:
+ description: If True, Enable graceful restart helper.
+ type: bool
+ shutdown:
+ description: Disable the OSPF instance.
+ type: bool
+ summary_address:
+ description: Summary route configuration.
+ type: dict
+ suboptions:
+ address:
+ description: IP summary address.
+ type: str
+ prefix:
+ description: Prefix.
+ type: str
+ mask:
+ description: Summary Mask.
+ type: str
+ attribute_map:
+ description: Set attributes of summary route.
+ type: str
+ not_advertise:
+ description: Do not advertise summary route.
+ type: bool
+ tag:
+ description: Set tag.
+ type: int
+ timers:
+ description: Configure OSPF timers.
+ type: list
+ elements: dict
+ suboptions:
+ lsa:
+ description: Configure OSPF LSA timers.
+ type: dict
+ suboptions:
+ rx:
+ description: Configure OSPF LSA receiving timers
+ type: dict
+ suboptions:
+ min_interval:
+ description: Configure OSPF LSA arrival timer.
+ type: int
+ tx:
+ description: Configure OSPF LSA transmission timers.
+ type: dict
+ suboptions:
+ delay:
+ description: Configure OSPF LSA transmission delay.
+ type: dict
+ suboptions:
+ initial:
+ description: Delay to generate first occurrence of LSA
+ in msecs.
+ type: int
+ min:
+ description: Min delay between originating the same LSA
+ in msecs.
+ type: int
+ max:
+ description: Maximum delay between originating the same
+ LSA in msecs.
+ type: int
+ out_delay:
+ description: Configure out-delay timer.
+ type: int
+ pacing:
+ description: Configure OSPF packet pacing.
+ type: int
+ spf:
+ description: Configure SPF timers
+ type: dict
+ suboptions:
+ seconds:
+ description: Seconds.
+ type: int
+ initial:
+ description: Initial SPF schedule delay in msecs.
+ type: int
+ min:
+ description: Min Hold time between two SPFs in msecs
+ type: int
+ max:
+ description: Max wait time between two SPFs in msecs.
+ type: int
+ fips_restrictions:
+ description: Use FIPS compliant algorithms
+ type: str
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ospf).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed]
+ default: merged
+
+"""
+EXAMPLES = """
+# Using merged
+
+# Before state:
+# ------------
+# localhost#show running-config | section ospf
+# localhost#
+
+ - name: replace Ospf configs
+ arista.eos.eos_ospfv2:
+ config:
+ - processes:
+ - process_id: 1
+ adjacency:
+ exchange_start:
+ threshold: 20045623
+ areas:
+ - filter:
+ address: "10.1.1.0/24"
+ id: "0.0.0.2"
+ - id: "0.0.0.50"
+ range:
+ address: "172.20.0.0/16"
+ cost: 34
+ default_information:
+ metric: 100
+ metric_type: 1
+ originate: True
+ distance:
+ intra_area: 85
+ max_lsa:
+ count: 8000
+ ignore_count: 3
+ ignore_time: 6
+ reset_time: 20
+ threshold: 40
+ networks:
+ - area: "0.0.0.0"
+ prefix: 10.10.2.0/24
+ - area: "0.0.0.0"
+ prefix: "10.10.3.0/24"
+ redistribute:
+ - routes: "static"
+ router_id: "170.21.0.4"
+ - process_id: 2
+ vrf: "vrf01"
+ areas:
+ - id: "0.0.0.9"
+ default_cost: 20
+ max_lsa:
+ count: 8000
+ ignore_count: 3
+ ignore_time: 6
+ reset_time: 20
+ threshold: 40
+ networks:
+ - area: "0.0.0.0"
+ prefix: 10.10.2.0/24
+ - area: "0.0.0.0"
+ prefix: "10.10.3.0/24"
+ redistribute:
+ - routes: "static"
+ router_id: "170.21.0.4"
+ - process_id: 2
+ vrf: "vrf01"
+ areas:
+ - id: "0.0.0.9"
+ default_cost: 20
+ max_lsa:
+ count: 8000
+ ignore_count: 3
+ ignore_time: 6
+ reset_time: 20
+ threshold: 40
+ - process_id: 3
+ vrf: "vrf02"
+ redistribute:
+ - routes: "connected"
+
+# After state:
+# localhost#show running-config | section ospf
+# router ospf 1
+# router-id 170.21.0.4
+# distance ospf intra-area 85
+# redistribute static
+# area 0.0.0.2 filter 10.1.1.0/24
+# area 0.0.0.50 range 172.20.0.0/16 cost 34
+# network 10.10.2.0/24 area 0.0.0.0
+# network 10.10.3.0/24 area 0.0.0.0
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# adjacency exchange-start threshold 20045623
+# default-information originate metric 100 metric-type 1
+#
+# router ospf 2 vrf vrf01
+# area 0.0.0.9 default-cost 20
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# !
+# router ospf 3 vrf vrf02
+# redistribute connected
+# max-lsa 12000
+# localhost#
+#
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+#
+
+
+# Using replaced:
+# --------------
+
+# Before State:
+
+# localhost#show running-config | section ospf
+# router ospf 1
+# router-id 170.21.0.4
+# distance ospf intra-area 85
+# redistribute static
+# area 0.0.0.2 filter 10.1.1.0/24
+# area 0.0.0.50 range 172.20.0.0/16 cost 34
+# network 10.10.2.0/24 area 0.0.0.0
+# network 10.10.3.0/24 area 0.0.0.0
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# adjacency exchange-start threshold 20045623
+# default-information originate metric 100 metric-type 1
+# !
+# router ospf 2 vrf vrf01
+# area 0.0.0.9 default-cost 20
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# !
+# router ospf 3 vrf vrf02
+# redistribute connected
+# max-lsa 12000
+# localhost#
+#
+# "before": [
+# {
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+#
+ - name: replace Ospf configs
+ arista.eos.eos_ospfv2:
+ config:
+ - processes:
+ - process_id: 2
+ vrf: "vrf01"
+ point_to_point: True
+ redistribute:
+ - routes: "isis"
+ isis_level: "level-1"
+
+ state: replaced
+
+# After State:
+# -----------
+# "router ospf 2 vrf vrf01",
+# "no area 0.0.0.9 default-cost 20",
+# "no max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20",
+# "point-to-point routes",
+# "redistribute isis level-1"
+#
+# "after": [
+# {
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "isis_level": "level-1",
+# "routes": "isis"
+# }
+# ],
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+#
+
+# Using overridden:
+# ----------------
+
+# Before State:
+# localhost#show running-config | section ospf
+# router ospf 1
+# router-id 170.21.0.4
+# distance ospf intra-area 85
+# redistribute static
+# area 0.0.0.2 filter 10.1.1.0/24
+# area 0.0.0.50 range 172.20.0.0/16 cost 34
+# network 10.10.2.0/24 area 0.0.0.0
+# network 10.10.3.0/24 area 0.0.0.0
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# adjacency exchange-start threshold 20045623
+# default-information originate metric 100 metric-type 1
+# !
+# router ospf 2 vrf vrf01
+# redistribute isis level-1
+# max-lsa 12000
+# !
+# router ospf 3 vrf vrf02
+# redistribute connected
+# max-lsa 12000
+# localhost#
+#
+# "before": [
+# {
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "isis_level": "level-1",
+# "routes": "isis"
+# }
+# ],
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+
+ - name: override Ospf configs
+ arista.eos.eos_ospfv2:
+ config:
+ - processes:
+ - process_id: 2
+ vrf: "vrf01"
+ redistribute:
+ - routes: "connected"
+
+ state: override
+
+# After State:
+
+# "no router ospf 1",
+# "no router ospf 3",
+# "router ospf 2 vrf vrf01",
+# "no max-lsa 12000",
+# "no redistribute isis level-1",
+# "redistribute connected"
+#
+# "after": [
+# {
+# "processes": [
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf01"
+# }
+# ]
+# }
+# ]
+
+# Using Deleted:
+
+# localhost#show running-config | section ospf
+# router ospf 1
+# router-id 170.21.0.4
+# distance ospf intra-area 85
+# redistribute static
+# area 0.0.0.2 filter 10.1.1.0/24
+# area 0.0.0.50 range 172.20.0.0/16 cost 34
+# network 10.10.2.0/24 area 0.0.0.0
+# network 10.10.3.0/24 area 0.0.0.0
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# adjacency exchange-start threshold 20045623
+# default-information originate metric 100 metric-type 1
+# !
+# router ospf 2 vrf vrf01
+# redistribute connected
+# area 0.0.0.9 default-cost 20
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# !
+# router ospf 3 vrf vrf02
+# redistribute connected
+# max-lsa 12000
+# localhost#
+#
+# "before": [
+# {
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+
+ - name: Delete Ospf configs
+ arista.eos.eos_ospfv2:
+ config:
+ - processes:
+ - process_id: 1
+
+ state: deleted
+
+# After State:
+# Commands:
+# "no router ospf 1"
+
+# "after": [
+# {
+# "processes": [
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+
+# Using gathered:
+# localhost#show running-config | section ospf
+# router ospf 2 vrf vrf01
+# redistribute connected
+# area 0.0.0.9 default-cost 20
+# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20
+# !
+# router ospf 3 vrf vrf02
+# redistribute connected
+# max-lsa 12000
+# localhost#
+
+ - name: replace Ospf configs
+ arista.eos.eos_ospfv2:
+ state: gathered
+
+# "gathered": [
+# {
+# "processes": [
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 8000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf01"
+# },
+# {
+# "max_lsa": {
+# "count": 12000
+# },
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+
+# Using parsed:
+# ------------
+
+# parsed.cfg
+# router ospf 1
+# adjacency exchange-start threshold 20045623
+# area 0.0.0.2 filter 10.1.1.0/24
+# area 0.0.0.50 range 172.20.0.0/16 cost 34
+# default-information originate metric 100 metric-type 1
+# distance ospf intra-area 85
+# max-lsa 80000 40 ignore-count 3 ignore-time 6 reset-time 20
+# network 10.10.2.0/24 area 0.0.0.0
+# network 10.10.3.0/24 area 0.0.0.0
+# redistribute static
+# router-id 170.21.0.4
+# router ospf 2 vrf vrf01,
+# area 0.0.0.9 default-cost 20
+# max-lsa 80000 40 ignore-count 3 ignore-time 6 reset-time 20
+# router ospf 3 vrf vrf02
+# redistribute static
+
+ - name: Parse Ospf configs
+ arista.eos.eos_ospfv2:
+ running_config: "{{ lookup('file', './parsed.cfg') }}"
+ state: parsed
+
+# "parsed": [
+# {
+# "processes": [
+# {
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 20045623
+# }
+# },
+# "areas": [
+# {
+# "filter": {
+# "address": "10.1.1.0/24"
+# },
+# "id": "0.0.0.2"
+# },
+# {
+# "id": "0.0.0.50",
+# "range": {
+# "address": "172.20.0.0/16",
+# "cost": 34
+# }
+# }
+# ],
+# "default_information": {
+# "metric": 100,
+# "metric_type": 1,
+# "originate": true
+# },
+# "distance": {
+# "intra_area": 85
+# },
+# "max_lsa": {
+# "count": 80000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "networks": [
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.2.0/24"
+# },
+# {
+# "area": "0.0.0.0",
+# "prefix": "10.10.3.0/24"
+# }
+# ],
+# "process_id": 1,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "router_id": "170.21.0.4"
+# },
+# {
+# "areas": [
+# {
+# "default_cost": 20,
+# "id": "0.0.0.9"
+# }
+# ],
+# "max_lsa": {
+# "count": 80000,
+# "ignore_count": 3,
+# "ignore_time": 6,
+# "reset_time": 20,
+# "threshold": 40
+# },
+# "process_id": 2,
+# "vrf": "vrf01,"
+# },
+# {
+# "process_id": 3,
+# "redistribute": [
+# {
+# "routes": "static"
+# }
+# ],
+# "vrf": "vrf02"
+# }
+# ]
+# }
+# ]
+
+# Using rendered:
+# --------------
+
+ - name: replace Ospf configs
+ arista.eos.eos_ospfv2:
+ config:
+ - processes:
+ - process_id: 1
+ adjacency:
+ exchange_start:
+ threshold: 20045623
+ areas:
+ - filter:
+ address: 10.1.1.0/24
+ id: 0.0.0.2
+ - id: 0.0.0.50
+ range:
+ address: 172.20.0.0/16
+ cost: 34
+ default_information:
+ metric: 100
+ metric_type: 1
+ originate: true
+ distance:
+ intra_area: 85
+ max_lsa:
+ count: 8000
+ ignore_count: 3
+ ignore_time: 6
+ reset_time: 20
+ threshold: 40
+ networks:
+ - area: 0.0.0.0
+ prefix: 10.10.2.0/24
+ - area: 0.0.0.0
+ prefix: 10.10.3.0/24
+ redistribute:
+ - routes: static
+ router_id: 170.21.0.4
+ state: rendered
+
+# "rendered": [
+# "router ospf 1",
+# "adjacency exchange-start threshold 20045623",
+# "area 0.0.0.2 filter 10.1.1.0/24",
+# "area 0.0.0.50 range 172.20.0.0/16 cost 34",
+# "default-information originate metric 100 metric-type 1",
+# "distance ospf intra-area 85",
+# "max-lsa 8000 40 ignore-count 3 ignore-time 6 reset-time 20",
+# "network 10.10.2.0/24 area 0.0.0.0",
+# "network 10.10.3.0/24 area 0.0.0.0",
+# "redistribute static",
+# "router-id 170.21.0.4"
+# ]
+#
+
+"""
+RETURN = """
+before:
+ description: The configuration prior to the model invocation.
+ returned: always
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+ type: list
+after:
+ description: The resulting configuration model invocation.
+ returned: when changed
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+ type: list
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ["router ospf 1",
+ "adjacency exchange-start threshold 20045623",
+ "area 0.0.0.2 filter 10.1.1.0/24",
+ "area 0.0.0.50 range 172.20.0.0/16 cost 34",
+ "default-information originate metric 100 metric-type 1",
+ "distance ospf intra-area 85",
+ "max-lsa 8000 40 ignore-count 3 ignore-time 6 reset-time 20",
+ "network 10.10.2.0/24 area 0.0.0.0",
+ "network 10.10.3.0/24 area 0.0.0.0",
+ "redistribute static",
+ "router-id 170.21.0.4"]
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv2.ospfv2 import (
+ Ospfv2Args,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospfv2.ospfv2 import (
+ Ospfv2,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+
+ module = AnsibleModule(
+ argument_spec=Ospfv2Args.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Ospfv2(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py
new file mode 100644
index 00000000..777e1ece
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py
@@ -0,0 +1,1528 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2020 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_ospfv3
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+module: eos_ospfv3
+short_description: OSPFv3 resource module
+description: This module configures and manages the attributes of ospfv3 on Arista
+ EOS platforms.
+version_added: 1.1.0
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+notes:
+- Tested against Arista EOS 4.23.0F
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A list of configurations for ospfv3.
+ type: dict
+ suboptions:
+ processes:
+ description: A list of dictionary specifying the ospfv3 processes.
+ type: list
+ elements: dict
+ suboptions:
+ vrf:
+ description: VRF name .
+ type: str
+ adjacency:
+ description: Configure adjacency options for OSPF instance.
+ type: dict
+ suboptions:
+ exchange_start:
+ description: Configure exchange-start options for OSPF instance.
+ type: dict
+ suboptions:
+ threshold:
+ description: Number of peers to bring up simultaneously.
+ type: int
+ auto_cost:
+ description: Set auto-cost.
+ type: dict
+ suboptions:
+ reference_bandwidth:
+ description: reference bandwidth in megabits per sec.
+ type: int
+ areas:
+ description: Specifies the configuration for OSPF areas
+ type: list
+ elements: dict
+ suboptions:
+ area_id:
+ description: Specifies a 32 bit number expressed in decimal or dotted-decimal
+ notation.
+ type: str
+ default_cost:
+ description: Specify the cost for default summary route in stub/NSSA
+ area.
+ type: int
+ authentication:
+ description: Configure authentication for the area incase of ospfv3.
+ type: dict
+ suboptions:
+ spi:
+ description: Specify the SPI value
+ type: int
+ algorithm:
+ description: Name of algorithm to be used.
+ type: str
+ choices: ['md5', 'sha1']
+ encrypt_key:
+ description: If False, key string is not encrypted
+ type: bool
+ hidden_key:
+ description: If True, Specifies that a HIDDEN key will follow.
+ type: bool
+ key:
+ description: 128 bit MD5 key or 140 bit SHA1 key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ encryption:
+ description: Configure encryption for the area
+ type: dict
+ suboptions:
+ spi:
+ description: Specify the SPI value
+ type: int
+ encryption:
+ description: name of encryption to be used.
+ type: str
+ choices: ['3des-cbc', 'aes-128-cbc', 'aes-192-cbc', 'aes-256-cbc', 'null']
+ algorithm:
+ description: name of the algorithm to be used.
+ type: str
+ choices: ['sha1', 'md5']
+ encrypt_key:
+ description: If False, key string is not encrypted
+ type: bool
+ hidden_key:
+ description: If True, Specifies that a HIDDEN key will follow.
+ type: bool
+ key:
+ description: 128 bit MD5 key or 140 bit SHA1 key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ nssa:
+ description: Configures NSSA parameters.
+ type: dict
+ suboptions:
+ default_information_originate:
+ description: Originate default Type 7 LSA.
+ type: dict
+ suboptions:
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ nssa_only:
+ description: Limit default advertisement to this NSSA area.
+ type: bool
+ set:
+ description: True if only default information orignate is set
+ type: bool
+ no_summary:
+ description: Filter all type-3 LSAs in the nssa area.
+ type: bool
+ nssa_only:
+ description: Disable Type-7 LSA p-bit setting
+ type: bool
+ translate:
+ description: Enable LSA translation.
+ type: bool
+ set:
+ description: True if only nssa is set
+ type: bool
+ stub:
+ description: Stub area.
+ type: dict
+ suboptions:
+ set:
+ description: True if only stub is set.
+ type: bool
+ summary_lsa:
+ description: If False , Filter all type-3 LSAs in the stub area.
+ type: bool
+
+ bfd:
+ description: Enable BFD.
+ type: dict
+ suboptions:
+ all_interfaces:
+ description: Enable BFD on all interfaces.
+ type: bool
+ fips_restrictions:
+ description: Use FIPS compliant algorithms
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: dict
+ suboptions:
+ grace_period:
+ description: Specify maximum time to wait for graceful-restart to
+ complete.
+ type: int
+ set:
+ description: When true sets the grace_fulrestart config alone.
+ type: bool
+ graceful_restart_helper:
+ description: If True, Enable graceful restart helper.
+ type: bool
+ log_adjacency_changes:
+ description: To configure link-state changes and transitions of OSPFv3
+ neighbors.
+ type: dict
+ suboptions:
+ detail:
+ description: If true , configures the switch to log all link-state
+ changes.
+ type: bool
+ set:
+ description: When true sets the log_adjacency_changes config alone.
+ type: bool
+ max_metric:
+ description: Set maximum metric.
+ type: dict
+ suboptions:
+ router_lsa:
+ description: Maximum metric in self-originated router-LSAs.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set router-lsa attribute.
+ type: bool
+ external_lsa:
+ description: Override external-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ include_stub:
+ description: Set maximum metric for stub links in router-LSAs.
+ type: bool
+ on_startup:
+ description: Set maximum metric temporarily after reboot.
+ type: dict
+ suboptions:
+ wait_period:
+ description:
+ - Wait period in seconds after startup.
+ type: int
+ wait_for_bgp:
+ description:
+ - Let BGP decide when to originate router-LSA with normal metric
+ type: bool
+ summary_lsa:
+ description: Override summary-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ passive_interface:
+ description: Include interface but without actively running OSPF.
+ type: bool
+ router_id:
+ description: 32-bit number assigned to a router running OSPFv3.
+ type: str
+ shutdown:
+ description: Disable the OSPF instance.
+ type: bool
+ timers:
+ description: Configure OSPF timers.
+ type: dict
+ suboptions:
+ lsa:
+ description: Configure OSPF LSA timers.
+ type: int
+ out_delay:
+ description: Configure out-delay timer.
+ type: int
+ pacing:
+ description: Configure OSPF packet pacing.
+ type: int
+ throttle:
+ description: Configure SPF timers
+ type: dict
+ suboptions:
+ lsa:
+ description: Configure threshold for retransmission of lsa
+ type: bool
+ spf:
+ description: Configure time between SPF calculations
+ type: bool
+ initial:
+ description: Initial SPF schedule delay in msecs.
+ type: int
+ min:
+ description: Min Hold time between two SPFs in msecs
+ type: int
+ max:
+ description: Max wait time between two SPFs in msecs.
+ type: int
+
+ address_family:
+ description: Enable address family and enter its config mode
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description: address family .
+ type: str
+ choices:
+ - ipv4
+ - ipv6
+ adjacency:
+ description: Configure adjacency options for OSPF instance.
+ type: dict
+ suboptions:
+ exchange_start:
+ description: Configure exchange-start options for OSPF instance.
+ type: dict
+ suboptions:
+ threshold:
+ description: Number of peers to bring up simultaneously.
+ type: int
+ auto_cost:
+ description: Set auto-cost.
+ type: dict
+ suboptions:
+ reference_bandwidth:
+ description: reference bandwidth in megabits per sec.
+ type: int
+ areas:
+ description: Specifies the configuration for OSPF areas
+ type: list
+ elements: dict
+ suboptions:
+ area_id:
+ description: Specifies a 32 bit number expressed in decimal or dotted-decimal
+ notation.
+ type: str
+ default_cost:
+ description: Specify the cost for default summary route in stub/NSSA
+ area.
+ type: int
+ authentication:
+ description: Configure authentication for the area incase of ospfv3.
+ type: dict
+ suboptions:
+ spi:
+ description: Specify the SPI value
+ type: int
+ algorithm:
+ description: Name of algorithm to be used.
+ type: str
+ choices: ['md5', 'sha1']
+ encrypt_key:
+ description: If False, key string is not encrypted
+ type: bool
+ hidden_key:
+ description: If True, Specifies that a HIDDEN key will follow.
+ type: bool
+ key:
+ description: 128 bit MD5 key or 140 bit SHA1 key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ encryption:
+ description: Configure encryption for the area
+ type: dict
+ suboptions:
+ spi:
+ description: Specify the SPI value
+ type: int
+ encryption:
+ description: name of encryption to be used.
+ type: str
+ choices: ['3des-cbc', 'aes-128-cbc', 'aes-192-cbc', 'aes-256-cbc', 'null']
+ algorithm:
+ description: name of the algorithm to be used.
+ type: str
+ choices: ['sha1', 'md5']
+ encrypt_key:
+ description: If False, key string is not encrypted
+ type: bool
+ hidden_key:
+ description: If True, Specifies that a HIDDEN key will follow.
+ type: bool
+ key:
+ description: 128 bit MD5 key or 140 bit SHA1 key.
+ type: str
+ passphrase:
+ description: Passphrase String for deriving keys for authentication and encryption.
+ type: str
+ nssa:
+ description: Configures NSSA parameters.
+ type: dict
+ suboptions:
+ default_information_originate:
+ description: Originate default Type 7 LSA.
+ type: dict
+ suboptions:
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ nssa_only:
+ description: Limit default advertisement to this NSSA area.
+ type: bool
+ set:
+ description: True if only default information orignate is set
+ type: bool
+ no_summary:
+ description: Filter all type-3 LSAs in the nssa area.
+ type: bool
+ nssa_only:
+ description: Disable Type-7 LSA p-bit setting
+ type: bool
+ translate:
+ description: Enable LSA translation.
+ type: bool
+ set:
+ description: True if only nssa is set
+ type: bool
+ ranges:
+ description: Configure route summarization.
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description: IP address.
+ type: str
+ subnet_address:
+ description: IP address with mask length
+ type: str
+ subnet_mask:
+ description: IP subnet mask
+ type: str
+ advertise:
+ description: Enable Advertisement of the range.
+ type: bool
+ cost:
+ description: Configures the metric.
+ type: int
+ stub:
+ description: Stub area.
+ type: dict
+ suboptions:
+ set:
+ description: True if only stub is set
+ type: bool
+ summary_lsa:
+ description: If False , Filter all type-3 LSAs in the stub area.
+ type: bool
+
+ bfd:
+ description: Enable BFD.
+ type: dict
+ suboptions:
+ all_interfaces:
+ description: Enable BFD on all interfaces.
+ type: bool
+ default_information:
+ description: Control distribution of default information.
+ type: dict
+ suboptions:
+ originate:
+ description: Distribute a default route.
+ type: bool
+ always:
+ description: Always advertise default route.
+ type: bool
+ metric:
+ description: Metric for default route.
+ type: int
+ metric_type:
+ description: Metric type for default route.
+ type: int
+ route_map:
+ description: Specify which route-map to use.
+ type: str
+ default_metric:
+ description: Configure the default metric for redistributed routes.
+ type: int
+ distance:
+ description: Specifies the administrative distance for routes.
+ type: int
+ fips_restrictions:
+ description: Use FIPS compliant algorithms
+ type: bool
+ graceful_restart:
+ description: Enable graceful restart mode.
+ type: dict
+ suboptions:
+ grace_period:
+ description: Specify maximum time to wait for graceful-restart to complete.
+ type: int
+ set:
+ description: When true sets the grace_fulrestart config alone.
+ type: bool
+ graceful_restart_helper:
+ description: If True, Enable graceful restart helper.
+ type: bool
+ log_adjacency_changes:
+ description: To configure link-state changes and transitions of OSPFv3
+ neighbors.
+ type: dict
+ suboptions:
+ detail:
+ description: If true , configures the switch to log all link-state
+ changes.
+ type: bool
+ set:
+ description: When true sets the log_adjacency_changes config alone.
+ type: bool
+ max_metric:
+ description: Set maximum metric.
+ type: dict
+ suboptions:
+ router_lsa:
+ description: Maximum metric in self-originated router-LSAs.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set router-lsa attribute.
+ type: bool
+ external_lsa:
+ description: Override external-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ include_stub:
+ description: Set maximum metric for stub links in router-LSAs.
+ type: bool
+ on_startup:
+ description: Set maximum metric temporarily after reboot.
+ type: dict
+ suboptions:
+ wait_period:
+ description:
+ - Wait period in seconds after startup.
+ type: int
+ wait_for_bgp:
+ description:
+ - Let BGP decide when to originate router-LSA with normal metric
+ type: bool
+ summary_lsa:
+ description: Override summary-lsa metric with max-metric value.
+ type: dict
+ suboptions:
+ set:
+ description:
+ - Set external-lsa attribute.
+ type: bool
+ max_metric_value:
+ description:
+ - Set max metric value for external LSAs.
+ type: int
+ maximum_paths:
+ description: Maximum number of next-hops in an ECMP route.
+ type: int
+ passive_interface:
+ description: Include interface but without actively running OSPF.
+ type: bool
+ redistribute:
+ description: Specifies the routes to be redistributed.
+ type: list
+ elements: dict
+ suboptions:
+ routes:
+ description: Route types (BGP,static,connected)
+ type: str
+ choices: ['bgp', 'connected', 'static']
+ route_map:
+ description: Specify which route map to use.
+ type: str
+ router_id:
+ description: 32-bit number assigned to a router running OSPFv3.
+ type: str
+ shutdown:
+ description: Disable the OSPF instance.
+ type: bool
+ timers:
+ description: Configure OSPF timers.
+ type: dict
+ suboptions:
+ lsa:
+ description: Configure OSPF LSA timers.
+ type: int
+ out_delay:
+ description: Configure out-delay timer.
+ type: int
+ pacing:
+ description: Configure OSPF packet pacing.
+ type: int
+ throttle:
+ description: Configure SPF timers
+ type: dict
+ suboptions:
+ lsa:
+ description: Configure threshold for retransmission of lsa
+ type: bool
+ spf:
+ description: Configure time between SPF calculations
+ type: bool
+ initial:
+ description: Initial SPF schedule delay in msecs.
+ type: int
+ min:
+ description: Min Hold time between two SPFs in msecs
+ type: int
+ max:
+ description: Max wait time between two SPFs in msecs.
+ type: int
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | section ospfv3).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed]
+ default: merged
+"""
+
+EXAMPLES = """
+
+# Using merged
+
+# Before state
+
+# veos#show running-config | section ospfv3
+# veos#
+
+
+ - arista.eos.eos_ospfv3:
+ config:
+ processes:
+ - address_family:
+ - timers:
+ lsa: 22
+ graceful_restart:
+ grace_period: 35
+ afi: "ipv6"
+ timers:
+ pacing: 55
+ fips_restrictions: True
+ router_id: "2.2.2.2"
+ vrf: "vrfmerge"
+
+
+# After state
+
+# veos#show running-config | section ospfv3
+# router ospfv3 vrf vrfmerge
+# router-id 2.2.2.2
+# fips restrictions
+# timers pacing flood 55
+# !
+# address-family ipv6
+# fips restrictions
+# timers lsa arrival 22
+# graceful-restart grace-period 35
+# veos#
+
+# Module Execution
+# "after": {
+# "processes": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "fips_restrictions": true,
+# "graceful_restart": {
+# "grace_period": 35
+# },
+# "timers": {
+# "lsa": 22
+# }
+# }
+# ],
+# "fips_restrictions": true,
+# "router_id": "2.2.2.2",
+# "timers": {
+# "pacing": 55
+# },
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "before": {},
+# "changed": true,
+# "commands": [
+# "router ospfv3 vrf vrfmerge",
+# "address-family ipv6",
+# "graceful-restart grace-period 35",
+# "timers lsa arrival 22",
+# "exit",
+# "timers pacing flood 55",
+# "fips restrictions",
+# "router-id 2.2.2.2",
+# "exit"
+# ],
+
+
+# using replaced
+
+# before state
+
+# veos#show running-config | section ospfv3
+# router ospfv3
+# fips restrictions
+# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=
+# !
+# router ospfv3 vrf vrfmerge
+# router-id 2.2.2.2
+# fips restrictions
+# timers pacing flood 55
+# !
+# address-family ipv6
+# fips restrictions
+# timers lsa arrival 22
+# graceful-restart grace-period 35
+# veos#
+
+
+ - arista.eos.eos_ospfv3:
+ config:
+ processes:
+ - areas:
+ - area_id: "0.0.0.0"
+ encryption:
+ spi: 43
+ encryption: "null"
+ algorithm: "md5"
+ encrypt_key: False
+ passphrase: "7hl8FV3lZ6H1mAKpjL47hQ=="
+ vrf: "default"
+ address_family:
+ - afi: "ipv4"
+ router_id: "7.1.1.1"
+ state: replaced
+
+# After state
+# veos#show running-config | section ospfv3
+# router ospfv3
+# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=
+# !
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+# Module execution
+
+# "after": {
+# "processes": [
+# {
+# "areas": [
+# {
+# "area_id": "0.0.0.0",
+# "encryption": {
+# "algorithm": "md5",
+# "encryption": "null",
+# "hidden_key": true,
+# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4="
+# }
+# }
+# ],
+# "vrf": "default"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "before": {
+# "processes": [
+# {
+# "areas": [
+# {
+# "area_id": "0.0.0.0",
+# "encryption": {
+# "algorithm": "md5",
+# "encryption": "null",
+# "hidden_key": true,
+# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4="
+# }
+# }
+# ],
+# "fips_restrictions": true,
+# "vrf": "default"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "fips_restrictions": true,
+# "graceful_restart": {
+# "grace_period": 35
+# },
+# "timers": {
+# "lsa": 22
+# }
+# }
+# ],
+# "fips_restrictions": true,
+# "router_id": "2.2.2.2",
+# "timers": {
+# "pacing": 55
+# },
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "router ospfv3 vrf vrfmerge",
+# "address-family ipv6",
+# "no fips restrictions",
+# "no graceful-restart",
+# "no timers lsa arrival 22",
+# "area 0.0.0.3 range 10.1.2.2/24 advertise",
+# "area 0.0.0.3 range 60.1.1.1 255.255.0.0 cost 30",
+# "exit",
+# "passive-interface default",
+# "no router-id",
+# "no fips restrictions",
+# "no timers pacing flood 55",
+# "exit"
+# ],
+
+
+# using overridden
+
+# before state
+
+# veos#show running-config | section ospfv3
+# router ospfv3
+# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=
+# !
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+
+ - arista.eos.eos_ospfv3:
+ config:
+ processes:
+ - address_family:
+ - areas:
+ - area_id: "0.0.0.3"
+ ranges:
+ - address: 10.1.2.2/24
+ advertise: True
+ - address: 60.1.1.1
+ subnet_mask: 255.255.0.0
+ cost: 30
+ afi: "ipv6"
+ passive_interface: True
+ vrf: "vrfmerge"
+ state: overridden
+
+# After state
+
+# veos#show running-config | section ospfv3
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+
+
+# Module execution
+
+# "after": {
+# "processes": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "before": {
+# "processes": [
+# {
+# "areas": [
+# {
+# "area_id": "0.0.0.0",
+# "encryption": {
+# "algorithm": "md5",
+# "encryption": "null",
+# "hidden_key": true,
+# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4="
+# }
+# }
+# ],
+# "vrf": "default"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "no router ospfv3",
+# "router ospfv3 vrf vrfmerge",
+# "address-family ipv6",
+# "no area 0.0.0.3 range 10.1.2.0/24",
+# "no area 0.0.0.3 range 60.1.0.0/16 cost 30",
+# "area 0.0.0.3 range 10.1.2.2/24 advertise",
+# "area 0.0.0.3 range 60.1.1.1 255.255.0.0 cost 30",
+# "exit",
+# "exit"
+# ],
+
+# using deleted
+
+# Before state
+
+# veos#show running-config | section ospfv3
+# router ospfv3
+# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=
+# !
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv4
+# redistribute connected
+# redistribute static route-map MAP01
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+
+ - arista.eos.eos_ospfv3:
+ config:
+ processes:
+ - vrf: "default"
+ state: deleted
+
+# After state
+
+# veos#show running-config | section ospfv3
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv4
+# redistribute connected
+# redistribute static route-map MAP01
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+
+# Module execution
+# "after": {
+# "processes": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ],
+# "redistribute": [
+# {
+# "routes": "connected"
+# },
+# {
+# "route_map": "MAP01",
+# "routes": "static"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "before": {
+# "processes": [
+# {
+# "areas": [
+# {
+# "area_id": "0.0.0.0",
+# "encryption": {
+# "algorithm": "md5",
+# "encryption": "null",
+# "hidden_key": true,
+# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4="
+# }
+# }
+# ],
+# "vrf": "default"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ],
+# "redistribute": [
+# {
+# "routes": "connected"
+# },
+# {
+# "route_map": "MAP01",
+# "routes": "static"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+# },
+# "changed": true,
+# "commands": [
+# "no router ospfv3"
+# ],
+
+# using parsed
+
+# parsed_ospfv3.cfg
+
+# router ospfv3
+# fips restrictions
+# area 0.0.0.20 stub
+# area 0.0.0.20 authentication ipsec spi 33 sha1 passphrase 7 4O8T3zo4xBdRWXBnsnK934o9SEb+jEhHUN6+xzZgCo2j9EnQBUvtwNxxLEmYmm6w
+# area 0.0.0.40 default-cost 45
+# area 0.0.0.40 stub
+# timers pacing flood 7
+# adjacency exchange-start threshold 11
+# !
+# address-family ipv4
+# fips restrictions
+# redistribute connected
+# !
+# address-family ipv6
+# router-id 10.1.1.1
+# fips restrictions
+# !
+# router ospfv3 vrf vrf01
+# bfd all-interfaces
+# fips restrictions
+# area 0.0.0.0 encryption ipsec spi 256 esp null sha1 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ==
+# log-adjacency-changes detail
+# !
+# address-family ipv4
+# passive-interface default
+# fips restrictions
+# redistribute connected route-map MAP01
+# maximum-paths 100
+# !
+# address-family ipv6
+# fips restrictions
+# area 0.0.0.10 nssa no-summary
+# default-information originate route-map DefaultRouteFilter
+# max-metric router-lsa external-lsa 25 summary-lsa
+# !
+# router ospfv3 vrf vrf02
+# fips restrictions
+# !
+# address-family ipv6
+# router-id 10.17.0.3
+# distance ospf intra-area 200
+# fips restrictions
+# area 0.0.0.1 stub
+# timers throttle spf 56 56 56
+# timers out-delay 10
+
+
+ - arista.eos.eos_ospfv3:
+ running_config: "{{ lookup('file', './parsed_ospfv3.cfg') }}"
+ state: parsed
+
+# Module execution
+
+# "parsed": {
+# "processes": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "fips_restrictions": true,
+# "redistribute": [
+# {
+# "routes": "connected"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "fips_restrictions": true,
+# "router_id": "10.1.1.1"
+# }
+# ],
+# "adjacency": {
+# "exchange_start": {
+# "threshold": 11
+# }
+# },
+# "areas": [
+# {
+# "area_id": "0.0.0.20",
+# "authentication": {
+# "algorithm": "sha1",
+# "hidden_key": true,
+# "passphrase": "4O8T3zo4xBdRWXBnsnK934o9SEb+jEhHUN6+xzZgCo2j9EnQBUvtwNxxLEmYmm6w",
+# "spi": 33
+# },
+# "stub": {
+# "set": true
+# }
+# },
+# {
+# "area_id": "0.0.0.40",
+# "default_cost": 45,
+# "stub": {
+# "set": true
+# }
+# }
+# ],
+# "fips_restrictions": true,
+# "timers": {
+# "pacing": 7
+# },
+# "vrf": "default"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "fips_restrictions": true,
+# "maximum_paths": 100,
+# "passive_interface": true,
+# "redistribute": [
+# {
+# "route_map": "MAP01",
+# "routes": "connected"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.10",
+# "nssa": {
+# "no_summary": true
+# }
+# }
+# ],
+# "default_information": {
+# "originate": true,
+# "route_map": "DefaultRouteFilter"
+# },
+# "fips_restrictions": true,
+# "max_metric": {
+# "router_lsa": {
+# "external_lsa": {
+# "max_metric_value": 25
+# },
+# "summary_lsa": {
+# "set": true
+# }
+# }
+# }
+# }
+# ],
+# "areas": [
+# {
+# "area_id": "0.0.0.0",
+# "encryption": {
+# "algorithm": "sha1",
+# "encryption": "null",
+# "hidden_key": true,
+# "passphrase": "7hl8FV3lZ6H1mAKpjL47hQ=="
+# }
+# }
+# ],
+# "bfd": {
+# "all_interfaces": true
+# },
+# "fips_restrictions": true,
+# "log_adjacency_changes": {
+# "detail": true
+# },
+# "vrf": "vrf01"
+# },
+# {
+# "address_family": [
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.1",
+# "stub": {
+# "set": true
+# }
+# }
+# ],
+# "distance": 200,
+# "fips_restrictions": true,
+# "router_id": "10.17.0.3",
+# "timers": {
+# "out_delay": 10,
+# "throttle": {
+# "initial": 56,
+# "max": 56,
+# "min": 56,
+# "spf": true
+# }
+# }
+# }
+# ],
+# "fips_restrictions": true,
+# "vrf": "vrf02"
+# }
+# ]
+
+# using gathered
+
+# native config
+
+# veos#show running-config | section ospfv3
+# router ospfv3 vrf vrfmerge
+# passive-interface default
+# !
+# address-family ipv4
+# redistribute connected
+# redistribute static route-map MAP01
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# !
+# address-family ipv6
+# area 0.0.0.3 range 10.1.2.0/24
+# area 0.0.0.3 range 60.1.0.0/16 cost 30
+# veos#
+
+
+ - arista.eos.eos_ospfv3:
+ state: gathered
+
+# module execution
+
+# "gathered": {
+# "processes": [
+# {
+# "address_family": [
+# {
+# "afi": "ipv4",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ],
+# "redistribute": [
+# {
+# "routes": "connected"
+# },
+# {
+# "route_map": "MAP01",
+# "routes": "static"
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "areas": [
+# {
+# "area_id": "0.0.0.3",
+# "ranges": [
+# {
+# "address": "10.1.2.0/24"
+# },
+# {
+# "address": "60.1.0.0/16",
+# "cost": 30
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "passive_interface": true,
+# "vrf": "vrfmerge"
+# }
+# ]
+
+# using rendered
+
+ - arista.eos.eos_ospfv3:
+ config:
+ processes:
+ - address_family:
+ - timers:
+ lsa: 22
+ graceful_restart:
+ grace_period: 35
+ afi: "ipv6"
+ timers:
+ pacing: 55
+ fips_restrictions: True
+ router_id: "2.2.2.2"
+ vrf: "vrfmerge"
+ state: rendered
+
+# module execution
+
+# "rendered": [
+# "router ospfv3 vrf vrfmerge",
+# "address-family ipv6",
+# "graceful-restart grace-period 35",
+# "timers lsa arrival 22",
+# "exit",
+# "timers pacing flood 55",
+# "fips restrictions",
+# "router-id 2.2.2.2",
+# "exit"
+# ]
+
+
+"""
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv3.ospfv3 import (
+ Ospfv3Args,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospfv3.ospfv3 import (
+ Ospfv3,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ module = AnsibleModule(
+ argument_spec=Ospfv3Args.argument_spec,
+ mutually_exclusive=[["config", "running_config"]],
+ required_if=[
+ ["state", "merged", ["config"]],
+ ["state", "replaced", ["config"]],
+ ["state", "overridden", ["config"]],
+ ["state", "rendered", ["config"]],
+ ["state", "parsed", ["running_config"]],
+ ],
+ supports_check_mode=False,
+ )
+
+ result = Ospfv3(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py
new file mode 100644
index 00000000..da6753a5
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py
@@ -0,0 +1,372 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# Copyright: (c) 2017, Ansible by Red Hat, inc
+# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_static_route
+author: Trishna Guha (@trishnaguha)
+short_description: (deprecated, removed after 2022-06-01) Manage static IP
+ routes on Arista EOS network devices
+description:
+- This module provides declarative management of static IP routes on Arista EOS network
+ devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_static_routes
+ why: Updated modules with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ address:
+ description:
+ - Network address with prefix of the static route.
+ aliases:
+ - prefix
+ type: str
+ next_hop:
+ description:
+ - Next hop IP of the static route.
+ type: str
+ vrf:
+ description:
+ - VRF for static route.
+ default: default
+ type: str
+ admin_distance:
+ description:
+ - Admin distance of the static route.
+ default: 1
+ type: int
+ aggregate:
+ description: List of static route definitions
+ type: list
+ elements: dict
+ suboptions:
+ address:
+ description:
+ - Network address with prefix of the static route.
+ aliases:
+ - prefix
+ type: str
+ required: True
+ next_hop:
+ description:
+ - Next hop IP of the static route.
+ type: str
+ vrf:
+ description:
+ - VRF for static route.
+ default: default
+ type: str
+ admin_distance:
+ description:
+ - Admin distance of the static route.
+ type: int
+ state:
+ description:
+ - State of the static route configuration.
+ choices:
+ - present
+ - absent
+ type: str
+ state:
+ description:
+ - State of the static route configuration.
+ default: present
+ choices:
+ - present
+ - absent
+ type: str
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: configure static route
+ arista.eos.eos_static_route:
+ address: 10.0.2.0/24
+ next_hop: 10.8.38.1
+ admin_distance: 2
+- name: delete static route
+ arista.eos.eos_static_route:
+ address: 10.0.2.0/24
+ next_hop: 10.8.38.1
+ state: absent
+- name: configure static routes using aggregate
+ arista.eos.eos_static_route:
+ aggregate:
+ - {address: 10.0.1.0/24, next_hop: 10.8.38.1}
+ - {address: 10.0.3.0/24, next_hop: 10.8.38.1}
+- name: Delete static route using aggregate
+ arista.eos.eos_static_route:
+ aggregate:
+ - {address: 10.0.1.0/24, next_hop: 10.8.38.1}
+ - {address: 10.0.3.0/24, next_hop: 10.8.38.1}
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - ip route 10.0.2.0/24 10.8.38.1 3
+ - no ip route 10.0.2.0/24 10.8.38.1
+"""
+
+import re
+
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ is_masklen,
+ validate_ip_address,
+)
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+ validate_prefix,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def is_address(value):
+ if value:
+ address = value.split("/")
+ if is_masklen(address[1]) and validate_ip_address(address[0]):
+ return True
+ return False
+
+
+def is_hop(value):
+ if value:
+ if validate_ip_address(value):
+ return True
+ return False
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+
+ for w in want:
+ address = w["address"]
+ next_hop = w["next_hop"]
+ admin_distance = w["admin_distance"]
+ vrf = w["vrf"]
+ state = w["state"]
+ del w["state"]
+
+ if state == "absent" and w in have:
+ if vrf == "default":
+ commands.append("no ip route %s %s" % (address, next_hop))
+ else:
+ commands.append(
+ "no ip route vrf %s %s %s" % (vrf, address, next_hop)
+ )
+ elif state == "present" and w not in have:
+ if vrf == "default":
+ commands.append(
+ "ip route %s %s %d" % (address, next_hop, admin_distance)
+ )
+ else:
+ commands.append(
+ "ip route vrf %s %s %s %d"
+ % (vrf, address, next_hop, admin_distance)
+ )
+
+ return commands
+
+
+def map_params_to_obj(module, required_together=None):
+ obj = []
+
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ module._check_required_together(required_together, item)
+ d = item.copy()
+
+ obj.append(d)
+ else:
+ obj.append(
+ {
+ "address": module.params["address"].strip(),
+ "next_hop": module.params["next_hop"].strip(),
+ "admin_distance": module.params["admin_distance"],
+ "state": module.params["state"],
+ "vrf": module.params["vrf"],
+ }
+ )
+
+ return obj
+
+
+def map_config_to_obj(module):
+ objs = []
+
+ try:
+ out = get_config(module, flags=["| include ip.route"])
+ except IndexError:
+ out = ""
+ if out:
+ lines = out.splitlines()
+ for line in lines:
+ obj = {}
+ add_match = re.search(r"ip route ([\d\./]+)", line, re.M)
+ if add_match:
+ obj["vrf"] = "default"
+ address = add_match.group(1)
+ if is_address(address):
+ obj["address"] = address
+ hop_match = re.search(
+ r"ip route {0} ([\d\./]+)".format(address), line, re.M
+ )
+ if hop_match:
+ hop = hop_match.group(1)
+ if is_hop(hop):
+ obj["next_hop"] = hop
+ dist_match = re.search(
+ r"ip route {0} {1} (\d+)".format(address, hop),
+ line,
+ re.M,
+ )
+ if dist_match:
+ distance = dist_match.group(1)
+ obj["admin_distance"] = int(distance)
+ else:
+ obj["admin_distance"] = 1
+
+ vrf_match = re.search(
+ r"ip route vrf ([\w]+) ([\d\./]+)", line, re.M
+ )
+ if vrf_match:
+ vrf = vrf_match.group(1)
+ obj["vrf"] = vrf
+ address = vrf_match.group(2)
+ if is_address(address):
+ obj["address"] = address
+ hop_vrf_match = re.search(
+ r"ip route vrf {0} {1} ([\d\./]+)".format(vrf, address),
+ line,
+ re.M,
+ )
+ if hop_vrf_match:
+ hop = hop_vrf_match.group(1)
+ if is_hop(hop):
+ obj["next_hop"] = hop
+ dist_vrf_match = re.search(
+ r"ip route vrf {0} {1} {2} (\d+)".format(
+ vrf, address, hop
+ ),
+ line,
+ re.M,
+ )
+ if dist_vrf_match:
+ distance = dist_vrf_match.group(1)
+ obj["admin_distance"] = int(distance)
+ else:
+ obj["admin_distance"] = 1
+
+ objs.append(obj)
+
+ return objs
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ address=dict(type="str", aliases=["prefix"]),
+ next_hop=dict(type="str"),
+ vrf=dict(type="str", default="default"),
+ admin_distance=dict(default=1, type="int"),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["address"] = dict(required=True, aliases=["prefix"])
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ aggregate_spec["vrf"].update(default="default")
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec)
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["aggregate", "address"]]
+ required_together = [["address", "next_hop"]]
+ mutually_exclusive = [["aggregate", "address"]]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_one_of=required_one_of,
+ required_together=required_together,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ address = module.params["address"]
+ if address is not None:
+ prefix = address.split("/")[-1]
+
+ if address and prefix:
+ if "/" not in address or not validate_ip_address(
+ address.split("/")[0]
+ ):
+ module.fail_json(
+ msg="{0} is not a valid IP address".format(address)
+ )
+
+ if not validate_prefix(prefix):
+ module.fail_json(
+ msg="Length of prefix should be between 0 and 32 bits"
+ )
+
+ warnings = list()
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py
new file mode 100644
index 00000000..516847a3
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py
@@ -0,0 +1,967 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_static_routes
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_static_routes
+short_description: Static routes resource module
+description: This module configures and manages the attributes of static routes on
+ Arista EOS platforms.
+version_added: 1.0.0
+author: Gomathi Selvi Srinivasan (@GomathiselviS)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description:
+ - A list of configurations for static routes.
+ type: list
+ elements: dict
+ suboptions:
+ vrf:
+ description:
+ - The VRF to which the static route(s) belong.
+ type: str
+ address_families:
+ description: A dictionary specifying the address family to which the static
+ route(s) belong.
+ type: list
+ elements: dict
+ suboptions:
+ afi:
+ description:
+ - Specifies the top level address family indicator.
+ type: str
+ choices:
+ - ipv4
+ - ipv6
+ required: true
+ routes:
+ description: A dictionary that specifies the static route configurations.
+ elements: dict
+ type: list
+ suboptions:
+ dest:
+ description:
+ - Destination IPv4 subnet (CIDR or address-mask notation).
+ - The address format is <v4/v6 address>/<mask> or <v4/v6 address>
+ <mask>.
+ - The mask is number in range 0-32 for IPv4 and in range 0-128 for
+ IPv6.
+ type: str
+ required: true
+ next_hops:
+ description:
+ - Details of route to be taken.
+ type: list
+ elements: dict
+ suboptions:
+ forward_router_address:
+ description:
+ - Forwarding router's address on destination interface.
+ type: str
+ interface:
+ description:
+ - Outgoing interface to take. For anything except 'null0', then
+ next hop IP address should also be configured.
+ - IP address of the next hop router or
+ - null0 Null0 interface or
+ - ethernet e_num Ethernet interface or
+ - loopback l_num Loopback interface or
+ - management m_num Management interface or
+ - port-channel p_num
+ - vlan v_num
+ - vxlan vx_num
+ - Nexthop-Group Specify nexthop group name
+ - Tunnel Tunnel interface
+ - vtep Configure VXLAN Tunnel End Points
+ type: str
+ nexthop_grp:
+ description:
+ - Nexthop group
+ type: str
+ admin_distance:
+ description:
+ - Preference or administrative distance of route (range 1-255).
+ type: int
+ description:
+ description:
+ - Name of the static route.
+ type: str
+ tag:
+ description:
+ - Route tag value (ranges from 0 to 4294967295).
+ type: int
+ track:
+ description:
+ - Track value (range 1 - 512). Track must already be configured
+ on the device before adding the route.
+ type: str
+ mpls_label:
+ description:
+ - MPLS label
+ type: int
+ vrf:
+ description:
+ - VRF of the destination.
+ type: str
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device by
+ executing the command B(show running-config | grep routes).
+ - The state I(parsed) reads the configuration from C(running_config) option and
+ transforms it into Ansible structured data as per the resource module's argspec
+ and the value is then returned in the I(parsed) key within the result.
+ type: str
+ state:
+ description:
+ - The state the configuration should be left in.
+ type: str
+ choices:
+ - deleted
+ - merged
+ - overridden
+ - replaced
+ - gathered
+ - rendered
+ - parsed
+ default: merged
+
+"""
+EXAMPLES = """
+# Using deleted
+
+# Before State:
+# ------------
+
+# veos(config)#show running-config | grep route
+# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute
+# ipv6 route 5222:5::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55
+# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1
+# veos(config)#
+
+- name: Delete afi
+ arista.eos.eos_static_routes:
+ config:
+ - vrf: testvrf
+ address_families:
+ - afi: ipv4
+ state: deleted
+
+# "after": [
+# {
+# "address_families": [
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5222:5::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "2222:6::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# },
+# {
+# "admin_distance": 55,
+# "interface": "Ethernet1"
+# },
+# {
+# "admin_distance": 90,
+# "description": "testroute1",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ],
+# "before": [
+# {
+# "address_families": [
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5222:5::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "22.65.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 90,
+# "description": "testroute",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "2222:6::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# },
+# {
+# "admin_distance": 55,
+# "interface": "Ethernet1"
+# },
+# {
+# "admin_distance": 90,
+# "description": "testroute1",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ],
+# "changed": true,
+# "commands": [
+# "no ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute"
+# ],
+
+# After State
+# ___________
+
+# veos(config)#show running-config | grep route
+# ipv6 route 5222:5::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55
+# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1
+
+#
+# Using merged
+
+# Before : [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "165.10.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 100,
+# "interface": "Ethernet1"
+# }
+# ]
+# },
+# {
+# "dest": "172.17.252.0/24",
+# "next_hops": [
+# {
+# "nexthop_grp": "testgroup"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5001::/64",
+# "next_hops": [
+# {
+# "admin_distance": 50,
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "130.1.122.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1",
+# "tag": 50
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ]
+#
+# Before State
+# -------------
+# veos(config)#show running-config | grep "route"
+# ip route 165.10.1.0/24 Ethernet1 100
+# ip route 172.17.252.0/24 Nexthop-Group testgroup
+# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50
+# ipv6 route 5001::/64 Ethernet1 50
+# veos(config)#
+
+- name: Merge new static route configuration
+ arista.eos.eos_static_routes:
+ config:
+ - vrf: testvrf
+ address_families:
+ - afi: ipv6
+ routes:
+ - dest: 2211::0/64
+ next_hop:
+ - forward_router_address: 100:1::2
+ interface: Ethernet1
+ state: merged
+
+# After State
+# -----------
+
+#After [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "165.10.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 100,
+# "interface": "Ethernet1"
+# }
+# ]
+# },
+# {
+# "dest": "172.17.252.0/24",
+# "next_hops": [
+# {
+# "nexthop_grp": "testgroup"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5001::/64",
+# "next_hops": [
+# {
+# "admin_distance": 50,
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "130.1.122.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1",
+# "tag": 50
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "2211::0/64",
+# "next_hops": [
+# {
+# "aforward_router_address": 100:1::2
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+
+# ],
+# "vrf": "testvrf"
+# }
+# ]
+#
+# veos(config)#show running-config | grep "route"
+# ip route 165.10.1.0/24 Ethernet1 100
+# ip route 172.17.252.0/24 Nexthop-Group testgroup
+# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50
+# ipv6 route 2211::/64 Ethernet1 100:1::2
+# ipv6 route 5001::/64 Ethernet1 50
+# veos(config)#
+
+
+# Using overridden
+
+
+# Before State
+# -------------
+
+# "before": [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "165.10.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 100,
+# "interface": "Ethernet1"
+# }
+# ]
+# },
+# {
+# "dest": "172.17.252.0/24",
+# "next_hops": [
+# {
+# "nexthop_grp": "testgroup"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5001::/64",
+# "next_hops": [
+# {
+# "admin_distance": 50,
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "130.1.122.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1",
+# "tag": 50
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ]
+# veos(config)#show running-config | grep "route"
+# ip route 165.10.1.0/24 Ethernet1 100
+# ip route 172.17.252.0/24 Nexthop-Group testgroup
+# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50
+# ipv6 route 5001::/64 Ethernet1 50
+# veos(config)#
+
+- name: Overridden static route configuration
+ arista.eos.eos_static_routes:
+ config:
+ - address_families:
+ - afi: ipv4
+ routes:
+ - dest: 10.2.2.0/24
+ next_hop:
+ - interface: Ethernet1
+ state: replaced
+
+# After State
+# -----------
+
+# "after": [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "10.2.2.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# veos(config)#show running-config | grep "route"
+# ip route 10.2.2.0/24 Ethernet1
+# veos(config)#
+
+
+# Using replaced
+
+# Before State
+# -------------
+
+# ip route 10.2.2.0/24 Ethernet1
+# ip route 10.2.2.0/24 64.1.1.1 label 17 33
+# ip route 33.33.33.0/24 Nexthop-Group testgrp
+# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute
+# ipv6 route 5222:5::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55
+# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1
+
+# [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "10.2.2.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1"
+# },
+# {
+# "admin_distance": 33,
+# "interface": "64.1.1.1",
+# "mpls_label": 17
+# }
+# ]
+# },
+# {
+# "dest": "33.33.33.0/24",
+# "next_hops": [
+# {
+# "nexthop_grp": "testgrp"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5222:5::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "22.65.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 90,
+# "description": "testroute",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "2222:6::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# },
+# {
+# "admin_distance": 90,
+# "description": "testroute1",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ]
+
+- name: Replace nexthop
+ arista.eos.eos_static_routes:
+ config:
+ - vrf: testvrf
+ address_families:
+ - afi: ipv6
+ routes:
+ - dest: 2222:6::/64
+ next_hops:
+ - admin_distance: 55
+ interface: Ethernet1
+ state: replaced
+
+# After State
+# -----------
+
+# veos(config)#show running-config | grep route
+# ip route 10.2.2.0/24 Ethernet1
+# ip route 10.2.2.0/24 64.1.1.1 label 17 33
+# ip route 33.33.33.0/24 Nexthop-Group testgrp
+# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute
+# ipv6 route 5222:5::/64 Management1 4312:100::1
+# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55
+
+# "after": [
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "10.2.2.0/24",
+# "next_hops": [
+# {
+# "interface": "Ethernet1"
+# },
+# {
+# "admin_distance": 33,
+# "interface": "64.1.1.1",
+# "mpls_label": 17
+# }
+# ]
+# },
+# {
+# "dest": "33.33.33.0/24",
+# "next_hops": [
+# {
+# "nexthop_grp": "testgrp"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "5222:5::/64",
+# "next_hops": [
+# {
+# "forward_router_address": "4312:100::1",
+# "interface": "Management1"
+# }
+# ]
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "address_families": [
+# {
+# "afi": "ipv4",
+# "routes": [
+# {
+# "dest": "22.65.1.0/24",
+# "next_hops": [
+# {
+# "admin_distance": 90,
+# "description": "testroute",
+# "interface": "Null0"
+# }
+# ]
+# }
+# ]
+# },
+# {
+# "afi": "ipv6",
+# "routes": [
+# {
+# "dest": "2222:6::/64",
+# "next_hops": [
+# {
+# "admin_distance": 55,
+# "interface": "Ethernet1"
+# }
+# ]
+# }
+# ]
+# }
+# ],
+# "vrf": "testvrf"
+# }
+# ]
+
+# Before State
+# -------------
+# veos(config)#show running-config | grep "route"
+# ip route 165.10.1.0/24 Ethernet1 10.1.1.2 100
+# ipv6 route 5001::/64 Ethernet1
+# veos(config)#
+
+
+- name: Gather the exisitng condiguration
+ arista.eos.eos_static_routes:
+ state: gathered
+
+# returns :
+# arista.eos.eos_static_routes:
+# config:
+# - address_families:
+# - afi: ipv4
+# routes:
+# - dest: 165.10.1.0/24
+# next_hop:
+# - forward_router_address: 10.1.1.2
+# interface: "Ethernet1"
+# admin_distance: 100
+# - afi: ipv6
+# routes:
+# - dest: 5001::/64
+# next_hop:
+# - interface: "Ethernet1"
+
+
+# Using rendered
+
+# arista.eos.eos_static_routes:
+# config:
+# - address_families:
+# - afi: ipv4
+# routes:
+# - dest: 165.10.1.0/24
+# next_hop:
+# - forward_router_address: 10.1.1.2
+# interface: "Ethernet1"
+# admin_distance: 100
+# - afi: ipv6
+# routes:
+# - dest: 5001::/64
+# next_hop:
+# - interface: "Ethernet1"
+
+# returns:
+
+# ip route 165.10.1.0/24 Ethernet1 10.1.1.2 100
+# ipv6 route 5001::/64 Ethernet1
+
+
+"""
+RETURN = """
+before:
+ description: The configuration prior to the model invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The resulting configuration model invocation.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample:
+ - ip route vrf vrf1 192.2.2.0/24 125.2.3.1 93
+rendered:
+ description: The set of CLI commands generated from the value in C(config) option
+ returned: When C(state) is I(rendered)
+ type: list
+ sample: >
+ "address_families": [
+ {
+ "afi": "ipv4",
+ "routes": [
+ {
+ "dest": "192.2.2.0/24",
+ "next_hops": [
+ {
+ "admin_distance": 93,
+ "description": null,
+ "forward_router_address": null,
+ "interface": "125.2.3.1",
+ "mpls_label": null,
+ "nexthop_grp": null,
+ "tag": null,
+ "track": null,
+ "vrf": null
+ }
+ ]
+ }
+ ]
+ }
+ ],
+ "vrf": "vrf1"
+ }
+ ],
+ "running_config": null,
+ "state": "rendered"
+ }
+gathered:
+ description: The configuration as structured data transformed for the running configuration
+ fetched from remote host
+ returned: When C(state) is I(gathered)
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+parsed:
+ description: The configuration as structured data transformed for the value of
+ C(running_config) option
+ returned: When C(state) is I(parsed)
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.static_routes.static_routes import (
+ Static_routesArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.static_routes.static_routes import (
+ Static_routes,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+
+ module = AnsibleModule(
+ argument_spec=Static_routesArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Static_routes(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py
new file mode 100644
index 00000000..fd010037
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py
@@ -0,0 +1,376 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_system
+author: Peter Sprygada (@privateip)
+short_description: Manage the system attributes on Arista EOS devices
+description:
+- This module provides declarative management of node system attributes on Arista
+ EOS devices. It provides an option to configure host system parameters or remove
+ those parameters from the device active configuration.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+notes:
+- Tested against EOS 4.15
+options:
+ hostname:
+ description:
+ - Configure the device hostname parameter. This option takes an ASCII string value.
+ type: str
+ domain_name:
+ description:
+ - Configure the IP domain name on the remote device to the provided value. Value
+ should be in the dotted name form and will be appended to the C(hostname) to
+ create a fully-qualified domain name.
+ type: str
+ domain_list:
+ description:
+ - Provides the list of domain suffixes to append to the hostname for the purpose
+ of doing name resolution. This argument accepts a list of names and will be
+ reconciled with the current active configuration on the running node.
+ aliases:
+ - domain_search
+ type: list
+ elements: str
+ lookup_source:
+ description:
+ - Provides one or more source interfaces to use for performing DNS lookups. The
+ interface provided in C(lookup_source) can only exist in a single VRF. This
+ argument accepts either a list of interface names or a list of hashes that configure
+ the interface name and VRF name. See examples.
+ elements: raw
+ type: list
+ name_servers:
+ description:
+ - List of DNS name servers by IP address to use to perform name resolution lookups. This
+ argument accepts either a list of DNS servers or a list of hashes that configure
+ the name server and VRF name. See examples.
+ type: list
+ elements: str
+ state:
+ description:
+ - State of the configuration values in the device's current active configuration. When
+ set to I(present), the values should be configured in the device active configuration
+ and when set to I(absent) the values should not be in the device active configuration
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+"""
+
+EXAMPLES = """
+- name: configure hostname and domain-name
+ arista.eos.eos_system:
+ hostname: eos01
+ domain_name: test.example.com
+
+- name: remove configuration
+ arista.eos.eos_system:
+ state: absent
+
+- name: configure DNS lookup sources
+ arista.eos.eos_system:
+ lookup_source: Management1
+
+- name: configure DNS lookup sources with VRF support
+ arista.eos.eos_system:
+ lookup_source:
+ - interface: Management1
+ vrf: mgmt
+ - interface: Ethernet1
+ vrf: myvrf
+
+- name: configure name servers
+ arista.eos.eos_system:
+ name_servers:
+ - 8.8.8.8
+ - 8.8.4.4
+
+- name: configure name servers with VRF support
+ arista.eos.eos_system:
+ name_servers:
+ - {server: 8.8.8.8, vrf: mgmt}
+ - {server: 8.8.4.4, vrf: mgmt}
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - hostname eos01
+ - ip domain-name test.example.com
+session_name:
+ description: The EOS config session name used to load the configuration
+ returned: changed
+ type: str
+ sample: ansible_1479315771
+"""
+import re
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ ComplexList,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ load_config,
+ get_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+_CONFIGURED_VRFS = None
+
+
+def has_vrf(module, vrf):
+ global _CONFIGURED_VRFS
+ if _CONFIGURED_VRFS is not None:
+ return vrf in _CONFIGURED_VRFS
+ config = get_config(module)
+ _CONFIGURED_VRFS = re.findall(r"vrf definition (\S+)", config)
+ _CONFIGURED_VRFS.append("default")
+ return vrf in _CONFIGURED_VRFS
+
+
+def map_obj_to_commands(want, have, module):
+ commands = list()
+ state = module.params["state"]
+
+ def needs_update(x):
+ return want.get(x) and (want.get(x) != have.get(x))
+
+ if state == "absent":
+ if have["domain_name"]:
+ commands.append("no ip domain-name")
+
+ if have["hostname"] != "localhost":
+ commands.append("no hostname")
+
+ if state == "present":
+ if needs_update("hostname"):
+ commands.append("hostname %s" % want["hostname"])
+
+ if needs_update("domain_name"):
+ commands.append("ip domain-name %s" % want["domain_name"])
+
+ if want["domain_list"]:
+ # handle domain_list items to be removed
+ for item in set(have["domain_list"]).difference(
+ want["domain_list"]
+ ):
+ commands.append("no ip domain-list %s" % item)
+
+ # handle domain_list items to be added
+ for item in set(want["domain_list"]).difference(
+ have["domain_list"]
+ ):
+ commands.append("ip domain-list %s" % item)
+
+ if want["lookup_source"]:
+ # handle lookup_source items to be removed
+ for item in have["lookup_source"]:
+ if item not in want["lookup_source"]:
+ if item["vrf"]:
+ if not has_vrf(module, item["vrf"]):
+ module.fail_json(
+ msg="vrf %s is not configured" % item["vrf"]
+ )
+ values = (item["vrf"], item["interface"])
+ commands.append(
+ "no ip domain lookup vrf %s source-interface %s"
+ % values
+ )
+ else:
+ commands.append(
+ "no ip domain lookup source-interface %s"
+ % item["interface"]
+ )
+
+ # handle lookup_source items to be added
+ for item in want["lookup_source"]:
+ if item not in have["lookup_source"]:
+ if item["vrf"]:
+ if not has_vrf(module, item["vrf"]):
+ module.fail_json(
+ msg="vrf %s is not configured" % item["vrf"]
+ )
+ values = (item["vrf"], item["interface"])
+ commands.append(
+ "ip domain lookup vrf %s source-interface %s"
+ % values
+ )
+ else:
+ commands.append(
+ "ip domain lookup source-interface %s"
+ % item["interface"]
+ )
+
+ if want["name_servers"]:
+ # handle name_servers items to be removed. Order does matter here
+ # since name servers can only be in one vrf at a time
+ for item in have["name_servers"]:
+ if item not in want["name_servers"]:
+ if not has_vrf(module, item["vrf"]):
+ module.fail_json(
+ msg="vrf %s is not configured" % item["vrf"]
+ )
+ if item["vrf"] not in ("default", None):
+ values = (item["vrf"], item["server"])
+ commands.append("no ip name-server vrf %s %s" % values)
+ else:
+ commands.append(
+ "no ip name-server %s" % item["server"]
+ )
+
+ # handle name_servers items to be added
+ for item in want["name_servers"]:
+ if item not in have["name_servers"]:
+ if not has_vrf(module, item["vrf"]):
+ module.fail_json(
+ msg="vrf %s is not configured" % item["vrf"]
+ )
+ if item["vrf"] not in ("default", None):
+ values = (item["vrf"], item["server"])
+ commands.append("ip name-server vrf %s %s" % values)
+ else:
+ commands.append("ip name-server %s" % item["server"])
+
+ return commands
+
+
+def parse_hostname(config):
+ match = re.search(r"^hostname (\S+)", config, re.M)
+ if match:
+ return match.group(1)
+
+
+def parse_domain_name(config):
+ match = re.search(r"^ip domain-name (\S+)", config, re.M)
+ if match:
+ return match.group(1)
+
+
+def parse_lookup_source(config):
+ objects = list()
+ regex = r"ip domain lookup (?:vrf (\S+) )*source-interface (\S+)"
+ for vrf, intf in re.findall(regex, config, re.M):
+ if len(vrf) == 0:
+ vrf = None
+ objects.append({"interface": intf, "vrf": vrf})
+ return objects
+
+
+def parse_name_servers(config):
+ objects = list()
+ for vrf, addr in re.findall(
+ r"ip name-server vrf (\S+) (\S+)", config, re.M
+ ):
+ objects.append({"server": addr, "vrf": vrf})
+ return objects
+
+
+def map_config_to_obj(module):
+ config = get_config(module)
+ return {
+ "hostname": parse_hostname(config),
+ "domain_name": parse_domain_name(config),
+ "domain_list": re.findall(r"^ip domain-list (\S+)", config, re.M),
+ "lookup_source": parse_lookup_source(config),
+ "name_servers": parse_name_servers(config),
+ }
+
+
+def map_params_to_obj(module):
+ obj = {
+ "hostname": module.params["hostname"],
+ "domain_name": module.params["domain_name"],
+ "domain_list": module.params["domain_list"],
+ }
+
+ lookup_source = ComplexList(
+ dict(interface=dict(key=True), vrf=dict()), module
+ )
+
+ name_servers = ComplexList(
+ dict(server=dict(key=True), vrf=dict(default="default")), module
+ )
+
+ for arg, cast in [
+ ("lookup_source", lookup_source),
+ ("name_servers", name_servers),
+ ]:
+ if module.params[arg] is not None:
+ obj[arg] = cast(module.params[arg])
+ else:
+ obj[arg] = None
+
+ return obj
+
+
+def main():
+ """ main entry point for module execution
+ """
+ argument_spec = dict(
+ hostname=dict(),
+ domain_name=dict(),
+ domain_list=dict(
+ type="list", aliases=["domain_search"], elements="str"
+ ),
+ # { interface: <str>, vrf: <str> }
+ lookup_source=dict(type="list", elements="raw"),
+ # { server: <str>; vrf: <str> }
+ name_servers=dict(type="list", elements="str"),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ argument_spec.update(eos_argument_spec)
+
+ module = AnsibleModule(
+ argument_spec=argument_spec, supports_check_mode=True
+ )
+
+ result = {"changed": False}
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands(want, have, module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py
new file mode 100644
index 00000000..a14331ae
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py
@@ -0,0 +1,477 @@
+#!/usr/bin/python
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_user
+author: Peter Sprygada (@privateip)
+short_description: Manage the collection of local users on EOS devices
+description:
+- This module provides declarative management of the local usernames configured on
+ Arista EOS devices. It allows playbooks to manage either individual usernames or
+ the collection of usernames in the current running config. It also supports purging
+ usernames from the configuration that are not explicitly defined.
+version_added: 1.0.0
+extends_documentation_fragment:
+- arista.eos.eos
+notes:
+- Tested against EOS 4.15
+options:
+ aggregate:
+ description:
+ - The set of username objects to be configured on the remote Arista EOS device. The
+ list entries can either be the username or a hash of username and properties. This
+ argument is mutually exclusive with the C(username) argument.
+ aliases:
+ - users
+ - collection
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - The username to be configured on the remote Arista EOS device. This argument
+ accepts a stringv value and is mutually exclusive with the C(aggregate) argument.
+ Please note that this option is not same as C(provider username).
+ type: str
+ configured_password:
+ description:
+ - The password to be configured on the remote Arista EOS device. The password
+ needs to be provided in clear and it will be encrypted on the device. Please
+ note that this option is not same as C(provider password).
+ type: str
+ update_password:
+ description:
+ - Since passwords are encrypted in the device running config, this argument will
+ instruct the module when to change the password. When set to C(always), the
+ password will always be updated in the device and when set to C(on_create) the
+ password will be updated only if the username is created.
+ type: str
+ choices:
+ - on_create
+ - always
+ privilege:
+ description:
+ - The C(privilege) argument configures the privilege level of the user when logged
+ into the system. This argument accepts integer values in the range of 1 to
+ 15.
+ type: int
+ role:
+ description:
+ - Configures the role for the username in the device running configuration. The
+ argument accepts a string value defining the role name. This argument does
+ not check if the role has been configured on the device.
+ type: str
+ sshkey:
+ description:
+ - Specifies the SSH public key to configure for the given username. This argument
+ accepts a valid SSH key value.
+ type: str
+ nopassword:
+ description:
+ - Defines the username without assigning a password. This will allow the user
+ to login to the system without being authenticated by a password.
+ type: bool
+ state:
+ description:
+ - Configures the state of the username definition as it relates to the device
+ operational configuration. When set to I(present), the username(s) should be
+ configured in the device active configuration and when set to I(absent) the
+ username(s) should not be in the device active configuration
+ type: str
+ choices:
+ - present
+ - absent
+ name:
+ description:
+ - The username to be configured on the remote Arista EOS device. This argument
+ accepts a stringv value and is mutually exclusive with the C(aggregate) argument.
+ Please note that this option is not same as C(provider username).
+ type: str
+ configured_password:
+ description:
+ - The password to be configured on the remote Arista EOS device. The password
+ needs to be provided in clear and it will be encrypted on the device. Please
+ note that this option is not same as C(provider password).
+ type: str
+ update_password:
+ description:
+ - Since passwords are encrypted in the device running config, this argument will
+ instruct the module when to change the password. When set to C(always), the
+ password will always be updated in the device and when set to C(on_create) the
+ password will be updated only if the username is created.
+ default: always
+ type: str
+ choices:
+ - on_create
+ - always
+ privilege:
+ description:
+ - The C(privilege) argument configures the privilege level of the user when logged
+ into the system. This argument accepts integer values in the range of 1 to
+ 15.
+ type: int
+ role:
+ description:
+ - Configures the role for the username in the device running configuration. The
+ argument accepts a string value defining the role name. This argument does
+ not check if the role has been configured on the device.
+ type: str
+ sshkey:
+ description:
+ - Specifies the SSH public key to configure for the given username. This argument
+ accepts a valid SSH key value.
+ type: str
+ nopassword:
+ description:
+ - Defines the username without assigning a password. This will allow the user
+ to login to the system without being authenticated by a password.
+ type: bool
+ purge:
+ description:
+ - Instructs the module to consider the resource definition absolute. It will
+ remove any previously configured usernames on the device with the exception
+ of the `admin` user which cannot be deleted per EOS constraints.
+ type: bool
+ default: false
+ state:
+ description:
+ - Configures the state of the username definition as it relates to the device
+ operational configuration. When set to I(present), the username(s) should be
+ configured in the device active configuration and when set to I(absent) the
+ username(s) should not be in the device active configuration
+ type: str
+ default: present
+ choices:
+ - present
+ - absent
+"""
+
+EXAMPLES = """
+- name: create a new user
+ arista.eos.eos_user:
+ name: ansible
+ sshkey: "{{ lookup('file', '~/.ssh/id_rsa.pub') }}"
+ state: present
+
+- name: remove all users except admin
+ arista.eos.eos_user:
+ purge: yes
+
+- name: set multiple users to privilege level 15
+ arista.eos.eos_user:
+ aggregate:
+ - name: netop
+ - name: netend
+ privilege: 15
+ state: present
+
+- name: Change Password for User netop
+ arista.eos.eos_user:
+ username: netop
+ configured_password: '{{ new_password }}'
+ update_password: always
+ state: present
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - name ansible secret password
+ - name admin secret admin
+session_name:
+ description: The EOS config session name used to load the configuration
+ returned: when changed is True
+ type: str
+ sample: ansible_1479315771
+"""
+
+import re
+
+from copy import deepcopy
+from functools import partial
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ get_config,
+ load_config,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+from ansible.module_utils.six import iteritems
+
+
+def validate_privilege(value, module):
+ if not 1 <= value <= 15:
+ module.fail_json(
+ msg="privilege must be between 1 and 15, got %s" % value
+ )
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ update_password = module.params["update_password"]
+
+ for update in updates:
+ want, have = update
+
+ def needs_update(x):
+ return want.get(x) and (want.get(x) != have.get(x))
+
+ def add(x):
+ return commands.append("username %s %s" % (want["name"], x))
+
+ if want["state"] == "absent":
+ commands.append("no username %s" % want["name"])
+ continue
+
+ if needs_update("configured_password"):
+ if update_password == "always" or not have:
+ add("secret %s" % want["configured_password"])
+
+ if needs_update("role"):
+ add("role %s" % want["role"])
+
+ if needs_update("privilege"):
+ add("privilege %s" % want["privilege"])
+
+ if needs_update("sshkey"):
+ add("sshkey %s" % want["sshkey"])
+
+ if needs_update("nopassword"):
+ if want["nopassword"]:
+ add("nopassword")
+ else:
+ add("no username %s nopassword" % want["name"])
+
+ if want.get("state") == "present" and want.get("name"):
+ value = [
+ want.get("configured_password"),
+ want.get("nopassword"),
+ want.get("sshkey"),
+ ]
+ if all(v is None for v in value) is True:
+ module.fail_json(
+ msg="configured_password, sshkey or nopassword should be provided"
+ )
+
+ return commands
+
+
+def parse_role(data):
+ match = re.search(r"role (\S+)", data, re.M)
+ if match:
+ return match.group(1)
+
+
+def parse_sshkey(data):
+ match = re.search(r"sshkey (.+)$", data, re.M)
+ if match:
+ return match.group(1)
+
+
+def parse_privilege(data):
+ match = re.search(r"privilege (\S+)", data, re.M)
+ if match:
+ return int(match.group(1))
+
+
+def map_config_to_obj(module):
+ data = get_config(module, flags=["section username"])
+
+ match = re.findall(r"^username (\S+)", data, re.M)
+ if not match:
+ return list()
+
+ instances = list()
+
+ for user in set(match):
+ regex = r"username %s .+$" % user
+ cfg = re.findall(regex, data, re.M)
+ cfg = "\n".join(cfg)
+ obj = {
+ "name": user,
+ "state": "present",
+ "nopassword": "nopassword" in cfg,
+ "configured_password": None,
+ "sshkey": parse_sshkey(cfg),
+ "privilege": parse_privilege(cfg),
+ "role": parse_role(cfg),
+ }
+ instances.append(obj)
+
+ return instances
+
+
+def get_param_value(key, item, module):
+ # if key doesn't exist in the item, get it from module.params
+ if not item.get(key):
+ value = module.params[key]
+
+ # if key does exist, do a type check on it to validate it
+ else:
+ value_type = module.argument_spec[key].get("type", "str")
+ type_checker = module._CHECK_ARGUMENT_TYPES_DISPATCHER[value_type]
+ type_checker(item[key])
+ value = item[key]
+
+ # validate the param value (if validator func exists)
+ validator = globals().get("validate_%s" % key)
+ if all((value, validator)):
+ validator(value, module)
+
+ return value
+
+
+def map_params_to_obj(module):
+ aggregate = module.params["aggregate"]
+ if not aggregate:
+ if not module.params["name"] and module.params["purge"]:
+ return list()
+ elif not module.params["name"]:
+ module.fail_json(msg="name is required")
+ else:
+ collection = [{"name": module.params["name"]}]
+ else:
+ collection = list()
+ for item in aggregate:
+ if not isinstance(item, dict):
+ collection.append({"name": item})
+ elif "name" not in item:
+ module.fail_json(msg="name is required")
+ else:
+ collection.append(item)
+
+ objects = list()
+
+ for item in collection:
+ get_value = partial(get_param_value, item=item, module=module)
+ item["configured_password"] = get_value("configured_password")
+ item["nopassword"] = get_value("nopassword")
+ item["privilege"] = get_value("privilege")
+ item["role"] = get_value("role")
+ item["sshkey"] = get_value("sshkey")
+ item["state"] = get_value("state")
+ objects.append(item)
+
+ return objects
+
+
+def update_objects(want, have):
+ updates = list()
+ for entry in want:
+ if "name" in entry:
+ item = next((i for i in have if i["name"] == entry["name"]), None)
+ if all((item is None, entry["state"] == "present")):
+ updates.append((entry, {}))
+ elif item:
+ for key, value in iteritems(entry):
+ if value and value != item[key]:
+ updates.append((entry, item))
+ return updates
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ name=dict(),
+ configured_password=dict(no_log=True),
+ nopassword=dict(type="bool"),
+ update_password=dict(
+ default="always", choices=["on_create", "always"]
+ ),
+ privilege=dict(type="int"),
+ role=dict(),
+ sshkey=dict(),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ argument_spec = dict(
+ aggregate=dict(
+ type="list",
+ elements="dict",
+ options=aggregate_spec,
+ aliases=["collection", "users"],
+ ),
+ purge=dict(type="bool", default=False),
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+ mutually_exclusive = [("name", "aggregate")]
+
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands(update_objects(want, have), module)
+
+ if module.params["purge"]:
+ want_users = [x["name"] for x in want]
+ have_users = [x["name"] for x in have]
+ for item in set(have_users).difference(want_users):
+ if item != "admin":
+ commands.append("no username %s" % item)
+
+ result["commands"] = commands
+
+ # the eos cli prevents this by rule so capture it and display
+ # a nice failure message
+ if "no username admin" in commands:
+ module.fail_json(msg="cannot delete the `admin` account")
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py
new file mode 100644
index 00000000..ea7d28f8
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py
@@ -0,0 +1,463 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# (c) 2017, Ansible by Red Hat, inc
+#
+# This file is part of Ansible by Red Hat
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+DOCUMENTATION = """
+module: eos_vlan
+author: Ricardo Carrillo Cruz (@rcarrillocruz)
+short_description: (deprecated, removed after 2022-06-01) Manage VLANs on Arista EOS
+ network devices
+description:
+- This module provides declarative management of VLANs on Arista EOS network devices.
+version_added: 1.0.0
+deprecated:
+ alternative: eos_vlans
+ why: Updated modules released with more functionality
+ removed_at_date: '2022-06-01'
+notes:
+- Tested against EOS 4.15
+options:
+ name:
+ description:
+ - Name of the VLAN.
+ type: str
+ vlan_id:
+ description:
+ - ID of the VLAN.
+ type: int
+ interfaces:
+ description:
+ - List of interfaces that should be associated to the VLAN. The name of interface
+ is case sensitive and should be in expanded format and not abbreviated.
+ type: list
+ elements: str
+ associated_interfaces:
+ description:
+ - This is a intent option and checks the operational state of the for given vlan
+ C(name) for associated interfaces. The name of interface is case sensitive and
+ should be in expanded format and not abbreviated. If the value in the C(associated_interfaces)
+ does not match with the operational state of vlan interfaces on device it will
+ result in failure.
+ type: list
+ elements: str
+ delay:
+ description:
+ - Delay the play should wait to check for declarative intent params values.
+ default: 10
+ type: int
+ aggregate:
+ description: List of VLANs definitions.
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the VLAN.
+ type: str
+ vlan_id:
+ description:
+ - ID of the VLAN.
+ required: true
+ type: int
+ interfaces:
+ description:
+ - List of interfaces that should be associated to the VLAN. The name of interface
+ is case sensitive and should be in expanded format and not abbreviated.
+ type: list
+ elements: str
+ associated_interfaces:
+ description:
+ - This is a intent option and checks the operational state of the for given vlan
+ C(name) for associated interfaces. The name of interface is case sensitive and
+ should be in expanded format and not abbreviated. If the value in the C(associated_interfaces)
+ does not match with the operational state of vlan interfaces on device it will
+ result in failure.
+ type: list
+ elements: str
+ delay:
+ description:
+ - Delay the play should wait to check for declarative intent params values.
+ default: 10
+ type: int
+ state:
+ description:
+ - State of the VLAN configuration.
+ type: str
+ default: present
+ choices:
+ - present
+ - absent
+ - active
+ - suspend
+ purge:
+ description:
+ - Purge VLANs not defined in the I(aggregate) parameter.
+ default: false
+ type: bool
+ state:
+ description:
+ - State of the VLAN configuration.
+ type: str
+ default: present
+ choices:
+ - present
+ - absent
+ - active
+ - suspend
+extends_documentation_fragment:
+- arista.eos.eos
+
+"""
+
+EXAMPLES = """
+- name: Create vlan
+ arista.eos.eos_vlan:
+ vlan_id: 4000
+ name: vlan-4000
+ state: present
+
+- name: Add interfaces to vlan
+ arista.eos.eos_vlan:
+ vlan_id: 4000
+ state: present
+ interfaces:
+ - Ethernet1
+ - Ethernet2
+
+- name: Check if interfaces is assigned to vlan
+ arista.eos.eos_vlan:
+ vlan_id: 4000
+ associated_interfaces:
+ - Ethernet1
+ - Ethernet2
+
+- name: Suspend vlan
+ arista.eos.eos_vlan:
+ vlan_id: 4000
+ state: suspend
+
+- name: Unsuspend vlan
+ arista.eos.eos_vlan:
+ vlan_id: 4000
+ state: active
+
+- name: Create aggregate of vlans
+ arista.eos.eos_vlan:
+ aggregate:
+ - vlan_id: 4000
+ - {vlan_id: 4001, name: vlan-4001}
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - vlan 20
+ - name test-vlan
+"""
+import time
+
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ load_config,
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def search_obj_in_list(vlan_id, lst):
+ for o in lst:
+ if o["vlan_id"] == vlan_id:
+ return o
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+ purge = module.params["purge"]
+
+ for w in want:
+ vlan_id = w["vlan_id"]
+ state = w["state"]
+
+ obj_in_have = search_obj_in_list(vlan_id, have)
+
+ if state == "absent":
+ if obj_in_have:
+ commands.append("no vlan %s" % w["vlan_id"])
+ elif state == "present":
+ if not obj_in_have:
+ commands.append("vlan %s" % w["vlan_id"])
+ if w["name"]:
+ commands.append("name %s" % w["name"])
+
+ if w["interfaces"]:
+ for i in w["interfaces"]:
+ commands.append("interface %s" % i)
+ commands.append(
+ "switchport access vlan %s" % w["vlan_id"]
+ )
+ else:
+ if w["name"] and w["name"] != obj_in_have["name"]:
+ commands.append("vlan %s" % w["vlan_id"])
+ commands.append("name %s" % w["name"])
+
+ if w["interfaces"]:
+ if not obj_in_have["interfaces"]:
+ for i in w["interfaces"]:
+ commands.append("vlan %s" % w["vlan_id"])
+ commands.append("interface %s" % i)
+ commands.append(
+ "switchport access vlan %s" % w["vlan_id"]
+ )
+ elif set(w["interfaces"]) != obj_in_have["interfaces"]:
+ missing_interfaces = list(
+ set(w["interfaces"])
+ - set(obj_in_have["interfaces"])
+ )
+ for i in missing_interfaces:
+ commands.append("vlan %s" % w["vlan_id"])
+ commands.append("interface %s" % i)
+ commands.append(
+ "switchport access vlan %s" % w["vlan_id"]
+ )
+
+ superfluous_interfaces = list(
+ set(obj_in_have["interfaces"])
+ - set(w["interfaces"])
+ )
+ for i in superfluous_interfaces:
+ commands.append("vlan %s" % w["vlan_id"])
+ commands.append("interface %s" % i)
+ commands.append(
+ "no switchport access vlan %s" % w["vlan_id"]
+ )
+ else:
+ if not obj_in_have:
+ commands.append("vlan %s" % w["vlan_id"])
+ if w["name"]:
+ commands.append("name %s" % w["name"])
+ commands.append("state %s" % w["state"])
+ elif (
+ w["name"] and obj_in_have["name"] != w["name"]
+ ) or obj_in_have["state"] != w["state"]:
+ commands.append("vlan %s" % w["vlan_id"])
+
+ if w["name"]:
+ if obj_in_have["name"] != w["name"]:
+ commands.append("name %s" % w["name"])
+
+ if obj_in_have["state"] != w["state"]:
+ commands.append("state %s" % w["state"])
+
+ if purge:
+ for h in have:
+ obj_in_want = search_obj_in_list(h["vlan_id"], want)
+ if not obj_in_want and h["vlan_id"] != "1":
+ commands.append("no vlan %s" % h["vlan_id"])
+
+ return commands
+
+
+def map_config_to_obj(module):
+ objs = []
+ vlans = run_commands(module, ["show vlan configured-ports | json"])
+
+ for vlan in vlans[0]["vlans"]:
+ obj = {}
+ obj["vlan_id"] = vlan
+ obj["name"] = vlans[0]["vlans"][vlan]["name"]
+ obj["state"] = vlans[0]["vlans"][vlan]["status"]
+ obj["interfaces"] = []
+
+ interfaces = vlans[0]["vlans"][vlan]
+
+ for interface in interfaces["interfaces"]:
+ obj["interfaces"].append(interface)
+
+ if obj["state"] == "suspended":
+ obj["state"] = "suspend"
+
+ objs.append(obj)
+
+ return objs
+
+
+def map_params_to_obj(module):
+ obj = []
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ if item.get("interfaces"):
+ item["interfaces"] = [
+ intf.replace(" ", "")
+ for intf in item.get("interfaces")
+ if intf
+ ]
+
+ if item.get("associated_interfaces"):
+ item["associated_interfaces"] = [
+ intf.replace(" ", "")
+ for intf in item.get("associated_interfaces")
+ if intf
+ ]
+
+ d = item.copy()
+ d["vlan_id"] = str(d["vlan_id"])
+
+ obj.append(d)
+ else:
+ obj.append(
+ {
+ "vlan_id": str(module.params["vlan_id"]),
+ "name": module.params["name"],
+ "state": module.params["state"],
+ "interfaces": [
+ intf.replace(" ", "")
+ for intf in module.params["interfaces"]
+ ]
+ if module.params["interfaces"]
+ else [],
+ "associated_interfaces": [
+ intf.replace(" ", "")
+ for intf in module.params["associated_interfaces"]
+ ]
+ if module.params["associated_interfaces"]
+ else [],
+ }
+ )
+
+ return obj
+
+
+def check_declarative_intent_params(want, module, result):
+ have = None
+ is_delay = False
+
+ for w in want:
+ if w.get("associated_interfaces") is None:
+ continue
+
+ if result["changed"] and not is_delay:
+ time.sleep(module.params["delay"])
+ is_delay = True
+
+ if have is None:
+ have = map_config_to_obj(module)
+
+ for i in w["associated_interfaces"]:
+ obj_in_have = search_obj_in_list(w["vlan_id"], have)
+
+ if (
+ obj_in_have
+ and "interfaces" in obj_in_have
+ and i not in obj_in_have["interfaces"]
+ ):
+ module.fail_json(
+ msg="Interface %s not configured on vlan %s"
+ % (i, w["vlan_id"])
+ )
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ vlan_id=dict(type="int"),
+ name=dict(),
+ interfaces=dict(type="list", elements="str"),
+ associated_interfaces=dict(type="list", elements="str"),
+ delay=dict(default=10, type="int"),
+ state=dict(
+ default="present",
+ choices=["present", "absent", "active", "suspend"],
+ ),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["vlan_id"] = dict(required=True, type="int")
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+
+ aggregate_spec["state"].update(default="present")
+ aggregate_spec["delay"].update(default=10)
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec),
+ purge=dict(default=False, type="bool"),
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["vlan_id", "aggregate"]]
+ mutually_exclusive = [["vlan_id", "aggregate"]]
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ supports_check_mode=True,
+ required_one_of=required_one_of,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ check_declarative_intent_params(want, module, result)
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py
new file mode 100644
index 00000000..8ee01fab
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py
@@ -0,0 +1,327 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+# Copyright 2019 Red Hat
+# GNU General Public License v3.0+
+# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
+
+#############################################
+# WARNING #
+#############################################
+#
+# This file is auto generated by the resource
+# module builder playbook.
+#
+# Do not edit this file manually.
+#
+# Changes to this file will be over written
+# by the resource module builder.
+#
+# Changes should be made in the model used to
+# generate this file or in the resource module
+# builder template.
+#
+#############################################
+
+"""
+The module file for eos_vlans
+"""
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+
+DOCUMENTATION = """
+module: eos_vlans
+short_description: VLANs resource module
+description: This module provides declarative management of VLANs on Arista EOS network
+ devices.
+version_added: 1.0.0
+author: Nathaniel Case (@qalthos)
+notes:
+- Tested against Arista EOS 4.20.10M
+- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html).
+options:
+ config:
+ description: A dictionary of VLANs options
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the VLAN.
+ type: str
+ vlan_id:
+ description:
+ - ID of the VLAN. Range 1-4094
+ type: int
+ required: true
+ state:
+ description:
+ - Operational state of the VLAN
+ type: str
+ choices:
+ - active
+ - suspend
+ running_config:
+ description:
+ - This option is used only with state I(parsed).
+ - The value of this option should be the output received from the EOS device
+ by executing the command B(show running-config | section vlan).
+ - The state I(parsed) reads the configuration from C(running_config) option
+ and transforms it into Ansible structured data as per the resource module's
+ argspec and the value
+ type: str
+ state:
+ description:
+ - The state of the configuration after module completion
+ type: str
+ choices:
+ - merged
+ - replaced
+ - overridden
+ - deleted
+ - rendered
+ - gathered
+ - parsed
+ default: merged
+
+"""
+EXAMPLES = """
+# Using deleted
+
+# Before state:
+# -------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+
+- name: Delete attributes of the given VLANs.
+ arista.eos.eos_vlans:
+ config:
+ - vlan_id: 20
+ state: deleted
+
+# After state:
+# ------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+
+
+# Using merged
+
+# Before state:
+# -------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+
+- name: Merge given VLAN attributes with device configuration
+ arista.eos.eos_vlans:
+ config:
+ - vlan_id: 20
+ state: suspend
+ state: merged
+
+# After state:
+# ------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+# state suspend
+
+
+# Using overridden
+
+# Before state:
+# -------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+
+- name: Override device configuration of all VLANs with provided configuration
+ arista.eos.eos_vlans:
+ config:
+ - vlan_id: 20
+ state: suspend
+ state: overridden
+
+# After state:
+# ------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 20
+# state suspend
+
+
+# Using replaced
+
+# Before state:
+# -------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+
+- name: Replace all attributes of specified VLANs with provided configuration
+ arista.eos.eos_vlans:
+ config:
+ - vlan_id: 20
+ state: suspend
+ state: replaced
+
+# After state:
+# ------------
+#
+# veos(config-vlan-20)#show running-config | section vlan
+# vlan 10
+# name ten
+# !
+# vlan 20
+# state suspend
+
+# using parsed
+
+# parsed.cfg
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+# state suspend
+
+- name: Use parsed to convert native configs to structured data
+ arista.eos.eos_vlans:
+ running_config: "{{ lookup('file', 'parsed.cfg') }}"
+ state: parsed
+
+# Output:
+# -------
+# parsed:
+# - vlan_id: 10
+# name: ten
+# - vlan_id: 20
+# state: suspend
+
+# Using rendered:
+
+- name: Use Rendered to convert the structured data to native config
+ arista.eos.eos_vlans:
+ config:
+ - vlan_id: 10
+ name: ten
+ - vlan_id: 20
+ state: suspend
+ state: rendered
+
+# Output:
+# ------
+# rendered:
+# - "vlan 10"
+# - "name ten"
+# - "vlan 20"
+# - "state suspend"
+
+# Using gathered:
+# native_config:
+# vlan 10
+# name ten
+# !
+# vlan 20
+# name twenty
+# state suspend
+
+- name: Gather vlans facts from the device
+ arista.eos.eos_vlans:
+ state: gathered
+
+# Output:
+# ------
+
+# gathered:
+# - vlan_id: 10
+# name: ten
+# - vlan_id: 20
+# state: suspend
+
+"""
+RETURN = """
+before:
+ description: The configuration as structured data prior to module invocation.
+ returned: always
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+after:
+ description: The configuration as structured data after module completion.
+ returned: when changed
+ type: list
+ sample: >
+ The configuration returned will always be in the same format
+ of the parameters above.
+commands:
+ description: The set of commands pushed to the remote device.
+ returned: always
+ type: list
+ sample: ['vlan 10', 'no name', 'vlan 11', 'name Eleven']
+"""
+
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.vlans.vlans import (
+ VlansArgs,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.vlans.vlans import (
+ Vlans,
+)
+
+
+def main():
+ """
+ Main entry point for module execution
+
+ :returns: the result form module invocation
+ """
+ required_if = [
+ ("state", "merged", ("config",)),
+ ("state", "replaced", ("config",)),
+ ("state", "overridden", ("config",)),
+ ("state", "rendered", ("config",)),
+ ("state", "parsed", ("running_config",)),
+ ]
+ mutually_exclusive = [("config", "running_config")]
+ module = AnsibleModule(
+ argument_spec=VlansArgs.argument_spec,
+ required_if=required_if,
+ supports_check_mode=True,
+ mutually_exclusive=mutually_exclusive,
+ )
+
+ result = Vlans(module).execute_module()
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py
new file mode 100644
index 00000000..23356bf7
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py
@@ -0,0 +1,432 @@
+#!/usr/bin/python
+# -*- coding: utf-8 -*-
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+# (c) 2017, Ansible by Red Hat, inc
+#
+# This file is part of Ansible by Red Hat
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+
+
+DOCUMENTATION = """
+module: eos_vrf
+author: Ricardo Carrillo Cruz (@rcarrillocruz)
+short_description: Manage VRFs on Arista EOS network devices
+description:
+- This module provides declarative management of VRFs on Arista EOS network devices.
+version_added: 1.0.0
+notes:
+- Tested against EOS 4.15
+options:
+ name:
+ description:
+ - Name of the VRF.
+ type: str
+ rd:
+ description:
+ - Route distinguisher of the VRF
+ type: str
+ interfaces:
+ description:
+ - Identifies the set of interfaces that should be configured in the VRF. Interfaces
+ must be routed interfaces in order to be placed into a VRF. The name of interface
+ should be in expanded format and not abbreviated.
+ type: list
+ elements: str
+ associated_interfaces:
+ description:
+ - This is a intent option and checks the operational state of the for given vrf
+ C(name) for associated interfaces. If the value in the C(associated_interfaces)
+ does not match with the operational state of vrf interfaces on device it will
+ result in failure.
+ type: list
+ elements: str
+ aggregate:
+ description: List of VRFs definitions
+ type: list
+ elements: dict
+ suboptions:
+ name:
+ description:
+ - Name of the VRF.
+ required: true
+ type: str
+ rd:
+ description:
+ - Route distinguisher of the VRF
+ type: str
+ interfaces:
+ description:
+ - Identifies the set of interfaces that should be configured in the VRF. Interfaces
+ must be routed interfaces in order to be placed into a VRF. The name of interface
+ should be in expanded format and not abbreviated.
+ type: list
+ elements: str
+ associated_interfaces:
+ description:
+ - This is a intent option and checks the operational state of the for given vrf
+ C(name) for associated interfaces. If the value in the C(associated_interfaces)
+ does not match with the operational state of vrf interfaces on device it will
+ result in failure.
+ type: list
+ elements: str
+ delay:
+ description:
+ - Time in seconds to wait before checking for the operational state on remote
+ device. This wait is applicable for operational state arguments.
+ default: 10
+ type: int
+ state:
+ description:
+ - State of the VRF configuration.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+ purge:
+ description:
+ - Purge VRFs not defined in the I(aggregate) parameter.
+ default: false
+ type: bool
+ delay:
+ description:
+ - Time in seconds to wait before checking for the operational state on remote
+ device. This wait is applicable for operational state arguments.
+ default: 10
+ type: int
+ state:
+ description:
+ - State of the VRF configuration.
+ default: present
+ type: str
+ choices:
+ - present
+ - absent
+extends_documentation_fragment:
+- arista.eos.eos
+"""
+
+EXAMPLES = """
+- name: Create vrf
+ arista.eos.eos_vrf:
+ name: test
+ rd: 1:200
+ interfaces:
+ - Ethernet2
+ state: present
+
+- name: Delete VRFs
+ arista.eos.eos_vrf:
+ name: test
+ state: absent
+
+- name: Create aggregate of VRFs with purge
+ arista.eos.eos_vrf:
+ aggregate:
+ - name: test4
+ rd: 1:204
+ - name: test5
+ rd: 1:205
+ state: present
+ purge: yes
+
+- name: Delete aggregate of VRFs
+ arista.eos.eos_vrf:
+ aggregate:
+ - name: test2
+ - name: test3
+ - name: test4
+ - name: test5
+ state: absent
+"""
+
+RETURN = """
+commands:
+ description: The list of configuration mode commands to send to the device
+ returned: always
+ type: list
+ sample:
+ - vrf definition test
+ - rd 1:100
+ - interface Ethernet1
+ - vrf forwarding test
+"""
+import re
+import time
+
+from copy import deepcopy
+
+from ansible.module_utils.basic import AnsibleModule
+from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import (
+ remove_default_spec,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ load_config,
+ run_commands,
+)
+from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import (
+ eos_argument_spec,
+)
+
+
+def search_obj_in_list(name, lst):
+ for o in lst:
+ if o["name"] == name:
+ return o
+
+
+def map_obj_to_commands(updates, module):
+ commands = list()
+ want, have = updates
+ state = module.params["state"]
+ purge = module.params["purge"]
+
+ for w in want:
+ name = w["name"]
+ rd = w["rd"]
+
+ obj_in_have = search_obj_in_list(name, have)
+
+ if state == "absent":
+ if obj_in_have:
+ commands.append("no vrf definition %s" % name)
+ elif state == "present":
+ if not obj_in_have:
+ commands.append("vrf definition %s" % name)
+
+ if rd is not None:
+ commands.append("rd %s" % rd)
+
+ if w["interfaces"]:
+ for i in w["interfaces"]:
+ commands.append("interface %s" % i)
+ commands.append("vrf forwarding %s" % w["name"])
+ else:
+ if w["rd"] is not None and w["rd"] != obj_in_have["rd"]:
+ commands.append("vrf definition %s" % w["name"])
+ commands.append("rd %s" % w["rd"])
+
+ if w["interfaces"]:
+ if not obj_in_have["interfaces"]:
+ for i in w["interfaces"]:
+ commands.append("interface %s" % i)
+ commands.append("vrf forwarding %s" % w["name"])
+ elif set(w["interfaces"]) != obj_in_have["interfaces"]:
+ missing_interfaces = list(
+ set(w["interfaces"])
+ - set(obj_in_have["interfaces"])
+ )
+
+ for i in missing_interfaces:
+ commands.append("interface %s" % i)
+ commands.append("vrf forwarding %s" % w["name"])
+
+ if purge:
+ for h in have:
+ obj_in_want = search_obj_in_list(h["name"], want)
+ if not obj_in_want:
+ commands.append("no vrf definition %s" % h["name"])
+
+ return commands
+
+
+def map_config_to_obj(module):
+ objs = []
+ output = run_commands(module, {"command": "show vrf", "output": "text"})
+
+ lines = output[0].strip().splitlines()[3:]
+
+ out_len = len(lines)
+ index = 0
+ while out_len > index:
+ line = lines[index]
+ if not line:
+ continue
+
+ splitted_line = re.split(r"\s{2,}", line.strip())
+
+ if len(splitted_line) == 1:
+ index += 1
+ continue
+ obj = dict()
+ obj["name"] = splitted_line[0]
+ obj["rd"] = splitted_line[1]
+ obj["interfaces"] = []
+
+ if len(splitted_line) > 4:
+ obj["interfaces"] = []
+ interfaces = splitted_line[4]
+ if interfaces.endswith(","):
+ while interfaces.endswith(","):
+ # gather all comma separated interfaces
+ if out_len <= index:
+ break
+ index += 1
+ line = lines[index]
+ vrf_line = re.split(r"\s{2,}", line.strip())
+ interfaces += vrf_line[-1]
+
+ for i in interfaces.split(","):
+ obj["interfaces"].append(i.strip().lower())
+ index += 1
+ objs.append(obj)
+
+ return objs
+
+
+def map_params_to_obj(module):
+ obj = []
+ aggregate = module.params.get("aggregate")
+ if aggregate:
+ for item in aggregate:
+ for key in item:
+ if item.get(key) is None:
+ item[key] = module.params[key]
+
+ if item.get("interfaces"):
+ item["interfaces"] = [
+ intf.replace(" ", "").lower()
+ for intf in item.get("interfaces")
+ if intf
+ ]
+
+ if item.get("associated_interfaces"):
+ item["associated_interfaces"] = [
+ intf.replace(" ", "").lower()
+ for intf in item.get("associated_interfaces")
+ if intf
+ ]
+
+ obj.append(item.copy())
+ else:
+ obj.append(
+ {
+ "name": module.params["name"],
+ "state": module.params["state"],
+ "rd": module.params["rd"],
+ "interfaces": [
+ intf.replace(" ", "").lower()
+ for intf in module.params["interfaces"]
+ ]
+ if module.params["interfaces"]
+ else [],
+ "associated_interfaces": [
+ intf.replace(" ", "").lower()
+ for intf in module.params["associated_interfaces"]
+ ]
+ if module.params["associated_interfaces"]
+ else [],
+ }
+ )
+
+ return obj
+
+
+def check_declarative_intent_params(want, module, result):
+ have = None
+ is_delay = False
+
+ for w in want:
+ if w.get("associated_interfaces") is None:
+ continue
+
+ if result["changed"] and not is_delay:
+ time.sleep(module.params["delay"])
+ is_delay = True
+
+ if have is None:
+ have = map_config_to_obj(module)
+
+ for i in w["associated_interfaces"]:
+ obj_in_have = search_obj_in_list(w["name"], have)
+
+ if obj_in_have:
+ interfaces = obj_in_have.get("interfaces")
+ if interfaces is not None and i not in interfaces:
+ module.fail_json(
+ msg="Interface %s not configured on vrf %s"
+ % (i, w["name"])
+ )
+
+
+def main():
+ """ main entry point for module execution
+ """
+ element_spec = dict(
+ name=dict(),
+ interfaces=dict(type="list", elements="str"),
+ associated_interfaces=dict(type="list", elements="str"),
+ delay=dict(default=10, type="int"),
+ rd=dict(),
+ state=dict(default="present", choices=["present", "absent"]),
+ )
+
+ aggregate_spec = deepcopy(element_spec)
+ aggregate_spec["name"] = dict(required=True)
+
+ # remove default in aggregate spec, to handle common arguments
+ remove_default_spec(aggregate_spec)
+ aggregate_spec["state"].update(default="present")
+ aggregate_spec["delay"].update(default=10)
+
+ argument_spec = dict(
+ aggregate=dict(type="list", elements="dict", options=aggregate_spec),
+ purge=dict(default=False, type="bool"),
+ )
+
+ argument_spec.update(element_spec)
+ argument_spec.update(eos_argument_spec)
+
+ required_one_of = [["name", "aggregate"]]
+ mutually_exclusive = [["name", "aggregate"]]
+ module = AnsibleModule(
+ argument_spec=argument_spec,
+ required_one_of=required_one_of,
+ mutually_exclusive=mutually_exclusive,
+ supports_check_mode=True,
+ )
+
+ warnings = list()
+
+ result = {"changed": False}
+
+ if warnings:
+ result["warnings"] = warnings
+
+ want = map_params_to_obj(module)
+ have = map_config_to_obj(module)
+
+ commands = map_obj_to_commands((want, have), module)
+ result["commands"] = commands
+
+ if commands:
+ commit = not module.check_mode
+ response = load_config(module, commands, commit=commit)
+ if response.get("diff") and module._diff:
+ result["diff"] = {"prepared": response.get("diff")}
+ result["session_name"] = response.get("session")
+ result["changed"] = True
+
+ check_declarative_intent_params(want, module, result)
+
+ module.exit_json(**result)
+
+
+if __name__ == "__main__":
+ main()
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py
new file mode 100644
index 00000000..ee575bc0
--- /dev/null
+++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py
@@ -0,0 +1,108 @@
+#
+# (c) 2016 Red Hat Inc.
+#
+# This file is part of Ansible
+#
+# Ansible is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+#
+# Ansible is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
+#
+
+from __future__ import absolute_import, division, print_function
+
+__metaclass__ = type
+
+import re
+import json
+
+from ansible.plugins.terminal import TerminalBase
+from ansible.errors import AnsibleConnectionFailure
+from ansible.module_utils._text import to_bytes, to_text
+
+
+class TerminalModule(TerminalBase):
+
+ terminal_stdout_re = [
+ re.compile(br"[\r\n]?[\w+\-\.:\/\[\]]+(?:\([^\)]+\)){,3}(?:>|#) ?$"),
+ re.compile(br"\[\w+\@[\w\-\.]+(?: [^\]])\] ?[>#\$] ?$"),
+ ]
+
+ terminal_stderr_re = [
+ re.compile(br"% ?Error"),
+ # re.compile(br"^% \w+", re.M),
+ re.compile(br"% User not present"),
+ re.compile(br"% ?Bad secret"),
+ re.compile(br"invalid input", re.I),
+ re.compile(br"(?:incomplete|ambiguous) command", re.I),
+ re.compile(br"connection timed out", re.I),
+ # Strings like this regarding VLANs are not errors
+ re.compile(br"[^\r\n]+ not found(?! in current VLAN)", re.I),
+ re.compile(br"'[^']' +returned error code: ?\d+"),
+ re.compile(br"[^\r\n](?<! shell )\/bin\/(?:ba)?sh"),
+ re.compile(br"% More than \d+ OSPF instance", re.I),
+ re.compile(br"% Subnet [0-9a-f.:/]+ overlaps", re.I),
+ re.compile(br"Maximum number of pending sessions has been reached"),
+ re.compile(br"% Prefix length must be less than"),
+ # returned in response to 'channel-group <name> mode <mode>'
+ re.compile(
+ br"% Cannot change mode; remove all members and try again."
+ ),
+ ]
+
+ def on_open_shell(self):
+ try:
+ for cmd in (b"terminal length 0", b"terminal width 512"):
+ self._exec_cli_command(cmd)
+ except AnsibleConnectionFailure:
+ raise AnsibleConnectionFailure("unable to set terminal parameters")
+
+ def on_become(self, passwd=None):
+ if self._get_prompt().endswith(b"#"):
+ return
+
+ cmd = {u"command": u"enable"}
+ if passwd:
+ cmd[u"prompt"] = to_text(
+ r"[\r\n]?password: $", errors="surrogate_or_strict"
+ )
+ cmd[u"answer"] = passwd
+ cmd[u"prompt_retry_check"] = True
+
+ try:
+ self._exec_cli_command(
+ to_bytes(json.dumps(cmd), errors="surrogate_or_strict")
+ )
+ prompt = self._get_prompt()
+ if prompt is None or not prompt.endswith(b"#"):
+ raise AnsibleConnectionFailure(
+ "failed to elevate privilege to enable mode still at prompt [%s]"
+ % prompt
+ )
+ except AnsibleConnectionFailure as e:
+ prompt = self._get_prompt()
+ raise AnsibleConnectionFailure(
+ "unable to elevate privilege to enable mode, at prompt [%s] with error: %s"
+ % (prompt, e.message)
+ )
+
+ def on_unbecome(self):
+ prompt = self._get_prompt()
+ if prompt is None:
+ # if prompt is None most likely the terminal is hung up at a prompt
+ return
+
+ if b"(config" in prompt:
+ self._exec_cli_command(b"end")
+ self._exec_cli_command(b"disable")
+
+ elif prompt.endswith(b"#"):
+ self._exec_cli_command(b"disable")