nautobot 2.4.13__py3-none-any.whl → 2.4.15__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of nautobot might be problematic. Click here for more details.

Files changed (463) hide show
  1. nautobot/apps/choices.py +8 -0
  2. nautobot/apps/ui.py +14 -0
  3. nautobot/core/api/views.py +2 -0
  4. nautobot/core/choices.py +4 -0
  5. nautobot/core/filters.py +21 -41
  6. nautobot/core/graphql/generators.py +8 -0
  7. nautobot/core/graphql/schema.py +30 -30
  8. nautobot/core/management/commands/check_job_approval_status.py +47 -0
  9. nautobot/core/management/commands/generate_test_data.py +1 -1
  10. nautobot/core/management/commands/migrate.py +90 -1
  11. nautobot/core/models/tree_queries.py +17 -0
  12. nautobot/core/settings.py +2 -2
  13. nautobot/core/settings.yaml +3 -3
  14. nautobot/core/tables.py +29 -6
  15. nautobot/core/templates/base_django.html +1 -1
  16. nautobot/core/templates/components/panel/header_extra_content_table.html +1 -1
  17. nautobot/core/templates/generic/object_list.html +17 -20
  18. nautobot/core/templates/inc/breadcrumbs.html +14 -0
  19. nautobot/core/templatetags/buttons.py +2 -4
  20. nautobot/core/templatetags/helpers.py +29 -6
  21. nautobot/core/templatetags/ui_framework.py +21 -0
  22. nautobot/core/testing/api.py +7 -0
  23. nautobot/core/testing/filters.py +20 -3
  24. nautobot/core/testing/forms.py +1 -1
  25. nautobot/core/tests/integration/test_filters.py +2 -2
  26. nautobot/core/tests/test_breadcrumbs.py +366 -0
  27. nautobot/core/tests/test_commands.py +40 -0
  28. nautobot/core/tests/test_filters.py +51 -1
  29. nautobot/core/tests/test_forms.py +1 -1
  30. nautobot/core/tests/test_graphql.py +4 -4
  31. nautobot/core/tests/test_titles.py +183 -0
  32. nautobot/core/tests/test_tree_queries.py +30 -0
  33. nautobot/core/tests/test_views.py +2 -2
  34. nautobot/core/tests/test_views_utils.py +1 -1
  35. nautobot/core/ui/breadcrumbs.py +538 -0
  36. nautobot/core/ui/bulk_buttons.py +53 -0
  37. nautobot/core/ui/object_detail.py +31 -8
  38. nautobot/core/ui/titles.py +127 -0
  39. nautobot/core/ui/utils.py +25 -0
  40. nautobot/core/utils/migrations.py +1 -1
  41. nautobot/core/views/__init__.py +1 -1
  42. nautobot/core/views/mixins.py +26 -1
  43. nautobot/core/views/renderers.py +20 -2
  44. nautobot/core/views/utils.py +14 -13
  45. nautobot/dcim/api/serializers.py +9 -0
  46. nautobot/dcim/choices.py +55 -0
  47. nautobot/dcim/constants.py +0 -16
  48. nautobot/dcim/factory.py +1 -1
  49. nautobot/dcim/filters/__init__.py +15 -3
  50. nautobot/dcim/forms.py +120 -7
  51. nautobot/dcim/management/commands/trace_paths.py +1 -1
  52. nautobot/dcim/migrations/0072_alter_powerfeed_options_and_more.py +97 -0
  53. nautobot/dcim/models/device_component_templates.py +8 -0
  54. nautobot/dcim/models/device_components.py +31 -12
  55. nautobot/dcim/models/devices.py +1 -1
  56. nautobot/dcim/models/power.py +171 -10
  57. nautobot/dcim/models/racks.py +7 -4
  58. nautobot/dcim/tables/devices.py +2 -0
  59. nautobot/dcim/tables/devicetypes.py +1 -0
  60. nautobot/dcim/tables/power.py +30 -2
  61. nautobot/dcim/templates/dcim/device.html +2 -2
  62. nautobot/dcim/templates/dcim/devicetype_retrieve.html +1 -214
  63. nautobot/dcim/templates/dcim/location_retrieve.html +2 -2
  64. nautobot/dcim/templates/dcim/powerfeed_edit.html +8 -0
  65. nautobot/dcim/templates/dcim/powerfeed_retrieve.html +1 -1
  66. nautobot/dcim/templates/dcim/rack.html +2 -318
  67. nautobot/dcim/templates/dcim/rack_edit.html +2 -47
  68. nautobot/dcim/templates/dcim/rack_retrieve.html +318 -0
  69. nautobot/dcim/templates/dcim/rack_update.html +47 -0
  70. nautobot/dcim/tests/integration/test_device_bulk_operations.py +61 -0
  71. nautobot/dcim/tests/test_api.py +24 -4
  72. nautobot/dcim/tests/test_filters.py +91 -13
  73. nautobot/dcim/tests/test_models.py +262 -0
  74. nautobot/dcim/tests/test_views.py +20 -12
  75. nautobot/dcim/urls.py +2 -27
  76. nautobot/dcim/utils.py +13 -30
  77. nautobot/dcim/views.py +428 -146
  78. nautobot/extras/choices.py +12 -4
  79. nautobot/extras/factory.py +19 -20
  80. nautobot/extras/filters/__init__.py +3 -2
  81. nautobot/extras/filters/mixins.py +23 -7
  82. nautobot/extras/forms/__init__.py +2 -1
  83. nautobot/extras/forms/forms.py +71 -0
  84. nautobot/extras/forms/mixins.py +4 -2
  85. nautobot/extras/managers.py +4 -1
  86. nautobot/extras/migrations/0062_collect_roles_from_related_apps_roles.py +30 -7
  87. nautobot/extras/migrations/0124_add_joblogentry_index.py +16 -0
  88. nautobot/extras/migrations/0125_jobresult_date_started.py +18 -0
  89. nautobot/extras/models/customfields.py +53 -5
  90. nautobot/extras/models/datasources.py +1 -2
  91. nautobot/extras/models/jobs.py +13 -3
  92. nautobot/extras/models/relationships.py +55 -6
  93. nautobot/extras/plugins/views.py +24 -1
  94. nautobot/extras/secrets/__init__.py +1 -1
  95. nautobot/extras/tables.py +9 -0
  96. nautobot/extras/templates/extras/customfield.html +2 -129
  97. nautobot/extras/templates/extras/customfield_edit.html +2 -108
  98. nautobot/extras/templates/extras/customfield_retrieve.html +129 -0
  99. nautobot/extras/templates/extras/customfield_update.html +108 -0
  100. nautobot/extras/templates/extras/graphqlquery.html +2 -97
  101. nautobot/extras/templates/extras/graphqlquery_list.html +1 -0
  102. nautobot/extras/templates/extras/graphqlquery_retrieve.html +97 -0
  103. nautobot/extras/templates/extras/inc/jobresult.html +7 -3
  104. nautobot/extras/templates/extras/jobresult.html +2 -155
  105. nautobot/extras/templates/extras/jobresult_retrieve.html +155 -0
  106. nautobot/extras/templates/extras/marketplace.html +5 -6
  107. nautobot/extras/templates/extras/note.html +2 -53
  108. nautobot/extras/templates/extras/note_retrieve.html +53 -0
  109. nautobot/extras/templates/extras/plugins_list.html +5 -6
  110. nautobot/extras/templates/extras/secretsgroup.html +2 -29
  111. nautobot/extras/templates/extras/secretsgroup_edit.html +2 -82
  112. nautobot/extras/templates/extras/secretsgroup_retrieve.html +29 -0
  113. nautobot/extras/templates/extras/secretsgroup_update.html +82 -0
  114. nautobot/extras/templatetags/custom_links.py +2 -2
  115. nautobot/extras/templatetags/job_buttons.py +1 -1
  116. nautobot/extras/templatetags/plugins.py +1 -1
  117. nautobot/extras/tests/integration/test_computedfields.py +2 -2
  118. nautobot/extras/tests/integration/test_customfields.py +14 -11
  119. nautobot/extras/tests/integration/test_dynamicgroups.py +1 -1
  120. nautobot/extras/tests/integration/test_notes.py +1 -1
  121. nautobot/extras/tests/integration/test_plugins.py +6 -6
  122. nautobot/extras/tests/integration/test_relationships.py +2 -2
  123. nautobot/extras/tests/test_customfields.py +115 -7
  124. nautobot/extras/tests/test_filters.py +9 -0
  125. nautobot/extras/tests/test_forms.py +2 -2
  126. nautobot/extras/tests/test_plugins.py +2 -3
  127. nautobot/extras/tests/test_relationships.py +14 -8
  128. nautobot/extras/tests/test_views.py +285 -2
  129. nautobot/extras/urls.py +5 -110
  130. nautobot/extras/utils.py +5 -2
  131. nautobot/extras/views.py +116 -311
  132. nautobot/ipam/api/views.py +69 -6
  133. nautobot/ipam/tables.py +8 -15
  134. nautobot/ipam/tests/migration/test_migrations.py +8 -8
  135. nautobot/ipam/tests/test_api.py +352 -2
  136. nautobot/ipam/tests/test_models.py +1 -1
  137. nautobot/project-static/docs/404.html +34 -34
  138. nautobot/project-static/docs/apps/index.html +34 -34
  139. nautobot/project-static/docs/apps/nautobot-apps.html +34 -34
  140. nautobot/project-static/docs/assets/_mkdocstrings.css +44 -6
  141. nautobot/project-static/docs/assets/javascripts/{bundle.56ea9cef.min.js → bundle.50899def.min.js} +2 -2
  142. nautobot/project-static/docs/assets/javascripts/{bundle.56ea9cef.min.js.map → bundle.50899def.min.js.map} +2 -2
  143. nautobot/project-static/docs/assets/stylesheets/{main.342714a4.min.css → main.7e37652d.min.css} +1 -1
  144. nautobot/project-static/docs/assets/stylesheets/{main.342714a4.min.css.map → main.7e37652d.min.css.map} +1 -1
  145. nautobot/project-static/docs/code-reference/nautobot/apps/__init__.html +39 -34
  146. nautobot/project-static/docs/code-reference/nautobot/apps/admin.html +36 -34
  147. nautobot/project-static/docs/code-reference/nautobot/apps/api.html +139 -54
  148. nautobot/project-static/docs/code-reference/nautobot/apps/change_logging.html +48 -38
  149. nautobot/project-static/docs/code-reference/nautobot/apps/choices.html +50 -40
  150. nautobot/project-static/docs/code-reference/nautobot/apps/config.html +36 -34
  151. nautobot/project-static/docs/code-reference/nautobot/apps/constants.html +35 -34
  152. nautobot/project-static/docs/code-reference/nautobot/apps/datasources.html +43 -39
  153. nautobot/project-static/docs/code-reference/nautobot/apps/events.html +52 -42
  154. nautobot/project-static/docs/code-reference/nautobot/apps/exceptions.html +50 -41
  155. nautobot/project-static/docs/code-reference/nautobot/apps/factory.html +54 -44
  156. nautobot/project-static/docs/code-reference/nautobot/apps/filters.html +85 -93
  157. nautobot/project-static/docs/code-reference/nautobot/apps/forms.html +154 -62
  158. nautobot/project-static/docs/code-reference/nautobot/apps/graphql.html +54 -46
  159. nautobot/project-static/docs/code-reference/nautobot/apps/jobs.html +146 -87
  160. nautobot/project-static/docs/code-reference/nautobot/apps/models.html +240 -70
  161. nautobot/project-static/docs/code-reference/nautobot/apps/querysets.html +38 -35
  162. nautobot/project-static/docs/code-reference/nautobot/apps/secrets.html +41 -35
  163. nautobot/project-static/docs/code-reference/nautobot/apps/tables.html +173 -52
  164. nautobot/project-static/docs/code-reference/nautobot/apps/testing.html +269 -85
  165. nautobot/project-static/docs/code-reference/nautobot/apps/ui.html +5987 -2643
  166. nautobot/project-static/docs/code-reference/nautobot/apps/urls.html +36 -34
  167. nautobot/project-static/docs/code-reference/nautobot/apps/utils.html +165 -89
  168. nautobot/project-static/docs/code-reference/nautobot/apps/views.html +161 -69
  169. nautobot/project-static/docs/development/apps/api/configuration-view.html +34 -34
  170. nautobot/project-static/docs/development/apps/api/database-backend-config.html +34 -34
  171. nautobot/project-static/docs/development/apps/api/models/django-admin.html +34 -34
  172. nautobot/project-static/docs/development/apps/api/models/global-search.html +34 -34
  173. nautobot/project-static/docs/development/apps/api/models/graphql.html +34 -34
  174. nautobot/project-static/docs/development/apps/api/models/index.html +34 -34
  175. nautobot/project-static/docs/development/apps/api/nautobot-app-config.html +34 -34
  176. nautobot/project-static/docs/development/apps/api/platform-features/custom-validators.html +34 -34
  177. nautobot/project-static/docs/development/apps/api/platform-features/filter-extensions.html +34 -34
  178. nautobot/project-static/docs/development/apps/api/platform-features/git-repository-content.html +34 -34
  179. nautobot/project-static/docs/development/apps/api/platform-features/index.html +34 -34
  180. nautobot/project-static/docs/development/apps/api/platform-features/jinja2-filters.html +34 -34
  181. nautobot/project-static/docs/development/apps/api/platform-features/jobs.html +34 -34
  182. nautobot/project-static/docs/development/apps/api/platform-features/populating-extensibility-features.html +34 -34
  183. nautobot/project-static/docs/development/apps/api/platform-features/secrets-providers.html +34 -34
  184. nautobot/project-static/docs/development/apps/api/platform-features/table-extensions.html +34 -34
  185. nautobot/project-static/docs/development/apps/api/platform-features/uniquely-identify-objects.html +34 -34
  186. nautobot/project-static/docs/development/apps/api/prometheus.html +34 -34
  187. nautobot/project-static/docs/development/apps/api/setup.html +34 -34
  188. nautobot/project-static/docs/development/apps/api/testing.html +34 -34
  189. nautobot/project-static/docs/development/apps/api/ui-extensions/banners.html +34 -34
  190. nautobot/project-static/docs/development/apps/api/ui-extensions/home-page.html +34 -34
  191. nautobot/project-static/docs/development/apps/api/ui-extensions/index.html +34 -34
  192. nautobot/project-static/docs/development/apps/api/ui-extensions/navigation.html +34 -34
  193. nautobot/project-static/docs/development/apps/api/ui-extensions/object-views.html +34 -34
  194. nautobot/project-static/docs/development/apps/api/views/base-template.html +34 -34
  195. nautobot/project-static/docs/development/apps/api/views/core-view-overrides.html +34 -34
  196. nautobot/project-static/docs/development/apps/api/views/django-generic-views.html +34 -34
  197. nautobot/project-static/docs/development/apps/api/views/help-documentation.html +34 -34
  198. nautobot/project-static/docs/development/apps/api/views/index.html +34 -34
  199. nautobot/project-static/docs/development/apps/api/views/nautobot-generic-views.html +34 -34
  200. nautobot/project-static/docs/development/apps/api/views/nautobotuiviewset.html +42 -36
  201. nautobot/project-static/docs/development/apps/api/views/nautobotuiviewsetrouter.html +34 -34
  202. nautobot/project-static/docs/development/apps/api/views/notes.html +34 -34
  203. nautobot/project-static/docs/development/apps/api/views/rest-api.html +34 -34
  204. nautobot/project-static/docs/development/apps/api/views/urls.html +34 -34
  205. nautobot/project-static/docs/development/apps/index.html +34 -34
  206. nautobot/project-static/docs/development/apps/migration/code-updates.html +34 -34
  207. nautobot/project-static/docs/development/apps/migration/dependency-updates.html +34 -34
  208. nautobot/project-static/docs/development/apps/migration/from-v1.html +34 -34
  209. nautobot/project-static/docs/development/apps/migration/model-updates/dcim.html +34 -34
  210. nautobot/project-static/docs/development/apps/migration/model-updates/extras.html +34 -34
  211. nautobot/project-static/docs/development/apps/migration/model-updates/global.html +34 -34
  212. nautobot/project-static/docs/development/apps/migration/model-updates/ipam.html +34 -34
  213. nautobot/project-static/docs/development/apps/migration/ui-component-framework/best-practices.html +37 -37
  214. nautobot/project-static/docs/development/apps/migration/ui-component-framework/breadcrumbs-titles.html +10544 -0
  215. nautobot/project-static/docs/development/apps/migration/ui-component-framework/custom-content.html +34 -34
  216. nautobot/project-static/docs/development/apps/migration/ui-component-framework/index.html +34 -34
  217. nautobot/project-static/docs/development/apps/migration/ui-component-framework/migration-steps.html +34 -34
  218. nautobot/project-static/docs/development/apps/porting-from-netbox.html +37 -37
  219. nautobot/project-static/docs/development/core/application-registry.html +162 -133
  220. nautobot/project-static/docs/development/core/best-practices.html +34 -34
  221. nautobot/project-static/docs/development/core/bootstrap-ui.html +34 -34
  222. nautobot/project-static/docs/development/core/caching.html +34 -34
  223. nautobot/project-static/docs/development/core/controllers.html +34 -34
  224. nautobot/project-static/docs/development/core/docker-compose-advanced-use-cases.html +34 -34
  225. nautobot/project-static/docs/development/core/generic-views.html +34 -34
  226. nautobot/project-static/docs/development/core/getting-started.html +34 -34
  227. nautobot/project-static/docs/development/core/homepage.html +34 -34
  228. nautobot/project-static/docs/development/core/index.html +34 -34
  229. nautobot/project-static/docs/development/core/minikube-dev-environment-for-k8s-jobs.html +34 -34
  230. nautobot/project-static/docs/development/core/model-checklist.html +34 -34
  231. nautobot/project-static/docs/development/core/model-features.html +34 -34
  232. nautobot/project-static/docs/development/core/natural-keys.html +34 -34
  233. nautobot/project-static/docs/development/core/navigation-menu.html +34 -34
  234. nautobot/project-static/docs/development/core/release-checklist.html +34 -34
  235. nautobot/project-static/docs/development/core/role-internals.html +34 -34
  236. nautobot/project-static/docs/development/core/settings.html +34 -34
  237. nautobot/project-static/docs/development/core/style-guide.html +34 -34
  238. nautobot/project-static/docs/development/core/templates.html +34 -34
  239. nautobot/project-static/docs/development/core/testing.html +34 -34
  240. nautobot/project-static/docs/development/core/ui-component-framework.html +724 -289
  241. nautobot/project-static/docs/development/core/user-preferences.html +34 -34
  242. nautobot/project-static/docs/development/index.html +34 -34
  243. nautobot/project-static/docs/development/jobs/getting-started.html +34 -34
  244. nautobot/project-static/docs/development/jobs/index.html +34 -34
  245. nautobot/project-static/docs/development/jobs/installation.html +34 -34
  246. nautobot/project-static/docs/development/jobs/job-extensions.html +34 -34
  247. nautobot/project-static/docs/development/jobs/job-logging.html +34 -34
  248. nautobot/project-static/docs/development/jobs/job-patterns.html +34 -34
  249. nautobot/project-static/docs/development/jobs/job-structure.html +34 -34
  250. nautobot/project-static/docs/development/jobs/migration/from-v1.html +34 -34
  251. nautobot/project-static/docs/development/jobs/testing.html +34 -34
  252. nautobot/project-static/docs/index.html +34 -34
  253. nautobot/project-static/docs/media/development/core/ui-component-framework/breadcrumbs-titles-data-flow.png +0 -0
  254. nautobot/project-static/docs/media/power_distribution.png +0 -0
  255. nautobot/project-static/docs/objects.inv +0 -0
  256. nautobot/project-static/docs/overview/application_stack.html +34 -34
  257. nautobot/project-static/docs/overview/design_philosophy.html +34 -34
  258. nautobot/project-static/docs/release-notes/index.html +34 -34
  259. nautobot/project-static/docs/release-notes/version-1.0.html +34 -34
  260. nautobot/project-static/docs/release-notes/version-1.1.html +34 -34
  261. nautobot/project-static/docs/release-notes/version-1.2.html +34 -34
  262. nautobot/project-static/docs/release-notes/version-1.3.html +34 -34
  263. nautobot/project-static/docs/release-notes/version-1.4.html +34 -34
  264. nautobot/project-static/docs/release-notes/version-1.5.html +34 -34
  265. nautobot/project-static/docs/release-notes/version-1.6.html +34 -34
  266. nautobot/project-static/docs/release-notes/version-2.0.html +34 -34
  267. nautobot/project-static/docs/release-notes/version-2.1.html +34 -34
  268. nautobot/project-static/docs/release-notes/version-2.2.html +34 -34
  269. nautobot/project-static/docs/release-notes/version-2.3.html +34 -34
  270. nautobot/project-static/docs/release-notes/version-2.4.html +402 -34
  271. nautobot/project-static/docs/requirements.txt +3 -3
  272. nautobot/project-static/docs/search/search_index.json +1 -1
  273. nautobot/project-static/docs/sitemap.xml +303 -299
  274. nautobot/project-static/docs/sitemap.xml.gz +0 -0
  275. nautobot/project-static/docs/user-guide/administration/configuration/authentication/ldap.html +34 -34
  276. nautobot/project-static/docs/user-guide/administration/configuration/authentication/remote.html +34 -34
  277. nautobot/project-static/docs/user-guide/administration/configuration/authentication/sso.html +34 -34
  278. nautobot/project-static/docs/user-guide/administration/configuration/index.html +34 -34
  279. nautobot/project-static/docs/user-guide/administration/configuration/redis.html +34 -34
  280. nautobot/project-static/docs/user-guide/administration/configuration/settings.html +37 -37
  281. nautobot/project-static/docs/user-guide/administration/configuration/time-zones.html +34 -34
  282. nautobot/project-static/docs/user-guide/administration/guides/celery-queues.html +34 -34
  283. nautobot/project-static/docs/user-guide/administration/guides/docker.html +34 -34
  284. nautobot/project-static/docs/user-guide/administration/guides/health-checks.html +34 -34
  285. nautobot/project-static/docs/user-guide/administration/guides/permissions.html +34 -34
  286. nautobot/project-static/docs/user-guide/administration/guides/prometheus-metrics.html +34 -34
  287. nautobot/project-static/docs/user-guide/administration/guides/replicating-nautobot.html +34 -34
  288. nautobot/project-static/docs/user-guide/administration/guides/request-profiling.html +34 -34
  289. nautobot/project-static/docs/user-guide/administration/guides/s3-django-storage.html +34 -34
  290. nautobot/project-static/docs/user-guide/administration/guides/selinux-troubleshooting.html +34 -34
  291. nautobot/project-static/docs/user-guide/administration/installation/app-install.html +34 -34
  292. nautobot/project-static/docs/user-guide/administration/installation/external-authentication.html +34 -34
  293. nautobot/project-static/docs/user-guide/administration/installation/http-server.html +34 -34
  294. nautobot/project-static/docs/user-guide/administration/installation/index.html +34 -34
  295. nautobot/project-static/docs/user-guide/administration/installation/install_system.html +34 -34
  296. nautobot/project-static/docs/user-guide/administration/installation/nautobot.html +34 -34
  297. nautobot/project-static/docs/user-guide/administration/installation/services.html +34 -34
  298. nautobot/project-static/docs/user-guide/administration/migration/migrating-from-netbox.html +34 -34
  299. nautobot/project-static/docs/user-guide/administration/migration/migrating-from-postgresql.html +34 -34
  300. nautobot/project-static/docs/user-guide/administration/security/index.html +34 -34
  301. nautobot/project-static/docs/user-guide/administration/security/notices.html +34 -34
  302. nautobot/project-static/docs/user-guide/administration/tools/nautobot-server.html +296 -251
  303. nautobot/project-static/docs/user-guide/administration/tools/nautobot-shell.html +34 -34
  304. nautobot/project-static/docs/user-guide/administration/upgrading/database-backup.html +34 -34
  305. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/after-you-upgrade.html +34 -34
  306. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/before-you-upgrade.html +34 -34
  307. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/for-developers.html +34 -34
  308. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/index.html +34 -34
  309. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/whats-changed.html +34 -34
  310. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/region-and-site-data-migration-guide.html +34 -34
  311. nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/upgrading-from-nautobot-v1.html +34 -34
  312. nautobot/project-static/docs/user-guide/administration/upgrading/upgrading.html +34 -34
  313. nautobot/project-static/docs/user-guide/core-data-model/circuits/circuit.html +34 -34
  314. nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittermination.html +34 -34
  315. nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittype.html +34 -34
  316. nautobot/project-static/docs/user-guide/core-data-model/circuits/provider.html +34 -34
  317. nautobot/project-static/docs/user-guide/core-data-model/circuits/providernetwork.html +34 -34
  318. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloud.html +34 -34
  319. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudaccount.html +34 -34
  320. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudnetwork.html +34 -34
  321. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudnetworkprefixassignment.html +34 -34
  322. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudresourcetype.html +34 -34
  323. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudservice.html +34 -34
  324. nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudservicenetworkassignment.html +34 -34
  325. nautobot/project-static/docs/user-guide/core-data-model/dcim/cable.html +34 -34
  326. nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleport.html +34 -34
  327. nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleporttemplate.html +34 -34
  328. nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverport.html +34 -34
  329. nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverporttemplate.html +34 -34
  330. nautobot/project-static/docs/user-guide/core-data-model/dcim/controller.html +34 -34
  331. nautobot/project-static/docs/user-guide/core-data-model/dcim/controllermanageddevicegroup.html +34 -34
  332. nautobot/project-static/docs/user-guide/core-data-model/dcim/device.html +34 -34
  333. nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebay.html +34 -34
  334. nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebaytemplate.html +34 -34
  335. nautobot/project-static/docs/user-guide/core-data-model/dcim/devicefamily.html +34 -34
  336. nautobot/project-static/docs/user-guide/core-data-model/dcim/deviceredundancygroup.html +34 -34
  337. nautobot/project-static/docs/user-guide/core-data-model/dcim/devicetype.html +34 -34
  338. nautobot/project-static/docs/user-guide/core-data-model/dcim/frontport.html +34 -34
  339. nautobot/project-static/docs/user-guide/core-data-model/dcim/frontporttemplate.html +34 -34
  340. nautobot/project-static/docs/user-guide/core-data-model/dcim/interface.html +37 -37
  341. nautobot/project-static/docs/user-guide/core-data-model/dcim/interfaceredundancygroup.html +34 -34
  342. nautobot/project-static/docs/user-guide/core-data-model/dcim/interfacetemplate.html +34 -34
  343. nautobot/project-static/docs/user-guide/core-data-model/dcim/inventoryitem.html +42 -52
  344. nautobot/project-static/docs/user-guide/core-data-model/dcim/location.html +34 -34
  345. nautobot/project-static/docs/user-guide/core-data-model/dcim/locationtype.html +34 -34
  346. nautobot/project-static/docs/user-guide/core-data-model/dcim/manufacturer.html +34 -34
  347. nautobot/project-static/docs/user-guide/core-data-model/dcim/module.html +34 -34
  348. nautobot/project-static/docs/user-guide/core-data-model/dcim/modulebay.html +37 -37
  349. nautobot/project-static/docs/user-guide/core-data-model/dcim/modulebaytemplate.html +34 -34
  350. nautobot/project-static/docs/user-guide/core-data-model/dcim/modulefamily.html +34 -34
  351. nautobot/project-static/docs/user-guide/core-data-model/dcim/moduletype.html +34 -34
  352. nautobot/project-static/docs/user-guide/core-data-model/dcim/platform.html +34 -34
  353. nautobot/project-static/docs/user-guide/core-data-model/dcim/powerfeed.html +316 -39
  354. nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlet.html +35 -35
  355. nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlettemplate.html +34 -34
  356. nautobot/project-static/docs/user-guide/core-data-model/dcim/powerpanel.html +147 -37
  357. nautobot/project-static/docs/user-guide/core-data-model/dcim/powerport.html +52 -35
  358. nautobot/project-static/docs/user-guide/core-data-model/dcim/powerporttemplate.html +51 -34
  359. nautobot/project-static/docs/user-guide/core-data-model/dcim/rack.html +34 -34
  360. nautobot/project-static/docs/user-guide/core-data-model/dcim/rackgroup.html +34 -34
  361. nautobot/project-static/docs/user-guide/core-data-model/dcim/rackreservation.html +34 -34
  362. nautobot/project-static/docs/user-guide/core-data-model/dcim/rearport.html +34 -34
  363. nautobot/project-static/docs/user-guide/core-data-model/dcim/rearporttemplate.html +34 -34
  364. nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareimagefile.html +34 -34
  365. nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareversion.html +34 -34
  366. nautobot/project-static/docs/user-guide/core-data-model/dcim/virtualchassis.html +34 -34
  367. nautobot/project-static/docs/user-guide/core-data-model/dcim/virtualdevicecontext.html +34 -34
  368. nautobot/project-static/docs/user-guide/core-data-model/extras/configcontext.html +34 -34
  369. nautobot/project-static/docs/user-guide/core-data-model/extras/configcontextschema.html +34 -34
  370. nautobot/project-static/docs/user-guide/core-data-model/extras/contact.html +34 -34
  371. nautobot/project-static/docs/user-guide/core-data-model/extras/team.html +34 -34
  372. nautobot/project-static/docs/user-guide/core-data-model/ipam/ipaddress.html +34 -34
  373. nautobot/project-static/docs/user-guide/core-data-model/ipam/namespace.html +34 -34
  374. nautobot/project-static/docs/user-guide/core-data-model/ipam/prefix.html +34 -34
  375. nautobot/project-static/docs/user-guide/core-data-model/ipam/rir.html +34 -34
  376. nautobot/project-static/docs/user-guide/core-data-model/ipam/routetarget.html +34 -34
  377. nautobot/project-static/docs/user-guide/core-data-model/ipam/service.html +34 -34
  378. nautobot/project-static/docs/user-guide/core-data-model/ipam/vlan.html +34 -34
  379. nautobot/project-static/docs/user-guide/core-data-model/ipam/vlangroup.html +34 -34
  380. nautobot/project-static/docs/user-guide/core-data-model/ipam/vrf.html +34 -34
  381. nautobot/project-static/docs/user-guide/core-data-model/overview/introduction.html +34 -34
  382. nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenant.html +34 -34
  383. nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenantgroup.html +34 -34
  384. nautobot/project-static/docs/user-guide/core-data-model/virtualization/cluster.html +34 -34
  385. nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustergroup.html +34 -34
  386. nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustertype.html +34 -34
  387. nautobot/project-static/docs/user-guide/core-data-model/virtualization/virtualmachine.html +34 -34
  388. nautobot/project-static/docs/user-guide/core-data-model/virtualization/vminterface.html +34 -34
  389. nautobot/project-static/docs/user-guide/core-data-model/wireless/index.html +34 -34
  390. nautobot/project-static/docs/user-guide/core-data-model/wireless/radioprofile.html +34 -34
  391. nautobot/project-static/docs/user-guide/core-data-model/wireless/supporteddatarate.html +34 -34
  392. nautobot/project-static/docs/user-guide/core-data-model/wireless/wirelessnetwork.html +34 -34
  393. nautobot/project-static/docs/user-guide/feature-guides/contacts-and-teams.html +34 -34
  394. nautobot/project-static/docs/user-guide/feature-guides/custom-fields.html +34 -34
  395. nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-devices.html +34 -34
  396. nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-location-types-and-locations.html +34 -34
  397. nautobot/project-static/docs/user-guide/feature-guides/getting-started/index.html +34 -34
  398. nautobot/project-static/docs/user-guide/feature-guides/getting-started/interfaces.html +34 -34
  399. nautobot/project-static/docs/user-guide/feature-guides/getting-started/ipam.html +34 -34
  400. nautobot/project-static/docs/user-guide/feature-guides/getting-started/platforms.html +34 -34
  401. nautobot/project-static/docs/user-guide/feature-guides/getting-started/search-bar.html +34 -34
  402. nautobot/project-static/docs/user-guide/feature-guides/getting-started/tenants.html +34 -34
  403. nautobot/project-static/docs/user-guide/feature-guides/getting-started/vlans-and-vlan-groups.html +34 -34
  404. nautobot/project-static/docs/user-guide/feature-guides/git-data-source.html +34 -34
  405. nautobot/project-static/docs/user-guide/feature-guides/graphql.html +34 -34
  406. nautobot/project-static/docs/user-guide/feature-guides/ip-address-merge-tool.html +34 -34
  407. nautobot/project-static/docs/user-guide/feature-guides/relationships.html +34 -34
  408. nautobot/project-static/docs/user-guide/feature-guides/software-image-files-and-versions.html +34 -34
  409. nautobot/project-static/docs/user-guide/feature-guides/wireless-networks-and-controllers.html +34 -34
  410. nautobot/project-static/docs/user-guide/index.html +34 -34
  411. nautobot/project-static/docs/user-guide/platform-functionality/change-logging.html +34 -34
  412. nautobot/project-static/docs/user-guide/platform-functionality/computedfield.html +34 -34
  413. nautobot/project-static/docs/user-guide/platform-functionality/customfield.html +34 -34
  414. nautobot/project-static/docs/user-guide/platform-functionality/customlink.html +34 -34
  415. nautobot/project-static/docs/user-guide/platform-functionality/dynamicgroup.html +34 -34
  416. nautobot/project-static/docs/user-guide/platform-functionality/events.html +34 -34
  417. nautobot/project-static/docs/user-guide/platform-functionality/exporttemplate.html +34 -34
  418. nautobot/project-static/docs/user-guide/platform-functionality/externalintegration.html +34 -34
  419. nautobot/project-static/docs/user-guide/platform-functionality/gitrepository.html +34 -34
  420. nautobot/project-static/docs/user-guide/platform-functionality/graphql.html +34 -34
  421. nautobot/project-static/docs/user-guide/platform-functionality/graphqlquery.html +34 -34
  422. nautobot/project-static/docs/user-guide/platform-functionality/imageattachment.html +34 -34
  423. nautobot/project-static/docs/user-guide/platform-functionality/jobs/index.html +34 -34
  424. nautobot/project-static/docs/user-guide/platform-functionality/jobs/job-scheduling-and-approvals.html +35 -35
  425. nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobbutton.html +34 -34
  426. nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobhook.html +34 -34
  427. nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobqueue.html +34 -34
  428. nautobot/project-static/docs/user-guide/platform-functionality/jobs/kubernetes-job-support.html +34 -34
  429. nautobot/project-static/docs/user-guide/platform-functionality/jobs/managing-jobs.html +34 -34
  430. nautobot/project-static/docs/user-guide/platform-functionality/jobs/models.html +34 -34
  431. nautobot/project-static/docs/user-guide/platform-functionality/napalm.html +34 -34
  432. nautobot/project-static/docs/user-guide/platform-functionality/note.html +34 -34
  433. nautobot/project-static/docs/user-guide/platform-functionality/objectmetadata.html +34 -34
  434. nautobot/project-static/docs/user-guide/platform-functionality/relationship.html +34 -34
  435. nautobot/project-static/docs/user-guide/platform-functionality/rendering-jinja-templates.html +34 -34
  436. nautobot/project-static/docs/user-guide/platform-functionality/rest-api/authentication.html +34 -34
  437. nautobot/project-static/docs/user-guide/platform-functionality/rest-api/filtering.html +34 -34
  438. nautobot/project-static/docs/user-guide/platform-functionality/rest-api/overview.html +34 -34
  439. nautobot/project-static/docs/user-guide/platform-functionality/rest-api/ui-related-endpoints.html +34 -34
  440. nautobot/project-static/docs/user-guide/platform-functionality/role.html +34 -34
  441. nautobot/project-static/docs/user-guide/platform-functionality/savedview.html +34 -34
  442. nautobot/project-static/docs/user-guide/platform-functionality/secret.html +34 -34
  443. nautobot/project-static/docs/user-guide/platform-functionality/staticgroupassociation.html +34 -34
  444. nautobot/project-static/docs/user-guide/platform-functionality/status.html +34 -34
  445. nautobot/project-static/docs/user-guide/platform-functionality/tag.html +34 -34
  446. nautobot/project-static/docs/user-guide/platform-functionality/template-filters.html +34 -34
  447. nautobot/project-static/docs/user-guide/platform-functionality/users/objectpermission.html +34 -34
  448. nautobot/project-static/docs/user-guide/platform-functionality/users/token.html +34 -34
  449. nautobot/project-static/docs/user-guide/platform-functionality/webhook.html +34 -34
  450. nautobot/tenancy/api/views.py +2 -1
  451. nautobot/users/tests/test_api.py +2 -2
  452. nautobot/virtualization/templates/virtualization/virtualmachine.html +2 -252
  453. nautobot/virtualization/templates/virtualization/virtualmachine_edit.html +2 -75
  454. nautobot/virtualization/templates/virtualization/virtualmachine_retrieve.html +252 -0
  455. nautobot/virtualization/templates/virtualization/virtualmachine_update.html +75 -0
  456. nautobot/virtualization/urls.py +3 -61
  457. nautobot/virtualization/views.py +48 -72
  458. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/METADATA +27 -27
  459. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/RECORD +463 -439
  460. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/LICENSE.txt +0 -0
  461. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/NOTICE +0 -0
  462. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/WHEEL +0 -0
  463. {nautobot-2.4.13.dist-info → nautobot-2.4.15.dist-info}/entry_points.txt +0 -0
