tables.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. from __future__ import unicode_literals
  2. import django_tables2 as tables
  3. from django_tables2.utils import Accessor
  4. from dcim.models import Interface
  5. from tenancy.tables import COL_TENANT
  6. from utilities.tables import BaseTable, ToggleColumn
  7. from .models import Aggregate, IPAddress, Prefix, RIR, Role, VLAN, VLANGroup, VRF
  8. RIR_UTILIZATION = """
  9. <div class="progress">
  10. {% if record.stats.total %}
  11. <div class="progress-bar" role="progressbar" style="width: {{ record.stats.percentages.active }}%;">
  12. <span class="sr-only">{{ record.stats.percentages.active }}%</span>
  13. </div>
  14. <div class="progress-bar progress-bar-info" role="progressbar" style="width: {{ record.stats.percentages.reserved }}%;">
  15. <span class="sr-only">{{ record.stats.percentages.reserved }}%</span>
  16. </div>
  17. <div class="progress-bar progress-bar-danger" role="progressbar" style="width: {{ record.stats.percentages.deprecated }}%;">
  18. <span class="sr-only">{{ record.stats.percentages.deprecated }}%</span>
  19. </div>
  20. <div class="progress-bar progress-bar-success" role="progressbar" style="width: {{ record.stats.percentages.available }}%;">
  21. <span class="sr-only">{{ record.stats.percentages.available }}%</span>
  22. </div>
  23. {% endif %}
  24. </div>
  25. """
  26. RIR_ACTIONS = """
  27. {% if perms.ipam.change_rir %}
  28. <a href="{% url 'ipam:rir_edit' slug=record.slug %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  29. {% endif %}
  30. """
  31. UTILIZATION_GRAPH = """
  32. {% load helpers %}
  33. {% if record.pk %}{% utilization_graph record.get_utilization %}{% else %}&mdash;{% endif %}
  34. """
  35. ROLE_PREFIX_COUNT = """
  36. <a href="{% url 'ipam:prefix_list' %}?role={{ record.slug }}">{{ value }}</a>
  37. """
  38. ROLE_VLAN_COUNT = """
  39. <a href="{% url 'ipam:vlan_list' %}?role={{ record.slug }}">{{ value }}</a>
  40. """
  41. ROLE_ACTIONS = """
  42. {% if perms.ipam.change_role %}
  43. <a href="{% url 'ipam:role_edit' slug=record.slug %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  44. {% endif %}
  45. """
  46. PREFIX_LINK = """
  47. {% if record.has_children %}
  48. <span class="text-nowrap" style="padding-left: {{ record.depth }}0px "><i class="fa fa-caret-right"></i></a>
  49. {% else %}
  50. <span class="text-nowrap" style="padding-left: {{ record.depth }}9px">
  51. {% endif %}
  52. <a href="{% if record.pk %}{% url 'ipam:prefix' pk=record.pk %}{% else %}{% url 'ipam:prefix_add' %}?prefix={{ record }}{% if parent.vrf %}&vrf={{ parent.vrf.pk }}{% endif %}{% if parent.site %}&site={{ parent.site.pk }}{% endif %}{% if parent.tenant %}&tenant_group={{ parent.tenant.group.pk }}&tenant={{ parent.tenant.pk }}{% endif %}{% endif %}">{{ record.prefix }}</a>
  53. </span>
  54. """
  55. PREFIX_ROLE_LINK = """
  56. {% if record.role %}
  57. <a href="{% url 'ipam:prefix_list' %}?role={{ record.role.slug }}">{{ record.role }}</a>
  58. {% else %}
  59. &mdash;
  60. {% endif %}
  61. """
  62. IPADDRESS_LINK = """
  63. {% if record.pk %}
  64. <a href="{{ record.get_absolute_url }}">{{ record.address }}</a>
  65. {% elif perms.ipam.add_ipaddress %}
  66. <a href="{% url 'ipam:ipaddress_add' %}?address={{ record.1 }}{% if prefix.vrf %}&vrf={{ prefix.vrf.pk }}{% endif %}{% if prefix.tenant %}&tenant={{ prefix.tenant.pk }}{% endif %}" class="btn btn-xs btn-success">{% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available</a>
  67. {% else %}
  68. {% if record.0 <= 65536 %}{{ record.0 }}{% else %}Many{% endif %} IP{{ record.0|pluralize }} available
  69. {% endif %}
  70. """
  71. IPADDRESS_ASSIGN_LINK = """
  72. <a href="{% url 'ipam:ipaddress_edit' pk=record.pk %}?interface={{ request.GET.interface }}&return_url={{ request.GET.return_url }}">{{ record }}</a>
  73. """
  74. IPADDRESS_PARENT = """
  75. {% if record.interface %}
  76. <a href="{{ record.interface.parent.get_absolute_url }}">{{ record.interface.parent }}</a>
  77. {% else %}
  78. &mdash;
  79. {% endif %}
  80. """
  81. VRF_LINK = """
  82. {% if record.vrf %}
  83. <a href="{{ record.vrf.get_absolute_url }}">{{ record.vrf }}</a>
  84. {% elif prefix.vrf %}
  85. {{ prefix.vrf }}
  86. {% else %}
  87. Global
  88. {% endif %}
  89. """
  90. STATUS_LABEL = """
  91. {% if record.pk %}
  92. <span class="label label-{{ record.get_status_class }}">{{ record.get_status_display }}</span>
  93. {% else %}
  94. <span class="label label-success">Available</span>
  95. {% endif %}
  96. """
  97. VLAN_PREFIXES = """
  98. {% for prefix in record.prefixes.all %}
  99. <a href="{% url 'ipam:prefix' pk=prefix.pk %}">{{ prefix }}</a>{% if not forloop.last %}<br />{% endif %}
  100. {% empty %}
  101. &mdash;
  102. {% endfor %}
  103. """
  104. VLAN_ROLE_LINK = """
  105. {% if record.role %}
  106. <a href="{% url 'ipam:vlan_list' %}?role={{ record.role.slug }}">{{ record.role }}</a>
  107. {% else %}
  108. &mdash;
  109. {% endif %}
  110. """
  111. VLANGROUP_ACTIONS = """
  112. {% with next_vid=record.get_next_available_vid %}
  113. {% if next_vid and perms.ipam.add_vlan %}
  114. <a href="{% url 'ipam:vlan_add' %}?site={{ record.site_id }}&group={{ record.pk }}&vid={{ next_vid }}" title="Add VLAN" class="btn btn-xs btn-success">
  115. <i class="glyphicon glyphicon-plus" aria-hidden="true"></i>
  116. </a>
  117. {% endif %}
  118. {% endwith %}
  119. {% if perms.ipam.change_vlangroup %}
  120. <a href="{% url 'ipam:vlangroup_edit' pk=record.pk %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  121. {% endif %}
  122. """
  123. VLAN_MEMBER_UNTAGGED = """
  124. {% if record.untagged_vlan_id == vlan.pk %}
  125. <i class="glyphicon glyphicon-ok">
  126. {% endif %}
  127. """
  128. VLAN_MEMBER_ACTIONS = """
  129. {% if perms.dcim.change_interface %}
  130. <a href="{% if record.device %}{% url 'dcim:interface_edit' pk=record.pk %}{% else %}{% url 'virtualization:interface_edit' pk=record.pk %}{% endif %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil"></i></a>
  131. {% endif %}
  132. """
  133. TENANT_LINK = """
  134. {% if record.tenant %}
  135. <a href="{% url 'tenancy:tenant' slug=record.tenant.slug %}" title="{{ record.tenant.description }}">{{ record.tenant }}</a>
  136. {% elif record.vrf.tenant %}
  137. <a href="{% url 'tenancy:tenant' slug=record.vrf.tenant.slug %}" title="{{ record.vrf.tenant.description }}">{{ record.vrf.tenant }}</a>*
  138. {% else %}
  139. &mdash;
  140. {% endif %}
  141. """
  142. #
  143. # VRFs
  144. #
  145. class VRFTable(BaseTable):
  146. pk = ToggleColumn()
  147. name = tables.LinkColumn()
  148. rd = tables.Column(verbose_name='RD')
  149. tenant = tables.TemplateColumn(template_code=COL_TENANT)
  150. class Meta(BaseTable.Meta):
  151. model = VRF
  152. fields = ('pk', 'name', 'rd', 'tenant', 'description')
  153. #
  154. # RIRs
  155. #
  156. class RIRTable(BaseTable):
  157. pk = ToggleColumn()
  158. name = tables.LinkColumn(verbose_name='Name')
  159. is_private = tables.BooleanColumn(verbose_name='Private')
  160. aggregate_count = tables.Column(verbose_name='Aggregates')
  161. actions = tables.TemplateColumn(template_code=RIR_ACTIONS, attrs={'td': {'class': 'text-right'}}, verbose_name='')
  162. class Meta(BaseTable.Meta):
  163. model = RIR
  164. fields = ('pk', 'name', 'is_private', 'aggregate_count', 'actions')
  165. class RIRDetailTable(RIRTable):
  166. stats_total = tables.Column(accessor='stats.total', verbose_name='Total',
  167. footer=lambda table: sum(r.stats['total'] for r in table.data))
  168. stats_active = tables.Column(accessor='stats.active', verbose_name='Active',
  169. footer=lambda table: sum(r.stats['active'] for r in table.data))
  170. stats_reserved = tables.Column(accessor='stats.reserved', verbose_name='Reserved',
  171. footer=lambda table: sum(r.stats['reserved'] for r in table.data))
  172. stats_deprecated = tables.Column(accessor='stats.deprecated', verbose_name='Deprecated',
  173. footer=lambda table: sum(r.stats['deprecated'] for r in table.data))
  174. stats_available = tables.Column(accessor='stats.available', verbose_name='Available',
  175. footer=lambda table: sum(r.stats['available'] for r in table.data))
  176. utilization = tables.TemplateColumn(template_code=RIR_UTILIZATION, verbose_name='Utilization')
  177. class Meta(RIRTable.Meta):
  178. fields = (
  179. 'pk', 'name', 'is_private', 'aggregate_count', 'stats_total', 'stats_active', 'stats_reserved',
  180. 'stats_deprecated', 'stats_available', 'utilization', 'actions',
  181. )
  182. #
  183. # Aggregates
  184. #
  185. class AggregateTable(BaseTable):
  186. pk = ToggleColumn()
  187. prefix = tables.LinkColumn(verbose_name='Aggregate')
  188. date_added = tables.DateColumn(format="Y-m-d", verbose_name='Added')
  189. class Meta(BaseTable.Meta):
  190. model = Aggregate
  191. fields = ('pk', 'prefix', 'rir', 'date_added', 'description')
  192. class AggregateDetailTable(AggregateTable):
  193. child_count = tables.Column(verbose_name='Prefixes')
  194. utilization = tables.TemplateColumn(UTILIZATION_GRAPH, orderable=False, verbose_name='Utilization')
  195. class Meta(AggregateTable.Meta):
  196. fields = ('pk', 'prefix', 'rir', 'child_count', 'utilization', 'date_added', 'description')
  197. #
  198. # Roles
  199. #
  200. class RoleTable(BaseTable):
  201. pk = ToggleColumn()
  202. prefix_count = tables.TemplateColumn(
  203. accessor=Accessor('prefixes.count'),
  204. template_code=ROLE_PREFIX_COUNT,
  205. orderable=False,
  206. verbose_name='Prefixes'
  207. )
  208. vlan_count = tables.TemplateColumn(
  209. accessor=Accessor('vlans.count'),
  210. template_code=ROLE_VLAN_COUNT,
  211. orderable=False,
  212. verbose_name='VLANs'
  213. )
  214. actions = tables.TemplateColumn(template_code=ROLE_ACTIONS, attrs={'td': {'class': 'text-right'}}, verbose_name='')
  215. class Meta(BaseTable.Meta):
  216. model = Role
  217. fields = ('pk', 'name', 'prefix_count', 'vlan_count', 'slug', 'actions')
  218. #
  219. # Prefixes
  220. #
  221. class PrefixTable(BaseTable):
  222. pk = ToggleColumn()
  223. prefix = tables.TemplateColumn(PREFIX_LINK, attrs={'th': {'style': 'padding-left: 17px'}})
  224. status = tables.TemplateColumn(STATUS_LABEL)
  225. vrf = tables.TemplateColumn(VRF_LINK, verbose_name='VRF')
  226. tenant = tables.TemplateColumn(template_code=TENANT_LINK)
  227. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  228. vlan = tables.LinkColumn('ipam:vlan', args=[Accessor('vlan.pk')], verbose_name='VLAN')
  229. role = tables.TemplateColumn(PREFIX_ROLE_LINK)
  230. class Meta(BaseTable.Meta):
  231. model = Prefix
  232. fields = ('pk', 'prefix', 'status', 'vrf', 'tenant', 'site', 'vlan', 'role', 'description')
  233. row_attrs = {
  234. 'class': lambda record: 'success' if not record.pk else '',
  235. }
  236. class PrefixDetailTable(PrefixTable):
  237. utilization = tables.TemplateColumn(UTILIZATION_GRAPH, orderable=False)
  238. class Meta(PrefixTable.Meta):
  239. fields = ('pk', 'prefix', 'status', 'vrf', 'utilization', 'tenant', 'site', 'vlan', 'role', 'description')
  240. #
  241. # IPAddresses
  242. #
  243. class IPAddressTable(BaseTable):
  244. pk = ToggleColumn()
  245. address = tables.TemplateColumn(IPADDRESS_LINK, verbose_name='IP Address')
  246. vrf = tables.TemplateColumn(VRF_LINK, verbose_name='VRF')
  247. status = tables.TemplateColumn(STATUS_LABEL)
  248. tenant = tables.TemplateColumn(template_code=TENANT_LINK)
  249. parent = tables.TemplateColumn(IPADDRESS_PARENT, orderable=False)
  250. interface = tables.Column(orderable=False)
  251. class Meta(BaseTable.Meta):
  252. model = IPAddress
  253. fields = ('pk', 'address', 'vrf', 'status', 'role', 'tenant', 'parent', 'interface', 'description')
  254. row_attrs = {
  255. 'class': lambda record: 'success' if not isinstance(record, IPAddress) else '',
  256. }
  257. class IPAddressDetailTable(IPAddressTable):
  258. nat_inside = tables.LinkColumn(
  259. 'ipam:ipaddress', args=[Accessor('nat_inside.pk')], orderable=False, verbose_name='NAT (Inside)'
  260. )
  261. class Meta(IPAddressTable.Meta):
  262. fields = (
  263. 'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'nat_inside', 'parent', 'interface', 'description',
  264. )
  265. class IPAddressAssignTable(BaseTable):
  266. address = tables.TemplateColumn(IPADDRESS_ASSIGN_LINK, verbose_name='IP Address')
  267. status = tables.TemplateColumn(STATUS_LABEL)
  268. parent = tables.TemplateColumn(IPADDRESS_PARENT, orderable=False)
  269. interface = tables.Column(orderable=False)
  270. class Meta(BaseTable.Meta):
  271. model = IPAddress
  272. fields = ('address', 'vrf', 'status', 'role', 'tenant', 'parent', 'interface', 'description')
  273. orderable = False
  274. #
  275. # VLAN groups
  276. #
  277. class VLANGroupTable(BaseTable):
  278. pk = ToggleColumn()
  279. name = tables.LinkColumn(verbose_name='Name')
  280. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  281. vlan_count = tables.Column(verbose_name='VLANs')
  282. slug = tables.Column(verbose_name='Slug')
  283. actions = tables.TemplateColumn(template_code=VLANGROUP_ACTIONS, attrs={'td': {'class': 'text-right'}},
  284. verbose_name='')
  285. class Meta(BaseTable.Meta):
  286. model = VLANGroup
  287. fields = ('pk', 'name', 'site', 'vlan_count', 'slug', 'actions')
  288. #
  289. # VLANs
  290. #
  291. class VLANTable(BaseTable):
  292. pk = ToggleColumn()
  293. vid = tables.LinkColumn('ipam:vlan', args=[Accessor('pk')], verbose_name='ID')
  294. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  295. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  296. tenant = tables.TemplateColumn(template_code=COL_TENANT)
  297. status = tables.TemplateColumn(STATUS_LABEL)
  298. role = tables.TemplateColumn(VLAN_ROLE_LINK)
  299. class Meta(BaseTable.Meta):
  300. model = VLAN
  301. fields = ('pk', 'vid', 'site', 'group', 'name', 'tenant', 'status', 'role', 'description')
  302. class VLANDetailTable(VLANTable):
  303. prefixes = tables.TemplateColumn(VLAN_PREFIXES, orderable=False, verbose_name='Prefixes')
  304. class Meta(VLANTable.Meta):
  305. fields = ('pk', 'vid', 'site', 'group', 'name', 'prefixes', 'tenant', 'status', 'role', 'description')
  306. class VLANMemberTable(BaseTable):
  307. parent = tables.LinkColumn(order_by=['device', 'virtual_machine'])
  308. name = tables.Column(verbose_name='Interface')
  309. untagged = tables.TemplateColumn(
  310. template_code=VLAN_MEMBER_UNTAGGED,
  311. orderable=False
  312. )
  313. actions = tables.TemplateColumn(
  314. template_code=VLAN_MEMBER_ACTIONS,
  315. attrs={'td': {'class': 'text-right'}},
  316. verbose_name=''
  317. )
  318. class Meta(BaseTable.Meta):
  319. model = Interface
  320. fields = ('parent', 'name', 'untagged', 'actions')