annet 0.16.15__tar.gz → 0.16.17__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of annet might be problematic. Click here for more details.

Files changed (169) hide show
  1. {annet-0.16.15/annet.egg-info → annet-0.16.17}/PKG-INFO +2 -2
  2. {annet-0.16.15 → annet-0.16.17}/annet/__init__.py +2 -2
  3. {annet-0.16.15 → annet-0.16.17}/annet/bgp_models.py +3 -3
  4. {annet-0.16.15 → annet-0.16.17}/annet/mesh/device_models.py +4 -4
  5. {annet-0.16.15 → annet-0.16.17}/annet/mesh/models_converter.py +0 -3
  6. {annet-0.16.15 → annet-0.16.17}/annet/mesh/peer_models.py +5 -1
  7. annet-0.16.17/annet/rulebook/cisco/misc.py +149 -0
  8. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/cisco.rul +1 -1
  9. {annet-0.16.15 → annet-0.16.17/annet.egg-info}/PKG-INFO +2 -2
  10. {annet-0.16.15 → annet-0.16.17}/setup.py +1 -1
  11. annet-0.16.15/annet/rulebook/cisco/misc.py +0 -57
  12. {annet-0.16.15 → annet-0.16.17}/AUTHORS +0 -0
  13. {annet-0.16.15 → annet-0.16.17}/LICENSE +0 -0
  14. {annet-0.16.15 → annet-0.16.17}/MANIFEST.in +0 -0
  15. {annet-0.16.15 → annet-0.16.17}/README.md +0 -0
  16. {annet-0.16.15 → annet-0.16.17}/annet/adapters/__init__.py +0 -0
  17. {annet-0.16.15 → annet-0.16.17}/annet/adapters/fetchers/__init__.py +0 -0
  18. {annet-0.16.15 → annet-0.16.17}/annet/adapters/fetchers/stub/__init__.py +0 -0
  19. {annet-0.16.15 → annet-0.16.17}/annet/adapters/fetchers/stub/fetcher.py +0 -0
  20. {annet-0.16.15 → annet-0.16.17}/annet/adapters/file/__init__.py +0 -0
  21. {annet-0.16.15 → annet-0.16.17}/annet/adapters/file/provider.py +0 -0
  22. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/__init__.py +0 -0
  23. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/__init__.py +0 -0
  24. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/client.py +0 -0
  25. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/manufacturer.py +0 -0
  26. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/models.py +0 -0
  27. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/query.py +0 -0
  28. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/status_client.py +0 -0
  29. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/common/storage_opts.py +0 -0
  30. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/provider.py +0 -0
  31. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/v24/__init__.py +0 -0
  32. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/v24/storage.py +0 -0
  33. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/v37/__init__.py +0 -0
  34. {annet-0.16.15 → annet-0.16.17}/annet/adapters/netbox/v37/storage.py +0 -0
  35. {annet-0.16.15 → annet-0.16.17}/annet/annet.py +0 -0
  36. {annet-0.16.15 → annet-0.16.17}/annet/annlib/__init__.py +0 -0
  37. {annet-0.16.15 → annet-0.16.17}/annet/annlib/command.py +0 -0
  38. {annet-0.16.15 → annet-0.16.17}/annet/annlib/diff.py +0 -0
  39. {annet-0.16.15 → annet-0.16.17}/annet/annlib/errors.py +0 -0
  40. {annet-0.16.15 → annet-0.16.17}/annet/annlib/filter_acl.py +0 -0
  41. {annet-0.16.15 → annet-0.16.17}/annet/annlib/jsontools.py +0 -0
  42. {annet-0.16.15 → annet-0.16.17}/annet/annlib/lib.py +0 -0
  43. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/__init__.py +0 -0
  44. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/db.py +0 -0
  45. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/devdb/__init__.py +0 -0
  46. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/devdb/data/devdb.json +0 -0
  47. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/views/__init__.py +0 -0
  48. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/views/dump.py +0 -0
  49. {annet-0.16.15 → annet-0.16.17}/annet/annlib/netdev/views/hardware.py +0 -0
  50. {annet-0.16.15 → annet-0.16.17}/annet/annlib/output.py +0 -0
  51. {annet-0.16.15 → annet-0.16.17}/annet/annlib/patching.py +0 -0
  52. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/__init__.py +0 -0
  53. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/acl.py +0 -0
  54. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/deploying.py +0 -0
  55. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/ordering.py +0 -0
  56. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/platform.py +0 -0
  57. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rbparser/syntax.py +0 -0
  58. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rulebook/__init__.py +0 -0
  59. {annet-0.16.15 → annet-0.16.17}/annet/annlib/rulebook/common.py +0 -0
  60. {annet-0.16.15 → annet-0.16.17}/annet/annlib/tabparser.py +0 -0
  61. {annet-0.16.15 → annet-0.16.17}/annet/annlib/types.py +0 -0
  62. {annet-0.16.15 → annet-0.16.17}/annet/api/__init__.py +0 -0
  63. {annet-0.16.15 → annet-0.16.17}/annet/argparse.py +0 -0
  64. {annet-0.16.15 → annet-0.16.17}/annet/cli.py +0 -0
  65. {annet-0.16.15 → annet-0.16.17}/annet/cli_args.py +0 -0
  66. {annet-0.16.15 → annet-0.16.17}/annet/configs/context.yml +0 -0
  67. {annet-0.16.15 → annet-0.16.17}/annet/configs/logging.yaml +0 -0
  68. {annet-0.16.15 → annet-0.16.17}/annet/connectors.py +0 -0
  69. {annet-0.16.15 → annet-0.16.17}/annet/deploy.py +0 -0
  70. {annet-0.16.15 → annet-0.16.17}/annet/diff.py +0 -0
  71. {annet-0.16.15 → annet-0.16.17}/annet/executor.py +0 -0
  72. {annet-0.16.15 → annet-0.16.17}/annet/filtering.py +0 -0
  73. {annet-0.16.15 → annet-0.16.17}/annet/gen.py +0 -0
  74. {annet-0.16.15 → annet-0.16.17}/annet/generators/__init__.py +0 -0
  75. {annet-0.16.15 → annet-0.16.17}/annet/generators/base.py +0 -0
  76. {annet-0.16.15 → annet-0.16.17}/annet/generators/common/__init__.py +0 -0
  77. {annet-0.16.15 → annet-0.16.17}/annet/generators/common/initial.py +0 -0
  78. {annet-0.16.15 → annet-0.16.17}/annet/generators/entire.py +0 -0
  79. {annet-0.16.15 → annet-0.16.17}/annet/generators/exceptions.py +0 -0
  80. {annet-0.16.15 → annet-0.16.17}/annet/generators/jsonfragment.py +0 -0
  81. {annet-0.16.15 → annet-0.16.17}/annet/generators/partial.py +0 -0
  82. {annet-0.16.15 → annet-0.16.17}/annet/generators/perf.py +0 -0
  83. {annet-0.16.15 → annet-0.16.17}/annet/generators/ref.py +0 -0
  84. {annet-0.16.15 → annet-0.16.17}/annet/generators/result.py +0 -0
  85. {annet-0.16.15 → annet-0.16.17}/annet/hardware.py +0 -0
  86. {annet-0.16.15 → annet-0.16.17}/annet/implicit.py +0 -0
  87. {annet-0.16.15 → annet-0.16.17}/annet/lib.py +0 -0
  88. {annet-0.16.15 → annet-0.16.17}/annet/mesh/__init__.py +0 -0
  89. {annet-0.16.15 → annet-0.16.17}/annet/mesh/basemodel.py +0 -0
  90. {annet-0.16.15 → annet-0.16.17}/annet/mesh/executor.py +0 -0
  91. {annet-0.16.15 → annet-0.16.17}/annet/mesh/match_args.py +0 -0
  92. {annet-0.16.15 → annet-0.16.17}/annet/mesh/registry.py +0 -0
  93. {annet-0.16.15 → annet-0.16.17}/annet/output.py +0 -0
  94. {annet-0.16.15 → annet-0.16.17}/annet/parallel.py +0 -0
  95. {annet-0.16.15 → annet-0.16.17}/annet/patching.py +0 -0
  96. {annet-0.16.15 → annet-0.16.17}/annet/reference.py +0 -0
  97. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/__init__.py +0 -0
  98. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/arista/__init__.py +0 -0
  99. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/arista/iface.py +0 -0
  100. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/aruba/__init__.py +0 -0
  101. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/aruba/ap_env.py +0 -0
  102. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/aruba/misc.py +0 -0
  103. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/b4com/__init__.py +0 -0
  104. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/b4com/file.py +0 -0
  105. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/cisco/__init__.py +0 -0
  106. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/cisco/iface.py +0 -0
  107. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/cisco/vlandb.py +0 -0
  108. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/common.py +0 -0
  109. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/deploying.py +0 -0
  110. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/__init__.py +0 -0
  111. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/aaa.py +0 -0
  112. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/bgp.py +0 -0
  113. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/iface.py +0 -0
  114. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/misc.py +0 -0
  115. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/huawei/vlandb.py +0 -0
  116. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/juniper/__init__.py +0 -0
  117. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/nexus/__init__.py +0 -0
  118. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/nexus/iface.py +0 -0
  119. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/patching.py +0 -0
  120. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/ribbon/__init__.py +0 -0
  121. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/routeros/__init__.py +0 -0
  122. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/routeros/file.py +0 -0
  123. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/arista.deploy +0 -0
  124. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/arista.order +0 -0
  125. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/arista.rul +0 -0
  126. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/aruba.deploy +0 -0
  127. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/aruba.order +0 -0
  128. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/aruba.rul +0 -0
  129. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/b4com.deploy +0 -0
  130. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/b4com.order +0 -0
  131. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/b4com.rul +0 -0
  132. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/cisco.deploy +0 -0
  133. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/cisco.order +0 -0
  134. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/huawei.deploy +0 -0
  135. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/huawei.order +0 -0
  136. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/huawei.rul +0 -0
  137. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/juniper.rul +0 -0
  138. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/nexus.deploy +0 -0
  139. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/nexus.order +0 -0
  140. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/nexus.rul +0 -0
  141. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/nokia.rul +0 -0
  142. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/optixtrans.deploy +0 -0
  143. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/optixtrans.order +0 -0
  144. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/optixtrans.rul +0 -0
  145. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/pc.deploy +0 -0
  146. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/pc.order +0 -0
  147. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/pc.rul +0 -0
  148. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/ribbon.deploy +0 -0
  149. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/ribbon.rul +0 -0
  150. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/routeros.order +0 -0
  151. {annet-0.16.15 → annet-0.16.17}/annet/rulebook/texts/routeros.rul +0 -0
  152. {annet-0.16.15 → annet-0.16.17}/annet/storage.py +0 -0
  153. {annet-0.16.15 → annet-0.16.17}/annet/tabparser.py +0 -0
  154. {annet-0.16.15 → annet-0.16.17}/annet/text_term_format.py +0 -0
  155. {annet-0.16.15 → annet-0.16.17}/annet/tracing.py +0 -0
  156. {annet-0.16.15 → annet-0.16.17}/annet/types.py +0 -0
  157. {annet-0.16.15 → annet-0.16.17}/annet.egg-info/SOURCES.txt +0 -0
  158. {annet-0.16.15 → annet-0.16.17}/annet.egg-info/dependency_links.txt +0 -0
  159. {annet-0.16.15 → annet-0.16.17}/annet.egg-info/entry_points.txt +0 -0
  160. {annet-0.16.15 → annet-0.16.17}/annet.egg-info/requires.txt +0 -0
  161. {annet-0.16.15 → annet-0.16.17}/annet.egg-info/top_level.txt +0 -0
  162. {annet-0.16.15 → annet-0.16.17}/annet_generators/__init__.py +0 -0
  163. {annet-0.16.15 → annet-0.16.17}/annet_generators/example/__init__.py +0 -0
  164. {annet-0.16.15 → annet-0.16.17}/annet_generators/example/lldp.py +0 -0
  165. {annet-0.16.15 → annet-0.16.17}/annet_generators/mesh_example/__init__.py +0 -0
  166. {annet-0.16.15 → annet-0.16.17}/annet_generators/mesh_example/bgp.py +0 -0
  167. {annet-0.16.15 → annet-0.16.17}/annet_generators/mesh_example/mesh_logic.py +0 -0
  168. {annet-0.16.15 → annet-0.16.17}/requirements.txt +0 -0
  169. {annet-0.16.15 → annet-0.16.17}/setup.cfg +0 -0