@@ -0,0 +1,366 @@
1
+ """
2
+ Unit tests for the updated breadcrumbs.py following Nautobot testing conventions.
3
+ """
4
+
5
+ from operator import itemgetter
6
+ from unittest.mock import patch
7
+
8
+ from django.template import Context
9
+ from django.utils.http import urlencode
10
+
11
+ from nautobot.core.testing import TestCase
12
+ from nautobot.core.ui.breadcrumbs import (
13
+ BaseBreadcrumbItem,
14
+ Breadcrumbs,
15
+ InstanceBreadcrumbItem,
16
+ ModelBreadcrumbItem,
17
+ ViewNameBreadcrumbItem,
18
+ )
19
+ from nautobot.dcim.models import Device, LocationType
20
+
21
+
22
+ class BreadcrumbItemsTestCase(TestCase):
23
+ """Test cases for BreadcrumbItem class."""
24
+
25
+ @classmethod
26
+ def setUpTestData(cls):
27
+ """Create test data."""
28
+ cls.location_type = LocationType.objects.create(name="Test Location Type Breadcrumbs")
29
+
30
+ def test_view_name_item(self):
31
+ """Test breadcrumb view name item."""
32
+ item = ViewNameBreadcrumbItem(view_name="home", label="Home")
33
+ context = Context({})
34
+
35
+ url, label = item.as_pair(context)
36
+
37
+ self.assertEqual(url, "/")
38
+ self.assertEqual(label, "Home")
39
+
40
+ def test_view_name_item_with_kwargs_and_query_params(self):
41
+ """Test breadcrumb view name item and kwargs."""
42
+ item = ViewNameBreadcrumbItem(
43
+ view_name="dcim:locationtype",
44
+ reverse_kwargs={"pk": self.location_type.pk},
45
+ reverse_query_params={"name": "test"},
46
+ label="Filtered Locations Types",
47
+ )
48
+ context = Context({})
49
+
50
+ url, label = item.as_pair(context)
51
+
52
+ self.assertEqual(url, f"/dcim/location-types/{self.location_type.pk}/?name=test")
53
+ self.assertEqual(label, "Filtered Locations Types")
54
+
55
+ def test_view_name_item_with_kwargs_and_query_params_callable(self):
56
+ """Test breadcrumb view name item and kwargs."""
57
+ item = ViewNameBreadcrumbItem(
58
+ view_name="dcim:locationtype",
59
+ reverse_kwargs=lambda c: {"pk": c["object"].pk},
60
+ reverse_query_params=lambda c: {"name": c["object"].name},
61
+ label="Filtered Locations Types",
62
+ )
63
+ context = Context({"object": self.location_type})
64
+
65
+ url, label = item.as_pair(context)
66
+
67
+ self.assertEqual(
68
+ url, f"/dcim/location-types/{self.location_type.pk}/?{urlencode({'name': self.location_type.name})}"
69
+ )
70
+ self.assertEqual(label, "Filtered Locations Types")
71
+
72
+ def test_callable_label_and_view_name(self):
73
+ """Test label and view_name as callables."""
74
+ item = ViewNameBreadcrumbItem(
75
+ view_name=lambda _: "home",
76
+ label=lambda context: f"Hi, {context['user']}!",
77
+ )
78
+ context = Context({"user": "Frodo"})
79
+ url, label = item.as_pair(context)
80
+ self.assertEqual(url, "/")
81
+ self.assertEqual(label, "Hi, Frodo!")
82
+
83
+ def test_model_items(self):
84
+ """Test breadcrumb model items."""
85
+ test_cases = [
86
+ {
87
+ "name": "model_class",
88
+ "kwargs": {"model": Device},
89
+ "expected_url": "/dcim/devices/",
90
+ "expected_label": "Devices",
91
+ },
92
+ {
93
+ "name": "model_instance",
94
+ "kwargs": {"model": self.location_type},
95
+ "expected_url": "/dcim/location-types/",
96
+ "expected_label": "Location Types",
97
+ },
98
+ {
99
+ "name": "model_instance_callable",
100
+ "kwargs": {"model": itemgetter("object")},
101
+ "expected_url": "/dcim/location-types/",
102
+ "expected_label": "Location Types",
103
+ },
104
+ {
105
+ "name": "model_str",
106
+ "kwargs": {"model": "dcim.device"},
107
+ "expected_url": "/dcim/devices/",
108
+ "expected_label": "Devices",
109
+ },
110
+ {
111
+ "name": "model_class_custom_url_action",
112
+ "kwargs": {"model": Device, "action": "add", "label_type": "singular"},
113
+ "expected_url": "/dcim/devices/add/",
114
+ "expected_label": "Device",
115
+ },
116
+ {
117
+ "name": "model_class_with_kwargs",
118
+ "kwargs": {
119
+ "model": Device,
120
+ "action": "",
121
+ "label_type": "singular",
122
+ "reverse_kwargs": {"pk": "947a8a80-9e62-5605-ab18-7a47c588f0ad"},
123
+ },
124
+ "expected_url": "/dcim/devices/947a8a80-9e62-5605-ab18-7a47c588f0ad/",
125
+ "expected_label": "Device",
126
+ },
127
+ {
128
+ "name": "model_class_with_query_params",
129
+ "kwargs": {"model": Device, "reverse_query_params": {"filter": "abc"}},
130
+ "expected_url": "/dcim/devices/?filter=abc",
131
+ "expected_label": "Devices",
132
+ },
133
+ {
134
+ "name": "model_class_with_query_params_callable",
135
+ "kwargs": {
136
+ "model": itemgetter("model_type"),
137
+ "reverse_query_params": lambda c: {"name": c["device_name"]},
138
+ },
139
+ "expected_url": "/dcim/devices/?name=abc",
140
+ "expected_label": "Devices",
141
+ },
142
+ ]
143
+ for test_case in test_cases:
144
+ with self.subTest(action=test_case["name"]):
145
+ item = ModelBreadcrumbItem(**test_case["kwargs"])
146
+ context = Context({"object": self.location_type, "model_type": Device, "device_name": "abc"})
147
+
148
+ url, label = item.as_pair(context)
149
+
150
+ self.assertEqual(url, test_case["expected_url"])
151
+ self.assertEqual(label, test_case["expected_label"])
152
+
153
+ def test_model_item_from_context(self):
154
+ """Test breadcrumb item with model from context."""
155
+ item = ModelBreadcrumbItem(model_key="object")
156
+ context = Context({"object": self.location_type})
157
+
158
+ url, label = item.as_pair(context)
159
+
160
+ self.assertEqual(url, "/dcim/location-types/")
161
+ self.assertEqual(label, "Location Types")
162
+
163
+ def test_instance_item(self):
164
+ """Test breadcrumb item with instance from context."""
165
+ item = InstanceBreadcrumbItem(instance_key="object")
166
+ context = Context({"object": self.location_type})
167
+
168
+ url, label = item.as_pair(context)
169
+
170
+ self.assertEqual(url, f"/dcim/location-types/{self.location_type.pk}/")
171
+ self.assertEqual(label, str(self.location_type))
172
+
173
+ def test_label_override(self):
174
+ """Test that explicit label overrides automatic label generation."""
175
+ items = [
176
+ ViewNameBreadcrumbItem(view_name="dcim:locationtype", label="Custom Label"),
177
+ ModelBreadcrumbItem(model=LocationType, label="Custom Label"),
178
+ InstanceBreadcrumbItem(label="Custom Label"),
179
+ ]
180
+
181
+ context = Context({"object": self.location_type})
182
+
183
+ for item in items:
184
+ with self.subTest():
185
+ _, label = item.as_pair(context)
186
+ self.assertEqual(label, "Custom Label")
187
+
188
+ def test_no_reverse_match(self):
189
+ """Test handling of NoReverseMatch exception."""
190
+ item = ViewNameBreadcrumbItem(view_name="nonexistent")
191
+ context = Context({})
192
+
193
+ url, label = item.as_pair(context)
194
+
195
+ self.assertEqual(url, "")
196
+ self.assertEqual(label, "")
197
+
198
+ def test_empty_context_keys(self):
199
+ """Test breadcrumb item when context keys are missing."""
200
+ context = Context({})
201
+ item = InstanceBreadcrumbItem(instance_key="missing_key")
202
+
203
+ url, label = item.as_pair(context)
204
+ self.assertEqual(url, "")
205
+ self.assertEqual(label, "")
206
+
207
+ item = ModelBreadcrumbItem(model_key="missing_key")
208
+ url, label = item.as_pair(context)
209
+ self.assertEqual(url, "")
210
+ self.assertEqual(label, "")
211
+
212
+
213
+ class BreadcrumbsTestCase(TestCase):
214
+ """Test cases for Breadcrumbs class."""
215
+
216
+ @classmethod
217
+ def setUpTestData(cls):
218
+ """Create test data."""
219
+ cls.location_type = LocationType.objects.create(name="Test Location Type Breadcrumbs")
220
+
221
+ def test_default_initialization(self):
222
+ """Test that Breadcrumbs initializes with default values."""
223
+ breadcrumbs = Breadcrumbs()
224
+
225
+ # Should have defaults for list and details
226
+ self.assertEqual(len(breadcrumbs.items["list"]), 2)
227
+ self.assertEqual(len(breadcrumbs.items["detail"]), 3)
228
+
229
+ # Verify adding items
230
+ new_item = BaseBreadcrumbItem()
231
+ breadcrumbs = Breadcrumbs(items={"detail": [new_item], "list": [new_item], "custom_action": [new_item]})
232
+
233
+ self.assertEqual(len(breadcrumbs.items["list"]), 1)
234
+ self.assertEqual(breadcrumbs.items["list"][0], new_item)
235
+
236
+ self.assertEqual(len(breadcrumbs.items["detail"]), 2)
237
+ self.assertEqual(breadcrumbs.items["detail"][0], new_item)
238
+
239
+ self.assertEqual(len(breadcrumbs.items["custom_action"]), 1)
240
+ self.assertEqual(breadcrumbs.items["custom_action"][0], new_item)
241
+
242
+ def test_custom_items(self):
243
+ """Test Breadcrumbs with custom items."""
244
+ custom_list_item = ViewNameBreadcrumbItem(view_name="home", label="Home")
245
+ custom_items = {
246
+ "list": [ViewNameBreadcrumbItem(view_name="home", label="Home")],
247
+ }
248
+ breadcrumbs = Breadcrumbs(items=custom_items)
249
+
250
+ self.assertEqual(len(breadcrumbs.items["list"]), 1)
251
+ self.assertEqual(breadcrumbs.items["list"][0], custom_list_item)
252
+
253
+ # Other defaults should still exist
254
+ self.assertIn("detail", breadcrumbs.items)
255
+ self.assertEqual(len(breadcrumbs.items["detail"]), 3)
256
+
257
+ def test_get_items_from_action_static_method(self):
258
+ """Test the _get_items_from_action static method."""
259
+ test_items = {
260
+ "list": [ViewNameBreadcrumbItem(view_name="", label="List Item")],
261
+ "detail": [ViewNameBreadcrumbItem(view_name="", label="Detail Item")],
262
+ }
263
+
264
+ # Test specific action found
265
+ result = Breadcrumbs.get_items_for_action(test_items, "list", False)
266
+ self.assertEqual(len(result), 1)
267
+ self.assertEqual(result[0].label, "List Item")
268
+
269
+ # Test fallback to detail when action not found and detail=True
270
+ result = Breadcrumbs.get_items_for_action(test_items, "nonexistent", True)
271
+ self.assertEqual(len(result), 1)
272
+ self.assertEqual(result[0].label, "Detail Item")
273
+
274
+ # Test no fallback when detail=False
275
+ result = Breadcrumbs.get_items_for_action(test_items, "nonexistent", False)
276
+ self.assertEqual(len(result), 0)
277
+ self.assertEqual(result, [])
278
+
279
+ def test_detail_fallback_behavior(self):
280
+ """Test that detail fallback works correctly in get_breadcrumbs_items."""
281
+ breadcrumbs = Breadcrumbs()
282
+ expected_items = [
283
+ ("/dcim/location-types/", "Location Types"),
284
+ (f"/dcim/location-types/{self.location_type.pk}/", str(self.location_type)),
285
+ ]
286
+
287
+ # Test with an action that doesn't exist but detail=True
288
+ context = Context(
289
+ {"view_action": "custom_action", "detail": True, "model": LocationType, "object": self.location_type}
290
+ )
291
+
292
+ items = breadcrumbs.get_breadcrumbs_items(context)
293
+
294
+ # Should get 2 items from detail fallback
295
+ self.assertEqual(len(items), 2)
296
+ self.assertEqual(items, expected_items)
297
+
298
+ def test_render_method(self):
299
+ """Test the render method."""
300
+ breadcrumbs = Breadcrumbs()
301
+ context = Context({"view_action": "list", "model": Device})
302
+
303
+ html = breadcrumbs.render(context)
304
+
305
+ expected_html = """<ol class="breadcrumb"><li><a href="/dcim/devices/">Devices</a></li></ol>"""
306
+ self.assertHTMLEqual(html, expected_html)
307
+
308
+ def test_get_extra_context(self):
309
+ """Test that get_extra_context can be extended."""
310
+
311
+ class CustomBreadcrumbs(Breadcrumbs):
312
+ def get_extra_context(self, context: Context):
313
+ return {"custom_key": "custom_value"}
314
+
315
+ render_context = {}
316
+
317
+ def capture_context(template, context, **kwargs):
318
+ nonlocal render_context
319
+ render_context = context.flatten()
320
+ return ""
321
+
322
+ breadcrumbs = CustomBreadcrumbs(items={"list": []})
323
+ context = Context({})
324
+
325
+ with patch("nautobot.core.ui.breadcrumbs.render_component_template", side_effect=capture_context):
326
+ breadcrumbs.render(context)
327
+
328
+ # Check that custom context was passed
329
+ self.assertEqual(render_context.get("custom_key"), "custom_value")
330
+
331
+ def test_should_render_skips_items(self):
332
+ """Breadcrumbs should skip items where should_render(context) is False."""
333
+
334
+ item_visible = ViewNameBreadcrumbItem(view_name="home", label="Visible", should_render=lambda _: True)
335
+ item_hidden = ViewNameBreadcrumbItem(view_name="home", label="Hidden", should_render=lambda _: False)
336
+ breadcrumbs = Breadcrumbs(items={"custom_action": [item_visible, item_hidden]})
337
+ context = Context({"view_action": "custom_action"})
338
+
339
+ items = breadcrumbs.get_breadcrumbs_items(context)
340
+ self.assertEqual(len(items), 1)
341
+ self.assertIn(("/", "Visible"), items)
342
+ self.assertNotIn(("/", "Hidden"), items)
343
+
344
+ def test_filter_breadcrumbs_items_removes_empty_pairs(self):
345
+ """filter_breadcrumbs_items should remove items where label is empty or None."""
346
+
347
+ breadcrumbs = Breadcrumbs()
348
+ # (url, label) pairs: only the last should remain
349
+ pairs = [
350
+ ("", ""), # empty
351
+ ("", " "), # whitespace
352
+ ("", "\t"), # whitespace
353
+ ("", "Non-empty"), # label not empty
354
+ ("/foo", ""), # url not empty
355
+ ("/foo", "Label"), # url, label not empty
356
+ (None, None), # both None
357
+ (None, "Label"), # label not None
358
+ ("/bar", None), # url not None
359
+ ]
360
+ expected = [
361
+ ("", "Non-empty"),
362
+ ("/foo", "Label"),
363
+ (None, "Label"),
364
+ ]
365
+ filtered = breadcrumbs.filter_breadcrumbs_items(pairs, Context({}))
366
+ self.assertEqual(filtered, expected)
@@ -1,9 +1,13 @@
1
1
  from io import StringIO
