tables.py 18 KB

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