@@ -1,10 +1,10 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: annet
3
- Version: 0.16.15
3
+ Version: 0.16.17
4
4
  Summary: annet
5
5
  Home-page: https://github.com/annetutil/annet
6
6
  License: MIT
7
- Requires-Python: >=3.9
7
+ Requires-Python: >=3.10
8
8
  License-File: LICENSE
9
9
  License-File: AUTHORS
10
10
  Requires-Dist: colorama>=0.4.6
@@ -56,6 +56,6 @@ def init(options: Namespace):
56
56
 
57
57
 
58
58
  def assert_python_version():
59
- if sys.version_info < (3, 9, 0):
60
- sys.stderr.write("Error: you need python 3.9.0 or higher\n")
59
+ if sys.version_info < (3, 10, 0):
60
+ sys.stderr.write("Error: you need python 3.10.0 or higher\n")
61
61
  sys.exit(1)
@@ -67,7 +67,7 @@ class BFDTimers:
67
67
  multiplier: int = 4
68
68
 
69
69
 
70
- Family = Literal["ipv4_unicast", "ipv6_unicast", "ipv4_labeled", "ipv6_labeled"]
70
+ Family = Literal["ipv4_unicast", "ipv6_unicast", "ipv4_labeled_unicast", "ipv6_labeled_unicast"]
71
71
 
