mirror of
https://github.com/netbox-community/netbox.git
synced 2025-08-09 00:58:16 -06:00
Merge branch 'netbox-community:develop' into script_rq_queue_name
This commit is contained in:
commit
8970c8dcdf
2
.github/ISSUE_TEMPLATE/bug_report.yaml
vendored
2
.github/ISSUE_TEMPLATE/bug_report.yaml
vendored
@ -26,7 +26,7 @@ body:
|
|||||||
attributes:
|
attributes:
|
||||||
label: NetBox Version
|
label: NetBox Version
|
||||||
description: What version of NetBox are you currently running?
|
description: What version of NetBox are you currently running?
|
||||||
placeholder: v4.0.5
|
placeholder: v4.0.7
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
|
2
.github/ISSUE_TEMPLATE/feature_request.yaml
vendored
2
.github/ISSUE_TEMPLATE/feature_request.yaml
vendored
@ -14,7 +14,7 @@ body:
|
|||||||
attributes:
|
attributes:
|
||||||
label: NetBox version
|
label: NetBox version
|
||||||
description: What version of NetBox are you currently running?
|
description: What version of NetBox are you currently running?
|
||||||
placeholder: v4.0.5
|
placeholder: v4.0.7
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
|
@ -17,7 +17,6 @@ NetBox exists to empower network engineers. Since its release in 2016, it has be
|
|||||||
<a href="#why-netbox">Why NetBox?</a> |
|
<a href="#why-netbox">Why NetBox?</a> |
|
||||||
<a href="#getting-started">Getting Started</a> |
|
<a href="#getting-started">Getting Started</a> |
|
||||||
<a href="#get-involved">Get Involved</a> |
|
<a href="#get-involved">Get Involved</a> |
|
||||||
<a href="#project-stats">Project Stats</a> |
|
|
||||||
<a href="#screenshots">Screenshots</a>
|
<a href="#screenshots">Screenshots</a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ Administrators are encouraged to adhere to industry best practices concerning th
|
|||||||
|
|
||||||
## Reporting a Suspected Vulnerability
|
## Reporting a Suspected Vulnerability
|
||||||
|
|
||||||
If you believe you've uncovered a security vulnerability and wish to report it confidentially, you may do so via email. Please note that any reported vulnerabilities **MUST** meet all the following conditions:
|
If you believe you've uncovered a security vulnerability and wish to report it confidentially, you may do so by emailing `security@netboxlabs.com`. Please ensure that your report meets all the following conditions:
|
||||||
|
|
||||||
* Affects the most recent stable release of NetBox, or a current beta release
|
* Affects the most recent stable release of NetBox, or a current beta release
|
||||||
* Affects a NetBox instance installed and configured per the official documentation
|
* Affects a NetBox instance installed and configured per the official documentation
|
||||||
@ -24,7 +24,7 @@ If you believe you've uncovered a security vulnerability and wish to report it c
|
|||||||
|
|
||||||
Please note that we **DO NOT** accept reports generated by automated tooling which merely suggest that a file or file(s) _may_ be vulnerable under certain conditions, as these are most often innocuous.
|
Please note that we **DO NOT** accept reports generated by automated tooling which merely suggest that a file or file(s) _may_ be vulnerable under certain conditions, as these are most often innocuous.
|
||||||
|
|
||||||
If you believe that you've found a vulnerability which meets all of these conditions, please [submit a draft security advisory](https://github.com/netbox-community/netbox/security/advisories/new) on GitHub. For any security concerns regarding NetBox deployed via Docker, please see the [netbox-docker](https://github.com/netbox-community/netbox-docker) project.
|
For any security concerns regarding the community-maintained Docker image for NetBox, please see the [netbox-docker](https://github.com/netbox-community/netbox-docker) project.
|
||||||
|
|
||||||
### Bug Bounties
|
### Bug Bounties
|
||||||
|
|
||||||
|
@ -110,7 +110,7 @@ Pillow
|
|||||||
|
|
||||||
# PostgreSQL database adapter for Python
|
# PostgreSQL database adapter for Python
|
||||||
# https://github.com/psycopg/psycopg/blob/master/docs/news.rst
|
# https://github.com/psycopg/psycopg/blob/master/docs/news.rst
|
||||||
psycopg[binary,pool]
|
psycopg[c,pool]
|
||||||
|
|
||||||
# YAML rendering library
|
# YAML rendering library
|
||||||
# https://github.com/yaml/pyyaml/blob/master/CHANGES
|
# https://github.com/yaml/pyyaml/blob/master/CHANGES
|
||||||
|
95605
contrib/openapi2.json
95605
contrib/openapi2.json
File diff suppressed because it is too large
Load Diff
69695
contrib/openapi2.yaml
69695
contrib/openapi2.yaml
File diff suppressed because it is too large
Load Diff
@ -40,3 +40,22 @@ REMOTE_AUTH_BACKEND = 'social_core.backends.google.GoogleOAuth2'
|
|||||||
NetBox supports single sign-on authentication via the [python-social-auth](https://github.com/python-social-auth) library. To enable SSO, specify the path to the desired authentication backend within the `social_core` Python package. Please see the complete list of [supported authentication backends](https://github.com/python-social-auth/social-core/tree/master/social_core/backends) for the available options.
|
NetBox supports single sign-on authentication via the [python-social-auth](https://github.com/python-social-auth) library. To enable SSO, specify the path to the desired authentication backend within the `social_core` Python package. Please see the complete list of [supported authentication backends](https://github.com/python-social-auth/social-core/tree/master/social_core/backends) for the available options.
|
||||||
|
|
||||||
Most remote authentication backends require some additional configuration through settings prefixed with `SOCIAL_AUTH_`. These will be automatically imported from NetBox's `configuration.py` file. Additionally, the [authentication pipeline](https://python-social-auth.readthedocs.io/en/latest/pipeline.html) can be customized via the `SOCIAL_AUTH_PIPELINE` parameter. (NetBox's default pipeline is defined in `netbox/settings.py` for your reference.)
|
Most remote authentication backends require some additional configuration through settings prefixed with `SOCIAL_AUTH_`. These will be automatically imported from NetBox's `configuration.py` file. Additionally, the [authentication pipeline](https://python-social-auth.readthedocs.io/en/latest/pipeline.html) can be customized via the `SOCIAL_AUTH_PIPELINE` parameter. (NetBox's default pipeline is defined in `netbox/settings.py` for your reference.)
|
||||||
|
|
||||||
|
#### Configuring the SSO module's appearance
|
||||||
|
|
||||||
|
The way a remote authentication backend is displayed to the user on the login
|
||||||
|
page may be adjusted via the `SOCIAL_AUTH_BACKEND_ATTRS` parameter, defaulting
|
||||||
|
to an empty dictionary. This dictionary maps a `social_core` module's name (ie.
|
||||||
|
`REMOTE_AUTH_BACKEND.name`) to a couple of parameters, `(display_name, icon)`.
|
||||||
|
|
||||||
|
The `display_name` is the name displayed to the user on the login page. The
|
||||||
|
icon may either be the URL of an icon; refer to a [Material Design
|
||||||
|
Icons](https://github.com/google/material-design-icons) icon's name; or be
|
||||||
|
`None` for no icon.
|
||||||
|
|
||||||
|
For instance, the OIDC backend may be customized with
|
||||||
|
```python
|
||||||
|
SOCIAL_AUTH_BACKEND_ATTRS = {
|
||||||
|
'oidc': ("My awesome SSO", "login"),
|
||||||
|
}
|
||||||
|
```
|
||||||
|
@ -31,6 +31,17 @@ The sampling rate for errors. Must be a value between 0 (disabled) and 1.0 (repo
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## SENTRY_SEND_DEFAULT_PII
|
||||||
|
|
||||||
|
Default: False
|
||||||
|
|
||||||
|
Maps to the Sentry SDK's [`send_default_pii`](https://docs.sentry.io/platforms/python/configuration/options/#send-default-pii) parameter. If enabled, certain personally identifiable information (PII) is added.
|
||||||
|
|
||||||
|
!!! warning "Sensitive data"
|
||||||
|
If you enable this option, be aware that sensitive data such as cookies and authentication tokens will be logged.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## SENTRY_TAGS
|
## SENTRY_TAGS
|
||||||
|
|
||||||
An optional dictionary of tag names and values to apply to Sentry error reports.For example:
|
An optional dictionary of tag names and values to apply to Sentry error reports.For example:
|
||||||
|
@ -177,7 +177,7 @@ The dotted path to the desired search backend class. `CachedValueSearchBackend`
|
|||||||
|
|
||||||
Default: None (local storage)
|
Default: None (local storage)
|
||||||
|
|
||||||
The backend storage engine for handling uploaded files (e.g. image attachments). NetBox supports integration with the [`django-storages`](https://django-storages.readthedocs.io/en/stable/) package, which provides backends for several popular file storage services. If not configured, local filesystem storage will be used.
|
The backend storage engine for handling uploaded files (e.g. image attachments). NetBox supports integration with the [`django-storages`](https://django-storages.readthedocs.io/en/stable/) and [`django-storage-swift`](https://github.com/dennisv/django-storage-swift) packages, which provide backends for several popular file storage services. If not configured, local filesystem storage will be used.
|
||||||
|
|
||||||
The configuration parameters for the specified storage backend are defined under the `STORAGE_CONFIG` setting.
|
The configuration parameters for the specified storage backend are defined under the `STORAGE_CONFIG` setting.
|
||||||
|
|
||||||
@ -187,7 +187,7 @@ The configuration parameters for the specified storage backend are defined under
|
|||||||
|
|
||||||
Default: Empty
|
Default: Empty
|
||||||
|
|
||||||
A dictionary of configuration parameters for the storage backend configured as `STORAGE_BACKEND`. The specific parameters to be used here are specific to each backend; see the [`django-storages` documentation](https://django-storages.readthedocs.io/en/stable/) for more detail.
|
A dictionary of configuration parameters for the storage backend configured as `STORAGE_BACKEND`. The specific parameters to be used here are specific to each backend; see the documentation for your selected backend ([`django-storages`](https://django-storages.readthedocs.io/en/stable/) or [`django-storage-swift`](https://github.com/dennisv/django-storage-swift)) for more detail.
|
||||||
|
|
||||||
If `STORAGE_BACKEND` is not defined, this setting will be ignored.
|
If `STORAGE_BACKEND` is not defined, this setting will be ignored.
|
||||||
|
|
||||||
|
@ -142,11 +142,11 @@ These two methods will load data in YAML or JSON format, respectively, from file
|
|||||||
|
|
||||||
The Script object provides a set of convenient functions for recording messages at different severity levels:
|
The Script object provides a set of convenient functions for recording messages at different severity levels:
|
||||||
|
|
||||||
* `log_debug(message, object=None)`
|
* `log_debug(message=None, obj=None)`
|
||||||
* `log_success(message, object=None)`
|
* `log_success(message=None, obj=None)`
|
||||||
* `log_info(message, object=None)`
|
* `log_info(message=None, obj=None)`
|
||||||
* `log_warning(message, object=None)`
|
* `log_warning(message=None, obj=None)`
|
||||||
* `log_failure(message, object=None)`
|
* `log_failure(message=None, obj=None)`
|
||||||
|
|
||||||
Log messages are returned to the user upon execution of the script. Markdown rendering is supported for log messages. A message may optionally be associated with a particular object by passing it as the second argument to the logging method.
|
Log messages are returned to the user upon execution of the script. Markdown rendering is supported for log messages. A message may optionally be associated with a particular object by passing it as the second argument to the logging method.
|
||||||
|
|
||||||
@ -156,6 +156,8 @@ A script can define one or more test methods to report on certain conditions. Al
|
|||||||
|
|
||||||
These methods are detected and run automatically when the script is executed, unless its `run()` method has been overridden. (When overriding `run()`, `run_tests()` can be called to run all test methods present in the script.)
|
These methods are detected and run automatically when the script is executed, unless its `run()` method has been overridden. (When overriding `run()`, `run_tests()` can be called to run all test methods present in the script.)
|
||||||
|
|
||||||
|
Calling any of these logging methods without a message will increment the relevant counter, but will not generate an output line in the script's log.
|
||||||
|
|
||||||
!!! info
|
!!! info
|
||||||
This functionality was ported from [legacy reports](./reports.md) in NetBox v4.0.
|
This functionality was ported from [legacy reports](./reports.md) in NetBox v4.0.
|
||||||
|
|
||||||
|
@ -113,7 +113,7 @@ Create a [new release](https://github.com/netbox-community/netbox/releases/new)
|
|||||||
* **Tag:** Current version (e.g. `v3.3.1`)
|
* **Tag:** Current version (e.g. `v3.3.1`)
|
||||||
* **Target:** `master`
|
* **Target:** `master`
|
||||||
* **Title:** Version and date (e.g. `v3.3.1 - 2022-08-25`)
|
* **Title:** Version and date (e.g. `v3.3.1 - 2022-08-25`)
|
||||||
* **Description:** Copy from the pull request body
|
* **Description:** Copy from the pull request body, then promote the `###` headers to `##` ones
|
||||||
|
|
||||||
Once created, the release will become available for users to install.
|
Once created, the release will become available for users to install.
|
||||||
|
|
||||||
@ -135,4 +135,6 @@ First, run the `build-site` action, by navigating to Actions > build-site > Run
|
|||||||
|
|
||||||
Once the documentation files have been compiled, they must be published by running the `deploy-kinsta` action. Select the desired deployment environment (staging or production) and specify `latest` as the deploy tag.
|
Once the documentation files have been compiled, they must be published by running the `deploy-kinsta` action. Select the desired deployment environment (staging or production) and specify `latest` as the deploy tag.
|
||||||
|
|
||||||
|
Clear the CDN cache from the [Kinsta](https://my.kinsta.com/) portal. Navigate to _Sites_ / _NetBox Labs_ / _Live_, select _Cache_ in the left-nav, click the _Clear Cache_ button, and confirm the clear operation.
|
||||||
|
|
||||||
Finally, verify that the documentation at <https://netboxlabs.com/docs/netbox/en/stable/> has been updated.
|
Finally, verify that the documentation at <https://netboxlabs.com/docs/netbox/en/stable/> has been updated.
|
||||||
|
@ -20,6 +20,8 @@ Then, commit the change and push to the `develop` branch on GitHub. Any new stri
|
|||||||
|
|
||||||
Typically, translated strings need to be updated only as part of the NetBox [release process](./release-checklist.md).
|
Typically, translated strings need to be updated only as part of the NetBox [release process](./release-checklist.md).
|
||||||
|
|
||||||
|
Check the Transifex dashboard for languages that are not marked _ready for use_, being sure to click _Show all languages_ if it appears at the bottom of the list. Use machine translation to round out any not-ready languages. It's not necessary to review the machine translation immediately as the translation teams will handle that aspect; the goal at this stage is to get translations included in the Transifex pull request.
|
||||||
|
|
||||||
To update translated strings, start by initiating a sync from Transifex. From the Transifex dashboard, navigate to Settings > Integrations > GitHub > Manage, and click the **Manual Sync** button at top right.
|
To update translated strings, start by initiating a sync from Transifex. From the Transifex dashboard, navigate to Settings > Integrations > GitHub > Manage, and click the **Manual Sync** button at top right.
|
||||||
|
|
||||||

|

|
||||||
|
@ -70,3 +70,19 @@ DROP TABLE
|
|||||||
netbox=> DROP TABLE pluginname_bar;
|
netbox=> DROP TABLE pluginname_bar;
|
||||||
DROP TABLE
|
DROP TABLE
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Remove the Django Migration Records
|
||||||
|
|
||||||
|
After removing the tables created by a plugin, the migrations that created the tables need to be removed from Django's migration history as well. This is necessary to make it possible to reinstall the plugin at a later time. If the migration history were left in place, Django would skip all migrations that were executed in the course of a previous installation, which would cause the plugin to fail after reinstallation.
|
||||||
|
|
||||||
|
```no-highlight
|
||||||
|
netbox=> SELECT * FROM django_migrations WHERE app='pluginname';
|
||||||
|
id | app | name | applied
|
||||||
|
-----+------------+------------------------+-------------------------------
|
||||||
|
492 | pluginname | 0001_initial | 2023-12-21 11:59:59.325995+00
|
||||||
|
493 | pluginname | 0002_add_foo | 2023-12-21 11:59:59.330026+00
|
||||||
|
netbox=> DELETE FROM django_migrations WHERE app='pluginname';
|
||||||
|
```
|
||||||
|
|
||||||
|
!!! warning
|
||||||
|
Exercise extreme caution when altering Django system tables. Users are strongly encouraged to perform a backup of their database immediately before taking these actions.
|
||||||
|
@ -1,18 +1,74 @@
|
|||||||
# NetBox v4.0
|
# NetBox v4.0
|
||||||
|
|
||||||
## v4.0.6 (FUTURE)
|
## v4.0.8 (FUTURE)
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
|
||||||
|
* [#15375](https://github.com/netbox-community/netbox/issues/15375) - Enable customization of SSO backend name & icon
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* [#16357](https://github.com/netbox-community/netbox/issues/16357) - Replicate assigned type & tenant for cable when clicking "create an add another"
|
||||||
|
* [#16760](https://github.com/netbox-community/netbox/issues/16760) - Fix data source syncing using git via a local path
|
||||||
|
* [#16838](https://github.com/netbox-community/netbox/issues/16838) - ActionsColumn should render extra buttons even when no stock actions are enabled
|
||||||
|
* [#16867](https://github.com/netbox-community/netbox/issues/16867) - Fix exception when a dashboard list widget references a model which has been removed
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v4.0.7 (2024-07-09)
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
|
||||||
|
* [#14554](https://github.com/netbox-community/netbox/issues/14554) - Add support for [django-storage-swift](https://github.com/dennisv/django-storage-swift) storage backend
|
||||||
|
* [#16424](https://github.com/netbox-community/netbox/issues/16424) - Enable filtering of devices by cluster and cluster group
|
||||||
|
* [#16716](https://github.com/netbox-community/netbox/issues/16716) - Display NAT address (if any) for OOB IP address under device view
|
||||||
|
* [#16725](https://github.com/netbox-community/netbox/issues/16725) - Always position the admin section last in the navigation menu
|
||||||
|
* [#16791](https://github.com/netbox-community/netbox/issues/16791) - Add 200 & 400 Gbps selections for circuit termination port speed
|
||||||
|
* [#16802](https://github.com/netbox-community/netbox/issues/16802) - Introduce `SENTRY_SEND_DEFAULT_PII` configuration parameter and disable PII export by default
|
||||||
|
* [#16817](https://github.com/netbox-community/netbox/issues/16817) - Add 200 & 400 Gbps selections for circuit commit rate
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* [#16523](https://github.com/netbox-community/netbox/issues/16523) - Restore highlighting of current device in virtual chassis members panel
|
||||||
|
* [#16654](https://github.com/netbox-community/netbox/issues/16654) - Fix parent item assignment for inventory item bulk import
|
||||||
|
* [#16657](https://github.com/netbox-community/netbox/issues/16657) - Fix translation of object types in global search
|
||||||
|
* [#16679](https://github.com/netbox-community/netbox/issues/16679) - Avoid overwriting custom JSON fields during bulk edit
|
||||||
|
* [#16689](https://github.com/netbox-community/netbox/issues/16689) - System configuration view should reflect static parameters when no config revisions exist
|
||||||
|
* [#16714](https://github.com/netbox-community/netbox/issues/16714) - Fix cloning of device types with 0U height
|
||||||
|
* [#16721](https://github.com/netbox-community/netbox/issues/16721) - Fix errant API request after deselecting a rack in device edit form
|
||||||
|
* [#16723](https://github.com/netbox-community/netbox/issues/16723) - Fix escaping of path to virtual environment in `upgrade.sh`
|
||||||
|
* [#16735](https://github.com/netbox-community/netbox/issues/16735) - Object list "results" tab should show a count of zero when empty
|
||||||
|
* [#16747](https://github.com/netbox-community/netbox/issues/16747) - Avoid clearing entire search cache when manually reindexing specific apps/models
|
||||||
|
* [#16758](https://github.com/netbox-community/netbox/issues/16758) - Ensure manually selected lagnuage persists across browser sessions
|
||||||
|
* [#16779](https://github.com/netbox-community/netbox/issues/16779) - Fix saved filter selection for child object lists
|
||||||
|
* [#16780](https://github.com/netbox-community/netbox/issues/16780) - IKE proposal created via REST API should not require authentication_algorithm
|
||||||
|
* [#16796](https://github.com/netbox-community/netbox/issues/16796) - Allow assignment of VM with no site to a cluster with a site
|
||||||
|
* [#16806](https://github.com/netbox-community/netbox/issues/16806) - Fix redirect URL when creating contact assignments with "add another" button
|
||||||
|
* [#16807](https://github.com/netbox-community/netbox/issues/16807) - Fix layout of VLAN edit form when custom fields are present
|
||||||
|
* [#16808](https://github.com/netbox-community/netbox/issues/16808) - Fix event rule triggering in scenario where objects are updated immediately prior to deletion
|
||||||
|
* [#16813](https://github.com/netbox-community/netbox/issues/16813) - Fix AttributeError exception when filtering bookmarks in dashboard widget by object type
|
||||||
|
* [#16843](https://github.com/netbox-community/netbox/issues/16843) - Permit creation of IKE policies via REST API without specifying an IKE mode
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v4.0.6 (2024-06-24)
|
||||||
|
|
||||||
### Enhancements
|
### Enhancements
|
||||||
|
|
||||||
* [#15348](https://github.com/netbox-community/netbox/issues/15348) - Show saved filters alongside quick search on object list views
|
* [#15348](https://github.com/netbox-community/netbox/issues/15348) - Show saved filters alongside quick search on object list views
|
||||||
* [#15794](https://github.com/netbox-community/netbox/issues/15794) - Dynamically populate related objects in UI views
|
* [#15794](https://github.com/netbox-community/netbox/issues/15794) - Dynamically populate related objects in UI views
|
||||||
* [#16256](https://github.com/netbox-community/netbox/issues/16256) - Enable alphabetical ordering of bookmarks on dashboard
|
* [#16256](https://github.com/netbox-community/netbox/issues/16256) - Enable alphabetical ordering of bookmarks on dashboard
|
||||||
|
* [#16307](https://github.com/netbox-community/netbox/issues/16307) - Enable calling `log_*()` methods on Script without passing a message
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|
||||||
* [#13925](https://github.com/netbox-community/netbox/issues/13925) - Fix support for "zulu" (UTC) timestamps for custom fields
|
* [#13925](https://github.com/netbox-community/netbox/issues/13925) - Fix support for "zulu" (UTC) timestamps for custom fields
|
||||||
* [#14829](https://github.com/netbox-community/netbox/issues/14829) - Fix support for simple conditions (without AND/OR) in event rules
|
* [#14829](https://github.com/netbox-community/netbox/issues/14829) - Fix support for simple conditions (without AND/OR) in event rules
|
||||||
|
* [#15717](https://github.com/netbox-community/netbox/issues/15717) - Allow assigning a device/VM in a site to a cluster with no site assigned
|
||||||
* [#16143](https://github.com/netbox-community/netbox/issues/16143) - Display timestamps in tables in the configured timezone
|
* [#16143](https://github.com/netbox-community/netbox/issues/16143) - Display timestamps in tables in the configured timezone
|
||||||
|
* [#16149](https://github.com/netbox-community/netbox/issues/16149) - Fix object linking in custom script logs
|
||||||
|
* [#16252](https://github.com/netbox-community/netbox/issues/16252) - Fix total count in tab at top of rack elevations view
|
||||||
|
* [#16273](https://github.com/netbox-community/netbox/issues/16273) - Restore global search bar on mobile
|
||||||
* [#16416](https://github.com/netbox-community/netbox/issues/16416) - Retain dark/light mode toggle on mobile view
|
* [#16416](https://github.com/netbox-community/netbox/issues/16416) - Retain dark/light mode toggle on mobile view
|
||||||
* [#16444](https://github.com/netbox-community/netbox/issues/16444) - Disable ordering circuits list by A/Z termination
|
* [#16444](https://github.com/netbox-community/netbox/issues/16444) - Disable ordering circuits list by A/Z termination
|
||||||
* [#16450](https://github.com/netbox-community/netbox/issues/16450) - Searching for rack unit in form dropdown should be case-insensitive
|
* [#16450](https://github.com/netbox-community/netbox/issues/16450) - Searching for rack unit in form dropdown should be case-insensitive
|
||||||
@ -21,6 +77,7 @@
|
|||||||
* [#16460](https://github.com/netbox-community/netbox/issues/16460) - Omit spaces from telephone number URLs
|
* [#16460](https://github.com/netbox-community/netbox/issues/16460) - Omit spaces from telephone number URLs
|
||||||
* [#16512](https://github.com/netbox-community/netbox/issues/16512) - Restore a user's preferred language (if any) on login
|
* [#16512](https://github.com/netbox-community/netbox/issues/16512) - Restore a user's preferred language (if any) on login
|
||||||
* [#16542](https://github.com/netbox-community/netbox/issues/16542) - Fix bulk form operations when HTMX is enabled
|
* [#16542](https://github.com/netbox-community/netbox/issues/16542) - Fix bulk form operations when HTMX is enabled
|
||||||
|
* [#16702](https://github.com/netbox-community/netbox/issues/16702) - Fix validation of `return_url` query parameter
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
@ -44,10 +44,20 @@ class LoginView(View):
|
|||||||
return super().dispatch(*args, **kwargs)
|
return super().dispatch(*args, **kwargs)
|
||||||
|
|
||||||
def gen_auth_data(self, name, url, params):
|
def gen_auth_data(self, name, url, params):
|
||||||
display_name, icon_name = get_auth_backend_display(name)
|
display_name, icon_source = get_auth_backend_display(name)
|
||||||
|
|
||||||
|
icon_name = None
|
||||||
|
icon_img = None
|
||||||
|
if icon_source:
|
||||||
|
if '://' in icon_source:
|
||||||
|
icon_img = icon_source
|
||||||
|
else:
|
||||||
|
icon_name = icon_source
|
||||||
|
|
||||||
return {
|
return {
|
||||||
'display_name': display_name,
|
'display_name': display_name,
|
||||||
'icon_name': icon_name,
|
'icon_name': icon_name,
|
||||||
|
'icon_img': icon_img,
|
||||||
'url': f'{url}?{urlencode(params)}',
|
'url': f'{url}?{urlencode(params)}',
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +121,7 @@ class LoginView(View):
|
|||||||
|
|
||||||
# Set the user's preferred language (if any)
|
# Set the user's preferred language (if any)
|
||||||
if language := request.user.config.get('locale.language'):
|
if language := request.user.config.get('locale.language'):
|
||||||
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, language)
|
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, language, max_age=request.session.get_expiry_age())
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
@ -206,7 +216,7 @@ class UserConfigView(LoginRequiredMixin, View):
|
|||||||
|
|
||||||
# Set/clear language cookie
|
# Set/clear language cookie
|
||||||
if language := form.cleaned_data['locale.language']:
|
if language := form.cleaned_data['locale.language']:
|
||||||
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, language)
|
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, language, max_age=request.session.get_expiry_age())
|
||||||
else:
|
else:
|
||||||
response.delete_cookie(settings.LANGUAGE_COOKIE_NAME)
|
response.delete_cookie(settings.LANGUAGE_COOKIE_NAME)
|
||||||
|
|
||||||
|
@ -38,6 +38,8 @@ class CircuitCommitRateChoices(ChoiceSet):
|
|||||||
(25000000, '25 Gbps'),
|
(25000000, '25 Gbps'),
|
||||||
(40000000, '40 Gbps'),
|
(40000000, '40 Gbps'),
|
||||||
(100000000, '100 Gbps'),
|
(100000000, '100 Gbps'),
|
||||||
|
(200000000, '200 Gbps'),
|
||||||
|
(400000000, '400 Gbps'),
|
||||||
(1544, 'T1 (1.544 Mbps)'),
|
(1544, 'T1 (1.544 Mbps)'),
|
||||||
(2048, 'E1 (2.048 Mbps)'),
|
(2048, 'E1 (2.048 Mbps)'),
|
||||||
]
|
]
|
||||||
@ -69,6 +71,8 @@ class CircuitTerminationPortSpeedChoices(ChoiceSet):
|
|||||||
(25000000, '25 Gbps'),
|
(25000000, '25 Gbps'),
|
||||||
(40000000, '40 Gbps'),
|
(40000000, '40 Gbps'),
|
||||||
(100000000, '100 Gbps'),
|
(100000000, '100 Gbps'),
|
||||||
|
(200000000, '200 Gbps'),
|
||||||
|
(400000000, '400 Gbps'),
|
||||||
(1544, 'T1 (1.544 Mbps)'),
|
(1544, 'T1 (1.544 Mbps)'),
|
||||||
(2048, 'E1 (2.048 Mbps)'),
|
(2048, 'E1 (2.048 Mbps)'),
|
||||||
]
|
]
|
||||||
|
@ -66,9 +66,6 @@ class CircuitTypeImportForm(NetBoxModelImportForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = CircuitType
|
model = CircuitType
|
||||||
fields = ('name', 'slug', 'color', 'description', 'tags')
|
fields = ('name', 'slug', 'color', 'description', 'tags')
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class CircuitImportForm(NetBoxModelImportForm):
|
class CircuitImportForm(NetBoxModelImportForm):
|
||||||
|
@ -84,9 +84,7 @@ class GitBackend(DataBackend):
|
|||||||
clone_args = {
|
clone_args = {
|
||||||
"branch": self.params.get('branch'),
|
"branch": self.params.get('branch'),
|
||||||
"config": self.config,
|
"config": self.config,
|
||||||
"depth": 1,
|
|
||||||
"errstream": porcelain.NoneStream(),
|
"errstream": porcelain.NoneStream(),
|
||||||
"quiet": True,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.url_scheme in ('http', 'https'):
|
if self.url_scheme in ('http', 'https'):
|
||||||
@ -97,6 +95,9 @@ class GitBackend(DataBackend):
|
|||||||
"password": self.params.get('password'),
|
"password": self.params.get('password'),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
if self.url_scheme:
|
||||||
|
clone_args["quiet"] = True
|
||||||
|
clone_args["depth"] = 1
|
||||||
|
|
||||||
logger.debug(f"Cloning git repo: {self.url}")
|
logger.debug(f"Cloning git repo: {self.url}")
|
||||||
try:
|
try:
|
||||||
|
@ -555,7 +555,7 @@ class SystemView(UserPassesTestMixin, View):
|
|||||||
config = ConfigRevision.objects.get(pk=cache.get('config_version'))
|
config = ConfigRevision.objects.get(pk=cache.get('config_version'))
|
||||||
except ConfigRevision.DoesNotExist:
|
except ConfigRevision.DoesNotExist:
|
||||||
# Fall back to using the active config data if no record is found
|
# Fall back to using the active config data if no record is found
|
||||||
config = ConfigRevision(data=get_config().defaults)
|
config = get_config()
|
||||||
|
|
||||||
# Raw data export
|
# Raw data export
|
||||||
if 'export' in request.GET:
|
if 'export' in request.GET:
|
||||||
|
@ -20,7 +20,7 @@ from utilities.filters import (
|
|||||||
ContentTypeFilter, MultiValueCharFilter, MultiValueMACAddressFilter, MultiValueNumberFilter, MultiValueWWNFilter,
|
ContentTypeFilter, MultiValueCharFilter, MultiValueMACAddressFilter, MultiValueNumberFilter, MultiValueWWNFilter,
|
||||||
NumericArrayFilter, TreeNodeMultipleChoiceFilter,
|
NumericArrayFilter, TreeNodeMultipleChoiceFilter,
|
||||||
)
|
)
|
||||||
from virtualization.models import Cluster
|
from virtualization.models import Cluster, ClusterGroup
|
||||||
from vpn.models import L2VPN
|
from vpn.models import L2VPN
|
||||||
from wireless.choices import WirelessRoleChoices, WirelessChannelChoices
|
from wireless.choices import WirelessRoleChoices, WirelessChannelChoices
|
||||||
from wireless.models import WirelessLAN, WirelessLink
|
from wireless.models import WirelessLAN, WirelessLink
|
||||||
@ -1018,6 +1018,17 @@ class DeviceFilterSet(
|
|||||||
queryset=Cluster.objects.all(),
|
queryset=Cluster.objects.all(),
|
||||||
label=_('VM cluster (ID)'),
|
label=_('VM cluster (ID)'),
|
||||||
)
|
)
|
||||||
|
cluster_group = django_filters.ModelMultipleChoiceFilter(
|
||||||
|
field_name='cluster__group__slug',
|
||||||
|
queryset=ClusterGroup.objects.all(),
|
||||||
|
to_field_name='slug',
|
||||||
|
label=_('Cluster group (slug)'),
|
||||||
|
)
|
||||||
|
cluster_group_id = django_filters.ModelMultipleChoiceFilter(
|
||||||
|
field_name='cluster__group',
|
||||||
|
queryset=ClusterGroup.objects.all(),
|
||||||
|
label=_('Cluster group (ID)'),
|
||||||
|
)
|
||||||
model = django_filters.ModelMultipleChoiceFilter(
|
model = django_filters.ModelMultipleChoiceFilter(
|
||||||
field_name='device_type__slug',
|
field_name='device_type__slug',
|
||||||
queryset=DeviceType.objects.all(),
|
queryset=DeviceType.objects.all(),
|
||||||
|
@ -174,9 +174,6 @@ class RackRoleImportForm(NetBoxModelImportForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = RackRole
|
model = RackRole
|
||||||
fields = ('name', 'slug', 'color', 'description', 'tags')
|
fields = ('name', 'slug', 'color', 'description', 'tags')
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class RackImportForm(NetBoxModelImportForm):
|
class RackImportForm(NetBoxModelImportForm):
|
||||||
@ -384,9 +381,6 @@ class DeviceRoleImportForm(NetBoxModelImportForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = DeviceRole
|
model = DeviceRole
|
||||||
fields = ('name', 'slug', 'color', 'vm_role', 'config_template', 'description', 'tags')
|
fields = ('name', 'slug', 'color', 'vm_role', 'config_template', 'description', 'tags')
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class PlatformImportForm(NetBoxModelImportForm):
|
class PlatformImportForm(NetBoxModelImportForm):
|
||||||
@ -1052,7 +1046,7 @@ class InventoryItemImportForm(NetBoxModelImportForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = InventoryItem
|
model = InventoryItem
|
||||||
fields = (
|
fields = (
|
||||||
'device', 'name', 'label', 'role', 'manufacturer', 'part_id', 'serial', 'asset_tag', 'discovered',
|
'device', 'name', 'label', 'role', 'manufacturer', 'parent', 'part_id', 'serial', 'asset_tag', 'discovered',
|
||||||
'description', 'tags', 'component_type', 'component_name',
|
'description', 'tags', 'component_type', 'component_name',
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -1104,9 +1098,6 @@ class InventoryItemRoleImportForm(NetBoxModelImportForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = InventoryItemRole
|
model = InventoryItemRole
|
||||||
fields = ('name', 'slug', 'color', 'description')
|
fields = ('name', 'slug', 'color', 'description')
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@ -1183,9 +1174,6 @@ class CableImportForm(NetBoxModelImportForm):
|
|||||||
'side_a_device', 'side_a_type', 'side_a_name', 'side_b_device', 'side_b_type', 'side_b_name', 'type',
|
'side_a_device', 'side_a_type', 'side_a_name', 'side_b_device', 'side_b_type', 'side_b_name', 'type',
|
||||||
'status', 'tenant', 'label', 'color', 'length', 'length_unit', 'description', 'comments', 'tags',
|
'status', 'tenant', 'label', 'color', 'length', 'length_unit', 'description', 'comments', 'tags',
|
||||||
]
|
]
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
def _clean_side(self, side):
|
def _clean_side(self, side):
|
||||||
"""
|
"""
|
||||||
|
@ -14,6 +14,7 @@ from utilities.forms import BOOLEAN_WITH_BLANK_CHOICES, FilterForm, add_blank_ch
|
|||||||
from utilities.forms.fields import ColorField, DynamicModelMultipleChoiceField, TagFilterField
|
from utilities.forms.fields import ColorField, DynamicModelMultipleChoiceField, TagFilterField
|
||||||
from utilities.forms.rendering import FieldSet
|
from utilities.forms.rendering import FieldSet
|
||||||
from utilities.forms.widgets import NumberWithOptions
|
from utilities.forms.widgets import NumberWithOptions
|
||||||
|
from virtualization.models import Cluster, ClusterGroup
|
||||||
from vpn.models import L2VPN
|
from vpn.models import L2VPN
|
||||||
from wireless.choices import *
|
from wireless.choices import *
|
||||||
|
|
||||||
@ -655,6 +656,7 @@ class DeviceFilterForm(
|
|||||||
'console_ports', 'console_server_ports', 'power_ports', 'power_outlets', 'interfaces', 'pass_through_ports',
|
'console_ports', 'console_server_ports', 'power_ports', 'power_outlets', 'interfaces', 'pass_through_ports',
|
||||||
name=_('Components')
|
name=_('Components')
|
||||||
),
|
),
|
||||||
|
FieldSet('cluster_group_id', 'cluster_id', name=_('Cluster')),
|
||||||
FieldSet(
|
FieldSet(
|
||||||
'has_primary_ip', 'has_oob_ip', 'virtual_chassis_member', 'config_template_id', 'local_context_data',
|
'has_primary_ip', 'has_oob_ip', 'virtual_chassis_member', 'config_template_id', 'local_context_data',
|
||||||
'has_virtual_device_context',
|
'has_virtual_device_context',
|
||||||
@ -821,6 +823,16 @@ class DeviceFilterForm(
|
|||||||
choices=BOOLEAN_WITH_BLANK_CHOICES
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
cluster_id = DynamicModelMultipleChoiceField(
|
||||||
|
queryset=Cluster.objects.all(),
|
||||||
|
required=False,
|
||||||
|
label=_('Cluster')
|
||||||
|
)
|
||||||
|
cluster_group_id = DynamicModelMultipleChoiceField(
|
||||||
|
queryset=ClusterGroup.objects.all(),
|
||||||
|
required=False,
|
||||||
|
label=_('Cluster group')
|
||||||
|
)
|
||||||
tag = TagFilterField(model)
|
tag = TagFilterField(model)
|
||||||
|
|
||||||
|
|
||||||
|
@ -88,6 +88,8 @@ class Cable(PrimaryModel):
|
|||||||
null=True
|
null=True
|
||||||
)
|
)
|
||||||
|
|
||||||
|
clone_fields = ('tenant', 'type',)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('pk',)
|
ordering = ('pk',)
|
||||||
verbose_name = _('cable')
|
verbose_name = _('cable')
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
import django_tables2 as tables
|
import django_tables2 as tables
|
||||||
from django_tables2.utils import Accessor
|
from django_tables2.utils import Accessor
|
||||||
|
from django.utils.html import escape
|
||||||
from django.utils.safestring import mark_safe
|
from django.utils.safestring import mark_safe
|
||||||
|
|
||||||
from dcim.models import Cable
|
from dcim.models import Cable
|
||||||
@ -35,7 +36,7 @@ class CableTerminationsColumn(tables.Column):
|
|||||||
|
|
||||||
def render(self, value):
|
def render(self, value):
|
||||||
links = [
|
links = [
|
||||||
f'<a href="{term.get_absolute_url()}">{term}</a>' for term in self._get_terminations(value)
|
f'<a href="{term.get_absolute_url()}">{escape(term)}</a>' for term in self._get_terminations(value)
|
||||||
]
|
]
|
||||||
return mark_safe('<br />'.join(links) or '—')
|
return mark_safe('<br />'.join(links) or '—')
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ from ipam.models import ASN, IPAddress, RIR, VRF
|
|||||||
from netbox.choices import ColorChoices
|
from netbox.choices import ColorChoices
|
||||||
from tenancy.models import Tenant, TenantGroup
|
from tenancy.models import Tenant, TenantGroup
|
||||||
from utilities.testing import ChangeLoggedFilterSetTests, create_test_device
|
from utilities.testing import ChangeLoggedFilterSetTests, create_test_device
|
||||||
from virtualization.models import Cluster, ClusterType
|
from virtualization.models import Cluster, ClusterType, ClusterGroup
|
||||||
from wireless.choices import WirelessChannelChoices, WirelessRoleChoices
|
from wireless.choices import WirelessChannelChoices, WirelessRoleChoices
|
||||||
|
|
||||||
User = get_user_model()
|
User = get_user_model()
|
||||||
@ -1959,10 +1959,16 @@ class DeviceTestCase(TestCase, ChangeLoggedFilterSetTests):
|
|||||||
Rack.objects.bulk_create(racks)
|
Rack.objects.bulk_create(racks)
|
||||||
|
|
||||||
cluster_type = ClusterType.objects.create(name='Cluster Type 1', slug='cluster-type-1')
|
cluster_type = ClusterType.objects.create(name='Cluster Type 1', slug='cluster-type-1')
|
||||||
|
cluster_groups = (
|
||||||
|
ClusterGroup(name='Cluster Group 1', slug='cluster-group-1'),
|
||||||
|
ClusterGroup(name='Cluster Group 2', slug='cluster-group-2'),
|
||||||
|
ClusterGroup(name='Cluster Group 3', slug='cluster-group-3'),
|
||||||
|
)
|
||||||
|
ClusterGroup.objects.bulk_create(cluster_groups)
|
||||||
clusters = (
|
clusters = (
|
||||||
Cluster(name='Cluster 1', type=cluster_type),
|
Cluster(name='Cluster 1', type=cluster_type, group=cluster_groups[0]),
|
||||||
Cluster(name='Cluster 2', type=cluster_type),
|
Cluster(name='Cluster 2', type=cluster_type, group=cluster_groups[1]),
|
||||||
Cluster(name='Cluster 3', type=cluster_type),
|
Cluster(name='Cluster 3', type=cluster_type, group=cluster_groups[2]),
|
||||||
)
|
)
|
||||||
Cluster.objects.bulk_create(clusters)
|
Cluster.objects.bulk_create(clusters)
|
||||||
|
|
||||||
@ -2213,6 +2219,13 @@ class DeviceTestCase(TestCase, ChangeLoggedFilterSetTests):
|
|||||||
params = {'cluster_id': [clusters[0].pk, clusters[1].pk]}
|
params = {'cluster_id': [clusters[0].pk, clusters[1].pk]}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
|
|
||||||
|
def test_cluster_group(self):
|
||||||
|
cluster_groups = ClusterGroup.objects.all()[:2]
|
||||||
|
params = {'cluster_group_id': [cluster_groups[0].pk, cluster_groups[1].pk]}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
|
params = {'cluster_group': [cluster_groups[0].slug, cluster_groups[1].slug]}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
|
|
||||||
def test_model(self):
|
def test_model(self):
|
||||||
params = {'model': ['model-1', 'model-2']}
|
params = {'model': ['model-1', 'model-2']}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
|
@ -31,6 +31,7 @@ from utilities.views import (
|
|||||||
GetRelatedModelsMixin, GetReturnURLMixin, ObjectPermissionRequiredMixin, ViewTab, register_model_view
|
GetRelatedModelsMixin, GetReturnURLMixin, ObjectPermissionRequiredMixin, ViewTab, register_model_view
|
||||||
)
|
)
|
||||||
from virtualization.filtersets import VirtualMachineFilterSet
|
from virtualization.filtersets import VirtualMachineFilterSet
|
||||||
|
from virtualization.forms import VirtualMachineFilterForm
|
||||||
from virtualization.models import VirtualMachine
|
from virtualization.models import VirtualMachine
|
||||||
from virtualization.tables import VirtualMachineTable
|
from virtualization.tables import VirtualMachineTable
|
||||||
from . import filtersets, forms, tables
|
from . import filtersets, forms, tables
|
||||||
@ -679,6 +680,7 @@ class RackRackReservationsView(generic.ObjectChildrenView):
|
|||||||
child_model = RackReservation
|
child_model = RackReservation
|
||||||
table = tables.RackReservationTable
|
table = tables.RackReservationTable
|
||||||
filterset = filtersets.RackReservationFilterSet
|
filterset = filtersets.RackReservationFilterSet
|
||||||
|
filterset_form = forms.RackReservationFilterForm
|
||||||
template_name = 'dcim/rack/reservations.html'
|
template_name = 'dcim/rack/reservations.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Reservations'),
|
label=_('Reservations'),
|
||||||
@ -697,6 +699,7 @@ class RackNonRackedView(generic.ObjectChildrenView):
|
|||||||
child_model = Device
|
child_model = Device
|
||||||
table = tables.DeviceTable
|
table = tables.DeviceTable
|
||||||
filterset = filtersets.DeviceFilterSet
|
filterset = filtersets.DeviceFilterSet
|
||||||
|
filterset_form = forms.DeviceFilterForm
|
||||||
template_name = 'dcim/rack/non_racked_devices.html'
|
template_name = 'dcim/rack/non_racked_devices.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Non-Racked Devices'),
|
label=_('Non-Racked Devices'),
|
||||||
@ -1835,6 +1838,7 @@ class DeviceConsolePortsView(DeviceComponentsView):
|
|||||||
child_model = ConsolePort
|
child_model = ConsolePort
|
||||||
table = tables.DeviceConsolePortTable
|
table = tables.DeviceConsolePortTable
|
||||||
filterset = filtersets.ConsolePortFilterSet
|
filterset = filtersets.ConsolePortFilterSet
|
||||||
|
filterset_form = forms.ConsolePortFilterForm
|
||||||
template_name = 'dcim/device/consoleports.html',
|
template_name = 'dcim/device/consoleports.html',
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Console Ports'),
|
label=_('Console Ports'),
|
||||||
@ -1850,6 +1854,7 @@ class DeviceConsoleServerPortsView(DeviceComponentsView):
|
|||||||
child_model = ConsoleServerPort
|
child_model = ConsoleServerPort
|
||||||
table = tables.DeviceConsoleServerPortTable
|
table = tables.DeviceConsoleServerPortTable
|
||||||
filterset = filtersets.ConsoleServerPortFilterSet
|
filterset = filtersets.ConsoleServerPortFilterSet
|
||||||
|
filterset_form = forms.ConsoleServerPortFilterForm
|
||||||
template_name = 'dcim/device/consoleserverports.html'
|
template_name = 'dcim/device/consoleserverports.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Console Server Ports'),
|
label=_('Console Server Ports'),
|
||||||
@ -1865,6 +1870,7 @@ class DevicePowerPortsView(DeviceComponentsView):
|
|||||||
child_model = PowerPort
|
child_model = PowerPort
|
||||||
table = tables.DevicePowerPortTable
|
table = tables.DevicePowerPortTable
|
||||||
filterset = filtersets.PowerPortFilterSet
|
filterset = filtersets.PowerPortFilterSet
|
||||||
|
filterset_form = forms.PowerPortFilterForm
|
||||||
template_name = 'dcim/device/powerports.html'
|
template_name = 'dcim/device/powerports.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Power Ports'),
|
label=_('Power Ports'),
|
||||||
@ -1880,6 +1886,7 @@ class DevicePowerOutletsView(DeviceComponentsView):
|
|||||||
child_model = PowerOutlet
|
child_model = PowerOutlet
|
||||||
table = tables.DevicePowerOutletTable
|
table = tables.DevicePowerOutletTable
|
||||||
filterset = filtersets.PowerOutletFilterSet
|
filterset = filtersets.PowerOutletFilterSet
|
||||||
|
filterset_form = forms.PowerOutletFilterForm
|
||||||
template_name = 'dcim/device/poweroutlets.html'
|
template_name = 'dcim/device/poweroutlets.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Power Outlets'),
|
label=_('Power Outlets'),
|
||||||
@ -1895,6 +1902,7 @@ class DeviceInterfacesView(DeviceComponentsView):
|
|||||||
child_model = Interface
|
child_model = Interface
|
||||||
table = tables.DeviceInterfaceTable
|
table = tables.DeviceInterfaceTable
|
||||||
filterset = filtersets.InterfaceFilterSet
|
filterset = filtersets.InterfaceFilterSet
|
||||||
|
filterset_form = forms.InterfaceFilterForm
|
||||||
template_name = 'dcim/device/interfaces.html'
|
template_name = 'dcim/device/interfaces.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Interfaces'),
|
label=_('Interfaces'),
|
||||||
@ -1916,6 +1924,7 @@ class DeviceFrontPortsView(DeviceComponentsView):
|
|||||||
child_model = FrontPort
|
child_model = FrontPort
|
||||||
table = tables.DeviceFrontPortTable
|
table = tables.DeviceFrontPortTable
|
||||||
filterset = filtersets.FrontPortFilterSet
|
filterset = filtersets.FrontPortFilterSet
|
||||||
|
filterset_form = forms.FrontPortFilterForm
|
||||||
template_name = 'dcim/device/frontports.html'
|
template_name = 'dcim/device/frontports.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Front Ports'),
|
label=_('Front Ports'),
|
||||||
@ -1931,6 +1940,7 @@ class DeviceRearPortsView(DeviceComponentsView):
|
|||||||
child_model = RearPort
|
child_model = RearPort
|
||||||
table = tables.DeviceRearPortTable
|
table = tables.DeviceRearPortTable
|
||||||
filterset = filtersets.RearPortFilterSet
|
filterset = filtersets.RearPortFilterSet
|
||||||
|
filterset_form = forms.RearPortFilterForm
|
||||||
template_name = 'dcim/device/rearports.html'
|
template_name = 'dcim/device/rearports.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Rear Ports'),
|
label=_('Rear Ports'),
|
||||||
@ -1946,6 +1956,7 @@ class DeviceModuleBaysView(DeviceComponentsView):
|
|||||||
child_model = ModuleBay
|
child_model = ModuleBay
|
||||||
table = tables.DeviceModuleBayTable
|
table = tables.DeviceModuleBayTable
|
||||||
filterset = filtersets.ModuleBayFilterSet
|
filterset = filtersets.ModuleBayFilterSet
|
||||||
|
filterset_form = forms.ModuleBayFilterForm
|
||||||
template_name = 'dcim/device/modulebays.html'
|
template_name = 'dcim/device/modulebays.html'
|
||||||
actions = {
|
actions = {
|
||||||
**DEFAULT_ACTION_PERMISSIONS,
|
**DEFAULT_ACTION_PERMISSIONS,
|
||||||
@ -1965,6 +1976,7 @@ class DeviceDeviceBaysView(DeviceComponentsView):
|
|||||||
child_model = DeviceBay
|
child_model = DeviceBay
|
||||||
table = tables.DeviceDeviceBayTable
|
table = tables.DeviceDeviceBayTable
|
||||||
filterset = filtersets.DeviceBayFilterSet
|
filterset = filtersets.DeviceBayFilterSet
|
||||||
|
filterset_form = forms.DeviceBayFilterForm
|
||||||
template_name = 'dcim/device/devicebays.html'
|
template_name = 'dcim/device/devicebays.html'
|
||||||
actions = {
|
actions = {
|
||||||
**DEFAULT_ACTION_PERMISSIONS,
|
**DEFAULT_ACTION_PERMISSIONS,
|
||||||
@ -1984,6 +1996,7 @@ class DeviceInventoryView(DeviceComponentsView):
|
|||||||
child_model = InventoryItem
|
child_model = InventoryItem
|
||||||
table = tables.DeviceInventoryItemTable
|
table = tables.DeviceInventoryItemTable
|
||||||
filterset = filtersets.InventoryItemFilterSet
|
filterset = filtersets.InventoryItemFilterSet
|
||||||
|
filterset_form = forms.InventoryItemFilterForm
|
||||||
template_name = 'dcim/device/inventory.html'
|
template_name = 'dcim/device/inventory.html'
|
||||||
actions = {
|
actions = {
|
||||||
**DEFAULT_ACTION_PERMISSIONS,
|
**DEFAULT_ACTION_PERMISSIONS,
|
||||||
@ -2062,6 +2075,7 @@ class DeviceVirtualMachinesView(generic.ObjectChildrenView):
|
|||||||
child_model = VirtualMachine
|
child_model = VirtualMachine
|
||||||
table = VirtualMachineTable
|
table = VirtualMachineTable
|
||||||
filterset = VirtualMachineFilterSet
|
filterset = VirtualMachineFilterSet
|
||||||
|
filterset_form = VirtualMachineFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Virtual Machines'),
|
label=_('Virtual Machines'),
|
||||||
badge=lambda obj: VirtualMachine.objects.filter(cluster=obj.cluster, device=obj).count(),
|
badge=lambda obj: VirtualMachine.objects.filter(cluster=obj.cluster, device=obj).count(),
|
||||||
@ -2944,6 +2958,7 @@ class InventoryItemChildrenView(generic.ObjectChildrenView):
|
|||||||
child_model = InventoryItem
|
child_model = InventoryItem
|
||||||
table = tables.InventoryItemTable
|
table = tables.InventoryItemTable
|
||||||
filterset = filtersets.InventoryItemFilterSet
|
filterset = filtersets.InventoryItemFilterSet
|
||||||
|
filterset_form = forms.InventoryItemFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Children'),
|
label=_('Children'),
|
||||||
badge=lambda obj: obj.child_items.count(),
|
badge=lambda obj: obj.child_items.count(),
|
||||||
@ -3410,8 +3425,9 @@ class VirtualChassisAddMemberView(ObjectPermissionRequiredMixin, GetReturnURLMix
|
|||||||
if membership_form.is_valid():
|
if membership_form.is_valid():
|
||||||
|
|
||||||
membership_form.save()
|
membership_form.save()
|
||||||
msg = f'Added member <a href="{device.get_absolute_url()}">{escape(device)}</a>'
|
messages.success(request, mark_safe(
|
||||||
messages.success(request, mark_safe(msg))
|
f'Added member <a href="{device.get_absolute_url()}">{escape(device)}</a>'
|
||||||
|
))
|
||||||
|
|
||||||
if '_addanother' in request.POST:
|
if '_addanother' in request.POST:
|
||||||
return redirect(request.get_full_path())
|
return redirect(request.get_full_path())
|
||||||
|
@ -251,6 +251,10 @@ class ObjectListWidget(DashboardWidget):
|
|||||||
def render(self, request):
|
def render(self, request):
|
||||||
app_label, model_name = self.config['model'].split('.')
|
app_label, model_name = self.config['model'].split('.')
|
||||||
model = ObjectType.objects.get_by_natural_key(app_label, model_name).model_class()
|
model = ObjectType.objects.get_by_natural_key(app_label, model_name).model_class()
|
||||||
|
if not model:
|
||||||
|
logger.debug(f"Dashboard Widget model_class not found: {app_label}:{model_name}")
|
||||||
|
return
|
||||||
|
|
||||||
viewname = get_viewname(model, action='list')
|
viewname = get_viewname(model, action='list')
|
||||||
|
|
||||||
# Evaluate user's permission. Note that this controls only whether the HTMX element is
|
# Evaluate user's permission. Note that this controls only whether the HTMX element is
|
||||||
@ -381,17 +385,17 @@ class BookmarksWidget(DashboardWidget):
|
|||||||
if request.user.is_anonymous:
|
if request.user.is_anonymous:
|
||||||
bookmarks = list()
|
bookmarks = list()
|
||||||
else:
|
else:
|
||||||
user_bookmarks = Bookmark.objects.filter(user=request.user)
|
bookmarks = Bookmark.objects.filter(user=request.user)
|
||||||
if self.config['order_by'] == BookmarkOrderingChoices.ORDERING_ALPHABETICAL_AZ:
|
|
||||||
bookmarks = sorted(user_bookmarks, key=lambda bookmark: bookmark.__str__().lower())
|
|
||||||
elif self.config['order_by'] == BookmarkOrderingChoices.ORDERING_ALPHABETICAL_ZA:
|
|
||||||
bookmarks = sorted(user_bookmarks, key=lambda bookmark: bookmark.__str__().lower(), reverse=True)
|
|
||||||
else:
|
|
||||||
bookmarks = user_bookmarks.order_by(self.config['order_by'])
|
|
||||||
if object_types := self.config.get('object_types'):
|
if object_types := self.config.get('object_types'):
|
||||||
models = get_models_from_content_types(object_types)
|
models = get_models_from_content_types(object_types)
|
||||||
content_types = ObjectType.objects.get_for_models(*models).values()
|
content_types = ObjectType.objects.get_for_models(*models).values()
|
||||||
bookmarks = bookmarks.filter(object_type__in=content_types)
|
bookmarks = bookmarks.filter(object_type__in=content_types)
|
||||||
|
if self.config['order_by'] == BookmarkOrderingChoices.ORDERING_ALPHABETICAL_AZ:
|
||||||
|
bookmarks = sorted(bookmarks, key=lambda bookmark: bookmark.__str__().lower())
|
||||||
|
elif self.config['order_by'] == BookmarkOrderingChoices.ORDERING_ALPHABETICAL_ZA:
|
||||||
|
bookmarks = sorted(bookmarks, key=lambda bookmark: bookmark.__str__().lower(), reverse=True)
|
||||||
|
else:
|
||||||
|
bookmarks = bookmarks.order_by(self.config['order_by'])
|
||||||
if max_items := self.config.get('max_items'):
|
if max_items := self.config.get('max_items'):
|
||||||
bookmarks = bookmarks[:max_items]
|
bookmarks = bookmarks[:max_items]
|
||||||
|
|
||||||
|
@ -63,6 +63,9 @@ def enqueue_object(queue, instance, user, request_id, action):
|
|||||||
if key in queue:
|
if key in queue:
|
||||||
queue[key]['data'] = serialize_for_event(instance)
|
queue[key]['data'] = serialize_for_event(instance)
|
||||||
queue[key]['snapshots']['postchange'] = get_snapshots(instance, action)['postchange']
|
queue[key]['snapshots']['postchange'] = get_snapshots(instance, action)['postchange']
|
||||||
|
# If the object is being deleted, update any prior "update" event to "delete"
|
||||||
|
if action == ObjectChangeActionChoices.ACTION_DELETE:
|
||||||
|
queue[key]['event'] = action
|
||||||
else:
|
else:
|
||||||
queue[key] = {
|
queue[key] = {
|
||||||
'content_type': ContentType.objects.get_for_model(instance),
|
'content_type': ContentType.objects.get_for_model(instance),
|
||||||
|
@ -228,9 +228,6 @@ class TagImportForm(CSVModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = Tag
|
model = Tag
|
||||||
fields = ('name', 'slug', 'color', 'description')
|
fields = ('name', 'slug', 'color', 'description')
|
||||||
help_texts = {
|
|
||||||
'color': mark_safe(_('RGB color in hexadecimal. Example:') + ' <code>00ff00</code>'),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class JournalEntryImportForm(NetBoxModelImportForm):
|
class JournalEntryImportForm(NetBoxModelImportForm):
|
||||||
|
@ -66,11 +66,16 @@ class Command(BaseCommand):
|
|||||||
raise CommandError(_("No indexers found!"))
|
raise CommandError(_("No indexers found!"))
|
||||||
self.stdout.write(f'Reindexing {len(indexers)} models.')
|
self.stdout.write(f'Reindexing {len(indexers)} models.')
|
||||||
|
|
||||||
# Clear all cached values for the specified models (if not being lazy)
|
# Clear cached values for the specified models (if not being lazy)
|
||||||
if not kwargs['lazy']:
|
if not kwargs['lazy']:
|
||||||
|
if model_labels:
|
||||||
|
content_types = [ContentType.objects.get_for_model(model) for model in indexers.keys()]
|
||||||
|
else:
|
||||||
|
content_types = None
|
||||||
|
|
||||||
self.stdout.write('Clearing cached values... ', ending='')
|
self.stdout.write('Clearing cached values... ', ending='')
|
||||||
self.stdout.flush()
|
self.stdout.flush()
|
||||||
deleted_count = search_backend.clear()
|
deleted_count = search_backend.clear(object_types=content_types)
|
||||||
self.stdout.write(f'{deleted_count} entries deleted.')
|
self.stdout.write(f'{deleted_count} entries deleted.')
|
||||||
|
|
||||||
# Index models
|
# Index models
|
||||||
|
@ -10,6 +10,7 @@ from django.contrib.postgres.fields import ArrayField
|
|||||||
from django.core.validators import RegexValidator, ValidationError
|
from django.core.validators import RegexValidator, ValidationError
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
from django.utils.html import escape
|
||||||
from django.utils.safestring import mark_safe
|
from django.utils.safestring import mark_safe
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
@ -489,7 +490,7 @@ class CustomField(CloningMixin, ExportTemplatesMixin, ChangeLoggedModel):
|
|||||||
|
|
||||||
# JSON
|
# JSON
|
||||||
elif self.type == CustomFieldTypeChoices.TYPE_JSON:
|
elif self.type == CustomFieldTypeChoices.TYPE_JSON:
|
||||||
field = JSONField(required=required, initial=json.dumps(initial) if initial else '')
|
field = JSONField(required=required, initial=json.dumps(initial) if initial else None)
|
||||||
|
|
||||||
# Object
|
# Object
|
||||||
elif self.type == CustomFieldTypeChoices.TYPE_OBJECT:
|
elif self.type == CustomFieldTypeChoices.TYPE_OBJECT:
|
||||||
@ -520,7 +521,7 @@ class CustomField(CloningMixin, ExportTemplatesMixin, ChangeLoggedModel):
|
|||||||
RegexValidator(
|
RegexValidator(
|
||||||
regex=self.validation_regex,
|
regex=self.validation_regex,
|
||||||
message=mark_safe(_("Values must match this regex: <code>{regex}</code>").format(
|
message=mark_safe(_("Values must match this regex: <code>{regex}</code>").format(
|
||||||
regex=self.validation_regex
|
regex=escape(self.validation_regex)
|
||||||
))
|
))
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
|
@ -484,6 +484,12 @@ class BaseScript:
|
|||||||
# A test method is currently active, so log the message using legacy Report logging
|
# A test method is currently active, so log the message using legacy Report logging
|
||||||
if self._current_test:
|
if self._current_test:
|
||||||
|
|
||||||
|
# Increment the event counter for this level
|
||||||
|
if level in self.tests[self._current_test]:
|
||||||
|
self.tests[self._current_test][level] += 1
|
||||||
|
|
||||||
|
# Record message (if any) to the report log
|
||||||
|
if message:
|
||||||
# TODO: Use a dataclass for test method logs
|
# TODO: Use a dataclass for test method logs
|
||||||
self.tests[self._current_test]['log'].append((
|
self.tests[self._current_test]['log'].append((
|
||||||
timezone.now().isoformat(),
|
timezone.now().isoformat(),
|
||||||
@ -493,10 +499,6 @@ class BaseScript:
|
|||||||
str(message),
|
str(message),
|
||||||
))
|
))
|
||||||
|
|
||||||
# Increment the event counter for this level
|
|
||||||
if level in self.tests[self._current_test]:
|
|
||||||
self.tests[self._current_test][level] += 1
|
|
||||||
|
|
||||||
elif message:
|
elif message:
|
||||||
|
|
||||||
# Record to the script's log
|
# Record to the script's log
|
||||||
@ -504,6 +506,8 @@ class BaseScript:
|
|||||||
'time': timezone.now().isoformat(),
|
'time': timezone.now().isoformat(),
|
||||||
'status': level,
|
'status': level,
|
||||||
'message': str(message),
|
'message': str(message),
|
||||||
|
'obj': str(obj) if obj else None,
|
||||||
|
'url': obj.get_absolute_url() if hasattr(obj, 'get_absolute_url') else None,
|
||||||
})
|
})
|
||||||
|
|
||||||
# Record to the system log
|
# Record to the system log
|
||||||
@ -511,19 +515,19 @@ class BaseScript:
|
|||||||
message = f"{obj}: {message}"
|
message = f"{obj}: {message}"
|
||||||
self.logger.log(LogLevelChoices.SYSTEM_LEVELS[level], message)
|
self.logger.log(LogLevelChoices.SYSTEM_LEVELS[level], message)
|
||||||
|
|
||||||
def log_debug(self, message, obj=None):
|
def log_debug(self, message=None, obj=None):
|
||||||
self._log(message, obj, level=LogLevelChoices.LOG_DEBUG)
|
self._log(message, obj, level=LogLevelChoices.LOG_DEBUG)
|
||||||
|
|
||||||
def log_success(self, message, obj=None):
|
def log_success(self, message=None, obj=None):
|
||||||
self._log(message, obj, level=LogLevelChoices.LOG_SUCCESS)
|
self._log(message, obj, level=LogLevelChoices.LOG_SUCCESS)
|
||||||
|
|
||||||
def log_info(self, message, obj=None):
|
def log_info(self, message=None, obj=None):
|
||||||
self._log(message, obj, level=LogLevelChoices.LOG_INFO)
|
self._log(message, obj, level=LogLevelChoices.LOG_INFO)
|
||||||
|
|
||||||
def log_warning(self, message, obj=None):
|
def log_warning(self, message=None, obj=None):
|
||||||
self._log(message, obj, level=LogLevelChoices.LOG_WARNING)
|
self._log(message, obj, level=LogLevelChoices.LOG_WARNING)
|
||||||
|
|
||||||
def log_failure(self, message, obj=None):
|
def log_failure(self, message=None, obj=None):
|
||||||
self._log(message, obj, level=LogLevelChoices.LOG_FAILURE)
|
self._log(message, obj, level=LogLevelChoices.LOG_FAILURE)
|
||||||
self.failed = True
|
self.failed = True
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import json
|
import json
|
||||||
|
|
||||||
import django_tables2 as tables
|
import django_tables2 as tables
|
||||||
|
from django.utils.html import format_html
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from extras.models import *
|
from extras.models import *
|
||||||
@ -545,6 +546,9 @@ class ScriptResultsTable(BaseTable):
|
|||||||
template_code="""{% load log_levels %}{% log_level record.status %}""",
|
template_code="""{% load log_levels %}{% log_level record.status %}""",
|
||||||
verbose_name=_('Level')
|
verbose_name=_('Level')
|
||||||
)
|
)
|
||||||
|
object = tables.Column(
|
||||||
|
verbose_name=_('Object')
|
||||||
|
)
|
||||||
message = columns.MarkdownColumn(
|
message = columns.MarkdownColumn(
|
||||||
verbose_name=_('Message')
|
verbose_name=_('Message')
|
||||||
)
|
)
|
||||||
@ -552,8 +556,17 @@ class ScriptResultsTable(BaseTable):
|
|||||||
class Meta(BaseTable.Meta):
|
class Meta(BaseTable.Meta):
|
||||||
empty_text = _(EMPTY_TABLE_TEXT)
|
empty_text = _(EMPTY_TABLE_TEXT)
|
||||||
fields = (
|
fields = (
|
||||||
'index', 'time', 'status', 'message',
|
'index', 'time', 'status', 'object', 'message',
|
||||||
)
|
)
|
||||||
|
default_columns = (
|
||||||
|
'index', 'time', 'status', 'object', 'message',
|
||||||
|
)
|
||||||
|
|
||||||
|
def render_object(self, value, record):
|
||||||
|
return format_html("<a href='{}'>{}</a>", record['url'], value)
|
||||||
|
|
||||||
|
def render_url(self, value):
|
||||||
|
return format_html("<a href='{}'>{}</a>", value, value)
|
||||||
|
|
||||||
|
|
||||||
class ReportResultsTable(BaseTable):
|
class ReportResultsTable(BaseTable):
|
||||||
@ -585,3 +598,9 @@ class ReportResultsTable(BaseTable):
|
|||||||
fields = (
|
fields = (
|
||||||
'index', 'method', 'time', 'status', 'object', 'url', 'message',
|
'index', 'method', 'time', 'status', 'object', 'url', 'message',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def render_object(self, value, record):
|
||||||
|
return format_html("<a href='{}'>{}</a>", record['url'], value)
|
||||||
|
|
||||||
|
def render_url(self, value):
|
||||||
|
return format_html("<a href='{}'>{}</a>", value, value)
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
from django import template
|
from django import template
|
||||||
|
from django.utils.html import escape
|
||||||
from django.utils.safestring import mark_safe
|
from django.utils.safestring import mark_safe
|
||||||
|
|
||||||
from core.models import ObjectType
|
from core.models import ObjectType
|
||||||
@ -59,8 +60,7 @@ def custom_links(context, obj):
|
|||||||
# Add non-grouped links
|
# Add non-grouped links
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
rendered = cl.render(link_context)
|
if rendered := cl.render(link_context):
|
||||||
if rendered:
|
|
||||||
template_code += LINK_BUTTON.format(
|
template_code += LINK_BUTTON.format(
|
||||||
rendered['link'], rendered['link_target'], cl.button_class, rendered['text']
|
rendered['link'], rendered['link_target'], cl.button_class, rendered['text']
|
||||||
)
|
)
|
||||||
@ -75,8 +75,7 @@ def custom_links(context, obj):
|
|||||||
|
|
||||||
for cl in links:
|
for cl in links:
|
||||||
try:
|
try:
|
||||||
rendered = cl.render(link_context)
|
if rendered := cl.render(link_context):
|
||||||
if rendered:
|
|
||||||
links_rendered.append(
|
links_rendered.append(
|
||||||
GROUP_LINK.format(rendered['link'], rendered['link_target'], rendered['text'])
|
GROUP_LINK.format(rendered['link'], rendered['link_target'], rendered['text'])
|
||||||
)
|
)
|
||||||
@ -88,7 +87,7 @@ def custom_links(context, obj):
|
|||||||
|
|
||||||
if links_rendered:
|
if links_rendered:
|
||||||
template_code += GROUP_BUTTON.format(
|
template_code += GROUP_BUTTON.format(
|
||||||
links[0].button_class, group, ''.join(links_rendered)
|
links[0].button_class, escape(group), ''.join(links_rendered)
|
||||||
)
|
)
|
||||||
|
|
||||||
return mark_safe(template_code)
|
return mark_safe(template_code)
|
||||||
|
@ -390,13 +390,36 @@ class EventRuleTest(APITestCase):
|
|||||||
request.id = uuid.uuid4()
|
request.id = uuid.uuid4()
|
||||||
request.user = self.user
|
request.user = self.user
|
||||||
|
|
||||||
self.assertEqual(self.queue.count, 0, msg="Unexpected jobs found in queue")
|
# Test create & update
|
||||||
|
|
||||||
with event_tracking(request):
|
with event_tracking(request):
|
||||||
site = Site(name='Site 1', slug='site-1')
|
site = Site(name='Site 1', slug='site-1')
|
||||||
site.save()
|
site.save()
|
||||||
|
site.description = 'foo'
|
||||||
# Save the site a second time
|
|
||||||
site.save()
|
site.save()
|
||||||
|
|
||||||
self.assertEqual(self.queue.count, 1, msg="Duplicate jobs found in queue")
|
self.assertEqual(self.queue.count, 1, msg="Duplicate jobs found in queue")
|
||||||
|
job = self.queue.get_jobs()[0]
|
||||||
|
self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_CREATE)
|
||||||
|
self.queue.empty()
|
||||||
|
|
||||||
|
# Test multiple updates
|
||||||
|
site = Site.objects.create(name='Site 2', slug='site-2')
|
||||||
|
with event_tracking(request):
|
||||||
|
site.description = 'foo'
|
||||||
|
site.save()
|
||||||
|
site.description = 'bar'
|
||||||
|
site.save()
|
||||||
|
self.assertEqual(self.queue.count, 1, msg="Duplicate jobs found in queue")
|
||||||
|
job = self.queue.get_jobs()[0]
|
||||||
|
self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_UPDATE)
|
||||||
|
self.queue.empty()
|
||||||
|
|
||||||
|
# Test update & delete
|
||||||
|
site = Site.objects.create(name='Site 3', slug='site-3')
|
||||||
|
with event_tracking(request):
|
||||||
|
site.description = 'foo'
|
||||||
|
site.save()
|
||||||
|
site.delete()
|
||||||
|
self.assertEqual(self.queue.count, 1, msg="Duplicate jobs found in queue")
|
||||||
|
job = self.queue.get_jobs()[0]
|
||||||
|
self.assertEqual(job.kwargs['event'], ObjectChangeActionChoices.ACTION_DELETE)
|
||||||
|
self.queue.empty()
|
||||||
|
@ -1202,6 +1202,8 @@ class ScriptResultView(TableMixin, generic.ObjectView):
|
|||||||
'time': log.get('time'),
|
'time': log.get('time'),
|
||||||
'status': log.get('status'),
|
'status': log.get('status'),
|
||||||
'message': log.get('message'),
|
'message': log.get('message'),
|
||||||
|
'object': log.get('obj'),
|
||||||
|
'url': log.get('url'),
|
||||||
}
|
}
|
||||||
data.append(result)
|
data.append(result)
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@ from django.utils.translation import gettext as _
|
|||||||
|
|
||||||
from circuits.models import Provider
|
from circuits.models import Provider
|
||||||
from dcim.filtersets import InterfaceFilterSet
|
from dcim.filtersets import InterfaceFilterSet
|
||||||
|
from dcim.forms import InterfaceFilterForm
|
||||||
from dcim.models import Interface, Site
|
from dcim.models import Interface, Site
|
||||||
from netbox.views import generic
|
from netbox.views import generic
|
||||||
from tenancy.views import ObjectContactsView
|
from tenancy.views import ObjectContactsView
|
||||||
@ -14,6 +15,7 @@ from utilities.query import count_related
|
|||||||
from utilities.tables import get_table_ordering
|
from utilities.tables import get_table_ordering
|
||||||
from utilities.views import GetRelatedModelsMixin, ViewTab, register_model_view
|
from utilities.views import GetRelatedModelsMixin, ViewTab, register_model_view
|
||||||
from virtualization.filtersets import VMInterfaceFilterSet
|
from virtualization.filtersets import VMInterfaceFilterSet
|
||||||
|
from virtualization.forms import VMInterfaceFilterForm
|
||||||
from virtualization.models import VMInterface
|
from virtualization.models import VMInterface
|
||||||
from . import filtersets, forms, tables
|
from . import filtersets, forms, tables
|
||||||
from .choices import PrefixStatusChoices
|
from .choices import PrefixStatusChoices
|
||||||
@ -206,6 +208,7 @@ class ASNRangeASNsView(generic.ObjectChildrenView):
|
|||||||
child_model = ASN
|
child_model = ASN
|
||||||
table = tables.ASNTable
|
table = tables.ASNTable
|
||||||
filterset = filtersets.ASNFilterSet
|
filterset = filtersets.ASNFilterSet
|
||||||
|
filterset_form = forms.ASNFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('ASNs'),
|
label=_('ASNs'),
|
||||||
badge=lambda x: x.get_child_asns().count(),
|
badge=lambda x: x.get_child_asns().count(),
|
||||||
@ -337,6 +340,7 @@ class AggregatePrefixesView(generic.ObjectChildrenView):
|
|||||||
child_model = Prefix
|
child_model = Prefix
|
||||||
table = tables.PrefixTable
|
table = tables.PrefixTable
|
||||||
filterset = filtersets.PrefixFilterSet
|
filterset = filtersets.PrefixFilterSet
|
||||||
|
filterset_form = forms.PrefixFilterForm
|
||||||
template_name = 'ipam/aggregate/prefixes.html'
|
template_name = 'ipam/aggregate/prefixes.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Prefixes'),
|
label=_('Prefixes'),
|
||||||
@ -523,6 +527,7 @@ class PrefixPrefixesView(generic.ObjectChildrenView):
|
|||||||
child_model = Prefix
|
child_model = Prefix
|
||||||
table = tables.PrefixTable
|
table = tables.PrefixTable
|
||||||
filterset = filtersets.PrefixFilterSet
|
filterset = filtersets.PrefixFilterSet
|
||||||
|
filterset_form = forms.PrefixFilterForm
|
||||||
template_name = 'ipam/prefix/prefixes.html'
|
template_name = 'ipam/prefix/prefixes.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Child Prefixes'),
|
label=_('Child Prefixes'),
|
||||||
@ -558,6 +563,7 @@ class PrefixIPRangesView(generic.ObjectChildrenView):
|
|||||||
child_model = IPRange
|
child_model = IPRange
|
||||||
table = tables.IPRangeTable
|
table = tables.IPRangeTable
|
||||||
filterset = filtersets.IPRangeFilterSet
|
filterset = filtersets.IPRangeFilterSet
|
||||||
|
filterset_form = forms.IPRangeFilterForm
|
||||||
template_name = 'ipam/prefix/ip_ranges.html'
|
template_name = 'ipam/prefix/ip_ranges.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Child Ranges'),
|
label=_('Child Ranges'),
|
||||||
@ -584,6 +590,7 @@ class PrefixIPAddressesView(generic.ObjectChildrenView):
|
|||||||
child_model = IPAddress
|
child_model = IPAddress
|
||||||
table = tables.IPAddressTable
|
table = tables.IPAddressTable
|
||||||
filterset = filtersets.IPAddressFilterSet
|
filterset = filtersets.IPAddressFilterSet
|
||||||
|
filterset_form = forms.IPAddressFilterForm
|
||||||
template_name = 'ipam/prefix/ip_addresses.html'
|
template_name = 'ipam/prefix/ip_addresses.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('IP Addresses'),
|
label=_('IP Addresses'),
|
||||||
@ -683,6 +690,7 @@ class IPRangeIPAddressesView(generic.ObjectChildrenView):
|
|||||||
child_model = IPAddress
|
child_model = IPAddress
|
||||||
table = tables.IPAddressTable
|
table = tables.IPAddressTable
|
||||||
filterset = filtersets.IPAddressFilterSet
|
filterset = filtersets.IPAddressFilterSet
|
||||||
|
filterset_form = forms.IPRangeFilterForm
|
||||||
template_name = 'ipam/iprange/ip_addresses.html'
|
template_name = 'ipam/iprange/ip_addresses.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('IP Addresses'),
|
label=_('IP Addresses'),
|
||||||
@ -885,6 +893,7 @@ class IPAddressRelatedIPsView(generic.ObjectChildrenView):
|
|||||||
child_model = IPAddress
|
child_model = IPAddress
|
||||||
table = tables.IPAddressTable
|
table = tables.IPAddressTable
|
||||||
filterset = filtersets.IPAddressFilterSet
|
filterset = filtersets.IPAddressFilterSet
|
||||||
|
filterset_form = forms.IPAddressFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Related IPs'),
|
label=_('Related IPs'),
|
||||||
badge=lambda x: x.get_related_ips().count(),
|
badge=lambda x: x.get_related_ips().count(),
|
||||||
@ -957,6 +966,7 @@ class VLANGroupVLANsView(generic.ObjectChildrenView):
|
|||||||
child_model = VLAN
|
child_model = VLAN
|
||||||
table = tables.VLANTable
|
table = tables.VLANTable
|
||||||
filterset = filtersets.VLANFilterSet
|
filterset = filtersets.VLANFilterSet
|
||||||
|
filterset_form = forms.VLANFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('VLANs'),
|
label=_('VLANs'),
|
||||||
badge=lambda x: x.get_child_vlans().count(),
|
badge=lambda x: x.get_child_vlans().count(),
|
||||||
@ -1112,6 +1122,7 @@ class VLANInterfacesView(generic.ObjectChildrenView):
|
|||||||
child_model = Interface
|
child_model = Interface
|
||||||
table = tables.VLANDevicesTable
|
table = tables.VLANDevicesTable
|
||||||
filterset = InterfaceFilterSet
|
filterset = InterfaceFilterSet
|
||||||
|
filterset_form = InterfaceFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Device Interfaces'),
|
label=_('Device Interfaces'),
|
||||||
badge=lambda x: x.get_interfaces().count(),
|
badge=lambda x: x.get_interfaces().count(),
|
||||||
@ -1129,6 +1140,7 @@ class VLANVMInterfacesView(generic.ObjectChildrenView):
|
|||||||
child_model = VMInterface
|
child_model = VMInterface
|
||||||
table = tables.VLANVirtualMachinesTable
|
table = tables.VLANVirtualMachinesTable
|
||||||
filterset = VMInterfaceFilterSet
|
filterset = VMInterfaceFilterSet
|
||||||
|
filterset_form = VMInterfaceFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('VM Interfaces'),
|
label=_('VM Interfaces'),
|
||||||
badge=lambda x: x.get_vminterfaces().count(),
|
badge=lambda x: x.get_vminterfaces().count(),
|
||||||
|
@ -49,12 +49,15 @@ AUTH_BACKEND_ATTRS = {
|
|||||||
'okta-openidconnect': ('Okta (OIDC)', None),
|
'okta-openidconnect': ('Okta (OIDC)', None),
|
||||||
'salesforce-oauth2': ('Salesforce', 'salesforce'),
|
'salesforce-oauth2': ('Salesforce', 'salesforce'),
|
||||||
}
|
}
|
||||||
|
# Override with potential user configuration
|
||||||
|
AUTH_BACKEND_ATTRS.update(getattr(settings, 'SOCIAL_AUTH_BACKEND_ATTRS', {}))
|
||||||
|
|
||||||
|
|
||||||
def get_auth_backend_display(name):
|
def get_auth_backend_display(name):
|
||||||
"""
|
"""
|
||||||
Return the user-friendly name and icon name for a remote authentication backend, if known. Defaults to the
|
Return the user-friendly name and icon name for a remote authentication backend, if
|
||||||
raw backend name and no icon.
|
known. Obtained from the defaults dictionary AUTH_BACKEND_ATTRS, overridden by the
|
||||||
|
setting `SOCIAL_AUTH_BACKEND_ATTRS`. Defaults to the raw backend name and no icon.
|
||||||
"""
|
"""
|
||||||
return AUTH_BACKEND_ATTRS.get(name, (name, None))
|
return AUTH_BACKEND_ATTRS.get(name, (name, None))
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import re
|
import re
|
||||||
|
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from netbox.search import LookupTypes
|
from netbox.search import LookupTypes
|
||||||
from netbox.search.backends import search_backend
|
from netbox.search.backends import search_backend
|
||||||
@ -36,7 +36,8 @@ class SearchForm(forms.Form):
|
|||||||
lookup = forms.ChoiceField(
|
lookup = forms.ChoiceField(
|
||||||
choices=LOOKUP_CHOICES,
|
choices=LOOKUP_CHOICES,
|
||||||
initial=LookupTypes.PARTIAL,
|
initial=LookupTypes.PARTIAL,
|
||||||
required=False
|
required=False,
|
||||||
|
label=_('Lookup')
|
||||||
)
|
)
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
|
@ -47,6 +47,11 @@ class CoreMiddleware:
|
|||||||
with event_tracking(request):
|
with event_tracking(request):
|
||||||
response = self.get_response(request)
|
response = self.get_response(request)
|
||||||
|
|
||||||
|
# Check if language cookie should be renewed
|
||||||
|
if request.user.is_authenticated and settings.SESSION_SAVE_EVERY_REQUEST:
|
||||||
|
if language := request.user.config.get('locale.language'):
|
||||||
|
response.set_cookie(settings.LANGUAGE_COOKIE_NAME, language, max_age=request.session.get_expiry_age())
|
||||||
|
|
||||||
# Attach the unique request ID as an HTTP header.
|
# Attach the unique request ID as an HTTP header.
|
||||||
response['X-Request-ID'] = request.id
|
response['X-Request-ID'] = request.id
|
||||||
|
|
||||||
|
@ -462,16 +462,13 @@ MENUS = [
|
|||||||
PROVISIONING_MENU,
|
PROVISIONING_MENU,
|
||||||
CUSTOMIZATION_MENU,
|
CUSTOMIZATION_MENU,
|
||||||
OPERATIONS_MENU,
|
OPERATIONS_MENU,
|
||||||
ADMIN_MENU,
|
|
||||||
]
|
]
|
||||||
|
|
||||||
#
|
# Add top-level plugin menus
|
||||||
# Add plugin menus
|
|
||||||
#
|
|
||||||
|
|
||||||
for menu in registry['plugins']['menus']:
|
for menu in registry['plugins']['menus']:
|
||||||
MENUS.append(menu)
|
MENUS.append(menu)
|
||||||
|
|
||||||
|
# Add the default "plugins" menu
|
||||||
if registry['plugins']['menu_items']:
|
if registry['plugins']['menu_items']:
|
||||||
|
|
||||||
# Build the default plugins menu
|
# Build the default plugins menu
|
||||||
@ -485,3 +482,6 @@ if registry['plugins']['menu_items']:
|
|||||||
groups=groups
|
groups=groups
|
||||||
)
|
)
|
||||||
MENUS.append(plugins_menu)
|
MENUS.append(plugins_menu)
|
||||||
|
|
||||||
|
# Add the admin menu last
|
||||||
|
MENUS.append(ADMIN_MENU)
|
||||||
|
@ -8,6 +8,7 @@ from django.db.models.fields.related import ForeignKey
|
|||||||
from django.db.models.functions import window
|
from django.db.models.functions import window
|
||||||
from django.db.models.signals import post_delete, post_save
|
from django.db.models.signals import post_delete, post_save
|
||||||
from django.utils.module_loading import import_string
|
from django.utils.module_loading import import_string
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
import netaddr
|
import netaddr
|
||||||
from netaddr.core import AddrFormatError
|
from netaddr.core import AddrFormatError
|
||||||
|
|
||||||
@ -39,7 +40,7 @@ class SearchBackend:
|
|||||||
# Organize choices by category
|
# Organize choices by category
|
||||||
categories = defaultdict(dict)
|
categories = defaultdict(dict)
|
||||||
for label, idx in registry['search'].items():
|
for label, idx in registry['search'].items():
|
||||||
categories[idx.get_category()][label] = title(idx.model._meta.verbose_name)
|
categories[idx.get_category()][label] = _(title(idx.model._meta.verbose_name))
|
||||||
|
|
||||||
# Compile a nested tuple of choices for form rendering
|
# Compile a nested tuple of choices for form rendering
|
||||||
results = (
|
results = (
|
||||||
|
@ -25,7 +25,7 @@ from utilities.string import trailing_slash
|
|||||||
# Environment setup
|
# Environment setup
|
||||||
#
|
#
|
||||||
|
|
||||||
VERSION = '4.0.6-dev'
|
VERSION = '4.0.8-dev'
|
||||||
HOSTNAME = platform.node()
|
HOSTNAME = platform.node()
|
||||||
# Set the base directory two levels up
|
# Set the base directory two levels up
|
||||||
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
|
||||||
@ -147,6 +147,7 @@ SECURE_SSL_REDIRECT = getattr(configuration, 'SECURE_SSL_REDIRECT', False)
|
|||||||
SENTRY_DSN = getattr(configuration, 'SENTRY_DSN', None)
|
SENTRY_DSN = getattr(configuration, 'SENTRY_DSN', None)
|
||||||
SENTRY_ENABLED = getattr(configuration, 'SENTRY_ENABLED', False)
|
SENTRY_ENABLED = getattr(configuration, 'SENTRY_ENABLED', False)
|
||||||
SENTRY_SAMPLE_RATE = getattr(configuration, 'SENTRY_SAMPLE_RATE', 1.0)
|
SENTRY_SAMPLE_RATE = getattr(configuration, 'SENTRY_SAMPLE_RATE', 1.0)
|
||||||
|
SENTRY_SEND_DEFAULT_PII = getattr(configuration, 'SENTRY_SEND_DEFAULT_PII', False)
|
||||||
SENTRY_TAGS = getattr(configuration, 'SENTRY_TAGS', {})
|
SENTRY_TAGS = getattr(configuration, 'SENTRY_TAGS', {})
|
||||||
SENTRY_TRACES_SAMPLE_RATE = getattr(configuration, 'SENTRY_TRACES_SAMPLE_RATE', 0)
|
SENTRY_TRACES_SAMPLE_RATE = getattr(configuration, 'SENTRY_TRACES_SAMPLE_RATE', 0)
|
||||||
SESSION_COOKIE_NAME = getattr(configuration, 'SESSION_COOKIE_NAME', 'sessionid')
|
SESSION_COOKIE_NAME = getattr(configuration, 'SESSION_COOKIE_NAME', 'sessionid')
|
||||||
@ -225,6 +226,23 @@ if STORAGE_BACKEND is not None:
|
|||||||
return globals().get(name, default)
|
return globals().get(name, default)
|
||||||
storages.utils.setting = _setting
|
storages.utils.setting = _setting
|
||||||
|
|
||||||
|
# django-storage-swift
|
||||||
|
elif STORAGE_BACKEND == 'swift.storage.SwiftStorage':
|
||||||
|
try:
|
||||||
|
import swift.utils # type: ignore
|
||||||
|
except ModuleNotFoundError as e:
|
||||||
|
if getattr(e, 'name') == 'swift':
|
||||||
|
raise ImproperlyConfigured(
|
||||||
|
f"STORAGE_BACKEND is set to {STORAGE_BACKEND} but django-storage-swift is not present. "
|
||||||
|
"It can be installed by running 'pip install django-storage-swift'."
|
||||||
|
)
|
||||||
|
raise e
|
||||||
|
|
||||||
|
# Load all SWIFT_* settings from the user configuration
|
||||||
|
for param, value in STORAGE_CONFIG.items():
|
||||||
|
if param.startswith('SWIFT_'):
|
||||||
|
globals()[param] = value
|
||||||
|
|
||||||
if STORAGE_CONFIG and STORAGE_BACKEND is None:
|
if STORAGE_CONFIG and STORAGE_BACKEND is None:
|
||||||
warnings.warn(
|
warnings.warn(
|
||||||
"STORAGE_CONFIG has been set in configuration.py but STORAGE_BACKEND is not defined. STORAGE_CONFIG will be "
|
"STORAGE_CONFIG has been set in configuration.py but STORAGE_BACKEND is not defined. STORAGE_CONFIG will be "
|
||||||
@ -536,7 +554,7 @@ if SENTRY_ENABLED:
|
|||||||
release=VERSION,
|
release=VERSION,
|
||||||
sample_rate=SENTRY_SAMPLE_RATE,
|
sample_rate=SENTRY_SAMPLE_RATE,
|
||||||
traces_sample_rate=SENTRY_TRACES_SAMPLE_RATE,
|
traces_sample_rate=SENTRY_TRACES_SAMPLE_RATE,
|
||||||
send_default_pii=True,
|
send_default_pii=SENTRY_SEND_DEFAULT_PII,
|
||||||
http_proxy=HTTP_PROXIES.get('http') if HTTP_PROXIES else None,
|
http_proxy=HTTP_PROXIES.get('http') if HTTP_PROXIES else None,
|
||||||
https_proxy=HTTP_PROXIES.get('https') if HTTP_PROXIES else None
|
https_proxy=HTTP_PROXIES.get('https') if HTTP_PROXIES else None
|
||||||
)
|
)
|
||||||
@ -551,7 +569,7 @@ if SENTRY_ENABLED:
|
|||||||
|
|
||||||
# Calculate a unique deployment ID from the secret key
|
# Calculate a unique deployment ID from the secret key
|
||||||
DEPLOYMENT_ID = hashlib.sha256(SECRET_KEY.encode('utf-8')).hexdigest()[:16]
|
DEPLOYMENT_ID = hashlib.sha256(SECRET_KEY.encode('utf-8')).hexdigest()[:16]
|
||||||
CENSUS_URL = 'https://census.netbox.dev/api/v1/'
|
CENSUS_URL = 'https://census.netbox.oss.netboxlabs.com/api/v1/'
|
||||||
CENSUS_PARAMS = {
|
CENSUS_PARAMS = {
|
||||||
'version': VERSION,
|
'version': VERSION,
|
||||||
'python_version': sys.version.split()[0],
|
'python_version': sys.version.split()[0],
|
||||||
|
@ -249,7 +249,7 @@ class ActionsColumn(tables.Column):
|
|||||||
|
|
||||||
def render(self, record, table, **kwargs):
|
def render(self, record, table, **kwargs):
|
||||||
# Skip dummy records (e.g. available VLANs) or those with no actions
|
# Skip dummy records (e.g. available VLANs) or those with no actions
|
||||||
if not getattr(record, 'pk', None) or not self.actions:
|
if not getattr(record, 'pk', None) or not (self.actions or self.extra_buttons):
|
||||||
return ''
|
return ''
|
||||||
|
|
||||||
model = table.Meta.model
|
model = table.Meta.model
|
||||||
@ -433,7 +433,7 @@ class LinkedCountColumn(tables.Column):
|
|||||||
f'{k}={getattr(record, v) or settings.FILTERS_NULL_CHOICE_VALUE}'
|
f'{k}={getattr(record, v) or settings.FILTERS_NULL_CHOICE_VALUE}'
|
||||||
for k, v in self.url_params.items()
|
for k, v in self.url_params.items()
|
||||||
])
|
])
|
||||||
return mark_safe(f'<a href="{url}">{value}</a>')
|
return mark_safe(f'<a href="{url}">{escape(value)}</a>')
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def value(self, value):
|
def value(self, value):
|
||||||
|
@ -176,7 +176,7 @@ class ObjectListView(BaseMultiObjectView, ActionsMixin, TableMixin):
|
|||||||
'model': model,
|
'model': model,
|
||||||
'table': table,
|
'table': table,
|
||||||
'actions': actions,
|
'actions': actions,
|
||||||
'filter_form': self.filterset_form(request.GET, label_suffix='') if self.filterset_form else None,
|
'filter_form': self.filterset_form(request.GET) if self.filterset_form else None,
|
||||||
'prerequisite_model': get_prerequisite_model(self.queryset),
|
'prerequisite_model': get_prerequisite_model(self.queryset),
|
||||||
**self.get_extra_context(request),
|
**self.get_extra_context(request),
|
||||||
}
|
}
|
||||||
|
@ -87,12 +87,14 @@ class ObjectChildrenView(ObjectView, ActionsMixin, TableMixin):
|
|||||||
child_model: The model class which represents the child objects
|
child_model: The model class which represents the child objects
|
||||||
table: The django-tables2 Table class used to render the child objects list
|
table: The django-tables2 Table class used to render the child objects list
|
||||||
filterset: A django-filter FilterSet that is applied to the queryset
|
filterset: A django-filter FilterSet that is applied to the queryset
|
||||||
|
filterset_form: The form class used to render filter options
|
||||||
actions: A mapping of supported actions to their required permissions. When adding custom actions, bulk
|
actions: A mapping of supported actions to their required permissions. When adding custom actions, bulk
|
||||||
action names must be prefixed with `bulk_`. (See ActionsMixin.)
|
action names must be prefixed with `bulk_`. (See ActionsMixin.)
|
||||||
"""
|
"""
|
||||||
child_model = None
|
child_model = None
|
||||||
table = None
|
table = None
|
||||||
filterset = None
|
filterset = None
|
||||||
|
filterset_form = None
|
||||||
template_name = 'generic/object_children.html'
|
template_name = 'generic/object_children.html'
|
||||||
|
|
||||||
def get_children(self, request, parent):
|
def get_children(self, request, parent):
|
||||||
@ -152,6 +154,7 @@ class ObjectChildrenView(ObjectView, ActionsMixin, TableMixin):
|
|||||||
'base_template': f'{instance._meta.app_label}/{instance._meta.model_name}.html',
|
'base_template': f'{instance._meta.app_label}/{instance._meta.model_name}.html',
|
||||||
'table': table,
|
'table': table,
|
||||||
'table_config': f'{table.name}_config',
|
'table_config': f'{table.name}_config',
|
||||||
|
'filter_form': self.filterset_form(request.GET) if self.filterset_form else None,
|
||||||
'actions': actions,
|
'actions': actions,
|
||||||
'tab': self.tab,
|
'tab': self.tab,
|
||||||
'return_url': request.get_full_path(),
|
'return_url': request.get_full_path(),
|
||||||
|
BIN
netbox/project-static/dist/netbox.js
vendored
BIN
netbox/project-static/dist/netbox.js
vendored
Binary file not shown.
BIN
netbox/project-static/dist/netbox.js.map
vendored
BIN
netbox/project-static/dist/netbox.js.map
vendored
Binary file not shown.
@ -27,10 +27,10 @@
|
|||||||
"bootstrap": "5.3.3",
|
"bootstrap": "5.3.3",
|
||||||
"clipboard": "2.0.11",
|
"clipboard": "2.0.11",
|
||||||
"flatpickr": "4.6.13",
|
"flatpickr": "4.6.13",
|
||||||
"gridstack": "10.2.0",
|
"gridstack": "10.3.0",
|
||||||
"htmx.org": "1.9.12",
|
"htmx.org": "1.9.12",
|
||||||
"query-string": "9.0.0",
|
"query-string": "9.0.0",
|
||||||
"sass": "1.77.4",
|
"sass": "1.77.6",
|
||||||
"tom-select": "2.3.1",
|
"tom-select": "2.3.1",
|
||||||
"typeface-inter": "3.18.1",
|
"typeface-inter": "3.18.1",
|
||||||
"typeface-roboto-mono": "1.1.13"
|
"typeface-roboto-mono": "1.1.13"
|
||||||
|
@ -74,20 +74,25 @@ export class DynamicTomSelect extends TomSelect {
|
|||||||
|
|
||||||
load(value: string) {
|
load(value: string) {
|
||||||
const self = this;
|
const self = this;
|
||||||
const url = self.getRequestUrl(value);
|
|
||||||
|
|
||||||
// Automatically clear any cached options. (Only options included
|
// Automatically clear any cached options. (Only options included
|
||||||
// in the API response should be present.)
|
// in the API response should be present.)
|
||||||
self.clearOptions();
|
self.clearOptions();
|
||||||
|
|
||||||
addClasses(self.wrapper, self.settings.loadingClass);
|
|
||||||
self.loading++;
|
|
||||||
|
|
||||||
// Populate the null option (if any) if not searching
|
// Populate the null option (if any) if not searching
|
||||||
if (self.nullOption && !value) {
|
if (self.nullOption && !value) {
|
||||||
self.addOption(self.nullOption);
|
self.addOption(self.nullOption);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Get the API request URL. If none is provided, abort as no request can be made.
|
||||||
|
const url = self.getRequestUrl(value);
|
||||||
|
if (!url) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
addClasses(self.wrapper, self.settings.loadingClass);
|
||||||
|
self.loading++;
|
||||||
|
|
||||||
// Make the API request
|
// Make the API request
|
||||||
fetch(url)
|
fetch(url)
|
||||||
.then(response => response.json())
|
.then(response => response.json())
|
||||||
@ -129,6 +134,9 @@ export class DynamicTomSelect extends TomSelect {
|
|||||||
for (const result of this.api_url.matchAll(new RegExp(`({{${key}}})`, 'g'))) {
|
for (const result of this.api_url.matchAll(new RegExp(`({{${key}}})`, 'g'))) {
|
||||||
if (value) {
|
if (value) {
|
||||||
url = replaceAll(url, result[1], value.toString());
|
url = replaceAll(url, result[1], value.toString());
|
||||||
|
} else {
|
||||||
|
// No value is available to replace the token; abort.
|
||||||
|
return '';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1754,10 +1754,10 @@ graphql@16.8.1:
|
|||||||
resolved "https://registry.yarnpkg.com/graphql/-/graphql-16.8.1.tgz#1930a965bef1170603702acdb68aedd3f3cf6f07"
|
resolved "https://registry.yarnpkg.com/graphql/-/graphql-16.8.1.tgz#1930a965bef1170603702acdb68aedd3f3cf6f07"
|
||||||
integrity sha512-59LZHPdGZVh695Ud9lRzPBVTtlX9ZCV150Er2W43ro37wVof0ctenSaskPPjN7lVTIN8mSZt8PHUNKZuNQUuxw==
|
integrity sha512-59LZHPdGZVh695Ud9lRzPBVTtlX9ZCV150Er2W43ro37wVof0ctenSaskPPjN7lVTIN8mSZt8PHUNKZuNQUuxw==
|
||||||
|
|
||||||
gridstack@10.2.0:
|
gridstack@10.3.0:
|
||||||
version "10.2.0"
|
version "10.3.0"
|
||||||
resolved "https://registry.yarnpkg.com/gridstack/-/gridstack-10.2.0.tgz#4ba9c7ee69a730851721a9f5cb33dc55026ded1f"
|
resolved "https://registry.yarnpkg.com/gridstack/-/gridstack-10.3.0.tgz#8fa065f896d0a880c5c54c24d189f3197184488a"
|
||||||
integrity sha512-svKAOq/dfinpvhe/nnxdyZOOEd9qynXiOPHvL96PALE0yWChWp/6lechnqKwud0tL/rRyAfMJ6Hh/z2fS13pBA==
|
integrity sha512-eGKsmU2TppV4coyDu9IIdIkm4qjgLLdjlEOFwQyQMuSwfOpzSfLdPc8du0HuebGr7CvAIrJxN4lBOmGrWSBg9g==
|
||||||
|
|
||||||
has-bigints@^1.0.1, has-bigints@^1.0.2:
|
has-bigints@^1.0.1, has-bigints@^1.0.2:
|
||||||
version "1.0.2"
|
version "1.0.2"
|
||||||
@ -2482,10 +2482,10 @@ safe-regex-test@^1.0.3:
|
|||||||
es-errors "^1.3.0"
|
es-errors "^1.3.0"
|
||||||
is-regex "^1.1.4"
|
is-regex "^1.1.4"
|
||||||
|
|
||||||
sass@1.77.4:
|
sass@1.77.6:
|
||||||
version "1.77.4"
|
version "1.77.6"
|
||||||
resolved "https://registry.yarnpkg.com/sass/-/sass-1.77.4.tgz#92059c7bfc56b827c56eb116778d157ec017a5cd"
|
resolved "https://registry.yarnpkg.com/sass/-/sass-1.77.6.tgz#898845c1348078c2e6d1b64f9ee06b3f8bd489e4"
|
||||||
integrity sha512-vcF3Ckow6g939GMA4PeU7b2K/9FALXk2KF9J87txdHzXbUF9XRQRwSxcAs/fGaTnJeBFd7UoV22j3lzMLdM0Pw==
|
integrity sha512-ByXE1oLD79GVq9Ht1PeHWCPMPB8XHpBuz1r85oByKHjZY6qV6rWnQovQzXJXuQ/XyE1Oj3iPk3lo28uzaRA2/Q==
|
||||||
dependencies:
|
dependencies:
|
||||||
chokidar ">=3.0.0 <4.0.0"
|
chokidar ">=3.0.0 <4.0.0"
|
||||||
immutable "^4.0.0"
|
immutable "^4.0.0"
|
||||||
|
@ -88,7 +88,7 @@
|
|||||||
<div class="col col-md-12">
|
<div class="col col-md-12">
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<h5 class="card-header">{% trans "Current Configuration" %}</h5>
|
<h5 class="card-header">{% trans "Current Configuration" %}</h5>
|
||||||
{% include 'core/inc/config_data.html' with config=config.data %}
|
{% include 'core/inc/config_data.html' %}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
@ -125,28 +125,30 @@
|
|||||||
</div>
|
</div>
|
||||||
</h5>
|
</h5>
|
||||||
<table class="table table-hover attr-table">
|
<table class="table table-hover attr-table">
|
||||||
<tr>
|
<thead>
|
||||||
|
<tr class="border-bottom">
|
||||||
<th>{% trans "Device" %}</th>
|
<th>{% trans "Device" %}</th>
|
||||||
<th>{% trans "Position" %}</th>
|
<th>{% trans "Position" %}</th>
|
||||||
<th>{% trans "Master" %}</th>
|
<th>{% trans "Master" %}</th>
|
||||||
<th>{% trans "Priority" %}</th>
|
<th>{% trans "Priority" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
{% for vc_member in vc_members %}
|
{% for vc_member in vc_members %}
|
||||||
<tr{% if vc_member == object %} class="info"{% endif %}>
|
<tr{% if vc_member == object %} class="table-primary"{% endif %}>
|
||||||
|
<td>{{ vc_member|linkify }}</td>
|
||||||
|
<td>{% badge vc_member.vc_position show_empty=True %}</td>
|
||||||
<td>
|
<td>
|
||||||
{{ vc_member|linkify }}
|
{% if object.virtual_chassis.master == vc_member %}
|
||||||
</td>
|
{% checkmark True %}
|
||||||
<td>
|
{% else %}
|
||||||
{% badge vc_member.vc_position show_empty=True %}
|
{{ ''|placeholder }}
|
||||||
</td>
|
{% endif %}
|
||||||
<td>
|
|
||||||
{% if object.virtual_chassis.master == vc_member %}<i class="mdi mdi-check-bold"></i>{% endif %}
|
|
||||||
</td>
|
|
||||||
<td>
|
|
||||||
{{ vc_member.vc_priority|placeholder }}
|
|
||||||
</td>
|
</td>
|
||||||
|
<td>{{ vc_member.vc_priority|placeholder }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -221,6 +223,11 @@
|
|||||||
<td>
|
<td>
|
||||||
{% if object.oob_ip %}
|
{% if object.oob_ip %}
|
||||||
<a href="{{ object.oob_ip.get_absolute_url }}" id="oob_ip">{{ object.oob_ip.address.ip }}</a>
|
<a href="{{ object.oob_ip.get_absolute_url }}" id="oob_ip">{{ object.oob_ip.address.ip }}</a>
|
||||||
|
{% if object.oob_ip.nat_inside %}
|
||||||
|
({% trans "NAT for" %} <a href="{{ object.oob_ip.nat_inside.get_absolute_url }}">{{ object.oob_ip.nat_inside.address.ip }}</a>)
|
||||||
|
{% elif object.oob_ip.nat_outside.exists %}
|
||||||
|
({% trans "NAT" %}: {% for nat in object.oob_ip.nat_outside.all %}<a href="{{ nat.get_absolute_url }}">{{ nat.address.ip }}</a>{% if not forloop.last %}, {% endif %}{% endfor %})
|
||||||
|
{% endif %}
|
||||||
{% copy_content "oob_ip" %}
|
{% copy_content "oob_ip" %}
|
||||||
{% else %}
|
{% else %}
|
||||||
{{ ''|placeholder }}
|
{{ ''|placeholder }}
|
||||||
|
@ -44,7 +44,7 @@
|
|||||||
{# Object table controls #}
|
{# Object table controls #}
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<div class="col-auto ms-auto d-print-none">
|
<div class="col-auto ms-auto d-print-none">
|
||||||
{% if request.user.is_authenticated %}
|
{% if request.user.is_authenticated and job.completed %}
|
||||||
<div class="table-configure input-group">
|
<div class="table-configure input-group">
|
||||||
<button type="button" data-bs-toggle="modal" title="{% trans "Configure Table" %}" data-bs-target="#ObjectTable_config"
|
<button type="button" data-bs-toggle="modal" title="{% trans "Configure Table" %}" data-bs-target="#ObjectTable_config"
|
||||||
class="btn">
|
class="btn">
|
||||||
|
@ -48,7 +48,7 @@ Context:
|
|||||||
<li class="nav-item" role="presentation">
|
<li class="nav-item" role="presentation">
|
||||||
<a class="nav-link active" id="object-list-tab" data-bs-toggle="tab" data-bs-target="#object-list" type="button" role="tab" aria-controls="edit-form" aria-selected="true">
|
<a class="nav-link active" id="object-list-tab" data-bs-toggle="tab" data-bs-target="#object-list" type="button" role="tab" aria-controls="edit-form" aria-selected="true">
|
||||||
{% trans "Results" %}
|
{% trans "Results" %}
|
||||||
<span class="badge text-bg-secondary total-object-count">{{ table.page.paginator.count }}</span>
|
<span class="badge text-bg-secondary total-object-count">{% if table.page.paginator.count %}{{ table.page.paginator.count }}{% else %}{{ total_count|default:"0" }}{% endif %}</span>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
{% if filter_form %}
|
{% if filter_form %}
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{% if filter_form %}
|
||||||
<div class="col-auto d-print-none">
|
<div class="col-auto d-print-none">
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<div class="input-group-text">
|
<div class="input-group-text">
|
||||||
@ -21,6 +22,7 @@
|
|||||||
{{ filter_form.filter_id }}
|
{{ filter_form.filter_id }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
<div class="col-auto ms-auto d-print-none">
|
<div class="col-auto ms-auto d-print-none">
|
||||||
{% if request.user.is_authenticated and table_modal %}
|
{% if request.user.is_authenticated and table_modal %}
|
||||||
|
@ -53,10 +53,6 @@
|
|||||||
{% endwith %}
|
{% endwith %}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="field-group my-5">
|
|
||||||
{% render_field form.comments %}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{% if form.custom_fields %}
|
{% if form.custom_fields %}
|
||||||
<div class="field-group my-5">
|
<div class="field-group my-5">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
@ -65,4 +61,8 @@
|
|||||||
{% render_custom_fields form %}
|
{% render_custom_fields form %}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="field-group my-5">
|
||||||
|
{% render_field form.comments %}
|
||||||
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -78,7 +78,8 @@
|
|||||||
{% for backend in auth_backends %}
|
{% for backend in auth_backends %}
|
||||||
<div class="col">
|
<div class="col">
|
||||||
<a href="{{ backend.url }}" class="btn w-100">
|
<a href="{{ backend.url }}" class="btn w-100">
|
||||||
{% if backend.icon_name %}<i class="mdi mdi-{{ backend.icon_name }}"></i>{% endif %}
|
{% if backend.icon_name %}<i class="mdi mdi-{{ backend.icon_name }}"></i>
|
||||||
|
{% elif backend.icon_img %}<img src="{{ backend.icon_img }}" height="24" class="me-2" />{% endif %}
|
||||||
{{ backend.display_name }}
|
{{ backend.display_name }}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
@ -13,6 +13,7 @@ class ObjectContactsView(generic.ObjectChildrenView):
|
|||||||
child_model = ContactAssignment
|
child_model = ContactAssignment
|
||||||
table = tables.ContactAssignmentTable
|
table = tables.ContactAssignmentTable
|
||||||
filterset = filtersets.ContactAssignmentFilterSet
|
filterset = filtersets.ContactAssignmentFilterSet
|
||||||
|
filterset_form = forms.ContactAssignmentFilterForm
|
||||||
template_name = 'tenancy/object_contacts.html'
|
template_name = 'tenancy/object_contacts.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Contacts'),
|
label=_('Contacts'),
|
||||||
@ -364,7 +365,7 @@ class ContactAssignmentEditView(generic.ObjectEditView):
|
|||||||
|
|
||||||
def get_extra_addanother_params(self, request):
|
def get_extra_addanother_params(self, request):
|
||||||
return {
|
return {
|
||||||
'content_type': request.GET.get('content_type'),
|
'object_type': request.GET.get('object_type'),
|
||||||
'object_id': request.GET.get('object_id'),
|
'object_id': request.GET.get('object_id'),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binary file not shown.
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
Binary file not shown.
File diff suppressed because it is too large
Load Diff
@ -39,7 +39,7 @@ def handle_protectederror(obj_list, request, e):
|
|||||||
if hasattr(dependent, 'get_absolute_url'):
|
if hasattr(dependent, 'get_absolute_url'):
|
||||||
dependent_objects.append(f'<a href="{dependent.get_absolute_url()}">{escape(dependent)}</a>')
|
dependent_objects.append(f'<a href="{dependent.get_absolute_url()}">{escape(dependent)}</a>')
|
||||||
else:
|
else:
|
||||||
dependent_objects.append(str(dependent))
|
dependent_objects.append(escape(str(dependent)))
|
||||||
err_message += ', '.join(dependent_objects)
|
err_message += ', '.join(dependent_objects)
|
||||||
|
|
||||||
messages.error(request, mark_safe(err_message))
|
messages.error(request, mark_safe(err_message))
|
||||||
|
@ -2,6 +2,7 @@ from collections import defaultdict
|
|||||||
|
|
||||||
from django.contrib.contenttypes.fields import GenericForeignKey
|
from django.contrib.contenttypes.fields import GenericForeignKey
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
from django.utils.safestring import mark_safe
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from utilities.ordering import naturalize
|
from utilities.ordering import naturalize
|
||||||
@ -26,6 +27,7 @@ class ColorField(models.CharField):
|
|||||||
|
|
||||||
def formfield(self, **kwargs):
|
def formfield(self, **kwargs):
|
||||||
kwargs['widget'] = ColorSelect
|
kwargs['widget'] = ColorSelect
|
||||||
|
kwargs['help_text'] = mark_safe(_('RGB color in hexadecimal. Example: ') + '<code>00ff00</code>')
|
||||||
return super().formfield(**kwargs)
|
return super().formfield(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
@ -55,7 +55,7 @@ def prepare_cloned_fields(instance):
|
|||||||
for key, value in attrs.items():
|
for key, value in attrs.items():
|
||||||
if type(value) in (list, tuple):
|
if type(value) in (list, tuple):
|
||||||
params.extend([(key, v) for v in value])
|
params.extend([(key, v) for v in value])
|
||||||
elif value not in (False, None):
|
elif value is not False and value is not None:
|
||||||
params.append((key, value))
|
params.append((key, value))
|
||||||
else:
|
else:
|
||||||
params.append((key, ''))
|
params.append((key, ''))
|
||||||
|
@ -1,7 +1,23 @@
|
|||||||
{% load helpers %}
|
{% load helpers %}
|
||||||
|
{% load i18n %}
|
||||||
{% load navigation %}
|
{% load navigation %}
|
||||||
|
|
||||||
<ul class="navbar-nav pt-lg-2" {% htmx_boost %}>
|
<ul class="navbar-nav pt-lg-2" {% htmx_boost %}>
|
||||||
|
<li class="nav-item d-block d-lg-none">
|
||||||
|
<form action="{% url 'search' %}" method="get" autocomplete="off" novalidate>
|
||||||
|
<div class="input-group mb-1 mt-2">
|
||||||
|
<div class="input-group-prepend">
|
||||||
|
<span class="input-group-text">
|
||||||
|
<i class="mdi mdi-magnify"></i>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<input type="text" name="q" value="" class="form-control" placeholder="{% trans "Search…" %}" aria-label="{% trans "Search NetBox" %}">
|
||||||
|
<div class="input-group-append">
|
||||||
|
<button type="submit" class="form-control">{% trans "Search" %}</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</li>
|
||||||
{% for menu, groups in nav_items %}
|
{% for menu, groups in nav_items %}
|
||||||
<li class="nav-item dropdown">
|
<li class="nav-item dropdown">
|
||||||
|
|
||||||
|
@ -58,7 +58,7 @@ def linkify(instance, attr=None):
|
|||||||
url = instance.get_absolute_url()
|
url = instance.get_absolute_url()
|
||||||
return mark_safe(f'<a href="{url}">{escape(text)}</a>')
|
return mark_safe(f'<a href="{url}">{escape(text)}</a>')
|
||||||
except (AttributeError, TypeError):
|
except (AttributeError, TypeError):
|
||||||
return text
|
return escape(text)
|
||||||
|
|
||||||
|
|
||||||
@register.filter()
|
@register.filter()
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
from django import template
|
from django import template
|
||||||
|
from django.utils.html import escape
|
||||||
from django.utils.safestring import mark_safe
|
from django.utils.safestring import mark_safe
|
||||||
|
|
||||||
register = template.Library()
|
register = template.Library()
|
||||||
@ -15,6 +16,6 @@ def nested_tree(obj):
|
|||||||
nodes = obj.get_ancestors(include_self=True)
|
nodes = obj.get_ancestors(include_self=True)
|
||||||
return mark_safe(
|
return mark_safe(
|
||||||
' / '.join(
|
' / '.join(
|
||||||
f'<a href="{node.get_absolute_url()}">{node}</a>' for node in nodes
|
f'<a href="{node.get_absolute_url()}">{escape(node)}</a>' for node in nodes
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
@ -4,6 +4,7 @@ from django.contrib.auth.mixins import AccessMixin
|
|||||||
from django.core.exceptions import ImproperlyConfigured
|
from django.core.exceptions import ImproperlyConfigured
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.urls.exceptions import NoReverseMatch
|
from django.urls.exceptions import NoReverseMatch
|
||||||
|
from django.utils.http import url_has_allowed_host_and_scheme
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from netbox.plugins import PluginConfig
|
from netbox.plugins import PluginConfig
|
||||||
@ -123,7 +124,7 @@ class GetReturnURLMixin:
|
|||||||
# First, see if `return_url` was specified as a query parameter or form data. Use this URL only if it's
|
# First, see if `return_url` was specified as a query parameter or form data. Use this URL only if it's
|
||||||
# considered safe.
|
# considered safe.
|
||||||
return_url = request.GET.get('return_url') or request.POST.get('return_url')
|
return_url = request.GET.get('return_url') or request.POST.get('return_url')
|
||||||
if return_url and return_url.startswith('/'):
|
if return_url and url_has_allowed_host_and_scheme(return_url, allowed_hosts=None):
|
||||||
return return_url
|
return return_url
|
||||||
|
|
||||||
# Next, check if the object being modified (if any) has an absolute URL.
|
# Next, check if the object being modified (if any) has an absolute URL.
|
||||||
|
@ -179,8 +179,8 @@ class VirtualMachine(ContactsMixin, ImageAttachmentsMixin, RenderConfigMixin, Co
|
|||||||
'cluster': _('A virtual machine must be assigned to a site and/or cluster.')
|
'cluster': _('A virtual machine must be assigned to a site and/or cluster.')
|
||||||
})
|
})
|
||||||
|
|
||||||
# Validate site for cluster & device
|
# Validate site for cluster & VM
|
||||||
if self.cluster and self.cluster.site is not None and self.cluster.site != self.site:
|
if self.cluster and self.site and self.cluster.site and self.cluster.site != self.site:
|
||||||
raise ValidationError({
|
raise ValidationError({
|
||||||
'cluster': _(
|
'cluster': _(
|
||||||
'The selected cluster ({cluster}) is not assigned to this site ({site}).'
|
'The selected cluster ({cluster}) is not assigned to this site ({site}).'
|
||||||
|
@ -10,6 +10,7 @@ from django.utils.translation import gettext as _
|
|||||||
from jinja2.exceptions import TemplateError
|
from jinja2.exceptions import TemplateError
|
||||||
|
|
||||||
from dcim.filtersets import DeviceFilterSet
|
from dcim.filtersets import DeviceFilterSet
|
||||||
|
from dcim.forms import DeviceFilterForm
|
||||||
from dcim.models import Device
|
from dcim.models import Device
|
||||||
from dcim.tables import DeviceTable
|
from dcim.tables import DeviceTable
|
||||||
from extras.views import ObjectConfigContextView
|
from extras.views import ObjectConfigContextView
|
||||||
@ -173,6 +174,7 @@ class ClusterVirtualMachinesView(generic.ObjectChildrenView):
|
|||||||
child_model = VirtualMachine
|
child_model = VirtualMachine
|
||||||
table = tables.VirtualMachineTable
|
table = tables.VirtualMachineTable
|
||||||
filterset = filtersets.VirtualMachineFilterSet
|
filterset = filtersets.VirtualMachineFilterSet
|
||||||
|
filterset_form = forms.VirtualMachineFilterForm
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Virtual Machines'),
|
label=_('Virtual Machines'),
|
||||||
badge=lambda obj: obj.virtual_machines.count(),
|
badge=lambda obj: obj.virtual_machines.count(),
|
||||||
@ -190,6 +192,7 @@ class ClusterDevicesView(generic.ObjectChildrenView):
|
|||||||
child_model = Device
|
child_model = Device
|
||||||
table = DeviceTable
|
table = DeviceTable
|
||||||
filterset = DeviceFilterSet
|
filterset = DeviceFilterSet
|
||||||
|
filterset_form = DeviceFilterForm
|
||||||
template_name = 'virtualization/cluster/devices.html'
|
template_name = 'virtualization/cluster/devices.html'
|
||||||
actions = {
|
actions = {
|
||||||
'add': {'add'},
|
'add': {'add'},
|
||||||
@ -350,6 +353,7 @@ class VirtualMachineInterfacesView(generic.ObjectChildrenView):
|
|||||||
child_model = VMInterface
|
child_model = VMInterface
|
||||||
table = tables.VirtualMachineVMInterfaceTable
|
table = tables.VirtualMachineVMInterfaceTable
|
||||||
filterset = filtersets.VMInterfaceFilterSet
|
filterset = filtersets.VMInterfaceFilterSet
|
||||||
|
filterset_form = forms.VMInterfaceFilterForm
|
||||||
template_name = 'virtualization/virtualmachine/interfaces.html'
|
template_name = 'virtualization/virtualmachine/interfaces.html'
|
||||||
actions = {
|
actions = {
|
||||||
**DEFAULT_ACTION_PERMISSIONS,
|
**DEFAULT_ACTION_PERMISSIONS,
|
||||||
@ -375,6 +379,7 @@ class VirtualMachineVirtualDisksView(generic.ObjectChildrenView):
|
|||||||
child_model = VirtualDisk
|
child_model = VirtualDisk
|
||||||
table = tables.VirtualMachineVirtualDiskTable
|
table = tables.VirtualMachineVirtualDiskTable
|
||||||
filterset = filtersets.VirtualDiskFilterSet
|
filterset = filtersets.VirtualDiskFilterSet
|
||||||
|
filterset_form = forms.VirtualDiskFilterForm
|
||||||
template_name = 'virtualization/virtualmachine/virtual_disks.html'
|
template_name = 'virtualization/virtualmachine/virtual_disks.html'
|
||||||
tab = ViewTab(
|
tab = ViewTab(
|
||||||
label=_('Virtual Disks'),
|
label=_('Virtual Disks'),
|
||||||
|
@ -25,7 +25,8 @@ class IKEProposalSerializer(NetBoxModelSerializer):
|
|||||||
choices=EncryptionAlgorithmChoices
|
choices=EncryptionAlgorithmChoices
|
||||||
)
|
)
|
||||||
authentication_algorithm = ChoiceField(
|
authentication_algorithm = ChoiceField(
|
||||||
choices=AuthenticationAlgorithmChoices
|
choices=AuthenticationAlgorithmChoices,
|
||||||
|
required=False
|
||||||
)
|
)
|
||||||
group = ChoiceField(
|
group = ChoiceField(
|
||||||
choices=DHGroupChoices
|
choices=DHGroupChoices
|
||||||
@ -49,7 +50,8 @@ class IKEPolicySerializer(NetBoxModelSerializer):
|
|||||||
choices=IKEVersionChoices
|
choices=IKEVersionChoices
|
||||||
)
|
)
|
||||||
mode = ChoiceField(
|
mode = ChoiceField(
|
||||||
choices=IKEModeChoices
|
choices=IKEModeChoices,
|
||||||
|
required=False
|
||||||
)
|
)
|
||||||
proposals = SerializedPKRelatedField(
|
proposals = SerializedPKRelatedField(
|
||||||
queryset=IKEProposal.objects.all(),
|
queryset=IKEProposal.objects.all(),
|
||||||
|
@ -1,37 +1,37 @@
|
|||||||
Django==5.0.6
|
Django==5.0.7
|
||||||
django-cors-headers==4.3.1
|
django-cors-headers==4.4.0
|
||||||
django-debug-toolbar==4.3.0
|
django-debug-toolbar==4.3.0
|
||||||
django-filter==24.2
|
django-filter==24.2
|
||||||
django-htmx==1.17.3
|
django-htmx==1.18.0
|
||||||
django-graphiql-debug-toolbar==0.2.0
|
django-graphiql-debug-toolbar==0.2.0
|
||||||
django-mptt==0.16.0
|
django-mptt==0.16.0
|
||||||
django-pglocks==1.0.4
|
django-pglocks==1.0.4
|
||||||
django-prometheus==2.3.1
|
django-prometheus==2.3.1
|
||||||
django-redis==5.4.0
|
django-redis==5.4.0
|
||||||
django-rich==1.8.0
|
django-rich==1.9.0
|
||||||
django-rq==2.10.2
|
django-rq==2.10.2
|
||||||
django-taggit==5.0.1
|
django-taggit==5.0.1
|
||||||
django-tables2==2.7.0
|
django-tables2==2.7.0
|
||||||
django-timezone-field==6.1.0
|
django-timezone-field==7.0
|
||||||
djangorestframework==3.15.1
|
djangorestframework==3.15.2
|
||||||
drf-spectacular==0.27.2
|
drf-spectacular==0.27.2
|
||||||
drf-spectacular-sidecar==2024.6.1
|
drf-spectacular-sidecar==2024.7.1
|
||||||
feedparser==6.0.11
|
feedparser==6.0.11
|
||||||
gunicorn==22.0.0
|
gunicorn==22.0.0
|
||||||
Jinja2==3.1.4
|
Jinja2==3.1.4
|
||||||
Markdown==3.6
|
Markdown==3.6
|
||||||
mkdocs-material==9.5.26
|
mkdocs-material==9.5.28
|
||||||
mkdocstrings[python-legacy]==0.25.1
|
mkdocstrings[python-legacy]==0.25.1
|
||||||
netaddr==1.3.0
|
netaddr==1.3.0
|
||||||
nh3==0.2.17
|
nh3==0.2.18
|
||||||
Pillow==10.3.0
|
Pillow==10.4.0
|
||||||
psycopg[c,pool]==3.1.19
|
psycopg[c,pool]==3.2.1
|
||||||
PyYAML==6.0.1
|
PyYAML==6.0.1
|
||||||
requests==2.32.3
|
requests==2.32.3
|
||||||
social-auth-app-django==5.4.1
|
social-auth-app-django==5.4.1
|
||||||
social-auth-core==4.5.4
|
social-auth-core==4.5.4
|
||||||
strawberry-graphql==0.234.0
|
strawberry-graphql==0.235.2
|
||||||
strawberry-graphql-django==0.42.0
|
strawberry-graphql-django==0.46.1
|
||||||
svgwrite==1.4.3
|
svgwrite==1.4.3
|
||||||
tablib==3.6.1
|
tablib==3.6.1
|
||||||
tzdata==2024.1
|
tzdata==2024.1
|
||||||
|
@ -33,7 +33,7 @@ echo "Using ${PYTHON_VERSION}"
|
|||||||
|
|
||||||
# Remove the existing virtual environment (if any)
|
# Remove the existing virtual environment (if any)
|
||||||
if [ -d "$VIRTUALENV" ]; then
|
if [ -d "$VIRTUALENV" ]; then
|
||||||
COMMAND="rm -rf ${VIRTUALENV}"
|
COMMAND="rm -rf \"${VIRTUALENV}\""
|
||||||
echo "Removing old virtual environment..."
|
echo "Removing old virtual environment..."
|
||||||
eval $COMMAND
|
eval $COMMAND
|
||||||
else
|
else
|
||||||
@ -41,7 +41,7 @@ else
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
# Create a new virtual environment
|
# Create a new virtual environment
|
||||||
COMMAND="${PYTHON} -m venv ${VIRTUALENV}"
|
COMMAND="${PYTHON} -m venv \"${VIRTUALENV}\""
|
||||||
echo "Creating a new virtual environment at ${VIRTUALENV}..."
|
echo "Creating a new virtual environment at ${VIRTUALENV}..."
|
||||||
eval $COMMAND || {
|
eval $COMMAND || {
|
||||||
echo "--------------------------------------------------------------------"
|
echo "--------------------------------------------------------------------"
|
||||||
|
Loading…
Reference in New Issue
Block a user