nautobot 2.2.8__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 +110 -14
- nautobot/core/forms/fields.py +10 -4
- nautobot/core/forms/forms.py +11 -3
- nautobot/core/forms/widgets.py +18 -1
- nautobot/core/graphql/generators.py +2 -2
- nautobot/core/graphql/schema.py +28 -7
- nautobot/core/jobs/__init__.py +20 -3
- 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 +15 -19
- nautobot/core/settings.yaml +48 -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 +28 -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_general_functionality.py +1 -1
- 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 +204 -2
- nautobot/core/tests/test_tables.py +3 -1
- nautobot/core/tests/test_templatetags_helpers.py +12 -5
- nautobot/core/tests/test_templatetags_netutils.py +3 -3
- 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 +282 -99
- nautobot/dcim/models/devices.py +472 -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 +248 -47
- 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/deviceredundancygroup_retrieve.html +6 -0
- 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 +1103 -8
- nautobot/dcim/tests/test_views.py +1525 -343
- nautobot/dcim/urls.py +17 -2
- nautobot/dcim/utils.py +2 -3
- nautobot/dcim/views.py +1109 -113
- 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 +73 -59
- 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 +26 -3
- nautobot/extras/jobs.py +2 -2
- 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 +87 -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 +109 -101
- nautobot/extras/tables.py +201 -17
- 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/job_detail.html +11 -0
- 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 +586 -8
- nautobot/extras/tests/test_webhooks.py +1 -1
- nautobot/extras/urls.py +5 -0
- nautobot/extras/utils.py +85 -16
- nautobot/extras/views.py +562 -122
- 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 +20 -20
- 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 +9 -2
- 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 +20 -34
- nautobot/project-static/css/base.css +21 -0
- nautobot/project-static/css/dark.css +11 -0
- nautobot/project-static/docs/404.html +894 -90
- nautobot/project-static/docs/apps/index.html +894 -90
- nautobot/project-static/docs/apps/nautobot-apps.html +894 -90
- 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 +921 -122
- nautobot/project-static/docs/code-reference/nautobot/apps/admin.html +906 -103
- nautobot/project-static/docs/code-reference/nautobot/apps/api.html +1620 -905
- nautobot/project-static/docs/code-reference/nautobot/apps/change_logging.html +937 -146
- nautobot/project-static/docs/code-reference/nautobot/apps/choices.html +979 -190
- nautobot/project-static/docs/code-reference/nautobot/apps/config.html +903 -101
- nautobot/project-static/docs/code-reference/nautobot/apps/constants.html +899 -95
- nautobot/project-static/docs/code-reference/nautobot/apps/datasources.html +993 -195
- nautobot/project-static/docs/code-reference/nautobot/apps/exceptions.html +976 -133
- nautobot/project-static/docs/code-reference/nautobot/apps/factory.html +1080 -274
- nautobot/project-static/docs/code-reference/nautobot/apps/filters.html +1244 -336
- nautobot/project-static/docs/code-reference/nautobot/apps/forms.html +1729 -877
- nautobot/project-static/docs/code-reference/nautobot/apps/graphql.html +1166 -383
- nautobot/project-static/docs/code-reference/nautobot/apps/jobs.html +2090 -1376
- nautobot/project-static/docs/code-reference/nautobot/apps/models.html +2248 -1424
- nautobot/project-static/docs/code-reference/nautobot/apps/querysets.html +914 -113
- nautobot/project-static/docs/code-reference/nautobot/apps/secrets.html +965 -165
- nautobot/project-static/docs/code-reference/nautobot/apps/tables.html +1012 -225
- nautobot/project-static/docs/code-reference/nautobot/apps/testing.html +1915 -1279
- nautobot/project-static/docs/code-reference/nautobot/apps/ui.html +1848 -1104
- nautobot/project-static/docs/code-reference/nautobot/apps/urls.html +906 -103
- nautobot/project-static/docs/code-reference/nautobot/apps/utils.html +2335 -1701
- nautobot/project-static/docs/code-reference/nautobot/apps/views.html +1804 -1026
- nautobot/project-static/docs/development/apps/api/configuration-view.html +894 -90
- nautobot/project-static/docs/development/apps/api/database-backend-config.html +894 -90
- nautobot/project-static/docs/development/apps/api/models/django-admin.html +894 -90
- nautobot/project-static/docs/development/apps/api/models/global-search.html +894 -90
- nautobot/project-static/docs/development/apps/api/models/graphql.html +894 -90
- nautobot/project-static/docs/development/apps/api/models/index.html +944 -92
- nautobot/project-static/docs/development/apps/api/nautobot-app-config.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/custom-validators.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/filter-extensions.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/git-repository-content.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/index.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/jinja2-filters.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/jobs.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/populating-extensibility-features.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/secrets-providers.html +894 -90
- nautobot/project-static/docs/development/apps/api/platform-features/uniquely-identify-objects.html +894 -90
- nautobot/project-static/docs/development/apps/api/prometheus.html +894 -90
- nautobot/project-static/docs/development/apps/api/setup.html +894 -90
- nautobot/project-static/docs/development/apps/api/testing.html +894 -90
- nautobot/project-static/docs/development/apps/api/ui-extensions/banners.html +894 -90
- nautobot/project-static/docs/development/apps/api/ui-extensions/home-page.html +894 -90
- nautobot/project-static/docs/development/apps/api/ui-extensions/index.html +894 -90
- nautobot/project-static/docs/development/apps/api/ui-extensions/navigation.html +894 -90
- nautobot/project-static/docs/development/apps/api/ui-extensions/object-views.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/base-template.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/core-view-overrides.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/django-generic-views.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/help-documentation.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/index.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/nautobot-generic-views.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/nautobotuiviewset.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/nautobotuiviewsetrouter.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/notes.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/rest-api.html +894 -90
- nautobot/project-static/docs/development/apps/api/views/urls.html +894 -90
- nautobot/project-static/docs/development/apps/index.html +894 -90
- nautobot/project-static/docs/development/apps/migration/code-updates.html +894 -90
- nautobot/project-static/docs/development/apps/migration/dependency-updates.html +894 -90
- nautobot/project-static/docs/development/apps/migration/from-v1.html +894 -90
- nautobot/project-static/docs/development/apps/migration/model-updates/dcim.html +894 -90
- nautobot/project-static/docs/development/apps/migration/model-updates/extras.html +894 -90
- nautobot/project-static/docs/development/apps/migration/model-updates/global.html +894 -90
- nautobot/project-static/docs/development/apps/migration/model-updates/ipam.html +894 -90
- nautobot/project-static/docs/development/apps/porting-from-netbox.html +894 -90
- nautobot/project-static/docs/development/core/application-registry.html +894 -90
- nautobot/project-static/docs/development/core/best-practices.html +895 -90
- nautobot/project-static/docs/development/core/bootstrap-ui.html +894 -90
- nautobot/project-static/docs/development/core/caching.html +894 -90
- nautobot/project-static/docs/development/core/controllers.html +894 -90
- nautobot/project-static/docs/development/core/docker-compose-advanced-use-cases.html +894 -90
- nautobot/project-static/docs/development/core/generic-views.html +894 -90
- nautobot/project-static/docs/development/core/getting-started.html +894 -90
- nautobot/project-static/docs/development/core/homepage.html +894 -90
- nautobot/project-static/docs/development/core/index.html +905 -90
- nautobot/project-static/docs/development/core/model-checklist.html +903 -91
- nautobot/project-static/docs/development/core/model-features.html +894 -90
- nautobot/project-static/docs/development/core/natural-keys.html +894 -90
- nautobot/project-static/docs/development/core/navigation-menu.html +894 -90
- nautobot/project-static/docs/development/core/release-checklist.html +897 -93
- nautobot/project-static/docs/development/core/role-internals.html +894 -90
- nautobot/project-static/docs/development/core/settings.html +894 -90
- nautobot/project-static/docs/development/core/style-guide.html +895 -91
- nautobot/project-static/docs/development/core/templates.html +906 -91
- nautobot/project-static/docs/development/core/testing.html +894 -90
- nautobot/project-static/docs/development/core/user-preferences.html +894 -90
- nautobot/project-static/docs/development/index.html +894 -90
- nautobot/project-static/docs/development/jobs/index.html +1271 -453
- nautobot/project-static/docs/development/jobs/migration/from-v1.html +894 -90
- nautobot/project-static/docs/index.html +9032 -13
- 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 +902 -91
- nautobot/project-static/docs/overview/design_philosophy.html +896 -92
- nautobot/project-static/docs/overview/index.html +13 -8228
- nautobot/project-static/docs/release-notes/index.html +1131 -94
- nautobot/project-static/docs/release-notes/version-1.0.html +894 -90
- nautobot/project-static/docs/release-notes/version-1.1.html +894 -90
- nautobot/project-static/docs/release-notes/version-1.2.html +894 -90
- nautobot/project-static/docs/release-notes/version-1.3.html +894 -90
- nautobot/project-static/docs/release-notes/version-1.4.html +894 -90
- nautobot/project-static/docs/release-notes/version-1.5.html +895 -91
- nautobot/project-static/docs/release-notes/version-1.6.html +895 -91
- nautobot/project-static/docs/release-notes/version-2.0.html +894 -90
- nautobot/project-static/docs/release-notes/version-2.1.html +894 -90
- nautobot/project-static/docs/release-notes/version-2.2.html +1137 -196
- 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 +335 -260
- nautobot/project-static/docs/sitemap.xml.gz +0 -0
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/ldap.html +894 -90
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/remote.html +894 -90
- nautobot/project-static/docs/user-guide/administration/configuration/authentication/sso.html +894 -90
- nautobot/project-static/docs/user-guide/administration/configuration/index.html +894 -90
- nautobot/project-static/docs/user-guide/administration/configuration/optional-settings.html +1025 -175
- nautobot/project-static/docs/user-guide/administration/configuration/required-settings.html +894 -90
- nautobot/project-static/docs/user-guide/administration/configuration/time-zones.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/caching.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/celery-queues.html +902 -90
- nautobot/project-static/docs/user-guide/administration/guides/healthcheck.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/permissions.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/prometheus-metrics.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/replicating-nautobot.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/request-profiling.html +894 -90
- nautobot/project-static/docs/user-guide/administration/guides/s3-django-storage.html +894 -90
- nautobot/project-static/docs/user-guide/administration/installation/app-install.html +894 -90
- nautobot/project-static/docs/user-guide/administration/installation/external-authentication.html +894 -90
- nautobot/project-static/docs/user-guide/administration/installation/http-server.html +946 -155
- nautobot/project-static/docs/user-guide/administration/installation/index.html +903 -95
- nautobot/project-static/docs/user-guide/administration/installation/install_system.html +936 -124
- nautobot/project-static/docs/user-guide/administration/installation/nautobot.html +956 -159
- nautobot/project-static/docs/user-guide/administration/installation/services.html +915 -114
- nautobot/project-static/docs/user-guide/administration/installation-extras/docker.html +910 -101
- nautobot/project-static/docs/user-guide/administration/installation-extras/health-checks.html +894 -90
- nautobot/project-static/docs/user-guide/administration/installation-extras/selinux-troubleshooting.html +894 -90
- nautobot/project-static/docs/user-guide/administration/migration/migrating-from-netbox.html +894 -90
- nautobot/project-static/docs/user-guide/administration/migration/migrating-from-postgresql.html +894 -90
- nautobot/project-static/docs/user-guide/administration/tools/nautobot-server.html +977 -121
- nautobot/project-static/docs/user-guide/administration/tools/nautobot-shell.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/database-backup.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/after-you-upgrade.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/before-you-upgrade.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/for-developers.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/index.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/ipam/whats-changed.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/region-and-site-data-migration-guide.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/from-v1/upgrading-from-nautobot-v1.html +894 -90
- nautobot/project-static/docs/user-guide/administration/upgrading/upgrading.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuit.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittermination.html +895 -91
- nautobot/project-static/docs/user-guide/core-data-model/circuits/circuittype.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/circuits/provider.html +898 -90
- nautobot/project-static/docs/user-guide/core-data-model/circuits/providernetwork.html +897 -93
- 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 +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleport.html +925 -107
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleporttemplate.html +925 -107
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverport.html +920 -102
- nautobot/project-static/docs/user-guide/core-data-model/dcim/consoleserverporttemplate.html +925 -107
- nautobot/project-static/docs/user-guide/core-data-model/dcim/controller.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/controllermanageddevicegroup.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/device.html +915 -107
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebay.html +922 -118
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicebaytemplate.html +923 -119
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicefamily.html +920 -116
- nautobot/project-static/docs/user-guide/core-data-model/dcim/deviceredundancygroup.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/devicetype.html +916 -107
- nautobot/project-static/docs/user-guide/core-data-model/dcim/frontport.html +928 -110
- nautobot/project-static/docs/user-guide/core-data-model/dcim/frontporttemplate.html +938 -120
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interface.html +930 -108
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interfaceredundancygroup.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/interfacetemplate.html +939 -121
- nautobot/project-static/docs/user-guide/core-data-model/dcim/inventoryitem.html +930 -112
- nautobot/project-static/docs/user-guide/core-data-model/dcim/location.html +920 -116
- nautobot/project-static/docs/user-guide/core-data-model/dcim/locationtype.html +923 -119
- nautobot/project-static/docs/user-guide/core-data-model/dcim/manufacturer.html +925 -117
- 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 +918 -114
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerfeed.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlet.html +942 -85
- nautobot/project-static/docs/user-guide/core-data-model/dcim/poweroutlettemplate.html +926 -108
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerpanel.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerport.html +945 -88
- nautobot/project-static/docs/user-guide/core-data-model/dcim/powerporttemplate.html +923 -105
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rack.html +931 -127
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rackgroup.html +920 -116
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rackreservation.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rearport.html +924 -106
- nautobot/project-static/docs/user-guide/core-data-model/dcim/rearporttemplate.html +926 -108
- nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareimagefile.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/softwareversion.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/dcim/virtualchassis.html +908 -104
- nautobot/project-static/docs/user-guide/core-data-model/extras/configcontext.html +938 -90
- nautobot/project-static/docs/user-guide/core-data-model/extras/configcontextschema.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/extras/contact.html +899 -91
- nautobot/project-static/docs/user-guide/core-data-model/extras/team.html +899 -91
- nautobot/project-static/docs/user-guide/core-data-model/ipam/ipaddress.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/namespace.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/prefix.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/rir.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/routetarget.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/service.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vlan.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vlangroup.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/ipam/vrf.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/overview/introduction.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenant.html +903 -98
- nautobot/project-static/docs/user-guide/core-data-model/tenancy/tenantgroup.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/cluster.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustergroup.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/clustertype.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/virtualmachine.html +894 -90
- nautobot/project-static/docs/user-guide/core-data-model/virtualization/vminterface.html +899 -91
- nautobot/project-static/docs/user-guide/feature-guides/contacts-and-teams.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/custom-fields.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-devices.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/creating-location-types-and-locations.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/index.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/interfaces.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/ipam.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/platforms.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/search-bar.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/tenants.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/getting-started/vlans-and-vlan-groups.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/git-data-source.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/graphql.html +894 -90
- 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 +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/relationships.html +894 -90
- nautobot/project-static/docs/user-guide/feature-guides/software-image-files-and-versions.html +894 -90
- nautobot/project-static/docs/user-guide/index.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/change-logging.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/computedfield.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/customfield.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/customlink.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/dynamicgroup.html +1260 -787
- nautobot/project-static/docs/user-guide/platform-functionality/exporttemplate.html +897 -93
- nautobot/project-static/docs/user-guide/platform-functionality/externalintegration.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/gitrepository.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/graphql.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/graphqlquery.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/imageattachment.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/index.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/job-scheduling-and-approvals.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobbutton.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/jobhook.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/jobs/models.html +898 -90
- nautobot/project-static/docs/user-guide/platform-functionality/napalm.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/note.html +897 -93
- nautobot/project-static/docs/user-guide/platform-functionality/objectmetadata.html +9061 -0
- nautobot/project-static/docs/user-guide/platform-functionality/relationship.html +897 -93
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/authentication.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/filtering.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/overview.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/rest-api/ui-related-endpoints.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/role.html +897 -93
- nautobot/project-static/docs/user-guide/platform-functionality/savedview.html +9137 -0
- nautobot/project-static/docs/user-guide/platform-functionality/secret.html +897 -93
- nautobot/project-static/docs/user-guide/platform-functionality/staticgroupassociation.html +8933 -0
- nautobot/project-static/docs/user-guide/platform-functionality/status.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/tag.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/template-filters.html +952 -123
- nautobot/project-static/docs/user-guide/platform-functionality/users/objectpermission.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/users/token.html +894 -90
- nautobot/project-static/docs/user-guide/platform-functionality/webhook.html +894 -90
- 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 +12 -8
- 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.8.dist-info → nautobot-2.3.0.dist-info}/METADATA +21 -19
- {nautobot-2.2.8.dist-info → nautobot-2.3.0.dist-info}/RECORD +684 -564
- 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.8.dist-info → nautobot-2.3.0.dist-info}/LICENSE.txt +0 -0
- {nautobot-2.2.8.dist-info → nautobot-2.3.0.dist-info}/NOTICE +0 -0
- {nautobot-2.2.8.dist-info → nautobot-2.3.0.dist-info}/WHEEL +0 -0
- {nautobot-2.2.8.dist-info → nautobot-2.3.0.dist-info}/entry_points.txt +0 -0
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
from datetime import datetime, timedelta, timezone
|
|
1
2
|
import os
|
|
2
3
|
import tempfile
|
|
3
4
|
from unittest import expectedFailure, mock
|
|
@@ -5,6 +6,7 @@ import uuid
|
|
|
5
6
|
import warnings
|
|
6
7
|
|
|
7
8
|
from django.conf import settings
|
|
9
|
+
from django.contrib.auth import get_user_model
|
|
8
10
|
from django.contrib.contenttypes.models import ContentType
|
|
9
11
|
from django.core.exceptions import ValidationError
|
|
10
12
|
from django.core.files.uploadedfile import SimpleUploadedFile
|
|
@@ -30,6 +32,7 @@ from nautobot.dcim.models import (
|
|
|
30
32
|
from nautobot.extras.choices import (
|
|
31
33
|
JobResultStatusChoices,
|
|
32
34
|
LogLevelChoices,
|
|
35
|
+
MetadataTypeDataTypeChoices,
|
|
33
36
|
ObjectChangeActionChoices,
|
|
34
37
|
ObjectChangeEventContextChoices,
|
|
35
38
|
SecretsGroupAccessTypeChoices,
|
|
@@ -46,6 +49,7 @@ from nautobot.extras.models import (
|
|
|
46
49
|
ComputedField,
|
|
47
50
|
ConfigContext,
|
|
48
51
|
ConfigContextSchema,
|
|
52
|
+
Contact,
|
|
49
53
|
DynamicGroup,
|
|
50
54
|
ExportTemplate,
|
|
51
55
|
ExternalIntegration,
|
|
@@ -55,13 +59,19 @@ from nautobot.extras.models import (
|
|
|
55
59
|
Job as JobModel,
|
|
56
60
|
JobLogEntry,
|
|
57
61
|
JobResult,
|
|
62
|
+
MetadataChoice,
|
|
63
|
+
MetadataType,
|
|
58
64
|
ObjectChange,
|
|
65
|
+
ObjectMetadata,
|
|
59
66
|
Role,
|
|
67
|
+
SavedView,
|
|
60
68
|
Secret,
|
|
61
69
|
SecretsGroup,
|
|
62
70
|
SecretsGroupAssociation,
|
|
71
|
+
StaticGroupAssociation,
|
|
63
72
|
Status,
|
|
64
73
|
Tag,
|
|
74
|
+
Team,
|
|
65
75
|
Webhook,
|
|
66
76
|
)
|
|
67
77
|
from nautobot.extras.models.statuses import StatusModel
|
|
@@ -78,6 +88,8 @@ from nautobot.virtualization.models import (
|
|
|
78
88
|
|
|
79
89
|
from example_app.jobs import ExampleJob
|
|
80
90
|
|
|
91
|
+
User = get_user_model()
|
|
92
|
+
|
|
81
93
|
|
|
82
94
|
class ComputedFieldTest(ModelTestCases.BaseModelTestCase):
|
|
83
95
|
"""
|
|
@@ -586,6 +598,10 @@ class ConfigContextTest(ModelTestCases.BaseModelTestCase):
|
|
|
586
598
|
dynamic_group_context.dynamic_groups.add(self.dynamic_groups)
|
|
587
599
|
dynamic_group_context_2.dynamic_groups.add(self.dynamic_group_2)
|
|
588
600
|
|
|
601
|
+
# Refresh caches
|
|
602
|
+
self.dynamic_groups.update_cached_members()
|
|
603
|
+
self.dynamic_group_2.update_cached_members()
|
|
604
|
+
|
|
589
605
|
self.assertIn("dynamic context 1", self.device.get_config_context().values())
|
|
590
606
|
self.assertNotIn("dynamic context 2", self.device.get_config_context().values())
|
|
591
607
|
self.assertIn("dynamic context 2", device2.get_config_context().values())
|
|
@@ -1078,8 +1094,8 @@ class JobModelTest(ModelTestCases.BaseModelTestCase):
|
|
|
1078
1094
|
self.assertEqual(self.app_job.class_path, self.app_job.job_class.class_path)
|
|
1079
1095
|
|
|
1080
1096
|
def test_latest_result(self):
|
|
1081
|
-
self.assertEqual(self.local_job.latest_result,
|
|
1082
|
-
self.assertEqual(self.app_job.latest_result,
|
|
1097
|
+
self.assertEqual(self.local_job.latest_result, self.local_job.job_results.only("status").first())
|
|
1098
|
+
self.assertEqual(self.app_job.latest_result, self.app_job.job_results.only("status").first())
|
|
1083
1099
|
# TODO(Glenn): create some JobResults and test that this works correctly for them as well.
|
|
1084
1100
|
|
|
1085
1101
|
def test_defaults(self):
|
|
@@ -1208,6 +1224,38 @@ class JobModelTest(ModelTestCases.BaseModelTestCase):
|
|
|
1208
1224
|
)
|
|
1209
1225
|
|
|
1210
1226
|
|
|
1227
|
+
class MetadataChoiceTest(ModelTestCases.BaseModelTestCase):
|
|
1228
|
+
model = MetadataChoice
|
|
1229
|
+
|
|
1230
|
+
def test_immutable_metadata_type(self):
|
|
1231
|
+
instance1 = MetadataChoice.objects.first()
|
|
1232
|
+
instance2 = MetadataChoice.objects.exclude(metadata_type=instance1.metadata_type).first()
|
|
1233
|
+
self.assertIsNotNone(instance2)
|
|
1234
|
+
with self.assertRaises(ValidationError):
|
|
1235
|
+
instance1.metadata_type = instance2.metadata_type
|
|
1236
|
+
instance1.validated_save()
|
|
1237
|
+
|
|
1238
|
+
def test_wrong_metadata_type(self):
|
|
1239
|
+
with self.assertRaises(ValidationError):
|
|
1240
|
+
instance = MetadataChoice(
|
|
1241
|
+
metadata_type=MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_TEXT).first(),
|
|
1242
|
+
value="Hello",
|
|
1243
|
+
weight=100,
|
|
1244
|
+
)
|
|
1245
|
+
self.assertIsNotNone(instance.metadata_type)
|
|
1246
|
+
instance.validated_save()
|
|
1247
|
+
|
|
1248
|
+
|
|
1249
|
+
class MetadataTypeTest(ModelTestCases.BaseModelTestCase):
|
|
1250
|
+
model = MetadataType
|
|
1251
|
+
|
|
1252
|
+
def test_immutable_data_type(self):
|
|
1253
|
+
instance = MetadataType.objects.exclude(data_type=MetadataTypeDataTypeChoices.TYPE_TEXT).first()
|
|
1254
|
+
with self.assertRaises(ValidationError):
|
|
1255
|
+
instance.data_type = MetadataTypeDataTypeChoices.TYPE_TEXT
|
|
1256
|
+
instance.validated_save()
|
|
1257
|
+
|
|
1258
|
+
|
|
1211
1259
|
class ObjectChangeTest(ModelTestCases.BaseModelTestCase):
|
|
1212
1260
|
model = ObjectChange
|
|
1213
1261
|
|
|
@@ -1282,6 +1330,407 @@ class ObjectChangeTest(ModelTestCases.BaseModelTestCase):
|
|
|
1282
1330
|
self.assertEqual("", log.absolute_url)
|
|
1283
1331
|
|
|
1284
1332
|
|
|
1333
|
+
class ObjectMetadataTest(ModelTestCases.BaseModelTestCase):
|
|
1334
|
+
model = ObjectMetadata
|
|
1335
|
+
|
|
1336
|
+
def test_immutable_metadata_type(self):
|
|
1337
|
+
instance1 = ObjectMetadata.objects.first()
|
|
1338
|
+
instance2 = ObjectMetadata.objects.exclude(metadata_type=instance1.metadata_type).first()
|
|
1339
|
+
self.assertIsNotNone(instance2)
|
|
1340
|
+
with self.assertRaises(ValidationError):
|
|
1341
|
+
instance1.metadata_type = instance2.metadata_type
|
|
1342
|
+
instance1.validated_save()
|
|
1343
|
+
|
|
1344
|
+
def test_invalid_assigned_object_type_not_allowed(self):
|
|
1345
|
+
type_location = MetadataType.objects.create(
|
|
1346
|
+
name="Location Metadata Type", data_type=MetadataTypeDataTypeChoices.TYPE_TEXT
|
|
1347
|
+
)
|
|
1348
|
+
type_location.content_types.add(ContentType.objects.get_for_model(Location))
|
|
1349
|
+
with self.assertRaises(ValidationError):
|
|
1350
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1351
|
+
metadata_type=type_location,
|
|
1352
|
+
value="Invalid assigned object type",
|
|
1353
|
+
scoped_fields=["status"],
|
|
1354
|
+
assigned_object_type=ContentType.objects.get_for_model(IPAddress),
|
|
1355
|
+
assigned_object_id=Contact.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1356
|
+
)
|
|
1357
|
+
obj_metadata.validated_save()
|
|
1358
|
+
|
|
1359
|
+
def test_contact_team_mutual_exclusive(self):
|
|
1360
|
+
type_contact_team = MetadataType.objects.create(
|
|
1361
|
+
name="TCT", data_type=MetadataTypeDataTypeChoices.TYPE_CONTACT_TEAM
|
|
1362
|
+
)
|
|
1363
|
+
type_contact_team.content_types.add(ContentType.objects.get_for_model(Contact))
|
|
1364
|
+
type_contact_team.content_types.add(ContentType.objects.get_for_model(Team))
|
|
1365
|
+
instance1 = ObjectMetadata(
|
|
1366
|
+
metadata_type=type_contact_team,
|
|
1367
|
+
contact=Contact.objects.first(),
|
|
1368
|
+
team=Team.objects.first(),
|
|
1369
|
+
scoped_fields=["address"],
|
|
1370
|
+
assigned_object_type=ContentType.objects.get_for_model(Contact),
|
|
1371
|
+
assigned_object_id=Contact.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1372
|
+
)
|
|
1373
|
+
instance2 = ObjectMetadata(
|
|
1374
|
+
metadata_type=type_contact_team,
|
|
1375
|
+
contact=None,
|
|
1376
|
+
team=None,
|
|
1377
|
+
scoped_fields=["phone"],
|
|
1378
|
+
assigned_object_type=ContentType.objects.get_for_model(Contact),
|
|
1379
|
+
assigned_object_id=Contact.objects.filter(associated_object_metadata__isnull=True).last().pk,
|
|
1380
|
+
)
|
|
1381
|
+
instance3 = ObjectMetadata(
|
|
1382
|
+
metadata_type=type_contact_team,
|
|
1383
|
+
contact=Contact.objects.first(),
|
|
1384
|
+
team=None,
|
|
1385
|
+
scoped_fields=["email"],
|
|
1386
|
+
assigned_object_type=ContentType.objects.get_for_model(Team),
|
|
1387
|
+
assigned_object_id=Team.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1388
|
+
)
|
|
1389
|
+
with self.assertRaises(ValidationError):
|
|
1390
|
+
instance1.validated_save()
|
|
1391
|
+
|
|
1392
|
+
with self.assertRaises(ValidationError):
|
|
1393
|
+
instance2.validated_save()
|
|
1394
|
+
|
|
1395
|
+
with self.assertRaises(ValidationError):
|
|
1396
|
+
instance3.value = "Value should be empty"
|
|
1397
|
+
instance3.validated_save()
|
|
1398
|
+
|
|
1399
|
+
def test_text_field_value(self):
|
|
1400
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1401
|
+
text_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_TEXT).first()
|
|
1402
|
+
text_metadata_type.content_types.add(obj_type)
|
|
1403
|
+
|
|
1404
|
+
# Create an ObjectMetadata
|
|
1405
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1406
|
+
metadata_type=text_metadata_type,
|
|
1407
|
+
value="Some text value",
|
|
1408
|
+
scoped_fields=["status", "parent"],
|
|
1409
|
+
assigned_object_type=obj_type,
|
|
1410
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1411
|
+
)
|
|
1412
|
+
obj_metadata.save()
|
|
1413
|
+
|
|
1414
|
+
# Assign a disallowed value (list) to obj_metadata
|
|
1415
|
+
with self.assertRaises(ValidationError) as context:
|
|
1416
|
+
obj_metadata.value = ["I", "am", "a", "list"]
|
|
1417
|
+
obj_metadata.validated_save()
|
|
1418
|
+
self.assertIn("Value must be a string", str(context.exception))
|
|
1419
|
+
|
|
1420
|
+
# Assign another disallowed value (int) to the first Location
|
|
1421
|
+
with self.assertRaises(ValidationError) as context:
|
|
1422
|
+
obj_metadata.value = 2
|
|
1423
|
+
obj_metadata.validated_save()
|
|
1424
|
+
self.assertIn("Value must be a string", str(context.exception))
|
|
1425
|
+
|
|
1426
|
+
# Assign another disallowed value (bool) to the first Location
|
|
1427
|
+
with self.assertRaises(ValidationError) as context:
|
|
1428
|
+
obj_metadata.value = True
|
|
1429
|
+
obj_metadata.validated_save()
|
|
1430
|
+
self.assertIn("Value must be a string", str(context.exception))
|
|
1431
|
+
obj_metadata.delete()
|
|
1432
|
+
|
|
1433
|
+
def test_integer_field_value(self):
|
|
1434
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1435
|
+
int_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_INTEGER).first()
|
|
1436
|
+
int_metadata_type.content_types.add(obj_type)
|
|
1437
|
+
# Create an ObjectMetadata
|
|
1438
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1439
|
+
metadata_type=int_metadata_type,
|
|
1440
|
+
value=15,
|
|
1441
|
+
scoped_fields=["status", "parent"],
|
|
1442
|
+
assigned_object_type=obj_type,
|
|
1443
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1444
|
+
)
|
|
1445
|
+
obj_metadata.validated_save()
|
|
1446
|
+
|
|
1447
|
+
# Assign another disallowed value (str) to the first Location
|
|
1448
|
+
with self.assertRaises(ValidationError) as context:
|
|
1449
|
+
obj_metadata.value = "I am not an integer"
|
|
1450
|
+
obj_metadata.validated_save()
|
|
1451
|
+
self.assertIn("Value must be an integer", str(context.exception))
|
|
1452
|
+
# Assign another disallowed value (str of a float) to the first Location
|
|
1453
|
+
with self.assertRaises(ValidationError) as context:
|
|
1454
|
+
obj_metadata.value = "2.0"
|
|
1455
|
+
obj_metadata.validated_save()
|
|
1456
|
+
self.assertIn("Value must be an integer", str(context.exception))
|
|
1457
|
+
|
|
1458
|
+
obj_metadata.value = 2.0
|
|
1459
|
+
obj_metadata.validated_save()
|
|
1460
|
+
self.assertEqual(obj_metadata.value, 2)
|
|
1461
|
+
obj_metadata.value = 15.0
|
|
1462
|
+
obj_metadata.validated_save()
|
|
1463
|
+
self.assertEqual(obj_metadata.value, 15)
|
|
1464
|
+
obj_metadata.value = 15.2
|
|
1465
|
+
obj_metadata.validated_save()
|
|
1466
|
+
self.assertEqual(obj_metadata.value, 15)
|
|
1467
|
+
obj_metadata.value = 15
|
|
1468
|
+
obj_metadata.validated_save()
|
|
1469
|
+
self.assertEqual(obj_metadata.value, 15)
|
|
1470
|
+
obj_metadata.value = "15"
|
|
1471
|
+
obj_metadata.validated_save()
|
|
1472
|
+
self.assertEqual(obj_metadata.value, 15)
|
|
1473
|
+
|
|
1474
|
+
# TODO add validation_minimum/validation_maximum tests
|
|
1475
|
+
obj_metadata.delete()
|
|
1476
|
+
|
|
1477
|
+
def test_float_field_value(self):
|
|
1478
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1479
|
+
float_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_FLOAT).first()
|
|
1480
|
+
float_metadata_type.content_types.add(obj_type)
|
|
1481
|
+
# Create an ObjectMetadata
|
|
1482
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1483
|
+
metadata_type=float_metadata_type,
|
|
1484
|
+
value=15.245,
|
|
1485
|
+
scoped_fields=["status", "parent"],
|
|
1486
|
+
assigned_object_type=obj_type,
|
|
1487
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1488
|
+
)
|
|
1489
|
+
obj_metadata.validated_save()
|
|
1490
|
+
|
|
1491
|
+
# Assign another disallowed value (str) to the first Location
|
|
1492
|
+
with self.assertRaises(ValidationError) as context:
|
|
1493
|
+
obj_metadata.value = "I am not a float"
|
|
1494
|
+
obj_metadata.validated_save()
|
|
1495
|
+
self.assertIn("Value must be a float", str(context.exception))
|
|
1496
|
+
|
|
1497
|
+
# Assign another disallowed value (int) to the first Location
|
|
1498
|
+
obj_metadata.value = 2
|
|
1499
|
+
obj_metadata.validated_save()
|
|
1500
|
+
self.assertEqual(obj_metadata.value, 2.0)
|
|
1501
|
+
obj_metadata.value = 15
|
|
1502
|
+
obj_metadata.validated_save()
|
|
1503
|
+
self.assertEqual(obj_metadata.value, 15.0)
|
|
1504
|
+
obj_metadata.value = 15.2
|
|
1505
|
+
obj_metadata.validated_save()
|
|
1506
|
+
self.assertEqual(obj_metadata.value, 15.2)
|
|
1507
|
+
obj_metadata.value = "3"
|
|
1508
|
+
obj_metadata.validated_save()
|
|
1509
|
+
self.assertEqual(obj_metadata.value, 3.0)
|
|
1510
|
+
obj_metadata.value = "15.2"
|
|
1511
|
+
obj_metadata.validated_save()
|
|
1512
|
+
self.assertEqual(obj_metadata.value, 15.2)
|
|
1513
|
+
|
|
1514
|
+
# TODO add validation_minimum/validation_maximum tests
|
|
1515
|
+
obj_metadata.delete()
|
|
1516
|
+
|
|
1517
|
+
def test_boolean_field_value(self):
|
|
1518
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1519
|
+
bool_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_BOOLEAN).first()
|
|
1520
|
+
bool_metadata_type.content_types.add(obj_type)
|
|
1521
|
+
|
|
1522
|
+
# Create an ObjectMetadata
|
|
1523
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1524
|
+
metadata_type=bool_metadata_type,
|
|
1525
|
+
value=False,
|
|
1526
|
+
scoped_fields=["status", "parent"],
|
|
1527
|
+
assigned_object_type=obj_type,
|
|
1528
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1529
|
+
)
|
|
1530
|
+
obj_metadata.validated_save()
|
|
1531
|
+
|
|
1532
|
+
# Assign a disallowed value (list) to obj_metadata
|
|
1533
|
+
with self.assertRaises(ValidationError) as context:
|
|
1534
|
+
obj_metadata.value = ["I", "am", "a", "list"]
|
|
1535
|
+
obj_metadata.validated_save()
|
|
1536
|
+
self.assertIn("Value must be true or false.", str(context.exception))
|
|
1537
|
+
|
|
1538
|
+
# Assign another disallowed value (str) to the first Location
|
|
1539
|
+
with self.assertRaises(ValidationError) as context:
|
|
1540
|
+
obj_metadata.value = "I am not an integer"
|
|
1541
|
+
obj_metadata.validated_save()
|
|
1542
|
+
self.assertIn("Value must be true or false.", str(context.exception))
|
|
1543
|
+
|
|
1544
|
+
# Assign another disallowed value (int) to the first Location
|
|
1545
|
+
with self.assertRaises(ValidationError) as context:
|
|
1546
|
+
obj_metadata.value = 2
|
|
1547
|
+
obj_metadata.validated_save()
|
|
1548
|
+
self.assertIn("Value must be true or false.", str(context.exception))
|
|
1549
|
+
obj_metadata.delete()
|
|
1550
|
+
|
|
1551
|
+
def test_date_field_value(self):
|
|
1552
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1553
|
+
date_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_DATE).first()
|
|
1554
|
+
date_metadata_type.content_types.add(obj_type)
|
|
1555
|
+
|
|
1556
|
+
# Create an ObjectMetadata
|
|
1557
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1558
|
+
metadata_type=date_metadata_type,
|
|
1559
|
+
value="1994-01-01",
|
|
1560
|
+
scoped_fields=["status", "parent"],
|
|
1561
|
+
assigned_object_type=obj_type,
|
|
1562
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1563
|
+
)
|
|
1564
|
+
obj_metadata.validated_save()
|
|
1565
|
+
|
|
1566
|
+
# Assign a disallowed value (invalidly formatted date) to obj_metadata
|
|
1567
|
+
with self.assertRaises(ValidationError) as context:
|
|
1568
|
+
obj_metadata.value = "01/01/1994"
|
|
1569
|
+
obj_metadata.validated_save()
|
|
1570
|
+
self.assertIn("Date values must be in the format YYYY-MM-DD.", str(context.exception))
|
|
1571
|
+
|
|
1572
|
+
# Assign another disallowed value (str) to the first Location
|
|
1573
|
+
with self.assertRaises(ValidationError) as context:
|
|
1574
|
+
obj_metadata.value = "I am not an integer"
|
|
1575
|
+
obj_metadata.validated_save()
|
|
1576
|
+
self.assertIn("Date values must be in the format YYYY-MM-DD.", str(context.exception))
|
|
1577
|
+
|
|
1578
|
+
# Assign another disallowed value (int) to the first Location
|
|
1579
|
+
with self.assertRaises(ValidationError) as context:
|
|
1580
|
+
obj_metadata.value = 2
|
|
1581
|
+
obj_metadata.validated_save()
|
|
1582
|
+
self.assertIn("Value must be a date or str object.", str(context.exception))
|
|
1583
|
+
# TODO add validation_minimum/validation_maximum tests
|
|
1584
|
+
obj_metadata.delete()
|
|
1585
|
+
|
|
1586
|
+
def test_datetime_field_value(self):
|
|
1587
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1588
|
+
datetime_metadata_type = MetadataType.objects.filter(
|
|
1589
|
+
data_type=MetadataTypeDataTypeChoices.TYPE_DATETIME
|
|
1590
|
+
).first()
|
|
1591
|
+
datetime_metadata_type.content_types.add(obj_type)
|
|
1592
|
+
|
|
1593
|
+
# Create an ObjectMetadata
|
|
1594
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1595
|
+
metadata_type=datetime_metadata_type,
|
|
1596
|
+
value="2024-06-27T17:58:47-0500",
|
|
1597
|
+
scoped_fields=["status", "parent"],
|
|
1598
|
+
assigned_object_type=obj_type,
|
|
1599
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1600
|
+
)
|
|
1601
|
+
obj_metadata.validated_save()
|
|
1602
|
+
|
|
1603
|
+
# Test valid formats of datetime value
|
|
1604
|
+
acceptable_datetime_formats = [
|
|
1605
|
+
"YYYY-MM-DDTHH:MM:SS",
|
|
1606
|
+
"YYYY-MM-DDTHH:MM:SS(+,-)zzzz",
|
|
1607
|
+
"YYYY-MM-DDTHH:MM:SS(+,-)zz:zz",
|
|
1608
|
+
]
|
|
1609
|
+
obj_metadata.value = "2024-06-27T17:58:47"
|
|
1610
|
+
obj_metadata.validated_save()
|
|
1611
|
+
self.assertEqual(obj_metadata.value, "2024-06-27T17:58:47+0000")
|
|
1612
|
+
obj_metadata.value = "2024-06-27T17:58:47+0500"
|
|
1613
|
+
obj_metadata.validated_save()
|
|
1614
|
+
self.assertEqual(obj_metadata.value, "2024-06-27T17:58:47+0500")
|
|
1615
|
+
obj_metadata.value = "2024-06-27T17:58:47+05:00"
|
|
1616
|
+
obj_metadata.validated_save()
|
|
1617
|
+
self.assertEqual(obj_metadata.value, "2024-06-27T17:58:47+05:00")
|
|
1618
|
+
obj_metadata.value = datetime(2020, 11, 1, 1)
|
|
1619
|
+
obj_metadata.validated_save()
|
|
1620
|
+
self.assertEqual(obj_metadata.value, "2020-11-01T01:00:00+00:00")
|
|
1621
|
+
obj_metadata.value = datetime(2020, 11, 1, 1, 35, 22)
|
|
1622
|
+
obj_metadata.validated_save()
|
|
1623
|
+
self.assertEqual(obj_metadata.value, "2020-11-01T01:35:22+00:00")
|
|
1624
|
+
obj_metadata.value = datetime(2020, 11, 1, 1, 35, 22, tzinfo=timezone(timedelta(hours=3)))
|
|
1625
|
+
obj_metadata.validated_save()
|
|
1626
|
+
self.assertEqual(obj_metadata.value, "2020-11-01T01:35:22+03:00")
|
|
1627
|
+
|
|
1628
|
+
error_message = f"Datetime values must be in the following formats {acceptable_datetime_formats}"
|
|
1629
|
+
with self.assertRaises(ValidationError) as context:
|
|
1630
|
+
obj_metadata.value = "01/01/1994"
|
|
1631
|
+
obj_metadata.validated_save()
|
|
1632
|
+
self.assertIn(error_message, str(context.exception))
|
|
1633
|
+
with self.assertRaises(ValidationError) as context:
|
|
1634
|
+
obj_metadata.value = "2024-06-27 17:58:47+0000"
|
|
1635
|
+
obj_metadata.validated_save()
|
|
1636
|
+
self.assertIn(error_message, str(context.exception))
|
|
1637
|
+
|
|
1638
|
+
with self.assertRaises(ValidationError) as context:
|
|
1639
|
+
obj_metadata.value = "I am not an integer"
|
|
1640
|
+
obj_metadata.validated_save()
|
|
1641
|
+
self.assertIn(error_message, str(context.exception))
|
|
1642
|
+
|
|
1643
|
+
with self.assertRaises(ValidationError) as context:
|
|
1644
|
+
obj_metadata.value = 2
|
|
1645
|
+
obj_metadata.validated_save()
|
|
1646
|
+
self.assertIn("Value must be a datetime or str object", str(context.exception))
|
|
1647
|
+
|
|
1648
|
+
# TODO add validation_minimum/validation_maximum tests
|
|
1649
|
+
obj_metadata.delete()
|
|
1650
|
+
|
|
1651
|
+
def test_select_field(self):
|
|
1652
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1653
|
+
select_metadata_type = MetadataType.objects.filter(data_type=MetadataTypeDataTypeChoices.TYPE_SELECT).first()
|
|
1654
|
+
select_metadata_type.content_types.add(obj_type)
|
|
1655
|
+
|
|
1656
|
+
MetadataChoice.objects.create(metadata_type=select_metadata_type, value="Option A")
|
|
1657
|
+
MetadataChoice.objects.create(metadata_type=select_metadata_type, value="Option B")
|
|
1658
|
+
MetadataChoice.objects.create(metadata_type=select_metadata_type, value="Option C")
|
|
1659
|
+
|
|
1660
|
+
# Create an ObjectMetadata
|
|
1661
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1662
|
+
metadata_type=select_metadata_type,
|
|
1663
|
+
value="Option A",
|
|
1664
|
+
scoped_fields=["status", "parent"],
|
|
1665
|
+
assigned_object_type=obj_type,
|
|
1666
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1667
|
+
)
|
|
1668
|
+
obj_metadata.validated_save()
|
|
1669
|
+
|
|
1670
|
+
with self.assertRaises(ValidationError) as context:
|
|
1671
|
+
obj_metadata.value = "Not valid option"
|
|
1672
|
+
obj_metadata.validated_save()
|
|
1673
|
+
self.assertIn("Invalid choice (Not valid option)", str(context.exception))
|
|
1674
|
+
|
|
1675
|
+
def test_multi_select_field(self):
|
|
1676
|
+
obj_type = ContentType.objects.get_for_model(Location)
|
|
1677
|
+
multi_select_metadata_type = MetadataType.objects.filter(
|
|
1678
|
+
data_type=MetadataTypeDataTypeChoices.TYPE_MULTISELECT
|
|
1679
|
+
).first()
|
|
1680
|
+
multi_select_metadata_type.content_types.add(obj_type)
|
|
1681
|
+
|
|
1682
|
+
MetadataChoice.objects.create(metadata_type=multi_select_metadata_type, value="Option A")
|
|
1683
|
+
MetadataChoice.objects.create(metadata_type=multi_select_metadata_type, value="Option B")
|
|
1684
|
+
MetadataChoice.objects.create(metadata_type=multi_select_metadata_type, value="Option C")
|
|
1685
|
+
|
|
1686
|
+
# Create an ObjectMetadata
|
|
1687
|
+
obj_metadata = ObjectMetadata.objects.create(
|
|
1688
|
+
metadata_type=multi_select_metadata_type,
|
|
1689
|
+
value=["Option A"],
|
|
1690
|
+
scoped_fields=["status", "parent"],
|
|
1691
|
+
assigned_object_type=obj_type,
|
|
1692
|
+
assigned_object_id=Location.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1693
|
+
)
|
|
1694
|
+
obj_metadata.validated_save()
|
|
1695
|
+
|
|
1696
|
+
invalid_options = ["Not A valid option", "NOT A VALID OPTION"]
|
|
1697
|
+
with self.assertRaises(ValidationError) as context:
|
|
1698
|
+
obj_metadata.value = invalid_options
|
|
1699
|
+
obj_metadata.validated_save()
|
|
1700
|
+
self.assertIn(f"Invalid choice(s) ({invalid_options})", str(context.exception))
|
|
1701
|
+
|
|
1702
|
+
def test_no_scoped_fields_overlap(self):
|
|
1703
|
+
"""
|
|
1704
|
+
Test that overlapping scoped_fields of ObjectMetadata with same metadata_type/assigned_object is not allowed.
|
|
1705
|
+
"""
|
|
1706
|
+
ObjectMetadata.objects.create(
|
|
1707
|
+
metadata_type=MetadataType.objects.first(),
|
|
1708
|
+
contact=Contact.objects.first(),
|
|
1709
|
+
scoped_fields=["host", "mask_length", "type", "role", "status"],
|
|
1710
|
+
assigned_object_type=ContentType.objects.get_for_model(IPAddress),
|
|
1711
|
+
assigned_object_id=IPAddress.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1712
|
+
)
|
|
1713
|
+
instance2 = ObjectMetadata.objects.create(
|
|
1714
|
+
metadata_type=MetadataType.objects.first(),
|
|
1715
|
+
contact=Contact.objects.first(),
|
|
1716
|
+
scoped_fields=[],
|
|
1717
|
+
assigned_object_type=ContentType.objects.get_for_model(IPAddress),
|
|
1718
|
+
assigned_object_id=IPAddress.objects.filter(associated_object_metadata__isnull=True).first().pk,
|
|
1719
|
+
)
|
|
1720
|
+
with self.assertRaises(ValidationError):
|
|
1721
|
+
# try scope all fields
|
|
1722
|
+
instance2.scoped_fields = []
|
|
1723
|
+
instance2.validated_save()
|
|
1724
|
+
|
|
1725
|
+
with self.assertRaises(ValidationError):
|
|
1726
|
+
instance2.scoped_fields = ["host", "mask_length"]
|
|
1727
|
+
instance2.validated_save()
|
|
1728
|
+
|
|
1729
|
+
with self.assertRaises(ValidationError):
|
|
1730
|
+
instance2.scoped_fields = ["role", "status", "type"]
|
|
1731
|
+
instance2.validated_save()
|
|
1732
|
+
|
|
1733
|
+
|
|
1285
1734
|
class RoleTest(ModelTestCases.BaseModelTestCase):
|
|
1286
1735
|
"""Tests for `Role` model class."""
|
|
1287
1736
|
|
|
@@ -1297,6 +1746,50 @@ class RoleTest(ModelTestCases.BaseModelTestCase):
|
|
|
1297
1746
|
self.assertQuerysetEqualAndNotEmpty(Role.objects.get_for_models([Device, IPAddress]), roles)
|
|
1298
1747
|
|
|
1299
1748
|
|
|
1749
|
+
class SavedViewTest(ModelTestCases.BaseModelTestCase):
|
|
1750
|
+
model = SavedView
|
|
1751
|
+
|
|
1752
|
+
def setUp(self):
|
|
1753
|
+
self.user = User.objects.create_user(username="Saved View test user")
|
|
1754
|
+
self.sv = SavedView.objects.create(name="Saved View", owner=self.user, view="dcim:location_list")
|
|
1755
|
+
|
|
1756
|
+
def test_is_global_default_saved_view_is_shared_automatically(self):
|
|
1757
|
+
self.sv.is_global_default = True
|
|
1758
|
+
self.sv.save()
|
|
1759
|
+
self.assertEqual(self.sv.is_shared, True)
|
|
1760
|
+
|
|
1761
|
+
def test_setting_new_global_default_saved_view_unset_old_global_default_saved_view(self):
|
|
1762
|
+
self.old_global_sv = SavedView.objects.create(
|
|
1763
|
+
name="Old Global Saved View", owner=self.user, view="dcim:location_list", is_global_default=True
|
|
1764
|
+
)
|
|
1765
|
+
|
|
1766
|
+
self.new_global_sv = SavedView.objects.create(
|
|
1767
|
+
name="New Global Saved View", owner=self.user, view="dcim:location_list"
|
|
1768
|
+
)
|
|
1769
|
+
self.new_global_sv.is_global_default = True
|
|
1770
|
+
self.new_global_sv.save()
|
|
1771
|
+
self.old_global_sv.refresh_from_db()
|
|
1772
|
+
self.assertEqual(self.new_global_sv.is_shared, True)
|
|
1773
|
+
self.assertEqual(self.old_global_sv.is_global_default, False)
|
|
1774
|
+
|
|
1775
|
+
def test_multiple_global_default_saved_views_can_exist_for_different_views(self):
|
|
1776
|
+
self.device_global_sv = SavedView.objects.create(
|
|
1777
|
+
name="Device Global Saved View", owner=self.user, view="dcim:device_list", is_global_default=True
|
|
1778
|
+
)
|
|
1779
|
+
self.device_global_sv.save()
|
|
1780
|
+
self.location_global_sv = SavedView.objects.create(
|
|
1781
|
+
name="Location Global Saved View", owner=self.user, view="dcim:location_list", is_global_default=True
|
|
1782
|
+
)
|
|
1783
|
+
self.location_global_sv.save()
|
|
1784
|
+
self.ipaddress_global_sv = SavedView.objects.create(
|
|
1785
|
+
name="IP Address Global Saved View", owner=self.user, view="ipam:ipaddress_list", is_global_default=True
|
|
1786
|
+
)
|
|
1787
|
+
self.ipaddress_global_sv.save()
|
|
1788
|
+
self.assertEqual(self.device_global_sv.is_shared, True)
|
|
1789
|
+
self.assertEqual(self.location_global_sv.is_shared, True)
|
|
1790
|
+
self.assertEqual(self.ipaddress_global_sv.is_shared, True)
|
|
1791
|
+
|
|
1792
|
+
|
|
1300
1793
|
class SecretTest(ModelTestCases.BaseModelTestCase):
|
|
1301
1794
|
"""
|
|
1302
1795
|
Tests for the `Secret` model class.
|
|
@@ -1685,6 +2178,10 @@ class SecretsGroupTest(ModelTestCases.BaseModelTestCase):
|
|
|
1685
2178
|
)
|
|
1686
2179
|
|
|
1687
2180
|
|
|
2181
|
+
class StaticGroupAssociationTest(ModelTestCases.BaseModelTestCase):
|
|
2182
|
+
model = StaticGroupAssociation
|
|
2183
|
+
|
|
2184
|
+
|
|
1688
2185
|
class StatusTest(ModelTestCases.BaseModelTestCase):
|
|
1689
2186
|
"""
|
|
1690
2187
|
Tests for the `Status` model class.
|
|
@@ -1797,8 +2294,8 @@ class JobLogEntryTest(TestCase): # TODO: change to BaseModelTestCase
|
|
|
1797
2294
|
)
|
|
1798
2295
|
log.save()
|
|
1799
2296
|
|
|
1800
|
-
self.assertEqual(JobLogEntry.objects.
|
|
1801
|
-
log_object = JobLogEntry.objects.first()
|
|
2297
|
+
self.assertEqual(JobLogEntry.objects.filter(job_result=self.job_result).count(), 1)
|
|
2298
|
+
log_object = JobLogEntry.objects.filter(job_result=self.job_result).first()
|
|
1802
2299
|
self.assertEqual(log_object.message, log.message)
|
|
1803
2300
|
self.assertEqual(log_object.log_level, log.log_level)
|
|
1804
2301
|
self.assertEqual(log_object.grouping, log.grouping)
|
|
@@ -1579,6 +1579,7 @@ class RequiredRelationshipTestMixin:
|
|
|
1579
1579
|
self.assertEqual(from_model.objects.count(), existing_count)
|
|
1580
1580
|
|
|
1581
1581
|
# 3. Try creating an object when all required data is present
|
|
1582
|
+
related_objects_data = {}
|
|
1582
1583
|
if interact_with == "ui":
|
|
1583
1584
|
related_objects_data = {related_field_name: required_object_pks}
|
|
1584
1585
|
|