72
72
 
73
73
  @dataclass(frozen=True)
@@ -182,6 +182,8 @@ class PeerGroup:
182
182
  internal_name: str = ""
183
183
  description: str = ""
184
184
  update_source: str = ""
185
+ import_policy: str = ""
186
+ export_policy: str = ""
185
187
 
186
188
  # more strict version of PeerOptions
187
189
  local_as: ASN = ASN(None)
@@ -242,8 +244,6 @@ class VrfOptions:
242
244
  ipv6_labeled_unicast: FamilyOptions
243
245
 
244
246
  vrf_name_global: Optional[str] = None
245
- import_policy: Optional[str] = None
246
- export_policy: Optional[str] = None
247
247
  rt_import: list[str] = field(default_factory=list)
248
248
  rt_export: list[str] = field(default_factory=list)
249
249
  rt_import_v4: list[str] = field(default_factory=list)
@@ -41,8 +41,8 @@ class _FamiliesMixin:
41
41
  def __init__(self, **kwargs):
42
42
  kwargs.setdefault("ipv4_unicast", FamilyOptions(family="ipv4_unicast"))
43
43
  kwargs.setdefault("ipv6_unicast", FamilyOptions(family="ipv6_unicast"))
44
- kwargs.setdefault("ipv4_labeled_unicast", FamilyOptions(family="ipv4_labeled"))
45
- kwargs.setdefault("ipv6_labeled_unicast", FamilyOptions(family="ipv6_labeled"))
44
+ kwargs.setdefault("ipv4_labeled_unicast", FamilyOptions(family="ipv4_labeled_unicast"))
45
+ kwargs.setdefault("ipv6_labeled_unicast", FamilyOptions(family="ipv6_labeled_unicast"))
46
46
  super().__init__(**kwargs)
