tables.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. import django_tables2 as tables
  2. from django_tables2.utils import Accessor
  3. from utilities.tables import BaseTable, ToggleColumn
  4. from .models import (
  5. ConsolePort, ConsolePortTemplate, ConsoleServerPortTemplate, Device, DeviceBayTemplate, DeviceRole, DeviceType,
  6. Interface, InterfaceTemplate, Manufacturer, Platform, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack,
  7. RackGroup, Region, Site,
  8. )
  9. REGION_LINK = """
  10. {% if record.get_children %}
  11. <span style="padding-left: {{ record.get_ancestors|length }}0px "><i class="fa fa-caret-right"></i></a>
  12. {% else %}
  13. <span style="padding-left: {{ record.get_ancestors|length }}9px">
  14. {% endif %}
  15. {{ record.name }}
  16. </span>
  17. """
  18. SITE_REGION_LINK = """
  19. {% if record.region %}
  20. <a href="{% url 'dcim:site_list' %}?region={{ record.region.slug }}">{{ record.region }}</a>
  21. {% else %}
  22. &mdash;
  23. {% endif %}
  24. """
  25. COLOR_LABEL = """
  26. <label class="label" style="background-color: #{{ record.color }}">{{ record }}</label>
  27. """
  28. DEVICE_LINK = """
  29. <a href="{% url 'dcim:device' pk=record.pk %}">
  30. {{ record.name|default:'<span class="label label-info">Unnamed device</span>' }}
  31. </a>
  32. """
  33. REGION_ACTIONS = """
  34. {% if perms.dcim.change_region %}
  35. <a href="{% url 'dcim:region_edit' pk=record.pk %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  36. {% endif %}
  37. """
  38. RACKGROUP_ACTIONS = """
  39. {% if perms.dcim.change_rackgroup %}
  40. <a href="{% url 'dcim:rackgroup_edit' pk=record.pk %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  41. {% endif %}
  42. """
  43. RACKROLE_ACTIONS = """
  44. {% if perms.dcim.change_rackrole %}
  45. <a href="{% url 'dcim:rackrole_edit' pk=record.pk %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  46. {% endif %}
  47. """
  48. RACK_ROLE = """
  49. {% if record.role %}
  50. <label class="label" style="background-color: #{{ record.role.color }}">{{ value }}</label>
  51. {% else %}
  52. &mdash;
  53. {% endif %}
  54. """
  55. DEVICEROLE_ACTIONS = """
  56. {% if perms.dcim.change_devicerole %}
  57. <a href="{% url 'dcim:devicerole_edit' slug=record.slug %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  58. {% endif %}
  59. """
  60. MANUFACTURER_ACTIONS = """
  61. {% if perms.dcim.change_manufacturer %}
  62. <a href="{% url 'dcim:manufacturer_edit' slug=record.slug %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  63. {% endif %}
  64. """
  65. PLATFORM_ACTIONS = """
  66. {% if perms.dcim.change_platform %}
  67. <a href="{% url 'dcim:platform_edit' slug=record.slug %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  68. {% endif %}
  69. """
  70. DEVICE_ROLE = """
  71. <label class="label" style="background-color: #{{ record.device_role.color }}">{{ value }}</label>
  72. """
  73. STATUS_ICON = """
  74. {% if record.status %}
  75. <span class="glyphicon glyphicon-ok-sign text-success" title="Active" aria-hidden="true"></span>
  76. {% else %}
  77. <span class="glyphicon glyphicon-minus-sign text-danger" title="Offline" aria-hidden="true"></span>
  78. {% endif %}
  79. """
  80. DEVICE_PRIMARY_IP = """
  81. {{ record.primary_ip6.address.ip|default:"" }}
  82. {% if record.primary_ip6 and record.primary_ip4 %}<br />{% endif %}
  83. {{ record.primary_ip4.address.ip|default:"" }}
  84. """
  85. UTILIZATION_GRAPH = """
  86. {% load helpers %}
  87. {% utilization_graph value %}
  88. """
  89. #
  90. # Regions
  91. #
  92. class RegionTable(BaseTable):
  93. pk = ToggleColumn()
  94. name = tables.TemplateColumn(template_code=REGION_LINK, orderable=False)
  95. site_count = tables.Column(verbose_name='Sites')
  96. slug = tables.Column(verbose_name='Slug')
  97. actions = tables.TemplateColumn(
  98. template_code=REGION_ACTIONS,
  99. attrs={'td': {'class': 'text-right'}},
  100. verbose_name=''
  101. )
  102. class Meta(BaseTable.Meta):
  103. model = Region
  104. fields = ('pk', 'name', 'site_count', 'slug', 'actions')
  105. #
  106. # Sites
  107. #
  108. class SiteTable(BaseTable):
  109. pk = ToggleColumn()
  110. name = tables.LinkColumn('dcim:site', args=[Accessor('slug')], verbose_name='Name')
  111. facility = tables.Column(verbose_name='Facility')
  112. region = tables.TemplateColumn(template_code=SITE_REGION_LINK, verbose_name='Region')
  113. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  114. asn = tables.Column(verbose_name='ASN')
  115. rack_count = tables.Column(accessor=Accessor('count_racks'), orderable=False, verbose_name='Racks')
  116. device_count = tables.Column(accessor=Accessor('count_devices'), orderable=False, verbose_name='Devices')
  117. prefix_count = tables.Column(accessor=Accessor('count_prefixes'), orderable=False, verbose_name='Prefixes')
  118. vlan_count = tables.Column(accessor=Accessor('count_vlans'), orderable=False, verbose_name='VLANs')
  119. circuit_count = tables.Column(accessor=Accessor('count_circuits'), orderable=False, verbose_name='Circuits')
  120. class Meta(BaseTable.Meta):
  121. model = Site
  122. fields = (
  123. 'pk', 'name', 'facility', 'region', 'tenant', 'asn', 'rack_count', 'device_count', 'prefix_count',
  124. 'vlan_count', 'circuit_count',
  125. )
  126. #
  127. # Rack groups
  128. #
  129. class RackGroupTable(BaseTable):
  130. pk = ToggleColumn()
  131. name = tables.LinkColumn(verbose_name='Name')
  132. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  133. rack_count = tables.Column(verbose_name='Racks')
  134. slug = tables.Column(verbose_name='Slug')
  135. actions = tables.TemplateColumn(template_code=RACKGROUP_ACTIONS, attrs={'td': {'class': 'text-right'}},
  136. verbose_name='')
  137. class Meta(BaseTable.Meta):
  138. model = RackGroup
  139. fields = ('pk', 'name', 'site', 'rack_count', 'slug', 'actions')
  140. #
  141. # Rack roles
  142. #
  143. class RackRoleTable(BaseTable):
  144. pk = ToggleColumn()
  145. name = tables.LinkColumn(verbose_name='Name')
  146. rack_count = tables.Column(verbose_name='Racks')
  147. color = tables.TemplateColumn(COLOR_LABEL, verbose_name='Color')
  148. slug = tables.Column(verbose_name='Slug')
  149. actions = tables.TemplateColumn(template_code=RACKROLE_ACTIONS, attrs={'td': {'class': 'text-right'}},
  150. verbose_name='')
  151. class Meta(BaseTable.Meta):
  152. model = RackGroup
  153. fields = ('pk', 'name', 'rack_count', 'color', 'slug', 'actions')
  154. #
  155. # Racks
  156. #
  157. class RackTable(BaseTable):
  158. pk = ToggleColumn()
  159. name = tables.LinkColumn('dcim:rack', args=[Accessor('pk')], verbose_name='Name')
  160. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  161. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  162. facility_id = tables.Column(verbose_name='Facility ID')
  163. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  164. role = tables.TemplateColumn(RACK_ROLE, verbose_name='Role')
  165. u_height = tables.TemplateColumn("{{ record.u_height }}U", verbose_name='Height')
  166. devices = tables.Column(accessor=Accessor('device_count'), verbose_name='Devices')
  167. get_utilization = tables.TemplateColumn(UTILIZATION_GRAPH, orderable=False, verbose_name='Utilization')
  168. class Meta(BaseTable.Meta):
  169. model = Rack
  170. fields = ('pk', 'name', 'site', 'group', 'facility_id', 'tenant', 'role', 'u_height', 'devices',
  171. 'get_utilization')
  172. class RackImportTable(BaseTable):
  173. name = tables.LinkColumn('dcim:rack', args=[Accessor('pk')], verbose_name='Name')
  174. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  175. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  176. facility_id = tables.Column(verbose_name='Facility ID')
  177. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  178. u_height = tables.Column(verbose_name='Height (U)')
  179. class Meta(BaseTable.Meta):
  180. model = Rack
  181. fields = ('site', 'group', 'name', 'facility_id', 'tenant', 'u_height')
  182. #
  183. # Manufacturers
  184. #
  185. class ManufacturerTable(BaseTable):
  186. pk = ToggleColumn()
  187. name = tables.LinkColumn(verbose_name='Name')
  188. devicetype_count = tables.Column(verbose_name='Device Types')
  189. slug = tables.Column(verbose_name='Slug')
  190. actions = tables.TemplateColumn(template_code=MANUFACTURER_ACTIONS, attrs={'td': {'class': 'text-right'}},
  191. verbose_name='')
  192. class Meta(BaseTable.Meta):
  193. model = Manufacturer
  194. fields = ('pk', 'name', 'devicetype_count', 'slug', 'actions')
  195. #
  196. # Device types
  197. #
  198. class DeviceTypeTable(BaseTable):
  199. pk = ToggleColumn()
  200. manufacturer = tables.Column(verbose_name='Manufacturer')
  201. model = tables.LinkColumn('dcim:devicetype', args=[Accessor('pk')], verbose_name='Device Type')
  202. part_number = tables.Column(verbose_name='Part Number')
  203. is_full_depth = tables.BooleanColumn(verbose_name='Full Depth')
  204. instance_count = tables.Column(verbose_name='Instances')
  205. class Meta(BaseTable.Meta):
  206. model = DeviceType
  207. fields = ('pk', 'model', 'manufacturer', 'part_number', 'u_height', 'is_full_depth', 'instance_count')
  208. #
  209. # Device type components
  210. #
  211. class ConsolePortTemplateTable(BaseTable):
  212. pk = ToggleColumn()
  213. class Meta(BaseTable.Meta):
  214. model = ConsolePortTemplate
  215. fields = ('pk', 'name')
  216. empty_text = "None"
  217. show_header = False
  218. class ConsoleServerPortTemplateTable(BaseTable):
  219. pk = ToggleColumn()
  220. class Meta(BaseTable.Meta):
  221. model = ConsoleServerPortTemplate
  222. fields = ('pk', 'name')
  223. empty_text = "None"
  224. show_header = False
  225. class PowerPortTemplateTable(BaseTable):
  226. pk = ToggleColumn()
  227. class Meta(BaseTable.Meta):
  228. model = PowerPortTemplate
  229. fields = ('pk', 'name')
  230. empty_text = "None"
  231. show_header = False
  232. class PowerOutletTemplateTable(BaseTable):
  233. pk = ToggleColumn()
  234. class Meta(BaseTable.Meta):
  235. model = PowerOutletTemplate
  236. fields = ('pk', 'name')
  237. empty_text = "None"
  238. show_header = False
  239. class InterfaceTemplateTable(BaseTable):
  240. pk = ToggleColumn()
  241. class Meta(BaseTable.Meta):
  242. model = InterfaceTemplate
  243. fields = ('pk', 'name', 'form_factor')
  244. empty_text = "None"
  245. show_header = False
  246. class DeviceBayTemplateTable(BaseTable):
  247. pk = ToggleColumn()
  248. class Meta(BaseTable.Meta):
  249. model = DeviceBayTemplate
  250. fields = ('pk', 'name')
  251. empty_text = "None"
  252. show_header = False
  253. #
  254. # Device roles
  255. #
  256. class DeviceRoleTable(BaseTable):
  257. pk = ToggleColumn()
  258. name = tables.LinkColumn(verbose_name='Name')
  259. device_count = tables.Column(verbose_name='Devices')
  260. color = tables.TemplateColumn(COLOR_LABEL, verbose_name='Color')
  261. slug = tables.Column(verbose_name='Slug')
  262. actions = tables.TemplateColumn(template_code=DEVICEROLE_ACTIONS, attrs={'td': {'class': 'text-right'}},
  263. verbose_name='')
  264. class Meta(BaseTable.Meta):
  265. model = DeviceRole
  266. fields = ('pk', 'name', 'device_count', 'color', 'slug', 'actions')
  267. #
  268. # Platforms
  269. #
  270. class PlatformTable(BaseTable):
  271. pk = ToggleColumn()
  272. name = tables.LinkColumn(verbose_name='Name')
  273. device_count = tables.Column(verbose_name='Devices')
  274. slug = tables.Column(verbose_name='Slug')
  275. actions = tables.TemplateColumn(template_code=PLATFORM_ACTIONS, attrs={'td': {'class': 'text-right'}},
  276. verbose_name='')
  277. class Meta(BaseTable.Meta):
  278. model = Platform
  279. fields = ('pk', 'name', 'device_count', 'slug', 'actions')
  280. #
  281. # Devices
  282. #
  283. class DeviceTable(BaseTable):
  284. pk = ToggleColumn()
  285. status = tables.TemplateColumn(template_code=STATUS_ICON, verbose_name='')
  286. name = tables.TemplateColumn(template_code=DEVICE_LINK, verbose_name='Name')
  287. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  288. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  289. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')], verbose_name='Rack')
  290. device_role = tables.TemplateColumn(DEVICE_ROLE, verbose_name='Role')
  291. device_type = tables.LinkColumn('dcim:devicetype', args=[Accessor('device_type.pk')], verbose_name='Type',
  292. text=lambda record: record.device_type.full_name)
  293. primary_ip = tables.TemplateColumn(orderable=False, verbose_name='IP Address',
  294. template_code=DEVICE_PRIMARY_IP)
  295. class Meta(BaseTable.Meta):
  296. model = Device
  297. fields = ('pk', 'name', 'status', 'tenant', 'site', 'rack', 'device_role', 'device_type', 'primary_ip')
  298. class DeviceImportTable(BaseTable):
  299. name = tables.TemplateColumn(template_code=DEVICE_LINK, verbose_name='Name')
  300. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  301. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  302. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')], verbose_name='Rack')
  303. position = tables.Column(verbose_name='Position')
  304. device_role = tables.Column(verbose_name='Role')
  305. device_type = tables.Column(verbose_name='Type')
  306. class Meta(BaseTable.Meta):
  307. model = Device
  308. fields = ('name', 'tenant', 'site', 'rack', 'position', 'device_role', 'device_type')
  309. empty_text = False
  310. #
  311. # Device connections
  312. #
  313. class ConsoleConnectionTable(BaseTable):
  314. console_server = tables.LinkColumn('dcim:device', accessor=Accessor('cs_port.device'),
  315. args=[Accessor('cs_port.device.pk')], verbose_name='Console server')
  316. cs_port = tables.Column(verbose_name='Port')
  317. device = tables.LinkColumn('dcim:device', args=[Accessor('device.pk')], verbose_name='Device')
  318. name = tables.Column(verbose_name='Console port')
  319. class Meta(BaseTable.Meta):
  320. model = ConsolePort
  321. fields = ('console_server', 'cs_port', 'device', 'name')
  322. class PowerConnectionTable(BaseTable):
  323. pdu = tables.LinkColumn('dcim:device', accessor=Accessor('power_outlet.device'),
  324. args=[Accessor('power_outlet.device.pk')], verbose_name='PDU')
  325. power_outlet = tables.Column(verbose_name='Outlet')
  326. device = tables.LinkColumn('dcim:device', args=[Accessor('device.pk')], verbose_name='Device')
  327. name = tables.Column(verbose_name='Power Port')
  328. class Meta(BaseTable.Meta):
  329. model = PowerPort
  330. fields = ('pdu', 'power_outlet', 'device', 'name')
  331. class InterfaceConnectionTable(BaseTable):
  332. device_a = tables.LinkColumn('dcim:device', accessor=Accessor('interface_a.device'),
  333. args=[Accessor('interface_a.device.pk')], verbose_name='Device A')
  334. interface_a = tables.Column(verbose_name='Interface A')
  335. device_b = tables.LinkColumn('dcim:device', accessor=Accessor('interface_b.device'),
  336. args=[Accessor('interface_b.device.pk')], verbose_name='Device B')
  337. interface_b = tables.Column(verbose_name='Interface B')
  338. class Meta(BaseTable.Meta):
  339. model = Interface
  340. fields = ('device_a', 'interface_a', 'device_b', 'interface_b')