2
0

rack.html 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. {% extends '_base.html' %}
  2. {% load buttons %}
  3. {% load custom_links %}
  4. {% load helpers %}
  5. {% load static %}
  6. {% block header %}
  7. <div class="row noprint">
  8. <div class="col-sm-8 col-md-9">
  9. <ol class="breadcrumb">
  10. <li><a href="{% url 'dcim:rack_list' %}">Racks</a></li>
  11. <li><a href="{% url 'dcim:rack_list' %}?site={{ rack.site.slug }}">{{ rack.site }}</a></li>
  12. <li>{{ rack }}</li>
  13. </ol>
  14. </div>
  15. <div class="col-sm-4 col-md-3">
  16. <form action="{% url 'dcim:rack_list' %}" method="get">
  17. <div class="input-group">
  18. <input type="text" name="q" class="form-control" placeholder="Search racks" />
  19. <span class="input-group-btn">
  20. <button type="submit" class="btn btn-primary">
  21. <span class="fa fa-search" aria-hidden="true"></span>
  22. </button>
  23. </span>
  24. </div>
  25. </form>
  26. </div>
  27. </div>
  28. <div class="pull-right noprint">
  29. <a {% if prev_rack %}href="{% url 'dcim:rack' pk=prev_rack.pk %}"{% else %}disabled="disabled"{% endif %} class="btn btn-primary">
  30. <span class="fa fa-chevron-left" aria-hidden="true"></span> Previous Rack
  31. </a>
  32. <a {% if next_rack %}href="{% url 'dcim:rack' pk=next_rack.pk %}"{% else %}disabled="disabled"{% endif %} class="btn btn-primary">
  33. <span class="fa fa-chevron-right" aria-hidden="true"></span> Next Rack
  34. </a>
  35. {% if perms.dcim.add_rack %}
  36. {% clone_button rack %}
  37. {% endif %}
  38. {% if perms.dcim.change_rack %}
  39. {% edit_button rack %}
  40. {% endif %}
  41. {% if perms.dcim.delete_rack %}
  42. {% delete_button rack %}
  43. {% endif %}
  44. </div>
  45. <h1>{% block title %}Rack {{ rack }}{% endblock %}</h1>
  46. {% include 'inc/created_updated.html' with obj=rack %}
  47. <div class="pull-right noprint">
  48. <button class="btn btn-sm btn-default toggle-images" selected="selected">
  49. <span class="glyphicon glyphicon-check" aria-hidden="true"></span> Show Images
  50. </button>
  51. {% custom_links rack %}
  52. </div>
  53. <ul class="nav nav-tabs">
  54. <li role="presentation"{% if not active_tab %} class="active"{% endif %}>
  55. <a href="{{ rack.get_absolute_url }}">Rack</a>
  56. </li>
  57. {% if perms.extras.view_objectchange %}
  58. <li role="presentation"{% if active_tab == 'changelog' %} class="active"{% endif %}>
  59. <a href="{% url 'dcim:rack_changelog' pk=rack.pk %}">Change Log</a>
  60. </li>
  61. {% endif %}
  62. </ul>
  63. {% endblock %}
  64. {% block content %}
  65. <div class="row">
  66. <div class="col-md-6">
  67. <div class="panel panel-default">
  68. <div class="panel-heading">
  69. <strong>Rack</strong>
  70. </div>
  71. <table class="table table-hover panel-body attr-table">
  72. <tr>
  73. <td>Site</td>
  74. <td>
  75. {% if rack.site.region %}
  76. <a href="{{ rack.site.region.get_absolute_url }}">{{ rack.site.region }}</a>
  77. <i class="fa fa-angle-right"></i>
  78. {% endif %}
  79. <a href="{% url 'dcim:site' slug=rack.site.slug %}">{{ rack.site }}</a>
  80. </td>
  81. </tr>
  82. <tr>
  83. <td>Group</td>
  84. <td>
  85. {% if rack.group %}
  86. <a href="{% url 'dcim:rack_list' %}?site={{ rack.site.slug }}&group={{ rack.group.slug }}">{{ rack.group }}</a>
  87. {% else %}
  88. <span class="text-muted">None</span>
  89. {% endif %}
  90. </td>
  91. </tr>
  92. <tr>
  93. <td>Facility ID</td>
  94. <td>{{ rack.facility_id|placeholder }}</td>
  95. </tr>
  96. <tr>
  97. <td>Tenant</td>
  98. <td>
  99. {% if rack.tenant %}
  100. {% if rack.tenant.group %}
  101. <a href="{{ rack.tenant.group.get_absolute_url }}">{{ rack.tenant.group }}</a>
  102. <i class="fa fa-angle-right"></i>
  103. {% endif %}
  104. <a href="{{ rack.tenant.get_absolute_url }}">{{ rack.tenant }}</a>
  105. {% else %}
  106. <span class="text-muted">None</span>
  107. {% endif %}
  108. </td>
  109. </tr>
  110. <tr>
  111. <td>Status</td>
  112. <td>
  113. {{ rack.get_status_display }}
  114. </td>
  115. </tr>
  116. <tr>
  117. <td>Role</td>
  118. <td>
  119. {% if rack.role %}
  120. <a href="{{ rack.role.get_absolute_url }}">{{ rack.role }}</a>
  121. {% else %}
  122. <span class="text-muted">None</span>
  123. {% endif %}
  124. </td>
  125. </tr>
  126. <tr>
  127. <td>Serial Number</td>
  128. <td>{{ rack.serial|placeholder }}</td>
  129. </tr>
  130. <tr>
  131. <td>Asset Tag</td>
  132. <td>{{ rack.asset_tag|placeholder }}</td>
  133. </tr>
  134. <tr>
  135. <td>Devices</td>
  136. <td>
  137. <a href="{% url 'dcim:device_list' %}?rack_id={{ rack.id }}">{{ rack.devices.count }}</a>
  138. </td>
  139. </tr>
  140. <tr>
  141. <td>Utilization</td>
  142. <td>{% utilization_graph rack.get_utilization %}</td>
  143. </tr>
  144. </table>
  145. </div>
  146. <div class="panel panel-default">
  147. <div class="panel-heading">
  148. <strong>Dimensions</strong>
  149. </div>
  150. <table class="table table-hover panel-body attr-table">
  151. <tr>
  152. <td>Type</td>
  153. <td>
  154. {% if rack.type %}
  155. {{ rack.get_type_display }}
  156. {% else %}
  157. <span class="text-muted">None</span>
  158. {% endif %}
  159. </td>
  160. </tr>
  161. <tr>
  162. <td>Width</td>
  163. <td>{{ rack.get_width_display }}</td>
  164. </tr>
  165. <tr>
  166. <td>Height</td>
  167. <td>{{ rack.u_height }}U ({% if rack.desc_units %}descending{% else %}ascending{% endif %})</td>
  168. </tr>
  169. <tr>
  170. <td>Outer Width</td>
  171. <td>
  172. {% if rack.outer_width %}
  173. <span>{{ rack.outer_width }} {{ rack.get_outer_unit_display }}</span>
  174. {% else %}
  175. <span class="text-muted">&mdash;</span>
  176. {% endif %}
  177. </td>
  178. </tr>
  179. <tr>
  180. <td>Outer Depth</td>
  181. <td>
  182. {% if rack.outer_depth %}
  183. <span>{{ rack.outer_depth }} {{ rack.get_outer_unit_display }}</span>
  184. {% else %}
  185. <span class="text-muted">&mdash;</span>
  186. {% endif %}
  187. </td>
  188. </tr>
  189. </table>
  190. </div>
  191. {% include 'inc/custom_fields_panel.html' with obj=rack %}
  192. {% include 'extras/inc/tags_panel.html' with tags=rack.tags.all url='dcim:rack_list' %}
  193. <div class="panel panel-default">
  194. <div class="panel-heading">
  195. <strong>Comments</strong>
  196. </div>
  197. <div class="panel-body rendered-markdown">
  198. {% if rack.comments %}
  199. {{ rack.comments|render_markdown }}
  200. {% else %}
  201. <span class="text-muted">None</span>
  202. {% endif %}
  203. </div>
  204. </div>
  205. {% if power_feeds %}
  206. <div class="panel panel-default">
  207. <div class="panel-heading">
  208. <strong>Power Feeds</strong>
  209. </div>
  210. <table class="table panel-body">
  211. <tr>
  212. <th>Panel</th>
  213. <th>Feed</th>
  214. <th>Status</th>
  215. <th>Type</th>
  216. <th>Utilization</th>
  217. </tr>
  218. {% for powerfeed in power_feeds %}
  219. <tr>
  220. <td>
  221. <a href="{{ powerfeed.power_panel.get_absolute_url }}">{{ powerfeed.power_panel.name }}</a>
  222. <td>
  223. <a href="{{ powerfeed.get_absolute_url }}">{{ powerfeed.name }}</a>
  224. </td>
  225. <td>
  226. <span class="label label-{{ powerfeed.get_status_class }}">{{ powerfeed.get_status_display }}</span>
  227. </td>
  228. <td>
  229. <span class="label label-{{ powerfeed.get_type_class }}">{{ powerfeed.get_type_display }}</span>
  230. </td>
  231. {% with power_port=powerfeed.connected_endpoint %}
  232. {% if power_port %}
  233. <td>{% utilization_graph power_port.get_power_draw.allocated|percentage:powerfeed.available_power %}</td>
  234. {% else %}
  235. <td class="text-muted">N/A</td>
  236. {% endif %}
  237. {% endwith %}
  238. </tr>
  239. {% endfor %}
  240. </table>
  241. </div>
  242. {% endif %}
  243. <div class="panel panel-default">
  244. <div class="panel-heading">
  245. <strong>Images</strong>
  246. </div>
  247. {% include 'inc/image_attachments.html' with images=rack.images.all %}
  248. {% if perms.extras.add_imageattachment %}
  249. <div class="panel-footer text-right noprint">
  250. <a href="{% url 'dcim:rack_add_image' object_id=rack.pk %}" class="btn btn-primary btn-xs">
  251. <span class="glyphicon glyphicon-plus" aria-hidden="true"></span>
  252. Attach an image
  253. </a>
  254. </div>
  255. {% endif %}
  256. </div>
  257. <div class="panel panel-default">
  258. <div class="panel-heading">
  259. <strong>Reservations</strong>
  260. </div>
  261. {% if reservations %}
  262. <table class="table table-hover panel-body">
  263. <tr>
  264. <th>Units</th>
  265. <th>Tenant</th>
  266. <th>Description</th>
  267. <th></th>
  268. </tr>
  269. {% for resv in reservations %}
  270. <tr>
  271. <td>
  272. <a href="{{ resv.get_absolute_url }}">{{ resv.unit_list }}</a>
  273. </td>
  274. <td>
  275. {% if resv.tenant %}
  276. <a href="{{ resv.tenant.get_absolute_url }}">{{ resv.tenant }}</a>
  277. {% else %}
  278. <span class="text-muted">None</span>
  279. {% endif %}
  280. </td>
  281. <td>
  282. {{ resv.description }}<br />
  283. <small>{{ resv.user }} &middot; {{ resv.created }}</small>
  284. </td>
  285. <td class="text-right noprint">
  286. {% if perms.dcim.change_rackreservation %}
  287. <a href="{% url 'dcim:rackreservation_edit' pk=resv.pk %}?return_url={{ rack.get_absolute_url }}" class="btn btn-warning btn-xs" title="Edit reservation">
  288. <i class="glyphicon glyphicon-pencil" aria-hidden="true"></i>
  289. </a>
  290. {% endif %}
  291. {% if perms.dcim.delete_rackreservation %}
  292. <a href="{% url 'dcim:rackreservation_delete' pk=resv.pk %}?return_url={{ rack.get_absolute_url }}" class="btn btn-danger btn-xs" title="Delete reservation">
  293. <i class="glyphicon glyphicon-trash" aria-hidden="true"></i>
  294. </a>
  295. {% endif %}
  296. </td>
  297. </tr>
  298. {% endfor %}
  299. </table>
  300. {% else %}
  301. <div class="panel-body text-muted">None</div>
  302. {% endif %}
  303. {% if perms.dcim.add_rackreservation %}
  304. <div class="panel-footer text-right noprint">
  305. <a href="{% url 'dcim:rackreservation_add' %}?rack={{ rack.pk }}&return_url={{ rack.get_absolute_url }}" class="btn btn-primary btn-xs">
  306. <span class="glyphicon glyphicon-plus" aria-hidden="true"></span>
  307. Add a reservation
  308. </a>
  309. </div>
  310. {% endif %}
  311. </div>
  312. </div>
  313. <div class="col-md-6">
  314. <div class="row" style="margin-bottom: 20px">
  315. <div class="col-md-6 col-sm-6 col-xs-12">
  316. <div class="rack_header">
  317. <h4>Front</h4>
  318. </div>
  319. {% include 'dcim/inc/rack_elevation.html' with face='front' %}
  320. </div>
  321. <div class="col-md-6 col-sm-6 col-xs-12">
  322. <div class="rack_header">
  323. <h4>Rear</h4>
  324. </div>
  325. {% include 'dcim/inc/rack_elevation.html' with face='rear' %}
  326. </div>
  327. </div>
  328. <div class="panel panel-default">
  329. <div class="panel-heading">
  330. <strong>Non-Racked Devices</strong>
  331. </div>
  332. {% if nonracked_devices %}
  333. <table class="table table-hover panel-body">
  334. <tr>
  335. <th>Name</th>
  336. <th>Role</th>
  337. <th>Type</th>
  338. <th>Parent</th>
  339. </tr>
  340. {% for device in nonracked_devices %}
  341. <tr{% if device.device_type.u_height %} class="warning"{% endif %}>
  342. <td>
  343. <a href="{% url 'dcim:device' pk=device.pk %}">{{ device }}</a>
  344. </td>
  345. <td>{{ device.device_role }}</td>
  346. <td>{{ device.device_type.display_name }}</td>
  347. <td>
  348. {% if device.parent_bay %}
  349. <a href="{{ device.parent_bay.device.get_absolute_url }}">{{ device.parent_bay }}</a>
  350. {% else %}
  351. <span class="text-muted">&mdash;</span>
  352. {% endif %}
  353. </td>
  354. </tr>
  355. {% endfor %}
  356. </table>
  357. {% else %}
  358. <div class="panel-body text-muted">None</div>
  359. {% endif %}
  360. {% if perms.dcim.add_device %}
  361. <div class="panel-footer text-right noprint">
  362. <a href="{% url 'dcim:device_add' %}?site={{ rack.site.pk }}&rack={{ rack.pk }}" class="btn btn-primary btn-xs">
  363. <span class="glyphicon glyphicon-plus" aria-hidden="true"></span>
  364. Add a non-racked device
  365. </a>
  366. </div>
  367. {% endif %}
  368. </div>
  369. </div>
  370. </div>
  371. {% endblock %}
  372. {% block javascript %}
  373. <script src="{% static 'js/rack_elevations.js' %}?v{{ settings.VERSION }}"></script>
  374. {% endblock %}