47
47
  ipv4_unicast: Annotated[FamilyOptions, Merge()]
48
48
  ipv6_unicast: Annotated[FamilyOptions, Merge()]
@@ -54,8 +54,8 @@ class VrfOptions(_FamiliesMixin, BaseMeshModel):
54
54
  def __init__(self, vrf_name: str, **kwargs):
55
55
  kwargs.setdefault("ipv4_unicast", FamilyOptions(family="ipv4_unicast", vrf_name=vrf_name))
56
56
  kwargs.setdefault("ipv6_unicast", FamilyOptions(family="ipv6_unicast", vrf_name=vrf_name))
57
- kwargs.setdefault("ipv4_labeled_unicast", FamilyOptions(family="ipv4_labeled", vrf_name=vrf_name))
58
- kwargs.setdefault("ipv6_labeled_unicast", FamilyOptions(family="ipv6_labeled", vrf_name=vrf_name))
57
+ kwargs.setdefault("ipv4_labeled_unicast", FamilyOptions(family="ipv4_labeled_unicast", vrf_name=vrf_name))
58
+ kwargs.setdefault("ipv6_labeled_unicast", FamilyOptions(family="ipv6_labeled_unicast", vrf_name=vrf_name))
59
59
  kwargs.setdefault("groups", KeyDefaultDict(lambda x: MeshPeerGroup(name=x)))