2
2
 
3
3
  from django.core.management import call_command
4
+ from django.utils.timezone import now
4
5
  import yaml
5
6
 
7
+ from nautobot.core.management.commands.check_job_approval_status import ApprovalRequiredScheduledJobsError
6
8
  from nautobot.core.testing import TestCase
9
+ from nautobot.extras.choices import JobExecutionType
10
+ from nautobot.extras.models import Job, ScheduledJob
7
11
 
8
12
 
9
13
  class ManagementCommandTestCase(TestCase):
@@ -29,3 +33,39 @@ class ManagementCommandTestCase(TestCase):
29
33
  self.assertHttpStatus(
30
34
  response, 200, f"{view_name}: {endpoint} returns status Code {response.status_code} instead of 200"
31
35
  )
36
+
37
+ def test_check_job_approval_status_no_jobs(self):
38
+ out = StringIO()
39
+ # update all jobs to not have approval_required=True
40
+ Job.objects.update(approval_required=False)
41
+ call_command("check_job_approval_status", stdout=out)
42
+ output = out.getvalue()
43
+ self.assertIn("No approval_required jobs or scheduled jobs found.", output)
44
+
45
+ def test_check_job_approval_status_with__with_approval_required_jobs(self):
46
+ out = StringIO()
47
+ self.assertTrue(Job.objects.filter(approval_required=True).exists())
48
+ self.assertFalse(ScheduledJob.objects.filter(approval_required=True).exists())
49
+ call_command("check_job_approval_status", stdout=out)
50
+ output = out.getvalue()
51
+ self.assertIn("Following jobs still have `approval_required=True`.", output)
52
+
53
+ def test_check_job_approval_status_with_approval_required_scheduled_jobs(self):
54
+ job = Job.objects.first()
55
+ scheduled_job = ScheduledJob.objects.create(
56
+ name="Scheduled Job",
57
+ task="test_managment_command.TestManagmentCommand",
58
+ job_model=job,
59
+ interval=JobExecutionType.TYPE_IMMEDIATELY,
60
+ user=self.user,
61
+ approval_required=True,
62
+ start_time=now(),
63
+ )
64
+ self.assertTrue(ScheduledJob.objects.filter(approval_required=True).exists())
65
+ with self.assertRaises(ApprovalRequiredScheduledJobsError) as cm:
66
+ call_command("check_job_approval_status")
67
+
68
+ self.assertIn(
69
+ "These need to be approved (and run) or denied before upgrading to Nautobot v3", str(cm.exception)
70
+ )
71
+ self.assertIn(scheduled_job.name, str(cm.exception))
@@ -19,10 +19,60 @@ from nautobot.core.utils import lookup
19
19
  from nautobot.dcim import choices as dcim_choices, filters as dcim_filters, models as dcim_models
