ip.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  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 = ('pk', 'id', 'name', 'slug', 'is_private', 'aggregate_count', 'description', 'tags', 'actions')
  85. default_columns = ('pk', 'name', 'is_private', 'aggregate_count', 'description', 'actions')
  86. #
  87. # ASNs
  88. #
  89. class ASNTable(BaseTable):
  90. pk = ToggleColumn()
  91. asn = tables.Column(
  92. linkify=True
  93. )
  94. def render_asn(self, value, record):
  95. return record.asn_with_asdot
  96. site_count = LinkedCountColumn(
  97. viewname='dcim:site_list',
  98. url_params={'asn_id': 'pk'},
  99. verbose_name='Sites'
  100. )
  101. actions = ButtonsColumn(ASN)
  102. class Meta(BaseTable.Meta):
  103. model = ASN
  104. fields = ('pk', 'asn', 'rir', 'site_count', 'tenant', 'description', 'actions')
  105. default_columns = ('pk', 'asn', 'rir', 'site_count', 'sites', 'tenant', 'actions')
  106. #
  107. # Aggregates
  108. #
  109. class AggregateTable(BaseTable):
  110. pk = ToggleColumn()
  111. prefix = tables.Column(
  112. linkify=True,
  113. verbose_name='Aggregate'
  114. )
  115. tenant = TenantColumn()
  116. date_added = tables.DateColumn(
  117. format="Y-m-d",
  118. verbose_name='Added'
  119. )
  120. child_count = tables.Column(
  121. verbose_name='Prefixes'
  122. )
  123. utilization = UtilizationColumn(
  124. accessor='get_utilization',
  125. orderable=False
  126. )
  127. tags = TagColumn(
  128. url_name='ipam:aggregate_list'
  129. )
  130. class Meta(BaseTable.Meta):
  131. model = Aggregate
  132. fields = ('pk', 'id', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description', 'tags')
  133. default_columns = ('pk', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description')
  134. #
  135. # Roles
  136. #
  137. class RoleTable(BaseTable):
  138. pk = ToggleColumn()
  139. name = tables.Column(
  140. linkify=True
  141. )
  142. prefix_count = LinkedCountColumn(
  143. viewname='ipam:prefix_list',
  144. url_params={'role_id': 'pk'},
  145. verbose_name='Prefixes'
  146. )
  147. vlan_count = LinkedCountColumn(
  148. viewname='ipam:vlan_list',
  149. url_params={'role_id': 'pk'},
  150. verbose_name='VLANs'
  151. )
  152. tags = TagColumn(
  153. url_name='ipam:role_list'
  154. )
  155. actions = ButtonsColumn(Role)
  156. class Meta(BaseTable.Meta):
  157. model = Role
  158. fields = ('pk', 'id', 'name', 'slug', 'prefix_count', 'vlan_count', 'description', 'weight', 'tags', 'actions')
  159. default_columns = ('pk', 'name', 'prefix_count', 'vlan_count', 'description', 'actions')
  160. #
  161. # Prefixes
  162. #
  163. class PrefixUtilizationColumn(UtilizationColumn):
  164. """
  165. Extend UtilizationColumn to allow disabling the warning & danger thresholds for prefixes
  166. marked as fully utilized.
  167. """
  168. template_code = """
  169. {% load helpers %}
  170. {% if record.pk and record.mark_utilized %}
  171. {% utilization_graph value warning_threshold=0 danger_threshold=0 %}
  172. {% elif record.pk %}
  173. {% utilization_graph value %}
  174. {% endif %}
  175. """
  176. class PrefixTable(BaseTable):
  177. pk = ToggleColumn()
  178. prefix = tables.TemplateColumn(
  179. template_code=PREFIX_LINK,
  180. attrs={'td': {'class': 'text-nowrap'}}
  181. )
  182. prefix_flat = tables.TemplateColumn(
  183. template_code=PREFIXFLAT_LINK,
  184. attrs={'td': {'class': 'text-nowrap'}},
  185. verbose_name='Prefix (Flat)',
  186. )
  187. depth = tables.Column(
  188. accessor=Accessor('_depth'),
  189. verbose_name='Depth'
  190. )
  191. children = LinkedCountColumn(
  192. accessor=Accessor('_children'),
  193. viewname='ipam:prefix_list',
  194. url_params={
  195. 'vrf_id': 'vrf_id',
  196. 'within': 'prefix',
  197. },
  198. verbose_name='Children'
  199. )
  200. status = ChoiceFieldColumn(
  201. default=AVAILABLE_LABEL
  202. )
  203. vrf = tables.TemplateColumn(
  204. template_code=VRF_LINK,
  205. verbose_name='VRF'
  206. )
  207. tenant = TenantColumn()
  208. site = tables.Column(
  209. linkify=True
  210. )
  211. vlan_group = tables.Column(
  212. accessor='vlan__group',
  213. linkify=True,
  214. verbose_name='VLAN Group'
  215. )
  216. vlan = tables.Column(
  217. linkify=True,
  218. verbose_name='VLAN'
  219. )
  220. role = tables.Column(
  221. linkify=True
  222. )
  223. is_pool = BooleanColumn(
  224. verbose_name='Pool'
  225. )
  226. mark_utilized = BooleanColumn(
  227. verbose_name='Marked Utilized'
  228. )
  229. utilization = PrefixUtilizationColumn(
  230. accessor='get_utilization',
  231. orderable=False
  232. )
  233. tags = TagColumn(
  234. url_name='ipam:prefix_list'
  235. )
  236. class Meta(BaseTable.Meta):
  237. model = Prefix
  238. fields = (
  239. 'pk', 'id', 'prefix', 'prefix_flat', 'status', 'children', 'vrf', 'utilization', 'tenant', 'site', 'vlan_group',
  240. 'vlan', 'role', 'is_pool', 'mark_utilized', 'description', 'tags',
  241. )
  242. default_columns = (
  243. 'pk', 'prefix', 'status', 'children', 'vrf', 'utilization', 'tenant', 'site', 'vlan', 'role', 'description',
  244. )
  245. row_attrs = {
  246. 'class': lambda record: 'success' if not record.pk else '',
  247. }
  248. #
  249. # IP ranges
  250. #
  251. class IPRangeTable(BaseTable):
  252. pk = ToggleColumn()
  253. start_address = tables.Column(
  254. linkify=True
  255. )
  256. vrf = tables.TemplateColumn(
  257. template_code=VRF_LINK,
  258. verbose_name='VRF'
  259. )
  260. status = ChoiceFieldColumn(
  261. default=AVAILABLE_LABEL
  262. )
  263. role = tables.Column(
  264. linkify=True
  265. )
  266. tenant = TenantColumn()
  267. utilization = UtilizationColumn(
  268. accessor='utilization',
  269. orderable=False
  270. )
  271. tags = TagColumn(
  272. url_name='ipam:iprange_list'
  273. )
  274. class Meta(BaseTable.Meta):
  275. model = IPRange
  276. fields = (
  277. 'pk', 'id', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
  278. 'utilization', 'tags',
  279. )
  280. default_columns = (
  281. 'pk', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
  282. )
  283. row_attrs = {
  284. 'class': lambda record: 'success' if not record.pk else '',
  285. }
  286. #
  287. # IPAddresses
  288. #
  289. class IPAddressTable(BaseTable):
  290. pk = ToggleColumn()
  291. address = tables.TemplateColumn(
  292. template_code=IPADDRESS_LINK,
  293. verbose_name='IP Address'
  294. )
  295. vrf = tables.TemplateColumn(
  296. template_code=VRF_LINK,
  297. verbose_name='VRF'
  298. )
  299. status = ChoiceFieldColumn(
  300. default=AVAILABLE_LABEL
  301. )
  302. role = ChoiceFieldColumn()
  303. tenant = TenantColumn()
  304. assigned_object = tables.Column(
  305. linkify=True,
  306. orderable=False,
  307. verbose_name='Interface'
  308. )
  309. assigned_object_parent = tables.Column(
  310. accessor='assigned_object.parent_object',
  311. linkify=True,
  312. orderable=False,
  313. verbose_name='Device/VM'
  314. )
  315. nat_inside = tables.Column(
  316. linkify=True,
  317. orderable=False,
  318. verbose_name='NAT (Inside)'
  319. )
  320. assigned = BooleanColumn(
  321. accessor='assigned_object_id',
  322. linkify=True,
  323. verbose_name='Assigned'
  324. )
  325. tags = TagColumn(
  326. url_name='ipam:ipaddress_list'
  327. )
  328. class Meta(BaseTable.Meta):
  329. model = IPAddress
  330. fields = (
  331. 'pk', 'id', 'address', 'vrf', 'status', 'role', 'tenant', 'nat_inside', 'assigned', 'dns_name', 'description',
  332. 'tags',
  333. )
  334. default_columns = (
  335. 'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'assigned', 'dns_name', 'description',
  336. )
  337. row_attrs = {
  338. 'class': lambda record: 'success' if not isinstance(record, IPAddress) else '',
  339. }
  340. class IPAddressAssignTable(BaseTable):
  341. address = tables.TemplateColumn(
  342. template_code=IPADDRESS_ASSIGN_LINK,
  343. verbose_name='IP Address'
  344. )
  345. status = ChoiceFieldColumn()
  346. assigned_object = tables.Column(
  347. orderable=False
  348. )
  349. class Meta(BaseTable.Meta):
  350. model = IPAddress
  351. fields = ('address', 'dns_name', 'vrf', 'status', 'role', 'tenant', 'assigned_object', 'description')
  352. exclude = ('id', )
  353. orderable = False
  354. class AssignedIPAddressesTable(BaseTable):
  355. """
  356. List IP addresses assigned to an object.
  357. """
  358. address = tables.Column(
  359. linkify=True,
  360. verbose_name='IP Address'
  361. )
  362. vrf = tables.TemplateColumn(
  363. template_code=VRF_LINK,
  364. verbose_name='VRF'
  365. )
  366. status = ChoiceFieldColumn()
  367. tenant = TenantColumn()
  368. actions = ButtonsColumn(
  369. model=IPAddress
  370. )
  371. class Meta(BaseTable.Meta):
  372. model = IPAddress
  373. fields = ('address', 'vrf', 'status', 'role', 'tenant', 'description')
  374. exclude = ('id', )