60
60
  super().__init__(vrf_name=vrf_name, **kwargs)
61
61
 
@@ -61,9 +61,6 @@ retort = Retort(
61
61
  }),
62
62
  loader(list[PeerGroup], lambda x: list(x.values()), Chain.FIRST),
63
63
  loader(PeerGroup, ObjMapping, Chain.FIRST),
64
- name_mapping(PeerGroup, map={
65
- "local_as": "asnum",
66
- }),
67
64
  ]
68
65
  )
69
66
 
@@ -3,7 +3,7 @@ from typing import Literal, Annotated, Union, Optional
3
3
  from .basemodel import BaseMeshModel, Concat
4
4
  from ..bgp_models import BFDTimers
5
5
 
6
- FamilyName = Literal["ipv4_unicast", "ipv6_unicast", "ipv4_labeled", "ipv6_labeled"]
6
+ FamilyName = Literal["ipv4_unicast", "ipv6_unicast", "ipv4_labeled_unicast", "ipv6_labeled_unicast"]
7
7
 
8
8
 
9
9
  class _SharedOptionsDTO(BaseMeshModel):
@@ -117,7 +117,11 @@ class VirtualPeerDTO(MeshSession):
117
117
  class MeshPeerGroup(_OptionsDTO):
118
118
  name: str
119
119
  families: Annotated[set[FamilyName], Concat()]
120
+ local_as: Union[int, str]
120
121
  remote_as: Union[int, str]
121
122
  internal_name: str
122
123
  update_source: str
123
124
  description: str