20
20
  from nautobot.dcim.models import Controller, Device
21
21
  from nautobot.extras import models as extras_models
22
- from nautobot.extras.utils import FeatureQuery
22
+ from nautobot.extras.utils import FeatureQuery, RoleModelsQuery
23
23
  from nautobot.ipam import models as ipam_models
24
24
 
25
25
 
26
+ class ContentTypeMultipleChoiceFilterTest(testing.TestCase):
27
+ class RoleFilterSet(filters.BaseFilterSet):
28
+ content_types = filters.ContentTypeMultipleChoiceFilter(
29
+ choices=RoleModelsQuery().get_choices,
30
+ conjoined=False,
31
+ )
32
+
33
+ class Meta:
34
+ model = extras_models.Role
35
+ fields = ["content_types"]
36
+
37
+ def test_filter_variations(self):
38
+ with self.subTest("single label"):
39
+ filterset = self.RoleFilterSet({"content_types": ["ipam.ipaddress"]}, extras_models.Role.objects.all())
40
+ qs = extras_models.Role.objects.filter(
41
+ content_types__in=[ContentType.objects.get_for_model(ipam_models.IPAddress)]
42
+ )
43
+ self.assertQuerysetEqualAndNotEmpty(filterset.qs, qs)
44
+
45
+ with self.subTest("multiple labels"):
46
+ filterset = self.RoleFilterSet(
47
+ {"content_types": ["ipam.ipaddress", "dcim.rack"]}, extras_models.Role.objects.all()
48
+ )
49
+ # remember, conjoined=False
50
+ qs = extras_models.Role.objects.filter(
51
+ django_models.Q(content_types__in=[ContentType.objects.get_for_model(ipam_models.IPAddress)])
52
+ | django_models.Q(content_types__in=[ContentType.objects.get_for_model(dcim_models.Rack)])
53
+ ).distinct()
54
+ self.assertQuerysetEqualAndNotEmpty(filterset.qs, qs)
55
+
56
+ with self.subTest("exclude single label"):
57
+ filterset = self.RoleFilterSet({"content_types__n": ["ipam.ipaddress"]}, extras_models.Role.objects.all())
58
+ qs = extras_models.Role.objects.exclude(
59
+ content_types__in=[ContentType.objects.get_for_model(ipam_models.IPAddress)]
60
+ )
61
+ self.assertQuerysetEqualAndNotEmpty(filterset.qs, qs)
62
+
63
+ with self.subTest("exclude multiple labels"):
64
+ filterset = self.RoleFilterSet(
65
+ {"content_types__n": ["ipam.ipaddress", "dcim.rack"]}, extras_models.Role.objects.all()
66
+ )
67
+ self.assertTrue(filterset.is_valid(), filterset.errors)
68
+ # remember, conjoined=False
69
+ qs = extras_models.Role.objects.exclude(
70
+ django_models.Q(content_types__in=[ContentType.objects.get_for_model(ipam_models.IPAddress)])
71
+ | django_models.Q(content_types__in=[ContentType.objects.get_for_model(dcim_models.Rack)])
72
+ ).distinct()
73
+ self.assertQuerysetEqualAndNotEmpty(filterset.qs, qs)
74
+
75
+
26
76
  class TreeNodeMultipleChoiceFilterTest(TestCase):
