ip.py 12 KB

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