125
+
126
+ import_policy: str
127
+ export_policy: str
@@ -0,0 +1,149 @@
1
+ import re
2
+ from collections import OrderedDict
3
+ from typing import Any
4
+
5
+ from annet.annlib.types import Op
6
+
7
+ from annet.rulebook import common
8
+
9
+
10
+ def ssh_key(rule, key, diff, hw, **_):
11
+ """
12
+ При включении ssh надо еще сгенерировать ключ. По конфигу никак не понять есть ли ключ на свитче или нет.
13
+ """
14
+ if diff[Op.ADDED]:
15
+ added = sorted([x["row"] for x in diff[Op.ADDED]])
16
+ if added == ["ip ssh version 2"]:
17
+ # Отсыпаем mpdaemon-у подсказок для дополнительной команды при наливке
18
+ comment = rule["comment"]
19
+ rule["comment"] = ["!!suppress_errors!!", "!!timeout=240!!"]
20
+ if hw.Cisco.C2960:
21
+ yield (False, "crypto key generate rsa modulus 2048", None)
22
+ else:
23
+ yield (False, "crypto key generate rsa general-keys modulus 2048", None)
24
+ rule["comment"] = comment
25
+ yield from common.default(rule, key, diff)
26
+
27
+
28
+ def no_ipv6_nd_suppress_ra(rule, key, diff, **_):
29
+ """
30
+ При конфигурации ipv6 nd на нексусах нужно добавлять
31
+ no ipv6 nd suppress-ra
32
+ иначе RA не будет включен.
33
+ К сожалению данной команды не видно в running-config.
34
+ Поэтому подмешиваем ее в патч вместо генератора
35
+ """
36
+ if diff[Op.ADDED]:
37
+ yield (False, "no ipv6 nd suppress-ra", None)
38
+ yield from common.default(rule, key, diff)
39
+
40
+
41
+ def no_ntp_distribute(rule, key, diff, **_):
42
+ """
43
+ Для того, чтобы удалить NTP из CFS, сначала нужно сбросить активные
44
+ NTP сессии.
45
+ """
46
+ if diff[Op.REMOVED]:
47
+ yield (False, "clear ntp session", None)
48
+ yield from common.default(rule, key, diff)
49
+
50
+
51
+ def banner_login(rule, key, diff, **_):
52
+ if diff[Op.REMOVED]:
53
+ yield (False, "no banner login", None)
54
+ elif diff[Op.ADDED]:
55
+ # Убираем дополнительный экранирующий сиимвол
56
+ key = re.sub(r"\^C", "^", key[0])
57
+ yield (False, f"banner login {key}", None)
58
+ else:
59
+ yield from common.default(rule, key, diff)
60
+
61
+
62
+ def bgp_diff(old, new, diff_pre, _pops=(Op.AFFECTED,)):
63
+ """
64
+ Some oder versions of Cisco IOS doesn't create subsection for address family block.
65
+
66
+ it looks like:
67
+
68
+ router bgp 65111
69
+ bgp router-id 1.1.1.1
70
+ bgp log-neighbor-changes
71
+ neighbor SPINE peer-group
72
+ !
73
+ address-family ipv4
74
+ neighbor SPINE send-community both
75
+ neighbor SPINE soft-reconfiguration inbound
76
+ neighbor SPINE route-map TOR_IMPORT_SPINE in
77
+ neighbor SPINE route-map TOR_EXPORT_SPINE out
78
+ exit-address-family
79
+
80
+ but should be
81
+
82
+ router bgp 65111
83
+ bgp router-id 1.1.1.1
84
+ bgp log-neighbor-changes
85
+ neighbor SPINE peer-group
86
+ !
87
+ address-family ipv4
88
+ neighbor SPINE send-community both
89
+ neighbor SPINE soft-reconfiguration inbound
90
+ neighbor SPINE route-map TOR_IMPORT_SPINE in
91
+ neighbor SPINE route-map TOR_EXPORT_SPINE out
92
+ exit-address-family
93
+
94
+ The diff_logic func do it before make diff.
95
+ """
96
+ corrected_old = _create_subsections(old, "address-family")
97
+
98
+ yield from common.default_diff(corrected_old, new, diff_pre, _pops)
99
+
100
+
101
+ def _create_subsections(data: OrderedDict[str, Any], sub_section_prefix: str) -> OrderedDict[str, Any]:
102
+ """
103
+ Reorganizes the given OrderedDict to nest commands under their respective
104
+ sub_section_prefix keys.
105
+
106
+ This function traverses the entries in the provided OrderedDict and groups
107
+ together all entries that are between keys with sub_section_prefix under those
108
+ keys as nested OrderedDicts. The reorganization keeps the order of entries
109
+ stable, only adding nesting where appropriate.
110
+
111
+ Args:
112
+ data (OrderedDict): The original configuration to be transformed.
113
+ sub_section_prefix (str): Prefix of subsection key
114
+
115
+ Returns:
116
+ OrderedDict: A new OrderedDict with nested 'address-family' sections.
117
+ """
118
+
119
+ result = OrderedDict()
120
+ sub_section = None
121
+ temp: OrderedDict = OrderedDict()
122
+
123
+ for key, value in data.items():
124
+ # make nested loop if found nested values
125
+ if value:
126
+ fixed_value: OrderedDict[str, Any] = _create_subsections(value, sub_section_prefix)
127
+ else:
128
+ fixed_value = value
129
+ if key.startswith(sub_section_prefix):
130
+ # in case of data has already had subsections
131
+ if value:
132
+ result[key] = fixed_value
133
+ continue
134
+ # if previous subsection present save collected data from temporary dict
135
+ if sub_section:
136
+ result[sub_section] = temp
137
+ # find a new subsection and initialize new dict
138
+ sub_section = key
139
+ temp = OrderedDict()
140
+ # put found data to temporary dict
141
+ elif sub_section:
142
+ temp[key] = fixed_value
143
+ else:
144
+ result[key] = fixed_value
145
+ # if data is finished save collected data from temporary dict
146
+ if sub_section:
147
+ result[sub_section] = temp
148
+
149
+ return result
@@ -73,7 +73,7 @@ interface */\w*Ethernet[0-9\/]+$/ %logic=common.permanent %diff_logic=cisco.
73
73
  storm-control * level