27
77
  class LocationFilterSet(filters.BaseFilterSet):
28
78
  parent = filters.TreeNodeMultipleChoiceFilter(queryset=dcim_models.Location.objects.all())
@@ -581,7 +581,7 @@ class JSONFieldTest(testing.TestCase):
581
581
  device_content_type = ContentType.objects.get_for_model(dcim_models.Device)
582
582
  custom_field.content_types.set([device_content_type])
583
583
  # Fetch URL with filter parameter
584
- response = self.client.get(f'{reverse("dcim:device_list")}?name=Foo%20Device')
584
+ response = self.client.get(f"{reverse('dcim:device_list')}?name=Foo%20Device")
585
585
  self.assertIn("Foo Device", str(response.content))
586
586
 
587
587
  def test_prepare_value_with_utf8(self):
@@ -527,18 +527,18 @@ class GraphQLAPIPermissionTest(GraphQLTestCaseBase):
527
527
  for i in range(2):
528
528
  # Rack permission
529
529
  rack_obj_permission = ObjectPermission.objects.create(
530
- name=f"Permission Rack {i+1}",
530
+ name=f"Permission Rack {i + 1}",
531
531
  actions=["view", "add", "change", "delete"],
532
- constraints={"location__name": f"Location {i+1}"},
532
+ constraints={"location__name": f"Location {i + 1}"},
533
533
  )
534
534
  rack_obj_permission.object_types.add(rack_object_type)
535
535
  rack_obj_permission.groups.add(cls.groups[i])
536
536
  rack_obj_permission.users.add(cls.users[i])
537
537
 
538
538
  location_obj_permission = ObjectPermission.objects.create(
539
- name=f"Permission Location {i+1}",
539
+ name=f"Permission Location {i + 1}",
540
540
  actions=["view", "add", "change", "delete"],
541
- constraints={"name": f"Location {i+1}"},
541
+ constraints={"name": f"Location {i + 1}"},
542
542
  )
543
543
  location_obj_permission.object_types.add(location_object_type)
544
544
  location_obj_permission.groups.add(cls.groups[i])