nautobot 2.2.9__py3-none-any.whl → 2.3.0__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.
- nautobot/apps/forms.py +4 -0
- nautobot/apps/models.py +10 -1
- nautobot/circuits/__init__.py +0 -1
- nautobot/circuits/apps.py +1 -0
- nautobot/circuits/factory.py +15 -3
- nautobot/circuits/filters.py +13 -0
- nautobot/circuits/forms.py +13 -0
- nautobot/circuits/migrations/0021_alter_circuit_status_alter_circuittermination__path.py +32 -0
- nautobot/circuits/migrations/0022_circuittermination_cloud_network.py +25 -0
- nautobot/circuits/models.py +16 -3
- nautobot/circuits/tables.py +16 -2
- nautobot/circuits/templates/circuits/circuittermination_create.html +10 -2
- nautobot/circuits/templates/circuits/circuittermination_retrieve.html +6 -0
- nautobot/circuits/templates/circuits/inc/circuit_termination.html +6 -1
- nautobot/circuits/tests/test_api.py +7 -5
- nautobot/circuits/tests/test_filters.py +12 -5
- nautobot/circuits/tests/test_models.py +33 -2
- nautobot/circuits/views.py +2 -3
- nautobot/cloud/__init__.py +0 -0
- nautobot/cloud/api/__init__.py +0 -0
- nautobot/cloud/api/serializers.py +54 -0
- nautobot/cloud/api/urls.py +16 -0
- nautobot/cloud/api/views.py +48 -0
- nautobot/cloud/apps.py +13 -0
- nautobot/cloud/factory.py +113 -0
- nautobot/cloud/filters.py +187 -0
- nautobot/cloud/forms.py +339 -0
- nautobot/cloud/homepage.py +43 -0
- nautobot/cloud/migrations/0001_initial.py +304 -0
- nautobot/cloud/migrations/__init__.py +0 -0
- nautobot/cloud/models.py +246 -0
- nautobot/cloud/navigation.py +85 -0
- nautobot/cloud/tables.py +157 -0
- nautobot/cloud/templates/cloud/cloudaccount_retrieve.html +43 -0
- nautobot/cloud/templates/cloud/cloudnetwork_retrieve.html +122 -0
- nautobot/cloud/templates/cloud/cloudnetwork_update.html +33 -0
- nautobot/cloud/templates/cloud/cloudresourcetype_retrieve.html +111 -0
- nautobot/cloud/templates/cloud/cloudservice_retrieve.html +69 -0
- nautobot/cloud/templates/cloud/cloudservice_update.html +25 -0
- nautobot/cloud/tests/__init__.py +0 -0
- nautobot/cloud/tests/test_api.py +248 -0
- nautobot/cloud/tests/test_filters.py +125 -0
- nautobot/cloud/tests/test_models.py +43 -0
- nautobot/cloud/tests/test_views.py +153 -0
- nautobot/cloud/urls.py +14 -0
- nautobot/cloud/views.py +181 -0
- nautobot/core/__init__.py +0 -3
- nautobot/core/api/metadata.py +1 -0
- nautobot/core/api/parsers.py +7 -1
- nautobot/core/api/urls.py +1 -0
- nautobot/core/api/utils.py +1 -0
- nautobot/core/api/views.py +4 -0
- nautobot/core/apps/__init__.py +6 -3
- nautobot/core/constants.py +8 -0
- nautobot/core/factory.py +32 -1
- nautobot/core/filters.py +95 -13
- nautobot/core/forms/fields.py +10 -4
- nautobot/core/forms/forms.py +11 -3
- nautobot/core/forms/widgets.py +18 -1
- nautobot/core/graphql/schema.py +26 -4
- nautobot/core/jobs/__init__.py +16 -2
- nautobot/core/jobs/cleanup.py +100 -0
- nautobot/core/jobs/groups.py +38 -0
- nautobot/core/management/commands/generate_test_data.py +116 -3
- nautobot/core/models/__init__.py +34 -9
- nautobot/core/models/generics.py +19 -3
- nautobot/core/models/name_color_content_types.py +7 -28
- nautobot/core/models/querysets.py +4 -3
- nautobot/core/models/tree_queries.py +1 -1
- nautobot/core/models/utils.py +21 -5
- nautobot/core/settings.py +2 -17
- nautobot/core/settings.yaml +34 -13
- nautobot/core/settings_funcs.py +103 -0
- nautobot/core/tables.py +130 -56
- nautobot/core/templates/admin/search_form.html +1 -1
- nautobot/core/templates/buttons/add.html +11 -3
- nautobot/core/templates/buttons/consolidated_bulk_action_buttons.html +13 -0
- nautobot/core/templates/buttons/consolidated_detail_view_action_buttons.html +13 -0
- nautobot/core/templates/buttons/export.html +101 -53
- nautobot/core/templates/buttons/job_import.html +11 -3
- nautobot/core/templates/generic/object_bulk_destroy.html +3 -1
- nautobot/core/templates/generic/object_bulk_update.html +3 -1
- nautobot/core/templates/generic/object_changelog.html +0 -9
- nautobot/core/templates/generic/object_list.html +156 -17
- nautobot/core/templates/generic/object_retrieve.html +80 -16
- nautobot/core/templates/inc/extras_features_edit_form_fields.html +8 -0
- nautobot/core/templates/inc/javascript.html +2 -0
- nautobot/core/templates/inc/media.html +2 -2
- nautobot/core/templates/inc/nav_menu.html +1 -0
- nautobot/core/templates/inc/paginator.html +7 -7
- nautobot/core/templates/inc/search_panel.html +2 -2
- nautobot/core/templates/inc/table.html +2 -2
- nautobot/core/templates/nautobot_config.py.j2 +13 -8
- nautobot/core/templates/utilities/templatetags/dynamic_group_assignment_modal.html +37 -0
- nautobot/core/templates/utilities/templatetags/filter_form_modal.html +2 -2
- nautobot/core/templates/utilities/templatetags/saved_view_modal.html +38 -0
- nautobot/core/templates/utilities/theme_preview.html +25 -8
- nautobot/core/templates/utilities/worker_status.html +152 -0
- nautobot/core/templatetags/buttons.py +335 -38
- nautobot/core/templatetags/form_helpers.py +1 -1
- nautobot/core/templatetags/helpers.py +181 -11
- nautobot/core/testing/api.py +5 -4
- nautobot/core/testing/filters.py +63 -14
- nautobot/core/testing/mixins.py +46 -0
- nautobot/core/testing/models.py +22 -0
- nautobot/core/testing/schema.py +4 -8
- nautobot/core/testing/views.py +31 -14
- nautobot/core/tests/integration/test_import_objects_ui.py +1 -0
- nautobot/core/tests/integration/test_swagger.py +1 -1
- nautobot/core/tests/nautobot_config.py +0 -1
- nautobot/core/tests/runner.py +2 -2
- nautobot/core/tests/test_api.py +1 -0
- nautobot/core/tests/test_authentication.py +7 -2
- nautobot/core/tests/test_filters.py +11 -9
- nautobot/core/tests/test_forms.py +9 -0
- nautobot/core/tests/test_graphql.py +27 -16
- nautobot/core/tests/test_jobs.py +122 -0
- nautobot/core/tests/test_tables.py +3 -1
- nautobot/core/tests/test_templatetags_helpers.py +12 -5
- nautobot/core/tests/test_utils.py +31 -20
- nautobot/core/tests/test_views.py +6 -6
- nautobot/core/urls.py +8 -3
- nautobot/core/utils/deprecation.py +29 -0
- nautobot/core/utils/filtering.py +12 -9
- nautobot/core/utils/lookup.py +37 -2
- nautobot/core/utils/requests.py +4 -1
- nautobot/core/views/__init__.py +137 -24
- nautobot/core/views/generic.py +119 -67
- nautobot/core/views/mixins.py +105 -36
- nautobot/core/views/paginator.py +9 -3
- nautobot/core/views/renderers.py +121 -56
- nautobot/core/views/utils.py +81 -1
- nautobot/dcim/__init__.py +0 -1
- nautobot/dcim/api/serializers.py +180 -44
- nautobot/dcim/api/urls.py +7 -3
- nautobot/dcim/api/views.py +53 -7
- nautobot/dcim/apps.py +3 -0
- nautobot/dcim/choices.py +25 -0
- nautobot/dcim/constants.py +7 -0
- nautobot/dcim/factory.py +252 -18
- nautobot/dcim/filters/__init__.py +373 -193
- nautobot/dcim/filters/mixins.py +274 -1
- nautobot/dcim/forms.py +834 -121
- nautobot/dcim/graphql/types.py +2 -2
- nautobot/dcim/homepage.py +1 -1
- nautobot/dcim/migrations/0059_add_role_field_to_interface_models.py +27 -0
- nautobot/dcim/migrations/0060_alter_cable_status_alter_consoleport__path_and_more.py +303 -0
- nautobot/dcim/migrations/0061_module_models.py +862 -0
- nautobot/dcim/migrations/0062_module_data_migration.py +25 -0
- nautobot/dcim/models/__init__.py +8 -0
- nautobot/dcim/models/cables.py +15 -0
- nautobot/dcim/models/device_component_templates.py +207 -53
- nautobot/dcim/models/device_components.py +275 -99
- nautobot/dcim/models/devices.py +468 -13
- nautobot/dcim/models/racks.py +0 -1
- nautobot/dcim/navigation.py +47 -0
- nautobot/dcim/signals.py +3 -3
- nautobot/dcim/tables/__init__.py +35 -23
- nautobot/dcim/tables/devices.py +229 -43
- nautobot/dcim/tables/devicetypes.py +65 -9
- nautobot/dcim/tables/racks.py +5 -1
- nautobot/dcim/tables/template_code.py +46 -26
- nautobot/dcim/templates/dcim/cable_connect.html +76 -3
- nautobot/dcim/templates/dcim/console_port_connection_list.html +7 -5
- nautobot/dcim/templates/dcim/device/base.html +14 -6
- nautobot/dcim/templates/dcim/device/consoleports.html +2 -3
- nautobot/dcim/templates/dcim/device/consoleserverports.html +2 -3
- nautobot/dcim/templates/dcim/device/devicebays.html +6 -7
- nautobot/dcim/templates/dcim/device/frontports.html +2 -3
- nautobot/dcim/templates/dcim/device/interfaces.html +2 -3
- nautobot/dcim/templates/dcim/device/inventory.html +2 -3
- nautobot/dcim/templates/dcim/device/modulebays.html +49 -0
- nautobot/dcim/templates/dcim/device/poweroutlets.html +2 -3
- nautobot/dcim/templates/dcim/device/powerports.html +2 -3
- nautobot/dcim/templates/dcim/device/rearports.html +2 -3
- nautobot/dcim/templates/dcim/device.html +45 -1
- nautobot/dcim/templates/dcim/device_component.html +13 -5
- nautobot/dcim/templates/dcim/device_list.html +2 -1
- nautobot/dcim/templates/dcim/devicetype.html +99 -98
- nautobot/dcim/templates/dcim/devicetype_list.html +8 -16
- nautobot/dcim/templates/dcim/inc/devicetype_component_table.html +1 -1
- nautobot/dcim/templates/dcim/inc/moduletype_component_table.html +39 -0
- nautobot/dcim/templates/dcim/interface.html +17 -2
- nautobot/dcim/templates/dcim/interface_connection_list.html +7 -5
- nautobot/dcim/templates/dcim/interface_edit.html +1 -0
- nautobot/dcim/templates/dcim/manufacturer.html +24 -0
- nautobot/dcim/templates/dcim/module/base.html +97 -0
- nautobot/dcim/templates/dcim/module_bulk_destroy.html +5 -0
- nautobot/dcim/templates/dcim/module_consoleports.html +53 -0
- nautobot/dcim/templates/dcim/module_consoleserverports.html +53 -0
- nautobot/dcim/templates/dcim/module_destroy.html +5 -0
- nautobot/dcim/templates/dcim/module_frontports.html +53 -0
- nautobot/dcim/templates/dcim/module_interfaces.html +57 -0
- nautobot/dcim/templates/dcim/module_list.html +20 -0
- nautobot/dcim/templates/dcim/module_modulebays.html +49 -0
- nautobot/dcim/templates/dcim/module_poweroutlets.html +53 -0
- nautobot/dcim/templates/dcim/module_powerports.html +53 -0
- nautobot/dcim/templates/dcim/module_rearports.html +53 -0
- nautobot/dcim/templates/dcim/module_retrieve.html +63 -0
- nautobot/dcim/templates/dcim/module_update.html +71 -0
- nautobot/dcim/templates/dcim/modulebay_bulk_destroy.html +5 -0
- nautobot/dcim/templates/dcim/modulebay_destroy.html +8 -0
- nautobot/dcim/templates/dcim/modulebay_retrieve.html +101 -0
- nautobot/dcim/templates/dcim/moduletype_list.html +11 -0
- nautobot/dcim/templates/dcim/moduletype_retrieve.html +159 -0
- nautobot/dcim/templates/dcim/power_port_connection_list.html +7 -5
- nautobot/dcim/templates/dcim/softwareimagefile_retrieve.html +65 -19
- nautobot/dcim/tests/integration/test_cable_connect_form.py +4 -4
- nautobot/dcim/tests/test_api.py +693 -208
- nautobot/dcim/tests/test_filters.py +843 -217
- nautobot/dcim/tests/test_models.py +1072 -8
- nautobot/dcim/tests/test_views.py +1510 -341
- nautobot/dcim/urls.py +17 -2
- nautobot/dcim/utils.py +2 -3
- nautobot/dcim/views.py +1106 -116
- nautobot/extras/__init__.py +0 -1
- nautobot/extras/api/serializers.py +115 -3
- nautobot/extras/api/urls.py +12 -0
- nautobot/extras/api/views.py +66 -0
- nautobot/extras/apps.py +2 -2
- nautobot/extras/choices.py +43 -0
- nautobot/extras/context_managers.py +13 -8
- nautobot/extras/datasources/git.py +2 -0
- nautobot/extras/factory.py +460 -9
- nautobot/extras/filters/__init__.py +174 -3
- nautobot/extras/filters/mixins.py +46 -43
- nautobot/extras/forms/base.py +24 -5
- nautobot/extras/forms/forms.py +227 -8
- nautobot/extras/forms/mixins.py +93 -0
- nautobot/extras/graphql/types.py +23 -10
- nautobot/extras/homepage.py +14 -1
- nautobot/extras/management/__init__.py +1 -0
- nautobot/extras/management/commands/refresh_dynamic_group_member_caches.py +1 -16
- nautobot/extras/migrations/0021_customfield_changelog_data.py +1 -0
- nautobot/extras/migrations/0109_dynamicgroup_group_type_dynamicgroup_tags_and_more.py +108 -0
- nautobot/extras/migrations/0110_alter_configcontext_cluster_groups_and_more.py +111 -0
- nautobot/extras/migrations/0111_metadata.py +162 -0
- nautobot/extras/migrations/0112_dynamic_group_group_type_data_migration.py +28 -0
- nautobot/extras/migrations/0113_saved_views.py +77 -0
- nautobot/extras/models/__init__.py +15 -1
- nautobot/extras/models/change_logging.py +3 -3
- nautobot/extras/models/contacts.py +4 -0
- nautobot/extras/models/customfields.py +18 -3
- nautobot/extras/models/groups.py +389 -225
- nautobot/extras/models/jobs.py +6 -3
- nautobot/extras/models/metadata.py +441 -0
- nautobot/extras/models/mixins.py +72 -62
- nautobot/extras/models/models.py +118 -9
- nautobot/extras/models/relationships.py +9 -2
- nautobot/extras/models/tags.py +13 -2
- nautobot/extras/navigation.py +57 -0
- nautobot/extras/plugins/__init__.py +3 -1
- nautobot/extras/querysets.py +30 -66
- nautobot/extras/signals.py +95 -100
- nautobot/extras/tables.py +165 -12
- nautobot/extras/templates/extras/dynamicgroup.html +44 -15
- nautobot/extras/templates/extras/dynamicgroup_edit.html +2 -0
- nautobot/extras/templates/extras/job.html +1 -1
- nautobot/extras/templates/extras/jobresult.html +61 -74
- nautobot/extras/templates/extras/metadatatype_create.html +89 -0
- nautobot/extras/templates/extras/metadatatype_retrieve.html +67 -0
- nautobot/extras/templates/extras/object_dynamicgroups.html +7 -0
- nautobot/extras/templates/extras/objectchange_list.html +0 -12
- nautobot/extras/templates/extras/plugins_list.html +1 -3
- nautobot/extras/templates/extras/role_retrieve.html +48 -0
- nautobot/extras/templates/extras/staticgroupassociation_retrieve.html +20 -0
- nautobot/extras/tests/integration/test_customfields.py +1 -0
- nautobot/extras/tests/test_api.py +509 -23
- nautobot/extras/tests/test_changelog.py +20 -9
- nautobot/extras/tests/test_context_managers.py +22 -15
- nautobot/extras/tests/test_datasources.py +13 -1
- nautobot/extras/tests/test_dynamicgroups.py +201 -171
- nautobot/extras/tests/test_filters.py +211 -12
- nautobot/extras/tests/test_jobs.py +6 -6
- nautobot/extras/tests/test_models.py +501 -4
- nautobot/extras/tests/test_relationships.py +1 -0
- nautobot/extras/tests/test_views.py +565 -8
- nautobot/extras/tests/test_webhooks.py +1 -1
- nautobot/extras/urls.py +5 -0
- nautobot/extras/utils.py +51 -11
- nautobot/extras/views.py +542 -76
- nautobot/ipam/__init__.py +0 -1
- nautobot/ipam/apps.py +1 -0
- nautobot/ipam/factory.py +17 -19
- nautobot/ipam/filters.py +13 -0
- nautobot/ipam/forms.py +8 -4
- nautobot/ipam/graphql/types.py +2 -2
- nautobot/ipam/migrations/0047_alter_ipaddress_role_alter_ipaddress_status_and_more.py +59 -0
- nautobot/ipam/models.py +11 -8
- nautobot/ipam/querysets.py +1 -1
- nautobot/ipam/signals.py +4 -2
- nautobot/ipam/tables.py +5 -0
- nautobot/ipam/templates/ipam/ipaddress_interfaces.html +1 -1
- nautobot/ipam/templates/ipam/ipaddress_vm_interfaces.html +1 -1
- nautobot/ipam/templates/ipam/prefix.html +1 -0
- nautobot/ipam/tests/test_api.py +37 -18
- nautobot/ipam/tests/test_filters.py +26 -2
- nautobot/ipam/tests/test_models.py +6 -0
- nautobot/ipam/tests/test_querysets.py +1 -1
- nautobot/ipam/tests/test_views.py +3 -2
- nautobot/ipam/urls.py +2 -2
- nautobot/ipam/views.py +18 -26
- nautobot/project-static/css/base.css +20 -0
- nautobot/project-static/css/dark.css +11 -0
- nautobot/project-static/docs/404.html +892 -88
- nautobot/project-static/docs/apps/index.html +892 -88
- nautobot/project-static/docs/apps/nautobot-apps.html +892 -88
- nautobot/project-static/docs/assets/_mkdocstrings.css +5 -0
- nautobot/project-static/docs/assets/stylesheets/main.3cba04c6.min.css +1 -0
- nautobot/project-static/docs/assets/stylesheets/main.3cba04c6.min.css.map +1 -0
- nautobot/project-static/docs/code-reference/nautobot/apps/__init__.html +919 -120
- nautobot/project-static/docs/code-reference/nautobot/apps/admin.html +904 -101
- nautobot/project-static/docs/code-reference/nautobot/apps/api.html +1618 -903
- nautobot/project-static/docs/code-reference/nautobot/apps/change_logging.html +935 -144
- nautobot/project-static/docs/code-reference/nautobot/apps/choices.html +977 -188
- nautobot/project-static/docs/code-reference/nautobot/apps/config.html +901 -99
- nautobot/project-static/docs/code-reference/nautobot/apps/constants.html +897 -93
- nautobot/project-static/docs/code-reference/nautobot/apps/datasources.html +991 -193
- nautobot/project-static/docs/code-reference/nautobot/apps/exceptions.html +974 -131
- nautobot/project-static/docs/code-reference/nautobot/apps/factory.html +1078 -272
- nautobot/project-static/docs/code-reference/nautobot/apps/filters.html +1242 -334
- nautobot/project-static/docs/code-reference/nautobot/apps/forms.html +1727 -875
- nautobot/project-static/docs/code-reference/nautobot/apps/graphql.html +1164 -381
- nautobot/project-static/docs/code-reference/nautobot/apps/jobs.html +2088 -1374
- nautobot/project-static/docs/code-reference/nautobot/apps/models.html +2246 -1422
- nautobot/project-static/docs/code-reference/nautobot/apps/querysets.html +912 -111
- nautobot/project-static/docs/code-reference/nautobot/apps/secrets.html +963 -163
- nautobot/project-static/docs/code-reference/nautobot/apps/tables.html +1010 -223
- nautobot/project-static/docs/code-reference/nautobot/apps/testing.html +1913 -1277
- nautobot/project-static/docs/code-reference/nautobot/apps/ui.html +1846 -1102
- nautobot/project-static/docs/code-reference/nautobot/apps/urls.html +904 -101
- nautobot/project-static/docs/code-reference/nautobot/apps/utils.html +2331 -1699
- nautobot/project-static/docs/code-reference/nautobot/apps/views.html +1802 -1024
- nautobot/project-static/docs/development/apps/api/configuration-view.html +892 -88
- nautobot/project-static/docs/development/apps/api/database-backend-config.html +892 -88
- nautobot/project-static/docs/development/apps/api/models/django-admin.html +892 -88
- nautobot/project-static/docs/development/apps/api/models/global-search.html +892 -88
- nautobot/project-static/docs/development/apps/api/models/graphql.html +892 -88
- nautobot/project-static/docs/development/apps/api/models/index.html +942 -90
- nautobot/project-static/docs/development/apps/api/nautobot-app-config.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/custom-validators.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/filter-extensions.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/git-repository-content.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/index.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/jinja2-filters.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/jobs.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/populating-extensibility-features.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/secrets-providers.html +892 -88
- nautobot/project-static/docs/development/apps/api/platform-features/uniquely-identify-objects.html +892 -88
- nautobot/project-static/docs/development/apps/api/prometheus.html +892 -88
- nautobot/project-static/docs/development/apps/api/setup.html +892 -88
- nautobot/project-static/docs/development/apps/api/testing.html +892 -88
- nautobot/project-static/docs/development/apps/api/ui-extensions/banners.html +892 -88
- nautobot/project-static/docs/development/apps/api/ui-extensions/home-page.html +892 -88
- nautobot/project-static/docs/development/apps/api/ui-extensions/index.html +892 -88
- nautobot/project-static/docs/development/apps/api/ui-extensions/navigation.html +892 -88
- nautobot/project-static/docs/development/apps/api/ui-extensions/object-views.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/base-template.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/core-view-overrides.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/django-generic-views.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/help-documentation.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/index.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/nautobot-generic-views.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/nautobotuiviewset.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/nautobotuiviewsetrouter.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/notes.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/rest-api.html +892 -88
- nautobot/project-static/docs/development/apps/api/views/urls.html +892 -88
- nautobot/project-static/docs/development/apps/index.html +892 -88
- nautobot/project-static/docs/development/apps/migration/code-updates.html +892 -88
- nautobot/project-static/docs/development/apps/migration/dependency-updates.html +892 -88
- nautobot/project-static/docs/development/apps/migration/from-v1.html +892 -88
- nautobot/project-static/docs/development/apps/migration/model-updates/dcim.html +892 -88
- nautobot/project-static/docs/development/apps/migration/model-updates/extras.html +892 -88
- nautobot/project-static/docs/development/apps/migration/model-updates/global.html +892 -88
- nautobot/project-static/docs/development/apps/migration/model-updates/ipam.html +892 -88
- nautobot/project-static/docs/development/apps/porting-from-netbox.html +892 -88
- nautobot/project-static/docs/development/core/application-registry.html +892 -88
- nautobot/project-static/docs/development/core/best-practices.html +893 -88
- nautobot/project-static/docs/development/core/bootstrap-ui.html +892 -88
- nautobot/project-static/docs/development/core/caching.html +892 -88
- nautobot/project-static/docs/development/core/controllers.html +892 -88
- nautobot/project-static/docs/development/core/docker-compose-advanced-use-cases.html +892 -88
- nautobot/project-static/docs/development/core/generic-views.html +892 -88
- nautobot/project-static/docs/development/core/getting-started.html +892 -88
- nautobot/project-static/docs/development/core/homepage.html +892 -88
- nautobot/project-static/docs/development/core/index.html +892 -88
- nautobot/project-static/docs/development/core/model-checklist.html +901 -89
- nautobot/project-static/docs/development/core/model-features.html +892 -88
- nautobot/project-static/docs/development/core/natural-keys.html +892 -88
- nautobot/project-static/docs/development/core/navigation-menu.html +892 -88
- nautobot/project-static/docs/development/core/release-checklist.html +895 -91
- nautobot/project-static/docs/development/core/role-internals.html +892 -88
- nautobot/project-static/docs/development/core/settings.html +892 -88
- nautobot/project-static/docs/development/core/style-guide.html +893 -89
- nautobot/project-static/docs/development/core/templates.html +904 -89
- nautobot/project-static/docs/development/core/testing.html +892 -88
- nautobot/project-static/docs/development/core/user-preferences.html +892 -88
- nautobot/project-static/docs/development/index.html +892 -88
- nautobot/project-static/docs/development/jobs/index.html +893 -89
- nautobot/project-static/docs/development/jobs/migration/from-v1.html +892 -88
- nautobot/project-static/docs/index.html +892 -88
- nautobot/project-static/docs/media/models/cloud_aws_direct_connect_dark.png +0 -0
- nautobot/project-static/docs/media/models/cloud_aws_direct_connect_light.png +0 -0
- nautobot/project-static/docs/models/cloud/cloudaccount.html +15 -0
- nautobot/project-static/docs/models/cloud/cloudnetwork.html +15 -0
- nautobot/project-static/docs/models/cloud/cloudnetworkprefixassignment.html +15 -0
- nautobot/project-static/docs/models/cloud/cloudresourcetype.html +15 -0
- nautobot/project-static/docs/models/cloud/cloudservice.html +15 -0
- nautobot/project-static/docs/models/cloud/cloudservicenetworkassignment.html +15 -0
- nautobot/project-static/docs/models/dcim/module.html +15 -0
- nautobot/project-static/docs/models/dcim/modulebay.html +15 -0
- nautobot/project-static/docs/models/dcim/modulebaytemplate.html +15 -0
- nautobot/project-static/docs/models/dcim/moduletype.html +15 -0
- nautobot/project-static/docs/models/extras/metadatachoice.html +15 -0
- nautobot/project-static/docs/models/extras/metadatatype.html +15 -0
- nautobot/project-static/docs/models/extras/objectmetadata.html +15 -0
- nautobot/project-static/docs/models/extras/role.html +15 -0
- nautobot/project-static/docs/models/extras/savedview.html +15 -0
- nautobot/project-static/docs/models/extras/staticgroupassociation.html +15 -0
- nautobot/project-static/docs/models/extras/status.html +15 -0
- nautobot/project-static/docs/objects.inv +0 -0
- nautobot/project-static/docs/overview/application_stack.html +900 -89
- nautobot/project-static/docs/overview/design_philosophy.html +892 -88
- nautobot/project-static/docs/release-notes/index.html +1129 -92
- nautobot/project-static/docs/release-notes/version-1.0.html +892 -88
- nautobot/project-static/docs/release-notes/version-1.1.html +892 -88
- nautobot/project-static/docs/release-notes/version-1.2.html +892 -88
- nautobot/project-static/docs/release-notes/version-1.3.html +892 -88
- nautobot/project-static/docs/release-notes/version-1.4.html +892 -88
- nautobot/project-static/docs/release-notes/version-1.5.html +893 -89
- nautobot/project-static/docs/release-notes/version-1.6.html +893 -89
- nautobot/project-static/docs/release-notes/version-2.0.html +892 -88
- nautobot/project-static/docs/release-notes/version-2.1.html +892 -88
- nautobot/project-static/docs/release-notes/version-2.2.html +895 -91
- nautobot/project-static/docs/release-notes/version-2.3.html +9954 -0
- nautobot/project-static/docs/requirements.txt +5 -5
- nautobot/project-static/docs/search/search_index.json +1 -1
- nautobot/project-static/docs/sitemap.xml +331 -256
- nautobot/project-static/docs/sitemap.xml.gz +0 -0
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/ldap.html +892 -88
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/remote.html +892 -88
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/sso.html +892 -88
- nautobot/project-static/docs/user-guide/administration/configuration/index.html +892 -88
- nautobot/project-static/docs/user-guide/administration/configuration/optional-settings.html +992 -174
- nautobot/project-static/docs/user-guide/administration/configuration/required-settings.html +892 -88
- nautobot/project-static/docs/user-guide/administration/configuration/time-zones.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/caching.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/celery-queues.html +896 -88
- nautobot/project-static/docs/user-guide/administration/guides/healthcheck.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/permissions.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/prometheus-metrics.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/replicating-nautobot.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/request-profiling.html +892 -88
- nautobot/project-static/docs/user-guide/administration/guides/s3-django-storage.html +892 -88
- nautobot/project-static/docs/user-guide/administration/installation/app-install.html +892 -88
- nautobot/project-static/docs/user-guide/administration/installation/external-authentication.html +892 -88
- nautobot/project-static/docs/user-guide/administration/installation/http-server.html +944 -153
- nautobot/project-static/docs/user-guide/administration/installation/index.html +901 -93
- nautobot/project-static/docs/user-guide/administration/installation/install_system.html +934 -122
- nautobot/project-static/docs/user-guide/administration/installation/nautobot.html +954 -157
- nautobot/project-static/docs/user-guide/administration/installation/services.html +913 -112
- nautobot/project-static/docs/user-guide/administration/installation-extras/docker.html +908 -99
- nautobot/project-static/docs/user-guide/administration/installation-extras/health-checks.html +892 -88
- nautobot/project-static/docs/user-guide/administration/installation-extras/selinux-troubleshooting.html +892 -88
- nautobot/project-static/docs/user-guide/administration/migration/migrating-from-netbox.html +892 -88
- nautobot/project-static/docs/user-guide/administration/migration/migrating-from-postgresql.html +892 -88
- nautobot/project-static/docs/user-guide/administration/tools/nautobot-server.html +893 -89
- nautobot/project-static/docs/user-guide/administration/tools/nautobot-shell.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/database-backup.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/after-you-upgrade.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/before-you-upgrade.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/for-developers.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/index.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/whats-changed.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/region-and-site-data-migration-guide.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/upgrading-from-nautobot-v1.html +892 -88
- nautobot/project-static/docs/user-guide/administration/upgrading/upgrading.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuit.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittermination.html +893 -89
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittype.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/circuits/provider.html +896 -88
- nautobot/project-static/docs/user-guide/core-data-model/circuits/providernetwork.html +895 -91
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloud.html +8984 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudaccount.html +8828 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudnetwork.html +8829 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudnetworkprefixassignment.html +8828 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudresourcetype.html +8829 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudservice.html +8833 -0
- nautobot/project-static/docs/user-guide/core-data-model/cloud/cloudservicenetworkassignment.html +8828 -0
- nautobot/project-static/docs/user-guide/core-data-model/dcim/cable.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleport.html +923 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleporttemplate.html +923 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverport.html +918 -100
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverporttemplate.html +923 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/controller.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/controllermanageddevicegroup.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/device.html +913 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebay.html +920 -116
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebaytemplate.html +921 -117
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicefamily.html +918 -114
- nautobot/project-static/docs/user-guide/core-data-model/dcim/deviceredundancygroup.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicetype.html +914 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/frontport.html +926 -108
- nautobot/project-static/docs/user-guide/core-data-model/dcim/frontporttemplate.html +936 -118
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interface.html +928 -106
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interfaceredundancygroup.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interfacetemplate.html +937 -119
- nautobot/project-static/docs/user-guide/core-data-model/dcim/inventoryitem.html +928 -110
- nautobot/project-static/docs/user-guide/core-data-model/dcim/location.html +918 -114
- nautobot/project-static/docs/user-guide/core-data-model/dcim/locationtype.html +921 -117
- nautobot/project-static/docs/user-guide/core-data-model/dcim/manufacturer.html +923 -115
- nautobot/project-static/docs/user-guide/core-data-model/dcim/module.html +8828 -0
- nautobot/project-static/docs/user-guide/core-data-model/dcim/modulebay.html +8846 -0
- nautobot/project-static/docs/user-guide/core-data-model/dcim/modulebaytemplate.html +8843 -0
- nautobot/project-static/docs/user-guide/core-data-model/dcim/moduletype.html +8823 -0
- nautobot/project-static/docs/user-guide/core-data-model/dcim/platform.html +916 -112
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerfeed.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlet.html +940 -83
- nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlettemplate.html +924 -106
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerpanel.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerport.html +943 -86
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerporttemplate.html +921 -103
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rack.html +929 -125
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rackgroup.html +918 -114
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rackreservation.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rearport.html +922 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rearporttemplate.html +924 -106
- nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareimagefile.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareversion.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/virtualchassis.html +906 -102
- nautobot/project-static/docs/user-guide/core-data-model/extras/configcontext.html +936 -88
- nautobot/project-static/docs/user-guide/core-data-model/extras/configcontextschema.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/extras/contact.html +897 -89
- nautobot/project-static/docs/user-guide/core-data-model/extras/team.html +897 -89
- nautobot/project-static/docs/user-guide/core-data-model/ipam/ipaddress.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/namespace.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/prefix.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/rir.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/routetarget.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/service.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vlan.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vlangroup.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vrf.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/overview/introduction.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenant.html +901 -96
- nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenantgroup.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/cluster.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustergroup.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustertype.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/virtualmachine.html +892 -88
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/vminterface.html +897 -89
- nautobot/project-static/docs/user-guide/feature-guides/contacts-and-teams.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/custom-fields.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-devices.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-location-types-and-locations.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/index.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/interfaces.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/ipam.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/platforms.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/search-bar.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/tenants.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/vlans-and-vlan-groups.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/git-data-source.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/graphql.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/clear-view-button.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/cleared-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/config-table-columns-to-locations.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/configure-button.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/create-saved-view-success.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/current-saved-view-drop-down-menu.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/default-location-list-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/dropdown-button-after-new-saved-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/filter-application-to-locations.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/filter-button.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/global-default-location-list-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/location-list-view-with-saved-views.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/navigation-menu.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/save-as-new-view-drop-down.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/save-view-modal.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-admin-edit-buttons.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-admin-edit-success.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-admin-edit-view-unchecked.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-admin-edit-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-different-user.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/saved-view-modal-unchecked.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/set-as-my-default-button.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/set-as-my-default-success.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/unsaved-saved-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/images/saved-views/updated-saved-view.png +0 -0
- nautobot/project-static/docs/user-guide/feature-guides/ip-address-merge-tool.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/relationships.html +892 -88
- nautobot/project-static/docs/user-guide/feature-guides/software-image-files-and-versions.html +892 -88
- nautobot/project-static/docs/user-guide/index.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/change-logging.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/computedfield.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/customfield.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/customlink.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/dynamicgroup.html +1258 -785
- nautobot/project-static/docs/user-guide/platform-functionality/exporttemplate.html +895 -91
- nautobot/project-static/docs/user-guide/platform-functionality/externalintegration.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/gitrepository.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/graphql.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/graphqlquery.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/imageattachment.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/index.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/job-scheduling-and-approvals.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobbutton.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobhook.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/models.html +896 -88
- nautobot/project-static/docs/user-guide/platform-functionality/napalm.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/note.html +895 -91
- nautobot/project-static/docs/user-guide/platform-functionality/objectmetadata.html +9061 -0
- nautobot/project-static/docs/user-guide/platform-functionality/relationship.html +895 -91
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/authentication.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/filtering.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/overview.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/ui-related-endpoints.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/role.html +895 -91
- nautobot/project-static/docs/user-guide/platform-functionality/savedview.html +9137 -0
- nautobot/project-static/docs/user-guide/platform-functionality/secret.html +895 -91
- nautobot/project-static/docs/user-guide/platform-functionality/staticgroupassociation.html +8933 -0
- nautobot/project-static/docs/user-guide/platform-functionality/status.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/tag.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/template-filters.html +950 -121
- nautobot/project-static/docs/user-guide/platform-functionality/users/objectpermission.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/users/token.html +892 -88
- nautobot/project-static/docs/user-guide/platform-functionality/webhook.html +892 -88
- nautobot/project-static/js/forms.js +71 -0
- nautobot/project-static/js/table_sorting_indicator.js +46 -0
- nautobot/project-static/js/tableconfig.js +6 -1
- nautobot/project-static/materialdesignicons-7.4.47/css/materialdesignicons.min.css +3 -0
- nautobot/project-static/{materialdesignicons-6.5.95 → materialdesignicons-7.4.47}/fonts/materialdesignicons-webfont.eot +0 -0
- nautobot/project-static/{materialdesignicons-6.5.95 → materialdesignicons-7.4.47}/fonts/materialdesignicons-webfont.ttf +0 -0
- nautobot/project-static/materialdesignicons-7.4.47/fonts/materialdesignicons-webfont.woff +0 -0
- nautobot/project-static/materialdesignicons-7.4.47/fonts/materialdesignicons-webfont.woff2 +0 -0
- nautobot/tenancy/__init__.py +0 -1
- nautobot/tenancy/apps.py +1 -0
- nautobot/tenancy/factory.py +3 -2
- nautobot/tenancy/filters/__init__.py +1 -0
- nautobot/tenancy/forms.py +1 -1
- nautobot/tenancy/templates/tenancy/tenant.html +24 -20
- nautobot/tenancy/views.py +11 -10
- nautobot/users/__init__.py +0 -1
- nautobot/users/api/serializers.py +1 -1
- nautobot/users/api/views.py +4 -2
- nautobot/users/apps.py +3 -2
- nautobot/users/factory.py +3 -3
- nautobot/users/migrations/0010_user_default_saved_views.py +20 -0
- nautobot/users/models.py +12 -0
- nautobot/users/tests/test_filters.py +6 -3
- nautobot/users/urls.py +8 -0
- nautobot/virtualization/__init__.py +0 -1
- nautobot/virtualization/apps.py +1 -0
- nautobot/virtualization/filters.py +6 -1
- nautobot/virtualization/forms.py +11 -3
- nautobot/virtualization/graphql/types.py +2 -2
- nautobot/virtualization/migrations/0029_add_role_field_to_interface_models.py +27 -0
- nautobot/virtualization/migrations/0030_alter_virtualmachine_local_config_context_data_owner_content_type_and_more.py +67 -0
- nautobot/virtualization/models.py +0 -2
- nautobot/virtualization/tables.py +10 -3
- nautobot/virtualization/templates/virtualization/virtualmachine.html +1 -1
- nautobot/virtualization/templates/virtualization/vminterface.html +7 -1
- nautobot/virtualization/templates/virtualization/vminterface_edit.html +1 -0
- nautobot/virtualization/tests/test_api.py +9 -4
- nautobot/virtualization/tests/test_filters.py +22 -0
- nautobot/virtualization/tests/test_models.py +7 -3
- nautobot/virtualization/tests/test_views.py +19 -3
- nautobot/virtualization/urls.py +2 -2
- nautobot/virtualization/views.py +10 -32
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/METADATA +20 -18
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/RECORD +677 -557
- nautobot/project-static/docs/assets/stylesheets/main.76a95c52.min.css +0 -1
- nautobot/project-static/docs/assets/stylesheets/main.76a95c52.min.css.map +0 -1
- nautobot/project-static/materialdesignicons-6.5.95/.github/ISSUE_TEMPLATE.md +0 -3
- nautobot/project-static/materialdesignicons-6.5.95/README.md +0 -25
- nautobot/project-static/materialdesignicons-6.5.95/css/materialdesignicons.css +0 -26654
- nautobot/project-static/materialdesignicons-6.5.95/css/materialdesignicons.css.map +0 -16
- nautobot/project-static/materialdesignicons-6.5.95/css/materialdesignicons.min.css +0 -3
- nautobot/project-static/materialdesignicons-6.5.95/css/materialdesignicons.min.css.map +0 -16
- nautobot/project-static/materialdesignicons-6.5.95/fonts/materialdesignicons-webfont.woff +0 -0
- nautobot/project-static/materialdesignicons-6.5.95/fonts/materialdesignicons-webfont.woff2 +0 -0
- nautobot/project-static/materialdesignicons-6.5.95/package.json +0 -28
- nautobot/project-static/materialdesignicons-6.5.95/preview.html +0 -717
- nautobot/project-static/materialdesignicons-6.5.95/scss/_animated.scss +0 -27
- nautobot/project-static/materialdesignicons-6.5.95/scss/_core.scss +0 -10
- nautobot/project-static/materialdesignicons-6.5.95/scss/_extras.scss +0 -65
- nautobot/project-static/materialdesignicons-6.5.95/scss/_functions.scss +0 -20
- nautobot/project-static/materialdesignicons-6.5.95/scss/_icons.scss +0 -10
- nautobot/project-static/materialdesignicons-6.5.95/scss/_path.scss +0 -10
- nautobot/project-static/materialdesignicons-6.5.95/scss/_variables.scss +0 -6606
- nautobot/project-static/materialdesignicons-6.5.95/scss/materialdesignicons.scss +0 -8
- /nautobot/project-static/{materialdesignicons-6.5.95 → materialdesignicons-7.4.47}/LICENSE +0 -0
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/LICENSE.txt +0 -0
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/NOTICE +0 -0
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/WHEEL +0 -0
- {nautobot-2.2.9.dist-info → nautobot-2.3.0.dist-info}/entry_points.txt +0 -0
|
@@ -15,7 +15,7 @@ from rest_framework import status
|
|
|
15
15
|
from nautobot.core.choices import ColorChoices
|
|
16
16
|
from nautobot.core.models.fields import slugify_dashes_to_underscores
|
|
17
17
|
from nautobot.core.testing import APITestCase, APIViewTestCases
|
|
18
|
-
from nautobot.core.testing.utils import disable_warnings
|
|
18
|
+
from nautobot.core.testing.utils import disable_warnings, get_deletable_objects
|
|
19
19
|
from nautobot.core.utils.lookup import get_route_for_model
|
|
20
20
|
from nautobot.core.utils.permissions import get_permission_for_model
|
|
21
21
|
from nautobot.dcim.models import (
|
|
@@ -32,8 +32,10 @@ from nautobot.dcim.tests import test_views
|
|
|
32
32
|
from nautobot.extras.api.serializers import ConfigContextSerializer, JobResultSerializer
|
|
33
33
|
from nautobot.extras.choices import (
|
|
34
34
|
DynamicGroupOperatorChoices,
|
|
35
|
+
DynamicGroupTypeChoices,
|
|
35
36
|
JobExecutionType,
|
|
36
37
|
JobResultStatusChoices,
|
|
38
|
+
MetadataTypeDataTypeChoices,
|
|
37
39
|
ObjectChangeActionChoices,
|
|
38
40
|
ObjectChangeEventContextChoices,
|
|
39
41
|
RelationshipTypeChoices,
|
|
@@ -61,18 +63,24 @@ from nautobot.extras.models import (
|
|
|
61
63
|
Job,
|
|
62
64
|
JobLogEntry,
|
|
63
65
|
JobResult,
|
|
66
|
+
MetadataChoice,
|
|
67
|
+
MetadataType,
|
|
64
68
|
Note,
|
|
65
69
|
ObjectChange,
|
|
70
|
+
ObjectMetadata,
|
|
66
71
|
Relationship,
|
|
67
72
|
RelationshipAssociation,
|
|
68
73
|
Role,
|
|
74
|
+
SavedView,
|
|
69
75
|
ScheduledJob,
|
|
70
76
|
Secret,
|
|
71
77
|
SecretsGroup,
|
|
72
78
|
SecretsGroupAssociation,
|
|
79
|
+
StaticGroupAssociation,
|
|
73
80
|
Status,
|
|
74
81
|
Tag,
|
|
75
82
|
Team,
|
|
83
|
+
UserSavedViewAssociation,
|
|
76
84
|
Webhook,
|
|
77
85
|
)
|
|
78
86
|
from nautobot.extras.models.jobs import JobButton, JobHook
|
|
@@ -80,6 +88,7 @@ from nautobot.extras.tests.constants import BIG_GRAPHQL_DEVICE_QUERY
|
|
|
80
88
|
from nautobot.extras.tests.test_relationships import RequiredRelationshipTestMixin
|
|
81
89
|
from nautobot.extras.utils import TaggableClassesQuery
|
|
82
90
|
from nautobot.ipam.models import IPAddress, Prefix, VLAN, VLANGroup
|
|
91
|
+
from nautobot.tenancy.models import Tenant
|
|
83
92
|
from nautobot.users.models import ObjectPermission
|
|
84
93
|
|
|
85
94
|
User = get_user_model()
|
|
@@ -394,6 +403,11 @@ class ContactTest(APIViewTestCases.APIViewTestCase):
|
|
|
394
403
|
|
|
395
404
|
@classmethod
|
|
396
405
|
def setUpTestData(cls):
|
|
406
|
+
# Contacts associated with ObjectMetadata objects are protected, create some deletable contacts
|
|
407
|
+
Contact.objects.create(name="Deletable contact 1")
|
|
408
|
+
Contact.objects.create(name="Deletable contact 2")
|
|
409
|
+
Contact.objects.create(name="Deletable contact 3")
|
|
410
|
+
|
|
397
411
|
cls.create_data = [
|
|
398
412
|
{
|
|
399
413
|
"name": "Contact 1",
|
|
@@ -790,29 +804,56 @@ class DynamicGroupTestMixin:
|
|
|
790
804
|
|
|
791
805
|
class DynamicGroupTest(DynamicGroupTestMixin, APIViewTestCases.APIViewTestCase):
|
|
792
806
|
model = DynamicGroup
|
|
793
|
-
choices_fields = ["content_type"]
|
|
794
|
-
|
|
795
|
-
|
|
796
|
-
|
|
797
|
-
|
|
798
|
-
|
|
799
|
-
|
|
800
|
-
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
807
|
-
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
|
|
807
|
+
choices_fields = ["content_type", "group_type"]
|
|
808
|
+
|
|
809
|
+
@classmethod
|
|
810
|
+
def setUpTestData(cls):
|
|
811
|
+
super().setUpTestData()
|
|
812
|
+
|
|
813
|
+
cls.create_data = [
|
|
814
|
+
{
|
|
815
|
+
"name": "API DynamicGroup 4",
|
|
816
|
+
"content_type": "dcim.device",
|
|
817
|
+
"filter": {"location": ["Location 1"]},
|
|
818
|
+
"tags": [tag.pk for tag in Tag.objects.get_for_model(DynamicGroup)],
|
|
819
|
+
"tenant": Tenant.objects.first().pk,
|
|
820
|
+
},
|
|
821
|
+
{
|
|
822
|
+
"name": "API DynamicGroup 5",
|
|
823
|
+
"content_type": "dcim.device",
|
|
824
|
+
"group_type": "dynamic-filter",
|
|
825
|
+
"filter": {"has_interfaces": False},
|
|
826
|
+
},
|
|
827
|
+
{
|
|
828
|
+
"name": "API DynamicGroup 6",
|
|
829
|
+
"content_type": "dcim.device",
|
|
830
|
+
"filter": {"location": ["Location 2"]},
|
|
831
|
+
},
|
|
832
|
+
{
|
|
833
|
+
"name": "API DynamicGroup 7",
|
|
834
|
+
"content_type": "dcim.device",
|
|
835
|
+
"group_type": "static",
|
|
836
|
+
},
|
|
837
|
+
]
|
|
838
|
+
cls.update_data = {
|
|
839
|
+
"name": "A new name",
|
|
840
|
+
"tags": [],
|
|
841
|
+
"tenant": Tenant.objects.last().pk,
|
|
842
|
+
"description": "a new description",
|
|
843
|
+
}
|
|
844
|
+
|
|
845
|
+
def test_changing_content_type_not_allowed(self):
|
|
846
|
+
self.add_permissions("extras.change_dynamicgroup")
|
|
847
|
+
data = {
|
|
848
|
+
"content_type": "circuits.circuittermination",
|
|
849
|
+
}
|
|
850
|
+
response = self.client.patch(self._get_detail_url(self.groups[0]), data, format="json", **self.header)
|
|
851
|
+
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
|
811
852
|
|
|
812
853
|
def test_get_members(self):
|
|
813
854
|
"""Test that the `/members/` API endpoint returns what is expected."""
|
|
814
855
|
self.add_permissions("extras.view_dynamicgroup")
|
|
815
|
-
instance =
|
|
856
|
+
instance = DynamicGroup.objects.filter(static_group_associations__isnull=False).distinct().first()
|
|
816
857
|
self.add_permissions(get_permission_for_model(instance.content_type.model_class(), "view"))
|
|
817
858
|
member_count = instance.members.count()
|
|
818
859
|
url = reverse("extras-api:dynamicgroup-members", kwargs={"pk": instance.pk})
|
|
@@ -823,7 +864,7 @@ class DynamicGroupTest(DynamicGroupTestMixin, APIViewTestCases.APIViewTestCase):
|
|
|
823
864
|
def test_get_members_with_constrained_permission(self):
|
|
824
865
|
"""Test that the `/members/` API endpoint enforces permissions on the member model."""
|
|
825
866
|
self.add_permissions("extras.view_dynamicgroup")
|
|
826
|
-
instance =
|
|
867
|
+
instance = DynamicGroup.objects.filter(static_group_associations__isnull=False).distinct().first()
|
|
827
868
|
obj1 = instance.members.first()
|
|
828
869
|
obj_perm = ObjectPermission(
|
|
829
870
|
name="Test permission",
|
|
@@ -837,7 +878,7 @@ class DynamicGroupTest(DynamicGroupTestMixin, APIViewTestCases.APIViewTestCase):
|
|
|
837
878
|
url = reverse("extras-api:dynamicgroup-members", kwargs={"pk": instance.pk})
|
|
838
879
|
response = self.client.get(url, **self.header)
|
|
839
880
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
840
|
-
self.assertEqual(
|
|
881
|
+
self.assertEqual(response.json()["count"], 1)
|
|
841
882
|
self.assertEqual(response.json()["results"][0]["id"], str(obj1.pk))
|
|
842
883
|
|
|
843
884
|
|
|
@@ -2267,7 +2308,110 @@ class JobLogEntryTest(
|
|
|
2267
2308
|
self.add_permissions("extras.view_jobresult")
|
|
2268
2309
|
url = reverse("extras-api:jobresult-logs", kwargs={"pk": self.job_result.pk})
|
|
2269
2310
|
response = self.client.get(url, **self.header)
|
|
2270
|
-
self.assertEqual(len(response.json()), JobLogEntry.objects.count())
|
|
2311
|
+
self.assertEqual(len(response.json()), JobLogEntry.objects.filter(job_result=self.job_result).count())
|
|
2312
|
+
|
|
2313
|
+
|
|
2314
|
+
class SavedViewTest(APIViewTestCases.APIViewTestCase):
|
|
2315
|
+
model = SavedView
|
|
2316
|
+
|
|
2317
|
+
def setUp(self):
|
|
2318
|
+
super().setUp()
|
|
2319
|
+
self.create_data = [
|
|
2320
|
+
{
|
|
2321
|
+
"owner": self.user.pk,
|
|
2322
|
+
"name": "Saved View 1",
|
|
2323
|
+
"view": "circuits:circuit_list",
|
|
2324
|
+
"config": {
|
|
2325
|
+
"filter_params": {"circuit_type": ["#047c4c", "#06cc23"], "status": ["Active", "Decommissioned"]}
|
|
2326
|
+
},
|
|
2327
|
+
"is_global_default": False,
|
|
2328
|
+
"is_shared": True,
|
|
2329
|
+
},
|
|
2330
|
+
{
|
|
2331
|
+
"owner": self.user.pk,
|
|
2332
|
+
"name": "Saved View 2",
|
|
2333
|
+
"view": "dcim:device_list",
|
|
2334
|
+
"config": {
|
|
2335
|
+
"filter_params": {
|
|
2336
|
+
"location": ["Campus-01", "Building-02", "Aisle-06"],
|
|
2337
|
+
"role": ["PossibleDangerous", "NervousDangerous"],
|
|
2338
|
+
"status": ["Active", "ExtremeOriginal"],
|
|
2339
|
+
}
|
|
2340
|
+
},
|
|
2341
|
+
"is_global_default": False,
|
|
2342
|
+
"is_shared": False,
|
|
2343
|
+
},
|
|
2344
|
+
{
|
|
2345
|
+
"owner": self.user.pk,
|
|
2346
|
+
"name": "Saved View 3",
|
|
2347
|
+
"view": "dcim:location_list",
|
|
2348
|
+
"config": {
|
|
2349
|
+
"filter_params": {
|
|
2350
|
+
"location_type": ["Campus", "Building", "Elevator"],
|
|
2351
|
+
"parent": ["Campus-01", "Building-02"],
|
|
2352
|
+
"q": "building-02",
|
|
2353
|
+
},
|
|
2354
|
+
"pagination_count": 50,
|
|
2355
|
+
"sort_order": [],
|
|
2356
|
+
"table_config": {
|
|
2357
|
+
"LocationTable": {
|
|
2358
|
+
"columns": ["name", "status", "location_type", "description", "parent", "tenant"]
|
|
2359
|
+
}
|
|
2360
|
+
},
|
|
2361
|
+
},
|
|
2362
|
+
"is_global_default": False,
|
|
2363
|
+
"is_shared": True,
|
|
2364
|
+
},
|
|
2365
|
+
]
|
|
2366
|
+
|
|
2367
|
+
|
|
2368
|
+
class UserSavedViewAssociationTest(APIViewTestCases.APIViewTestCase):
|
|
2369
|
+
model = UserSavedViewAssociation
|
|
2370
|
+
|
|
2371
|
+
@classmethod
|
|
2372
|
+
def setUpTestData(cls):
|
|
2373
|
+
cls.saved_view_views_distinct = SavedView.objects.values("view").distinct()
|
|
2374
|
+
cls.users = User.objects.all()
|
|
2375
|
+
|
|
2376
|
+
cls.create_data = []
|
|
2377
|
+
for i, saved_view in enumerate(cls.saved_view_views_distinct[:3]):
|
|
2378
|
+
sv = SavedView.objects.filter(view=saved_view["view"]).first()
|
|
2379
|
+
cls.create_data.append(
|
|
2380
|
+
{
|
|
2381
|
+
"user": cls.users[i].pk,
|
|
2382
|
+
"saved_view": sv.pk,
|
|
2383
|
+
"view_name": sv.view,
|
|
2384
|
+
}
|
|
2385
|
+
)
|
|
2386
|
+
for i, saved_view in enumerate(cls.saved_view_views_distinct[4:7]):
|
|
2387
|
+
sv = SavedView.objects.filter(view=saved_view["view"]).first()
|
|
2388
|
+
UserSavedViewAssociation.objects.create(
|
|
2389
|
+
user=cls.users[i],
|
|
2390
|
+
saved_view=sv,
|
|
2391
|
+
view_name=sv.view,
|
|
2392
|
+
)
|
|
2393
|
+
|
|
2394
|
+
def test_creating_invalid_user_to_saved_view(self):
|
|
2395
|
+
# Add object-level permission
|
|
2396
|
+
duplicate_view_name = self.saved_view_views_distinct[0]["view"]
|
|
2397
|
+
saved_view = SavedView.objects.filter(view=duplicate_view_name).first()
|
|
2398
|
+
user = self.users[0]
|
|
2399
|
+
UserSavedViewAssociation.objects.create(
|
|
2400
|
+
saved_view=saved_view,
|
|
2401
|
+
user=user,
|
|
2402
|
+
view_name=saved_view.view,
|
|
2403
|
+
)
|
|
2404
|
+
duplicate_user_to_savedview_create_data = {
|
|
2405
|
+
"user": user.pk,
|
|
2406
|
+
"saved_view": saved_view.pk,
|
|
2407
|
+
"view_name": duplicate_view_name,
|
|
2408
|
+
}
|
|
2409
|
+
self.add_permissions("extras.add_usersavedviewassociation")
|
|
2410
|
+
response = self.client.post(
|
|
2411
|
+
self._get_list_url(), duplicate_user_to_savedview_create_data, format="json", **self.header
|
|
2412
|
+
)
|
|
2413
|
+
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
|
2414
|
+
self.assertIn("User saved view association with this User and View name already exists.", str(response.content))
|
|
2271
2415
|
|
|
2272
2416
|
|
|
2273
2417
|
class ScheduledJobTest(
|
|
@@ -2493,6 +2637,168 @@ class JobApprovalTest(APITestCase):
|
|
|
2493
2637
|
self.assertHttpStatus(response, status.HTTP_405_METHOD_NOT_ALLOWED)
|
|
2494
2638
|
|
|
2495
2639
|
|
|
2640
|
+
class MetadataTypeTest(APIViewTestCases.APIViewTestCase):
|
|
2641
|
+
model = MetadataType
|
|
2642
|
+
choices_fields = ["data_type"]
|
|
2643
|
+
create_data = [
|
|
2644
|
+
{
|
|
2645
|
+
"name": "System of Record",
|
|
2646
|
+
"description": "The SoR that this record or field originates from",
|
|
2647
|
+
"data_type": MetadataTypeDataTypeChoices.TYPE_TEXT,
|
|
2648
|
+
"content_types": ["dcim.device", "dcim.interface", "ipam.ipaddress"],
|
|
2649
|
+
},
|
|
2650
|
+
{
|
|
2651
|
+
"name": "Last Synced",
|
|
2652
|
+
"description": "The last time this record or field was synced from the SoR",
|
|
2653
|
+
"data_type": MetadataTypeDataTypeChoices.TYPE_DATETIME,
|
|
2654
|
+
"content_types": ["dcim.device", "dcim.interface", "ipam.ipaddress"],
|
|
2655
|
+
},
|
|
2656
|
+
{
|
|
2657
|
+
"name": "Data Owner",
|
|
2658
|
+
"data_type": MetadataTypeDataTypeChoices.TYPE_CONTACT_TEAM,
|
|
2659
|
+
"content_types": ["extras.customfield"],
|
|
2660
|
+
},
|
|
2661
|
+
]
|
|
2662
|
+
update_data = {
|
|
2663
|
+
"name": "Something new",
|
|
2664
|
+
"description": "A new name for existing metadata.",
|
|
2665
|
+
"content_types": ["dcim.interface", "ipam.vrf"],
|
|
2666
|
+
}
|
|
2667
|
+
|
|
2668
|
+
def get_deletable_object(self):
|
|
2669
|
+
return MetadataType.objects.create(name="Delete Me", data_type=MetadataTypeDataTypeChoices.TYPE_SELECT)
|
|
2670
|
+
|
|
2671
|
+
def get_deletable_object_pks(self):
|
|
2672
|
+
mdts = [
|
|
2673
|
+
MetadataType.objects.create(name="SoR", data_type=MetadataTypeDataTypeChoices.TYPE_SELECT),
|
|
2674
|
+
MetadataType.objects.create(name="Colors", data_type=MetadataTypeDataTypeChoices.TYPE_MULTISELECT),
|
|
2675
|
+
MetadataType.objects.create(
|
|
2676
|
+
name="Location Metadata Type", data_type=MetadataTypeDataTypeChoices.TYPE_SELECT
|
|
2677
|
+
),
|
|
2678
|
+
]
|
|
2679
|
+
return [mdt.pk for mdt in mdts]
|
|
2680
|
+
|
|
2681
|
+
|
|
2682
|
+
class MetadataChoiceTest(APIViewTestCases.APIViewTestCase):
|
|
2683
|
+
model = MetadataChoice
|
|
2684
|
+
|
|
2685
|
+
update_data = {
|
|
2686
|
+
"value": "Something new",
|
|
2687
|
+
"weight": 0,
|
|
2688
|
+
}
|
|
2689
|
+
|
|
2690
|
+
@classmethod
|
|
2691
|
+
def setUpTestData(cls):
|
|
2692
|
+
mdts = [
|
|
2693
|
+
MetadataType.objects.create(name="SoR", data_type=MetadataTypeDataTypeChoices.TYPE_SELECT),
|
|
2694
|
+
MetadataType.objects.create(name="Colors", data_type=MetadataTypeDataTypeChoices.TYPE_MULTISELECT),
|
|
2695
|
+
]
|
|
2696
|
+
|
|
2697
|
+
cls.create_data = [
|
|
2698
|
+
{
|
|
2699
|
+
"metadata_type": mdts[0].pk,
|
|
2700
|
+
"value": "ServiceNow",
|
|
2701
|
+
"weight": 200,
|
|
2702
|
+
},
|
|
2703
|
+
{
|
|
2704
|
+
"metadata_type": mdts[0].pk,
|
|
2705
|
+
"value": "IPFabric",
|
|
2706
|
+
},
|
|
2707
|
+
{
|
|
2708
|
+
"metadata_type": mdts[1].pk,
|
|
2709
|
+
"value": "red",
|
|
2710
|
+
"weight": 250,
|
|
2711
|
+
},
|
|
2712
|
+
{
|
|
2713
|
+
"metadata_type": mdts[1].pk,
|
|
2714
|
+
"value": "green",
|
|
2715
|
+
"weight": 250,
|
|
2716
|
+
},
|
|
2717
|
+
]
|
|
2718
|
+
|
|
2719
|
+
|
|
2720
|
+
class ObjectMetadataTest(APIViewTestCases.APIViewTestCase):
|
|
2721
|
+
model = ObjectMetadata
|
|
2722
|
+
choices_fields = ["assigned_object_type"]
|
|
2723
|
+
# ObjectMetadata records created for SoftwareImageFile records will contain a `hashing_algorithm` key;
|
|
2724
|
+
# presence of strings like "md5" and "sha256" in the API response for ObjectMetadatas is therefore *not* a failure
|
|
2725
|
+
VERBOTEN_STRINGS = ("password",)
|
|
2726
|
+
|
|
2727
|
+
@classmethod
|
|
2728
|
+
def setUpTestData(cls):
|
|
2729
|
+
mdts = [
|
|
2730
|
+
MetadataType.objects.create(name="Location Metadata Type", data_type=MetadataTypeDataTypeChoices.TYPE_TEXT),
|
|
2731
|
+
MetadataType.objects.create(name="Device Metadata Type", data_type=MetadataTypeDataTypeChoices.TYPE_TEXT),
|
|
2732
|
+
MetadataType.objects.create(
|
|
2733
|
+
name="Contact/Team Metadata Type", data_type=MetadataTypeDataTypeChoices.TYPE_CONTACT_TEAM
|
|
2734
|
+
),
|
|
2735
|
+
]
|
|
2736
|
+
mdts[0].content_types.set(list(ContentType.objects.values_list("pk", flat=True)))
|
|
2737
|
+
mdts[1].content_types.set(list(ContentType.objects.values_list("pk", flat=True)))
|
|
2738
|
+
mdts[2].content_types.set(list(ContentType.objects.values_list("pk", flat=True)))
|
|
2739
|
+
ObjectMetadata.objects.create(
|
|
2740
|
+
metadata_type=mdts[0],
|
|
2741
|
+
value="Hey",
|
|
2742
|
+
scoped_fields=["parent", "status"],
|
|
2743
|
+
assigned_object_type=ContentType.objects.get_for_model(IPAddress),
|
|
2744
|
+
assigned_object_id=IPAddress.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
2745
|
+
)
|
|
2746
|
+
ObjectMetadata.objects.create(
|
|
2747
|
+
metadata_type=mdts[0],
|
|
2748
|
+
value="Hello",
|
|
2749
|
+
scoped_fields=["namespace"],
|
|
2750
|
+
assigned_object_type=ContentType.objects.get_for_model(Prefix),
|
|
2751
|
+
assigned_object_id=Prefix.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
2752
|
+
)
|
|
2753
|
+
ObjectMetadata.objects.create(
|
|
2754
|
+
metadata_type=mdts[2],
|
|
2755
|
+
contact=Contact.objects.first(),
|
|
2756
|
+
scoped_fields=["status"],
|
|
2757
|
+
assigned_object_type=ContentType.objects.get_for_model(Prefix),
|
|
2758
|
+
assigned_object_id=Prefix.objects.filter(associated_object_metadata__isnull=True).last().pk,
|
|
2759
|
+
)
|
|
2760
|
+
cls.create_data = [
|
|
2761
|
+
{
|
|
2762
|
+
"metadata_type": mdts[0].pk,
|
|
2763
|
+
"scoped_fields": ["location_type"],
|
|
2764
|
+
"value": "random words",
|
|
2765
|
+
"assigned_object_type": "dcim.location",
|
|
2766
|
+
"assigned_object_id": Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
2767
|
+
},
|
|
2768
|
+
{
|
|
2769
|
+
"metadata_type": mdts[1].pk,
|
|
2770
|
+
"scoped_fields": ["name"],
|
|
2771
|
+
"value": "random words",
|
|
2772
|
+
"assigned_object_type": "dcim.location",
|
|
2773
|
+
"assigned_object_id": Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
2774
|
+
},
|
|
2775
|
+
{
|
|
2776
|
+
"metadata_type": mdts[2].pk,
|
|
2777
|
+
"scoped_fields": [],
|
|
2778
|
+
"contact": Contact.objects.first().pk,
|
|
2779
|
+
"assigned_object_type": "dcim.device",
|
|
2780
|
+
"assigned_object_id": Device.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
2781
|
+
},
|
|
2782
|
+
{
|
|
2783
|
+
"metadata_type": mdts[2].pk,
|
|
2784
|
+
"scoped_fields": ["interfaces"],
|
|
2785
|
+
"team": Team.objects.first().pk,
|
|
2786
|
+
"assigned_object_type": "dcim.device",
|
|
2787
|
+
"assigned_object_id": Device.objects.filter(associated_object_metadata__isnull=True).last().pk,
|
|
2788
|
+
},
|
|
2789
|
+
]
|
|
2790
|
+
cls.update_data = {
|
|
2791
|
+
"scoped_fields": ["pk"],
|
|
2792
|
+
}
|
|
2793
|
+
|
|
2794
|
+
def get_deletable_object(self):
|
|
2795
|
+
# TODO: CSV round-trip doesn't work for empty scoped_fields values at present. :-(
|
|
2796
|
+
instance = get_deletable_objects(self.model, self._get_queryset().exclude(scoped_fields=[])).first()
|
|
2797
|
+
if instance is None:
|
|
2798
|
+
self.fail("Couldn't find a single deletable object with non-empty scoped_fields")
|
|
2799
|
+
return instance
|
|
2800
|
+
|
|
2801
|
+
|
|
2496
2802
|
class NoteTest(APIViewTestCases.APIViewTestCase):
|
|
2497
2803
|
model = Note
|
|
2498
2804
|
choices_fields = ["assigned_object_type"]
|
|
@@ -2559,6 +2865,10 @@ class NoteTest(APIViewTestCases.APIViewTestCase):
|
|
|
2559
2865
|
class ObjectChangeTest(APIViewTestCases.GetObjectViewTestCase, APIViewTestCases.ListObjectsViewTestCase):
|
|
2560
2866
|
model = ObjectChange
|
|
2561
2867
|
|
|
2868
|
+
# ObjectChange records created for SoftwareImageFile records will contain a `hashing_algorithm` key;
|
|
2869
|
+
# presence of strings like "md5" and "sha256" in the API response for ObjectChanges is therefore *not* a failure
|
|
2870
|
+
VERBOTEN_STRINGS = ("password",)
|
|
2871
|
+
|
|
2562
2872
|
@classmethod
|
|
2563
2873
|
def setUpTestData(cls):
|
|
2564
2874
|
cc = ConfigContext.objects.create(name="Config Context 1", weight=100, data={"foo": 123})
|
|
@@ -3513,6 +3823,177 @@ class SecretsGroupAssociationTest(APIViewTestCases.APIViewTestCase):
|
|
|
3513
3823
|
]
|
|
3514
3824
|
|
|
3515
3825
|
|
|
3826
|
+
class StaticGroupAssociationTest(APIViewTestCases.APIViewTestCase):
|
|
3827
|
+
model = StaticGroupAssociation
|
|
3828
|
+
choices_fields = ["associated_object_type"]
|
|
3829
|
+
|
|
3830
|
+
# StaticGroupAssociation records created for SoftwareImageFile records will contain a `hashing_algorithm` key;
|
|
3831
|
+
# presence of strings like "md5" and "sha256" in the API response for StaticGroupAssociation is *not* a failure
|
|
3832
|
+
VERBOTEN_STRINGS = ("password",)
|
|
3833
|
+
|
|
3834
|
+
@classmethod
|
|
3835
|
+
def setUpTestData(cls):
|
|
3836
|
+
cls.dg1 = DynamicGroup.objects.create(
|
|
3837
|
+
name="Locations",
|
|
3838
|
+
content_type=ContentType.objects.get_for_model(Location),
|
|
3839
|
+
group_type=DynamicGroupTypeChoices.TYPE_STATIC,
|
|
3840
|
+
)
|
|
3841
|
+
cls.dg2 = DynamicGroup.objects.create(
|
|
3842
|
+
name="Devices",
|
|
3843
|
+
content_type=ContentType.objects.get_for_model(Device),
|
|
3844
|
+
group_type=DynamicGroupTypeChoices.TYPE_STATIC,
|
|
3845
|
+
)
|
|
3846
|
+
cls.dg3 = DynamicGroup.objects.create(
|
|
3847
|
+
name="VLANs",
|
|
3848
|
+
content_type=ContentType.objects.get_for_model(VLAN),
|
|
3849
|
+
group_type=DynamicGroupTypeChoices.TYPE_STATIC,
|
|
3850
|
+
)
|
|
3851
|
+
location_pks = list(Location.objects.values_list("pk", flat=True)[:4])
|
|
3852
|
+
device_pks = list(Device.objects.values_list("pk", flat=True)[:4])
|
|
3853
|
+
StaticGroupAssociation.objects.create(
|
|
3854
|
+
dynamic_group=cls.dg1,
|
|
3855
|
+
associated_object_type=ContentType.objects.get_for_model(Location),
|
|
3856
|
+
associated_object_id=location_pks[0],
|
|
3857
|
+
)
|
|
3858
|
+
StaticGroupAssociation.objects.create(
|
|
3859
|
+
dynamic_group=cls.dg1,
|
|
3860
|
+
associated_object_type=ContentType.objects.get_for_model(Location),
|
|
3861
|
+
associated_object_id=location_pks[1],
|
|
3862
|
+
)
|
|
3863
|
+
StaticGroupAssociation.objects.create(
|
|
3864
|
+
dynamic_group=cls.dg2,
|
|
3865
|
+
associated_object_type=ContentType.objects.get_for_model(Device),
|
|
3866
|
+
associated_object_id=device_pks[0],
|
|
3867
|
+
)
|
|
3868
|
+
|
|
3869
|
+
cls.create_data = [
|
|
3870
|
+
{
|
|
3871
|
+
"dynamic_group": cls.dg1.pk,
|
|
3872
|
+
"associated_object_type": "dcim.location",
|
|
3873
|
+
"associated_object_id": location_pks[2],
|
|
3874
|
+
},
|
|
3875
|
+
{
|
|
3876
|
+
"dynamic_group": cls.dg1.pk,
|
|
3877
|
+
"associated_object_type": "dcim.location",
|
|
3878
|
+
"associated_object_id": location_pks[3],
|
|
3879
|
+
},
|
|
3880
|
+
{
|
|
3881
|
+
"dynamic_group": cls.dg2.pk,
|
|
3882
|
+
"associated_object_type": "dcim.device",
|
|
3883
|
+
"associated_object_id": device_pks[2],
|
|
3884
|
+
},
|
|
3885
|
+
{
|
|
3886
|
+
"dynamic_group": cls.dg2.pk,
|
|
3887
|
+
"associated_object_type": "dcim.device",
|
|
3888
|
+
"associated_object_id": device_pks[3],
|
|
3889
|
+
},
|
|
3890
|
+
]
|
|
3891
|
+
# TODO: this isn't really valid since we're changing the associated_object_type but not the associated_object_id
|
|
3892
|
+
# Should we disallow bulk-updates of StaticGroupAssociation? Or maybe skip the bulk-update tests at least?
|
|
3893
|
+
cls.bulk_update_data = {
|
|
3894
|
+
"dynamic_group": cls.dg3.pk,
|
|
3895
|
+
"associated_object_type": "ipam.vlan",
|
|
3896
|
+
}
|
|
3897
|
+
|
|
3898
|
+
def test_content_type_mismatch(self):
|
|
3899
|
+
self.add_permissions("extras.add_staticgroupassociation")
|
|
3900
|
+
data = {
|
|
3901
|
+
"dynamic_group": self.dg1.pk,
|
|
3902
|
+
"associated_object_type": "ipam.ipaddress",
|
|
3903
|
+
"associated_object_id": IPAddress.objects.first().pk,
|
|
3904
|
+
}
|
|
3905
|
+
response = self.client.post(self._get_list_url(), data, format="json", **self.header)
|
|
3906
|
+
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
|
3907
|
+
|
|
3908
|
+
def test_list_omits_hidden_by_default(self):
|
|
3909
|
+
"""Test that the list view defaults to omitting associations of non-static groups."""
|
|
3910
|
+
sga1 = StaticGroupAssociation.all_objects.filter(
|
|
3911
|
+
dynamic_group__group_type=DynamicGroupTypeChoices.TYPE_STATIC
|
|
3912
|
+
).first()
|
|
3913
|
+
self.assertIsNotNone(sga1)
|
|
3914
|
+
sga2 = StaticGroupAssociation.all_objects.exclude(
|
|
3915
|
+
dynamic_group__group_type=DynamicGroupTypeChoices.TYPE_STATIC
|
|
3916
|
+
).first()
|
|
3917
|
+
self.assertIsNotNone(sga2)
|
|
3918
|
+
|
|
3919
|
+
self.add_permissions("extras.view_staticgroupassociation")
|
|
3920
|
+
response = self.client.get(self._get_list_url(), **self.header)
|
|
3921
|
+
self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
3922
|
+
self.assertIsInstance(response.data, dict)
|
|
3923
|
+
self.assertIn("results", response.data)
|
|
3924
|
+
found_sga1 = False
|
|
3925
|
+
found_sga2 = False
|
|
3926
|
+
for record in response.data["results"]:
|
|
3927
|
+
if record["id"] == str(sga1.id):
|
|
3928
|
+
found_sga1 = True
|
|
3929
|
+
elif record["id"] == str(sga2.id):
|
|
3930
|
+
found_sga2 = True
|
|
3931
|
+
self.assertTrue(found_sga1)
|
|
3932
|
+
self.assertFalse(found_sga2)
|
|
3933
|
+
|
|
3934
|
+
def test_list_hidden_with_filter(self):
|
|
3935
|
+
"""Test that the list view can show hidden associations with the appropriate filter."""
|
|
3936
|
+
sga1 = StaticGroupAssociation.all_objects.exclude(
|
|
3937
|
+
dynamic_group__group_type=DynamicGroupTypeChoices.TYPE_STATIC
|
|
3938
|
+
).first()
|
|
3939
|
+
self.assertIsNotNone(sga1)
|
|
3940
|
+
|
|
3941
|
+
self.add_permissions("extras.view_staticgroupassociation")
|
|
3942
|
+
response = self.client.get(f"{self._get_list_url()}?dynamic_group={sga1.dynamic_group.pk}", **self.header)
|
|
3943
|
+
self.assertHttpStatus(response, status.HTTP_200_OK)
|
|
3944
|
+
self.assertIsInstance(response.data, dict)
|
|
3945
|
+
self.assertIn("results", response.data)
|
|
3946
|
+
found_sga1 = False
|
|
3947
|
+
for record in response.data["results"]:
|
|
3948
|
+
if record["id"] == str(sga1.id):
|
|
3949
|
+
found_sga1 = True
|
|
3950
|
+
self.assertTrue(found_sga1)
|
|
3951
|
+
|
|
3952
|
+
def test_changes_to_hidden_groups_not_permitted(self):
|
|
3953
|
+
"""Test that the REST API cannot create/update/delete hidden associations."""
|
|
3954
|
+
self.add_permissions(
|
|
3955
|
+
"extras.view_staticgroupassociation",
|
|
3956
|
+
"extras.add_staticgroupassociation",
|
|
3957
|
+
"extras.delete_staticgroupassociation",
|
|
3958
|
+
"extras.change_staticgroupassociation",
|
|
3959
|
+
)
|
|
3960
|
+
|
|
3961
|
+
with self.subTest("create hidden association"):
|
|
3962
|
+
dg = DynamicGroup.objects.exclude(group_type=DynamicGroupTypeChoices.TYPE_STATIC).first()
|
|
3963
|
+
self.assertIsNotNone(dg)
|
|
3964
|
+
create_data = {
|
|
3965
|
+
"dynamic_group": str(dg.pk),
|
|
3966
|
+
"associated_object_type": f"{dg.content_type.app_label}.{dg.content_type.model}",
|
|
3967
|
+
"associated_object_id": "00000000-0000-0000-0000-000000000000",
|
|
3968
|
+
}
|
|
3969
|
+
response = self.client.post(
|
|
3970
|
+
f"{self._get_list_url()}?dynamic_group={dg.pk}", create_data, format="json", **self.header
|
|
3971
|
+
)
|
|
3972
|
+
self.assertHttpStatus(response, [status.HTTP_400_BAD_REQUEST, status.HTTP_403_FORBIDDEN])
|
|
3973
|
+
|
|
3974
|
+
with self.subTest("update hidden association"):
|
|
3975
|
+
sga = StaticGroupAssociation.all_objects.exclude(
|
|
3976
|
+
dynamic_group__group_type=DynamicGroupTypeChoices.TYPE_STATIC
|
|
3977
|
+
).first()
|
|
3978
|
+
self.assertIsNotNone(sga)
|
|
3979
|
+
url = self._get_detail_url(sga) + f"?dynamic_group={sga.dynamic_group.pk}"
|
|
3980
|
+
update_data = {"associated_object_id": "00000000-0000-0000-0000-000000000000"}
|
|
3981
|
+
response = self.client.patch(url, update_data, format="json", **self.header)
|
|
3982
|
+
self.assertHttpStatus(response, status.HTTP_404_NOT_FOUND)
|
|
3983
|
+
sga.refresh_from_db()
|
|
3984
|
+
self.assertNotEqual(sga.associated_object_id, "00000000-0000-0000-0000-000000000000")
|
|
3985
|
+
|
|
3986
|
+
with self.subTest("delete hidden association"):
|
|
3987
|
+
sga = StaticGroupAssociation.all_objects.exclude(
|
|
3988
|
+
dynamic_group__group_type=DynamicGroupTypeChoices.TYPE_STATIC
|
|
3989
|
+
).first()
|
|
3990
|
+
self.assertIsNotNone(sga)
|
|
3991
|
+
url = self._get_detail_url(sga) + f"?dynamic_group={sga.dynamic_group.pk}"
|
|
3992
|
+
response = self.client.delete(url, **self.header)
|
|
3993
|
+
self.assertHttpStatus(response, status.HTTP_404_NOT_FOUND)
|
|
3994
|
+
self.assertTrue(StaticGroupAssociation.all_objects.filter(pk=sga.pk).exists())
|
|
3995
|
+
|
|
3996
|
+
|
|
3516
3997
|
class StatusTest(APIViewTestCases.APIViewTestCase):
|
|
3517
3998
|
model = Status
|
|
3518
3999
|
bulk_update_data = {
|
|
@@ -3637,6 +4118,11 @@ class TeamTest(APIViewTestCases.APIViewTestCase):
|
|
|
3637
4118
|
|
|
3638
4119
|
@classmethod
|
|
3639
4120
|
def setUpTestData(cls):
|
|
4121
|
+
# Teams associated with ObjectMetadata objects are protected, create some deletable teams
|
|
4122
|
+
Team.objects.create(name="Deletable team 1")
|
|
4123
|
+
Team.objects.create(name="Deletable team 2")
|
|
4124
|
+
Team.objects.create(name="Deletable team 3")
|
|
4125
|
+
|
|
3640
4126
|
cls.create_data = [
|
|
3641
4127
|
{
|
|
3642
4128
|
"name": "Team 1",
|