tables.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. import django_tables2 as tables
  2. from django_tables2.utils import Accessor
  3. from utilities.tables import BaseTable, SearchTable, ToggleColumn
  4. from .models import (
  5. ConsolePort, ConsolePortTemplate, ConsoleServerPortTemplate, Device, DeviceBayTemplate, DeviceRole, DeviceType,
  6. Interface, InterfaceTemplate, Manufacturer, Platform, PowerOutletTemplate, PowerPort, PowerPortTemplate, Rack,
  7. RackGroup, RackReservation, 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>
  12. {% else %}
  13. <span style="padding-left: {{ record.get_ancestors|length }}9px">
  14. {% endif %}
  15. <a href="{% url 'dcim:site_list' %}?region={{ record.slug }}">{{ record.name }}</a>
  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. RACKRESERVATION_ACTIONS = """
  56. {% if perms.dcim.change_rackreservation %}
  57. <a href="{% url 'dcim:rackreservation_edit' pk=record.pk %}" class="btn btn-xs btn-warning"><i class="glyphicon glyphicon-pencil" aria-hidden="true"></i></a>
  58. {% endif %}
  59. """
  60. DEVICEROLE_ACTIONS = """
  61. {% if perms.dcim.change_devicerole %}
  62. <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>
  63. {% endif %}
  64. """
  65. MANUFACTURER_ACTIONS = """
  66. {% if perms.dcim.change_manufacturer %}
  67. <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>
  68. {% endif %}
  69. """
  70. PLATFORM_ACTIONS = """
  71. {% if perms.dcim.change_platform %}
  72. <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>
  73. {% endif %}
  74. """
  75. DEVICE_ROLE = """
  76. <label class="label" style="background-color: #{{ record.device_role.color }}">{{ value }}</label>
  77. """
  78. STATUS_ICON = """
  79. {% if record.status %}
  80. <span class="glyphicon glyphicon-ok-sign text-success" title="Active" aria-hidden="true"></span>
  81. {% else %}
  82. <span class="glyphicon glyphicon-minus-sign text-danger" title="Offline" aria-hidden="true"></span>
  83. {% endif %}
  84. """
  85. DEVICE_PRIMARY_IP = """
  86. {{ record.primary_ip6.address.ip|default:"" }}
  87. {% if record.primary_ip6 and record.primary_ip4 %}<br />{% endif %}
  88. {{ record.primary_ip4.address.ip|default:"" }}
  89. """
  90. SUBDEVICE_ROLE_TEMPLATE = """
  91. {% if record.subdevice_role == True %}Parent{% elif record.subdevice_role == False %}Child{% else %}&mdash;{% endif %}
  92. """
  93. UTILIZATION_GRAPH = """
  94. {% load helpers %}
  95. {% utilization_graph value %}
  96. """
  97. #
  98. # Regions
  99. #
  100. class RegionTable(BaseTable):
  101. pk = ToggleColumn()
  102. name = tables.TemplateColumn(template_code=REGION_LINK, orderable=False)
  103. site_count = tables.Column(verbose_name='Sites')
  104. slug = tables.Column(verbose_name='Slug')
  105. actions = tables.TemplateColumn(
  106. template_code=REGION_ACTIONS,
  107. attrs={'td': {'class': 'text-right'}},
  108. verbose_name=''
  109. )
  110. class Meta(BaseTable.Meta):
  111. model = Region
  112. fields = ('pk', 'name', 'site_count', 'slug', 'actions')
  113. #
  114. # Sites
  115. #
  116. class SiteTable(BaseTable):
  117. pk = ToggleColumn()
  118. name = tables.LinkColumn()
  119. region = tables.TemplateColumn(template_code=SITE_REGION_LINK)
  120. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  121. rack_count = tables.Column(accessor=Accessor('count_racks'), orderable=False, verbose_name='Racks')
  122. device_count = tables.Column(accessor=Accessor('count_devices'), orderable=False, verbose_name='Devices')
  123. prefix_count = tables.Column(accessor=Accessor('count_prefixes'), orderable=False, verbose_name='Prefixes')
  124. vlan_count = tables.Column(accessor=Accessor('count_vlans'), orderable=False, verbose_name='VLANs')
  125. circuit_count = tables.Column(accessor=Accessor('count_circuits'), orderable=False, verbose_name='Circuits')
  126. class Meta(BaseTable.Meta):
  127. model = Site
  128. fields = (
  129. 'pk', 'name', 'facility', 'region', 'tenant', 'asn', 'rack_count', 'device_count', 'prefix_count',
  130. 'vlan_count', 'circuit_count',
  131. )
  132. class SiteSearchTable(SearchTable):
  133. name = tables.LinkColumn()
  134. region = tables.TemplateColumn(template_code=SITE_REGION_LINK)
  135. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  136. class Meta(SearchTable.Meta):
  137. model = Site
  138. fields = ('name', 'facility', 'region', 'tenant', 'asn')
  139. #
  140. # Rack groups
  141. #
  142. class RackGroupTable(BaseTable):
  143. pk = ToggleColumn()
  144. name = tables.LinkColumn(verbose_name='Name')
  145. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  146. rack_count = tables.Column(verbose_name='Racks')
  147. slug = tables.Column(verbose_name='Slug')
  148. actions = tables.TemplateColumn(template_code=RACKGROUP_ACTIONS, attrs={'td': {'class': 'text-right'}},
  149. verbose_name='')
  150. class Meta(BaseTable.Meta):
  151. model = RackGroup
  152. fields = ('pk', 'name', 'site', 'rack_count', 'slug', 'actions')
  153. #
  154. # Rack roles
  155. #
  156. class RackRoleTable(BaseTable):
  157. pk = ToggleColumn()
  158. name = tables.LinkColumn(verbose_name='Name')
  159. rack_count = tables.Column(verbose_name='Racks')
  160. color = tables.TemplateColumn(COLOR_LABEL, verbose_name='Color')
  161. slug = tables.Column(verbose_name='Slug')
  162. actions = tables.TemplateColumn(template_code=RACKROLE_ACTIONS, attrs={'td': {'class': 'text-right'}},
  163. verbose_name='')
  164. class Meta(BaseTable.Meta):
  165. model = RackGroup
  166. fields = ('pk', 'name', 'rack_count', 'color', 'slug', 'actions')
  167. #
  168. # Racks
  169. #
  170. class RackTable(BaseTable):
  171. pk = ToggleColumn()
  172. name = tables.LinkColumn()
  173. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  174. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  175. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  176. role = tables.TemplateColumn(RACK_ROLE)
  177. u_height = tables.TemplateColumn("{{ record.u_height }}U", verbose_name='Height')
  178. devices = tables.Column(accessor=Accessor('device_count'))
  179. get_utilization = tables.TemplateColumn(UTILIZATION_GRAPH, orderable=False, verbose_name='Utilization')
  180. class Meta(BaseTable.Meta):
  181. model = Rack
  182. fields = (
  183. 'pk', 'name', 'site', 'group', 'facility_id', 'tenant', 'role', 'u_height', 'devices', 'get_utilization'
  184. )
  185. class RackSearchTable(SearchTable):
  186. name = tables.LinkColumn()
  187. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  188. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  189. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  190. role = tables.TemplateColumn(RACK_ROLE)
  191. u_height = tables.TemplateColumn("{{ record.u_height }}U", verbose_name='Height')
  192. class Meta(SearchTable.Meta):
  193. model = Rack
  194. fields = ('name', 'site', 'group', 'facility_id', 'tenant', 'role', 'u_height')
  195. class RackImportTable(BaseTable):
  196. name = tables.LinkColumn('dcim:rack', args=[Accessor('pk')], verbose_name='Name')
  197. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  198. group = tables.Column(accessor=Accessor('group.name'), verbose_name='Group')
  199. facility_id = tables.Column(verbose_name='Facility ID')
  200. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  201. u_height = tables.Column(verbose_name='Height (U)')
  202. class Meta(BaseTable.Meta):
  203. model = Rack
  204. fields = ('site', 'group', 'name', 'facility_id', 'tenant', 'u_height')
  205. #
  206. # Rack reservations
  207. #
  208. class RackReservationTable(BaseTable):
  209. pk = ToggleColumn()
  210. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')])
  211. unit_list = tables.Column(orderable=False, verbose_name='Units')
  212. actions = tables.TemplateColumn(
  213. template_code=RACKRESERVATION_ACTIONS, attrs={'td': {'class': 'text-right'}}, verbose_name=''
  214. )
  215. class Meta(BaseTable.Meta):
  216. model = RackReservation
  217. fields = ('pk', 'rack', 'unit_list', 'user', 'created', 'description', 'actions')
  218. #
  219. # Manufacturers
  220. #
  221. class ManufacturerTable(BaseTable):
  222. pk = ToggleColumn()
  223. name = tables.LinkColumn(verbose_name='Name')
  224. devicetype_count = tables.Column(verbose_name='Device Types')
  225. slug = tables.Column(verbose_name='Slug')
  226. actions = tables.TemplateColumn(template_code=MANUFACTURER_ACTIONS, attrs={'td': {'class': 'text-right'}},
  227. verbose_name='')
  228. class Meta(BaseTable.Meta):
  229. model = Manufacturer
  230. fields = ('pk', 'name', 'devicetype_count', 'slug', 'actions')
  231. #
  232. # Device types
  233. #
  234. class DeviceTypeTable(BaseTable):
  235. pk = ToggleColumn()
  236. model = tables.LinkColumn('dcim:devicetype', args=[Accessor('pk')], verbose_name='Device Type')
  237. is_full_depth = tables.BooleanColumn(verbose_name='Full Depth')
  238. is_console_server = tables.BooleanColumn(verbose_name='CS')
  239. is_pdu = tables.BooleanColumn(verbose_name='PDU')
  240. is_network_device = tables.BooleanColumn(verbose_name='Net')
  241. subdevice_role = tables.TemplateColumn(SUBDEVICE_ROLE_TEMPLATE, verbose_name='Subdevice Role')
  242. instance_count = tables.Column(verbose_name='Instances')
  243. class Meta(BaseTable.Meta):
  244. model = DeviceType
  245. fields = (
  246. 'pk', 'model', 'manufacturer', 'part_number', 'u_height', 'is_full_depth', 'is_console_server', 'is_pdu',
  247. 'is_network_device', 'subdevice_role', 'instance_count'
  248. )
  249. class DeviceTypeSearchTable(SearchTable):
  250. model = tables.LinkColumn('dcim:devicetype', args=[Accessor('pk')], verbose_name='Device Type')
  251. is_full_depth = tables.BooleanColumn(verbose_name='Full Depth')
  252. is_console_server = tables.BooleanColumn(verbose_name='CS')
  253. is_pdu = tables.BooleanColumn(verbose_name='PDU')
  254. is_network_device = tables.BooleanColumn(verbose_name='Net')
  255. subdevice_role = tables.TemplateColumn(SUBDEVICE_ROLE_TEMPLATE, verbose_name='Subdevice Role')
  256. class Meta(SearchTable.Meta):
  257. model = DeviceType
  258. fields = (
  259. 'model', 'manufacturer', 'part_number', 'u_height', 'is_full_depth', 'is_console_server', 'is_pdu',
  260. 'is_network_device', 'subdevice_role',
  261. )
  262. #
  263. # Device type components
  264. #
  265. class ConsolePortTemplateTable(BaseTable):
  266. pk = ToggleColumn()
  267. class Meta(BaseTable.Meta):
  268. model = ConsolePortTemplate
  269. fields = ('pk', 'name')
  270. empty_text = "None"
  271. show_header = False
  272. class ConsoleServerPortTemplateTable(BaseTable):
  273. pk = ToggleColumn()
  274. class Meta(BaseTable.Meta):
  275. model = ConsoleServerPortTemplate
  276. fields = ('pk', 'name')
  277. empty_text = "None"
  278. show_header = False
  279. class PowerPortTemplateTable(BaseTable):
  280. pk = ToggleColumn()
  281. class Meta(BaseTable.Meta):
  282. model = PowerPortTemplate
  283. fields = ('pk', 'name')
  284. empty_text = "None"
  285. show_header = False
  286. class PowerOutletTemplateTable(BaseTable):
  287. pk = ToggleColumn()
  288. class Meta(BaseTable.Meta):
  289. model = PowerOutletTemplate
  290. fields = ('pk', 'name')
  291. empty_text = "None"
  292. show_header = False
  293. class InterfaceTemplateTable(BaseTable):
  294. pk = ToggleColumn()
  295. class Meta(BaseTable.Meta):
  296. model = InterfaceTemplate
  297. fields = ('pk', 'name', 'form_factor')
  298. empty_text = "None"
  299. show_header = False
  300. class DeviceBayTemplateTable(BaseTable):
  301. pk = ToggleColumn()
  302. class Meta(BaseTable.Meta):
  303. model = DeviceBayTemplate
  304. fields = ('pk', 'name')
  305. empty_text = "None"
  306. show_header = False
  307. #
  308. # Device roles
  309. #
  310. class DeviceRoleTable(BaseTable):
  311. pk = ToggleColumn()
  312. name = tables.LinkColumn(verbose_name='Name')
  313. device_count = tables.Column(verbose_name='Devices')
  314. color = tables.TemplateColumn(COLOR_LABEL, verbose_name='Color')
  315. slug = tables.Column(verbose_name='Slug')
  316. actions = tables.TemplateColumn(template_code=DEVICEROLE_ACTIONS, attrs={'td': {'class': 'text-right'}},
  317. verbose_name='')
  318. class Meta(BaseTable.Meta):
  319. model = DeviceRole
  320. fields = ('pk', 'name', 'device_count', 'color', 'slug', 'actions')
  321. #
  322. # Platforms
  323. #
  324. class PlatformTable(BaseTable):
  325. pk = ToggleColumn()
  326. name = tables.LinkColumn(verbose_name='Name')
  327. device_count = tables.Column(verbose_name='Devices')
  328. slug = tables.Column(verbose_name='Slug')
  329. rpc_client = tables.Column(accessor='get_rpc_client_display', orderable=False, verbose_name='RPC Client')
  330. actions = tables.TemplateColumn(template_code=PLATFORM_ACTIONS, attrs={'td': {'class': 'text-right'}},
  331. verbose_name='')
  332. class Meta(BaseTable.Meta):
  333. model = Platform
  334. fields = ('pk', 'name', 'device_count', 'slug', 'rpc_client', 'actions')
  335. #
  336. # Devices
  337. #
  338. class DeviceTable(BaseTable):
  339. pk = ToggleColumn()
  340. name = tables.TemplateColumn(template_code=DEVICE_LINK)
  341. status = tables.TemplateColumn(template_code=STATUS_ICON, verbose_name='')
  342. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  343. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  344. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')])
  345. device_role = tables.TemplateColumn(DEVICE_ROLE, verbose_name='Role')
  346. device_type = tables.LinkColumn(
  347. 'dcim:devicetype', args=[Accessor('device_type.pk')], verbose_name='Type',
  348. text=lambda record: record.device_type.full_name
  349. )
  350. primary_ip = tables.TemplateColumn(
  351. orderable=False, verbose_name='IP Address', template_code=DEVICE_PRIMARY_IP
  352. )
  353. class Meta(BaseTable.Meta):
  354. model = Device
  355. fields = ('pk', 'name', 'status', 'tenant', 'site', 'rack', 'device_role', 'device_type', 'primary_ip')
  356. class DeviceSearchTable(SearchTable):
  357. name = tables.TemplateColumn(template_code=DEVICE_LINK)
  358. status = tables.TemplateColumn(template_code=STATUS_ICON, verbose_name='')
  359. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')])
  360. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')])
  361. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')])
  362. device_role = tables.TemplateColumn(DEVICE_ROLE, verbose_name='Role')
  363. device_type = tables.LinkColumn(
  364. 'dcim:devicetype', args=[Accessor('device_type.pk')], verbose_name='Type',
  365. text=lambda record: record.device_type.full_name
  366. )
  367. class Meta(SearchTable.Meta):
  368. model = Device
  369. fields = ('name', 'status', 'tenant', 'site', 'rack', 'device_role', 'device_type')
  370. class DeviceImportTable(BaseTable):
  371. name = tables.TemplateColumn(template_code=DEVICE_LINK, verbose_name='Name')
  372. tenant = tables.LinkColumn('tenancy:tenant', args=[Accessor('tenant.slug')], verbose_name='Tenant')
  373. site = tables.LinkColumn('dcim:site', args=[Accessor('site.slug')], verbose_name='Site')
  374. rack = tables.LinkColumn('dcim:rack', args=[Accessor('rack.pk')], verbose_name='Rack')
  375. position = tables.Column(verbose_name='Position')
  376. device_role = tables.Column(verbose_name='Role')
  377. device_type = tables.Column(verbose_name='Type')
  378. class Meta(BaseTable.Meta):
  379. model = Device
  380. fields = ('name', 'tenant', 'site', 'rack', 'position', 'device_role', 'device_type')
  381. empty_text = False
  382. #
  383. # Device connections
  384. #
  385. class ConsoleConnectionTable(BaseTable):
  386. console_server = tables.LinkColumn('dcim:device', accessor=Accessor('cs_port.device'),
  387. args=[Accessor('cs_port.device.pk')], verbose_name='Console server')
  388. cs_port = tables.Column(verbose_name='Port')
  389. device = tables.LinkColumn('dcim:device', args=[Accessor('device.pk')], verbose_name='Device')
  390. name = tables.Column(verbose_name='Console port')
  391. class Meta(BaseTable.Meta):
  392. model = ConsolePort
  393. fields = ('console_server', 'cs_port', 'device', 'name')
  394. class PowerConnectionTable(BaseTable):
  395. pdu = tables.LinkColumn('dcim:device', accessor=Accessor('power_outlet.device'),
  396. args=[Accessor('power_outlet.device.pk')], verbose_name='PDU')
  397. power_outlet = tables.Column(verbose_name='Outlet')
  398. device = tables.LinkColumn('dcim:device', args=[Accessor('device.pk')], verbose_name='Device')
  399. name = tables.Column(verbose_name='Power Port')
  400. class Meta(BaseTable.Meta):
  401. model = PowerPort
  402. fields = ('pdu', 'power_outlet', 'device', 'name')
  403. class InterfaceConnectionTable(BaseTable):
  404. device_a = tables.LinkColumn('dcim:device', accessor=Accessor('interface_a.device'),
  405. args=[Accessor('interface_a.device.pk')], verbose_name='Device A')
  406. interface_a = tables.Column(verbose_name='Interface A')
  407. device_b = tables.LinkColumn('dcim:device', accessor=Accessor('interface_b.device'),
  408. args=[Accessor('interface_b.device.pk')], verbose_name='Device B')
  409. interface_b = tables.Column(verbose_name='Interface B')
  410. class Meta(BaseTable.Meta):
  411. model = Interface
  412. fields = ('device_a', 'interface_a', 'device_b', 'interface_b')