ip.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. from django.utils.translation import gettext_lazy as _
  2. import django_tables2 as tables
  3. from django.utils.safestring import mark_safe
  4. from django_tables2.utils import Accessor
  5. from ipam.models import *
  6. from netbox.tables import NetBoxTable, columns
  7. from tenancy.tables import TenancyColumnsMixin, TenantColumn
  8. __all__ = (
  9. 'AggregateTable',
  10. 'AssignedIPAddressesTable',
  11. 'IPAddressAssignTable',
  12. 'IPAddressTable',
  13. 'IPRangeTable',
  14. 'PrefixTable',
  15. 'RIRTable',
  16. 'RoleTable',
  17. )
  18. AVAILABLE_LABEL = mark_safe('<span class="badge text-bg-success">Available</span>')
  19. AGGREGATE_COPY_BUTTON = """
  20. {% copy_content record.pk prefix="aggregate_" %}
  21. """
  22. PREFIX_LINK = """
  23. {% if record.pk %}
  24. <a href="{{ record.get_absolute_url }}" id="prefix_{{ record.pk }}">{{ record.prefix }}</a>
  25. {% else %}
  26. <a href="{% url 'ipam:prefix_add' %}?prefix={{ record }}{% if object.vrf %}&vrf={{ object.vrf.pk }}{% endif %}{% if object.site %}&site={{ object.site.pk }}{% endif %}{% if object.tenant %}&tenant_group={{ object.tenant.group.pk }}&tenant={{ object.tenant.pk }}{% endif %}">{{ record.prefix }}</a>
  27. {% endif %}
  28. """
  29. PREFIX_COPY_BUTTON = """
  30. {% copy_content record.pk prefix="prefix_" %}
  31. """
  32. PREFIX_LINK_WITH_DEPTH = """
  33. {% load helpers %}
  34. {% if record.depth %}
  35. <div class="record-depth">
  36. {% for i in record.depth|as_range %}
  37. <span>•</span>
  38. {% endfor %}
  39. </div>
  40. {% endif %}
  41. """ + PREFIX_LINK
  42. IPADDRESS_LINK = """
  43. {% if record.pk %}
  44. <a href="{{ record.get_absolute_url }}" id="ipaddress_{{ record.pk }}">{{ record.address }}</a>
  45. {% elif perms.ipam.add_ipaddress %}
  46. <a href="{% url 'ipam:ipaddress_add' %}?address={{ record.1 }}{% if object.vrf %}&vrf={{ object.vrf.pk }}{% endif %}{% if object.tenant %}&tenant={{ object.tenant.pk }}{% endif %}" class="btn btn-sm btn-success">{% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available</a>
  47. {% else %}
  48. {% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available
  49. {% endif %}
  50. """
  51. IPADDRESS_COPY_BUTTON = """
  52. {% copy_content record.pk prefix="ipaddress_" %}
  53. """
  54. IPADDRESS_ASSIGN_LINK = """
  55. <a href="{% url 'ipam:ipaddress_edit' pk=record.pk %}?{% if request.GET.interface %}interface={{ request.GET.interface }}{% elif request.GET.vminterface %}vminterface={{ request.GET.vminterface }}{% endif %}&return_url={{ request.GET.return_url }}">{{ record }}</a>
  56. """
  57. VRF_LINK = """
  58. {% if value %}
  59. <a href="{{ record.vrf.get_absolute_url }}">{{ record.vrf }}</a>
  60. {% elif object.vrf %}
  61. <a href="{{ object.vrf.get_absolute_url }}">{{ object.vrf }}</a>
  62. {% else %}
  63. Global
  64. {% endif %}
  65. """
  66. #
  67. # RIRs
  68. #
  69. class RIRTable(NetBoxTable):
  70. name = tables.Column(
  71. verbose_name=_('Name'),
  72. linkify=True
  73. )
  74. is_private = columns.BooleanColumn(
  75. verbose_name=_('Private')
  76. )
  77. aggregate_count = columns.LinkedCountColumn(
  78. viewname='ipam:aggregate_list',
  79. url_params={'rir_id': 'pk'},
  80. verbose_name=_('Aggregates')
  81. )
  82. tags = columns.TagColumn(
  83. url_name='ipam:rir_list'
  84. )
  85. class Meta(NetBoxTable.Meta):
  86. model = RIR
  87. fields = (
  88. 'pk', 'id', 'name', 'slug', 'is_private', 'aggregate_count', 'description', 'tags', 'created',
  89. 'last_updated', 'actions',
  90. )
  91. default_columns = ('pk', 'name', 'is_private', 'aggregate_count', 'description')
  92. #
  93. # Aggregates
  94. #
  95. class AggregateTable(TenancyColumnsMixin, NetBoxTable):
  96. prefix = tables.Column(
  97. linkify=True,
  98. verbose_name=_('Aggregate'),
  99. attrs={
  100. # Allow the aggregate to be copied to the clipboard
  101. 'a': {'id': lambda record: f"aggregate_{record.pk}"}
  102. }
  103. )
  104. date_added = tables.DateColumn(
  105. format="Y-m-d",
  106. verbose_name=_('Added')
  107. )
  108. child_count = tables.Column(
  109. verbose_name=_('Prefixes')
  110. )
  111. utilization = columns.UtilizationColumn(
  112. verbose_name=_('Utilization'),
  113. accessor='get_utilization',
  114. orderable=False
  115. )
  116. comments = columns.MarkdownColumn(
  117. verbose_name=_('Comments'),
  118. )
  119. tags = columns.TagColumn(
  120. url_name='ipam:aggregate_list'
  121. )
  122. actions = columns.ActionsColumn(
  123. extra_buttons=AGGREGATE_COPY_BUTTON
  124. )
  125. class Meta(NetBoxTable.Meta):
  126. model = Aggregate
  127. fields = (
  128. 'pk', 'id', 'prefix', 'rir', 'tenant', 'tenant_group', 'child_count', 'utilization', 'date_added',
  129. 'description', 'comments', 'tags', 'created', 'last_updated',
  130. )
  131. default_columns = ('pk', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description')
  132. #
  133. # Roles
  134. #
  135. class RoleTable(NetBoxTable):
  136. name = tables.Column(
  137. verbose_name=_('Name'),
  138. linkify=True
  139. )
  140. prefix_count = columns.LinkedCountColumn(
  141. viewname='ipam:prefix_list',
  142. url_params={'role_id': 'pk'},
  143. verbose_name=_('Prefixes')
  144. )
  145. iprange_count = columns.LinkedCountColumn(
  146. viewname='ipam:iprange_list',
  147. url_params={'role_id': 'pk'},
  148. verbose_name=_('IP Ranges')
  149. )
  150. vlan_count = columns.LinkedCountColumn(
  151. viewname='ipam:vlan_list',
  152. url_params={'role_id': 'pk'},
  153. verbose_name=_('VLANs')
  154. )
  155. tags = columns.TagColumn(
  156. url_name='ipam:role_list'
  157. )
  158. class Meta(NetBoxTable.Meta):
  159. model = Role
  160. fields = (
  161. 'pk', 'id', 'name', 'slug', 'prefix_count', 'iprange_count', 'vlan_count', 'description', 'weight', 'tags',
  162. 'created', 'last_updated', 'actions',
  163. )
  164. default_columns = ('pk', 'name', 'prefix_count', 'iprange_count', 'vlan_count', 'description')
  165. #
  166. # Prefixes
  167. #
  168. class PrefixUtilizationColumn(columns.UtilizationColumn):
  169. """
  170. Extend UtilizationColumn to allow disabling the warning & danger thresholds for prefixes
  171. marked as fully utilized.
  172. """
  173. template_code = """
  174. {% load helpers %}
  175. {% if record.pk and record.mark_utilized %}
  176. {% utilization_graph value warning_threshold=0 danger_threshold=0 %}
  177. {% elif record.pk %}
  178. {% utilization_graph value %}
  179. {% endif %}
  180. """
  181. class PrefixTable(TenancyColumnsMixin, NetBoxTable):
  182. prefix = columns.TemplateColumn(
  183. verbose_name=_('Prefix'),
  184. template_code=PREFIX_LINK_WITH_DEPTH,
  185. export_raw=True,
  186. attrs={'td': {'class': 'text-nowrap'}}
  187. )
  188. prefix_flat = columns.TemplateColumn(
  189. accessor=Accessor('prefix'),
  190. template_code=PREFIX_LINK,
  191. export_raw=True,
  192. verbose_name=_('Prefix (Flat)')
  193. )
  194. depth = tables.Column(
  195. accessor=Accessor('_depth'),
  196. verbose_name=_('Depth')
  197. )
  198. children = columns.LinkedCountColumn(
  199. accessor=Accessor('_children'),
  200. viewname='ipam:prefix_list',
  201. url_params={
  202. 'vrf_id': 'vrf_id',
  203. 'within': 'prefix',
  204. },
  205. verbose_name=_('Children')
  206. )
  207. status = columns.ChoiceFieldColumn(
  208. verbose_name=_('Status'),
  209. default=AVAILABLE_LABEL
  210. )
  211. vrf = tables.TemplateColumn(
  212. template_code=VRF_LINK,
  213. verbose_name=_('VRF')
  214. )
  215. site = tables.Column(
  216. verbose_name=_('Site'),
  217. linkify=True
  218. )
  219. vlan_group = tables.Column(
  220. accessor='vlan__group',
  221. linkify=True,
  222. verbose_name=_('VLAN Group')
  223. )
  224. vlan = tables.Column(
  225. linkify=True,
  226. verbose_name=_('VLAN')
  227. )
  228. role = tables.Column(
  229. verbose_name=_('Role'),
  230. linkify=True
  231. )
  232. is_pool = columns.BooleanColumn(
  233. verbose_name=_('Pool')
  234. )
  235. mark_utilized = columns.BooleanColumn(
  236. verbose_name=_('Marked Utilized')
  237. )
  238. utilization = PrefixUtilizationColumn(
  239. verbose_name=_('Utilization'),
  240. accessor='get_utilization',
  241. orderable=False
  242. )
  243. comments = columns.MarkdownColumn(
  244. verbose_name=_('Comments'),
  245. )
  246. tags = columns.TagColumn(
  247. url_name='ipam:prefix_list'
  248. )
  249. actions = columns.ActionsColumn(
  250. extra_buttons=PREFIX_COPY_BUTTON
  251. )
  252. class Meta(NetBoxTable.Meta):
  253. model = Prefix
  254. fields = (
  255. 'pk', 'id', 'prefix', 'prefix_flat', 'status', 'children', 'vrf', 'utilization', 'tenant', 'tenant_group',
  256. 'site', 'vlan_group', 'vlan', 'role', 'is_pool', 'mark_utilized', 'description', 'comments', 'tags',
  257. 'created', 'last_updated',
  258. )
  259. default_columns = (
  260. 'pk', 'prefix', 'status', 'children', 'vrf', 'utilization', 'tenant', 'site', 'vlan', 'role', 'description',
  261. )
  262. row_attrs = {
  263. 'class': lambda record: 'success' if not record.pk else '',
  264. }
  265. #
  266. # IP ranges
  267. #
  268. class IPRangeTable(TenancyColumnsMixin, NetBoxTable):
  269. start_address = tables.Column(
  270. verbose_name=_('Start address'),
  271. linkify=True
  272. )
  273. vrf = tables.TemplateColumn(
  274. template_code=VRF_LINK,
  275. verbose_name=_('VRF')
  276. )
  277. status = columns.ChoiceFieldColumn(
  278. verbose_name=_('Status'),
  279. default=AVAILABLE_LABEL
  280. )
  281. role = tables.Column(
  282. verbose_name=_('Role'),
  283. linkify=True
  284. )
  285. mark_utilized = columns.BooleanColumn(
  286. verbose_name=_('Marked Utilized')
  287. )
  288. utilization = columns.UtilizationColumn(
  289. verbose_name=_('Utilization'),
  290. accessor='utilization',
  291. orderable=False
  292. )
  293. comments = columns.MarkdownColumn(
  294. verbose_name=_('Comments'),
  295. )
  296. tags = columns.TagColumn(
  297. url_name='ipam:iprange_list'
  298. )
  299. class Meta(NetBoxTable.Meta):
  300. model = IPRange
  301. fields = (
  302. 'pk', 'id', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'tenant_group',
  303. 'mark_utilized', 'utilization', 'description', 'comments', 'tags', 'created', 'last_updated',
  304. )
  305. default_columns = (
  306. 'pk', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
  307. )
  308. row_attrs = {
  309. 'class': lambda record: 'success' if not record.pk else '',
  310. }
  311. #
  312. # IPAddresses
  313. #
  314. class IPAddressTable(TenancyColumnsMixin, NetBoxTable):
  315. address = tables.TemplateColumn(
  316. template_code=IPADDRESS_LINK,
  317. verbose_name=_('IP Address')
  318. )
  319. vrf = tables.TemplateColumn(
  320. template_code=VRF_LINK,
  321. verbose_name=_('VRF')
  322. )
  323. status = columns.ChoiceFieldColumn(
  324. verbose_name=_('Status'),
  325. default=AVAILABLE_LABEL
  326. )
  327. role = columns.ChoiceFieldColumn(
  328. verbose_name=_('Role'),
  329. )
  330. assigned_object = tables.Column(
  331. linkify=True,
  332. orderable=False,
  333. verbose_name=_('Interface')
  334. )
  335. assigned_object_parent = tables.Column(
  336. accessor='assigned_object__parent_object',
  337. linkify=True,
  338. orderable=False,
  339. verbose_name=_('Parent')
  340. )
  341. nat_inside = tables.Column(
  342. linkify=True,
  343. orderable=False,
  344. verbose_name=_('NAT (Inside)')
  345. )
  346. nat_outside = tables.ManyToManyColumn(
  347. linkify_item=True,
  348. orderable=False,
  349. verbose_name=_('NAT (Outside)')
  350. )
  351. assigned = columns.BooleanColumn(
  352. accessor='assigned_object_id',
  353. linkify=lambda record: record.assigned_object.get_absolute_url(),
  354. verbose_name=_('Assigned')
  355. )
  356. comments = columns.MarkdownColumn(
  357. verbose_name=_('Comments'),
  358. )
  359. tags = columns.TagColumn(
  360. url_name='ipam:ipaddress_list'
  361. )
  362. actions = columns.ActionsColumn(
  363. extra_buttons=IPADDRESS_COPY_BUTTON
  364. )
  365. class Meta(NetBoxTable.Meta):
  366. model = IPAddress
  367. fields = (
  368. 'pk', 'id', 'address', 'vrf', 'status', 'role', 'tenant', 'tenant_group', 'nat_inside', 'nat_outside',
  369. 'assigned', 'dns_name', 'description', 'comments', 'tags', 'created', 'last_updated',
  370. )
  371. default_columns = (
  372. 'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'assigned', 'dns_name', 'description',
  373. )
  374. row_attrs = {
  375. 'class': lambda record: 'success' if not isinstance(record, IPAddress) else '',
  376. }
  377. class IPAddressAssignTable(NetBoxTable):
  378. address = tables.TemplateColumn(
  379. template_code=IPADDRESS_ASSIGN_LINK,
  380. verbose_name=_('IP Address')
  381. )
  382. status = columns.ChoiceFieldColumn(
  383. verbose_name=_('Status'),
  384. )
  385. assigned_object = tables.Column(
  386. verbose_name=_('Assigned Object'),
  387. orderable=False
  388. )
  389. class Meta(NetBoxTable.Meta):
  390. model = IPAddress
  391. fields = ('address', 'dns_name', 'vrf', 'status', 'role', 'tenant', 'assigned_object', 'description')
  392. exclude = ('id', )
  393. orderable = False
  394. class AssignedIPAddressesTable(NetBoxTable):
  395. """
  396. List IP addresses assigned to an object.
  397. """
  398. address = tables.Column(
  399. linkify=True,
  400. verbose_name=_('IP Address')
  401. )
  402. vrf = tables.TemplateColumn(
  403. template_code=VRF_LINK,
  404. verbose_name=_('VRF')
  405. )
  406. status = columns.ChoiceFieldColumn(
  407. verbose_name=_('Status'),
  408. )
  409. tenant = TenantColumn(
  410. verbose_name=_('Tenant'),
  411. )
  412. class Meta(NetBoxTable.Meta):
  413. model = IPAddress
  414. fields = ('address', 'vrf', 'status', 'role', 'tenant', 'description')
  415. exclude = ('id', )