74
74
  spanning-tree portfast
75
75
 
76
- router bgp *
76
+ router bgp * %diff_logic=cisco.misc.bgp_diff
77
77
  router-id
78
78
  vrf *
79
79
  router-id
@@ -1,10 +1,10 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: annet
3
- Version: 0.16.15
3
+ Version: 0.16.17
4
4
  Summary: annet
5
5
  Home-page: https://github.com/annetutil/annet
6
6
  License: MIT
7
- Requires-Python: >=3.9
7
+ Requires-Python: >=3.10
8
8
  License-File: LICENSE
9
9
  License-File: AUTHORS
10
10
  Requires-Dist: colorama>=0.4.6
@@ -46,7 +46,7 @@ if __name__ == "__main__":
46
46
  extras_require={
47
47
  "netbox": ["annetbox[sync]>=0.1.8"],
48
48
  },
49
- python_requires=">=3.9",
49
+ python_requires=">=3.10",
50
50
  install_requires=requirements(),
51
51
  include_package_data=True,
52
52
  )
@@ -1,57 +0,0 @@
1
- import re
2
-
3
- from annet.annlib.types import Op
4
-
5
- from annet.rulebook import common
6
-
7
-
8
- def ssh_key(rule, key, diff, hw, **_):
9
- """
10
- При включении ssh надо еще сгенерировать ключ. По конфигу никак не понять есть ли ключ на свитче или нет.
11
- """
12
- if diff[Op.ADDED]:
13
- added = sorted([x["row"] for x in diff[Op.ADDED]])
14
- if added == ["ip ssh version 2"]:
15
- # Отсыпаем mpdaemon-у подсказок для дополнительной команды при наливке
16
- comment = rule["comment"]
17
- rule["comment"] = ["!!suppress_errors!!", "!!timeout=240!!"]
18
- if hw.Cisco.C2960:
19
- yield (False, "crypto key generate rsa modulus 2048", None)
20
- else:
21
- yield (False, "crypto key generate rsa general-keys modulus 2048", None)
22
- rule["comment"] = comment
23
- yield from common.default(rule, key, diff)
24
-
25
-
26
- def no_ipv6_nd_suppress_ra(rule, key, diff, **_):
27
- """
28
- При конфигурации ipv6 nd на нексусах нужно добавлять
29
- no ipv6 nd suppress-ra
30
- иначе RA не будет включен.
31
- К сожалению данной команды не видно в running-config.
32
- Поэтому подмешиваем ее в патч вместо генератора
33
- """
34
- if diff[Op.ADDED]:
35
- yield (False, "no ipv6 nd suppress-ra", None)
36
- yield from common.default(rule, key, diff)
37
-
38
-
39
- def no_ntp_distribute(rule, key, diff, **_):
40
- """
41
- Для того, чтобы удалить NTP из CFS, сначала нужно сбросить активные
42
- NTP сессии.
43
- """
44
- if diff[Op.REMOVED]:
45
- yield (False, "clear ntp session", None)
46
- yield from common.default(rule, key, diff)
47
-
48
-
49
- def banner_login(rule, key, diff, **_):
50
- if diff[Op.REMOVED]:
51
- yield (False, "no banner login", None)
52
- elif diff[Op.ADDED]:
53
- # Убираем дополнительный экранирующий сиимвол
54
- key = re.sub(r"\^C", "^", key[0])
55
- yield (False, f"banner login {key}", None)
56
- else:
57
- yield from common.default(rule, key, diff)
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes