tables.py 12 KB

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