mirror of
https://github.com/netbox-community/netbox.git
synced 2025-12-15 04:49:36 -06:00
Merge branch 'main' into feature
Some checks failed
CI / build (20.x, 3.12) (push) Has been cancelled
CI / build (20.x, 3.13) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, actions) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, javascript-typescript) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, python) (push) Has been cancelled
Some checks failed
CI / build (20.x, 3.12) (push) Has been cancelled
CI / build (20.x, 3.13) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, actions) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, javascript-typescript) (push) Has been cancelled
CodeQL / Analyze (${{ matrix.language }}) (none, python) (push) Has been cancelled
This commit is contained in:
commit
068d493cc6
@ -2,7 +2,7 @@
|
|||||||
name: ✨ Feature Request
|
name: ✨ Feature Request
|
||||||
type: Feature
|
type: Feature
|
||||||
description: Propose a new NetBox feature or enhancement
|
description: Propose a new NetBox feature or enhancement
|
||||||
labels: ["type: feature", "status: needs triage"]
|
labels: ["netbox", "type: feature", "status: needs triage"]
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
@ -15,7 +15,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.4.4
|
placeholder: v4.4.5
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
|
|||||||
4
.github/ISSUE_TEMPLATE/02-bug_report.yaml
vendored
4
.github/ISSUE_TEMPLATE/02-bug_report.yaml
vendored
@ -2,7 +2,7 @@
|
|||||||
name: 🐛 Bug Report
|
name: 🐛 Bug Report
|
||||||
type: Bug
|
type: Bug
|
||||||
description: Report a reproducible bug in the current release of NetBox
|
description: Report a reproducible bug in the current release of NetBox
|
||||||
labels: ["type: bug", "status: needs triage"]
|
labels: ["netbox", "type: bug", "status: needs triage"]
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
@ -27,7 +27,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.4.4
|
placeholder: v4.4.5
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
|
|||||||
@ -2,7 +2,7 @@
|
|||||||
name: 📖 Documentation Change
|
name: 📖 Documentation Change
|
||||||
type: Documentation
|
type: Documentation
|
||||||
description: Suggest an addition or modification to the NetBox documentation
|
description: Suggest an addition or modification to the NetBox documentation
|
||||||
labels: ["type: documentation", "status: needs triage"]
|
labels: ["netbox", "type: documentation", "status: needs triage"]
|
||||||
body:
|
body:
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
attributes:
|
attributes:
|
||||||
|
|||||||
2
.github/ISSUE_TEMPLATE/04-translation.yaml
vendored
2
.github/ISSUE_TEMPLATE/04-translation.yaml
vendored
@ -2,7 +2,7 @@
|
|||||||
name: 🌍 Translation
|
name: 🌍 Translation
|
||||||
type: Translation
|
type: Translation
|
||||||
description: Request support for a new language in the user interface
|
description: Request support for a new language in the user interface
|
||||||
labels: ["type: translation"]
|
labels: ["netbox", "type: translation"]
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
|
|||||||
2
.github/ISSUE_TEMPLATE/05-housekeeping.yaml
vendored
2
.github/ISSUE_TEMPLATE/05-housekeeping.yaml
vendored
@ -2,7 +2,7 @@
|
|||||||
name: 🏡 Housekeeping
|
name: 🏡 Housekeeping
|
||||||
type: Housekeeping
|
type: Housekeeping
|
||||||
description: A change pertaining to the codebase itself (developers only)
|
description: A change pertaining to the codebase itself (developers only)
|
||||||
labels: ["type: housekeeping"]
|
labels: ["netbox", "type: housekeeping"]
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
|
|||||||
2
.github/ISSUE_TEMPLATE/06-deprecation.yaml
vendored
2
.github/ISSUE_TEMPLATE/06-deprecation.yaml
vendored
@ -2,7 +2,7 @@
|
|||||||
name: 🗑️ Deprecation
|
name: 🗑️ Deprecation
|
||||||
type: Deprecation
|
type: Deprecation
|
||||||
description: The removal of an existing feature or resource
|
description: The removal of an existing feature or resource
|
||||||
labels: ["type: deprecation"]
|
labels: ["netbox", "type: deprecation"]
|
||||||
body:
|
body:
|
||||||
- type: textarea
|
- type: textarea
|
||||||
attributes:
|
attributes:
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
rev: v0.6.9
|
rev: v0.14.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff
|
- id: ruff
|
||||||
name: "Ruff linter"
|
name: "Ruff linter"
|
||||||
|
|||||||
@ -2,7 +2,7 @@
|
|||||||
"openapi": "3.0.3",
|
"openapi": "3.0.3",
|
||||||
"info": {
|
"info": {
|
||||||
"title": "NetBox REST API",
|
"title": "NetBox REST API",
|
||||||
"version": "4.4.4",
|
"version": "4.4.5",
|
||||||
"license": {
|
"license": {
|
||||||
"name": "Apache v2 License"
|
"name": "Apache v2 License"
|
||||||
}
|
}
|
||||||
@ -61458,6 +61458,14 @@
|
|||||||
"operationId": "dcim_mac_addresses_list",
|
"operationId": "dcim_mac_addresses_list",
|
||||||
"description": "Get a list of MAC address objects.",
|
"description": "Get a list of MAC address objects.",
|
||||||
"parameters": [
|
"parameters": [
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "assigned",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"description": "Is assigned"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"in": "query",
|
"in": "query",
|
||||||
"name": "assigned_object_id",
|
"name": "assigned_object_id",
|
||||||
@ -62293,6 +62301,14 @@
|
|||||||
"explode": true,
|
"explode": true,
|
||||||
"style": "form"
|
"style": "form"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "primary",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"description": "Is primary"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"in": "query",
|
"in": "query",
|
||||||
"name": "q",
|
"name": "q",
|
||||||
@ -75760,6 +75776,157 @@
|
|||||||
"operationId": "dcim_power_outlet_templates_list",
|
"operationId": "dcim_power_outlet_templates_list",
|
||||||
"description": "Get a list of power outlet template objects.",
|
"description": "Get a list of power outlet template objects.",
|
||||||
"parameters": [
|
"parameters": [
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__empty",
|
||||||
|
"schema": {
|
||||||
|
"type": "boolean"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__ic",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__ie",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__iew",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__iregex",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__isw",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__n",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__nic",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__nie",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__niew",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__nisw",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"in": "query",
|
||||||
|
"name": "color__regex",
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"explode": true,
|
||||||
|
"style": "form"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"in": "query",
|
"in": "query",
|
||||||
"name": "created",
|
"name": "created",
|
||||||
@ -242458,6 +242625,11 @@
|
|||||||
"x-spec-enum-id": "8f9617d2648ab261",
|
"x-spec-enum-id": "8f9617d2648ab261",
|
||||||
"nullable": true
|
"nullable": true
|
||||||
},
|
},
|
||||||
|
"color": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[0-9a-f]{6}$",
|
||||||
|
"maxLength": 6
|
||||||
|
},
|
||||||
"power_port": {
|
"power_port": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
@ -247155,6 +247327,11 @@
|
|||||||
},
|
},
|
||||||
"nullable": true
|
"nullable": true
|
||||||
},
|
},
|
||||||
|
"color": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[0-9a-f]{6}$",
|
||||||
|
"maxLength": 6
|
||||||
|
},
|
||||||
"power_port": {
|
"power_port": {
|
||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
@ -247371,6 +247548,11 @@
|
|||||||
"x-spec-enum-id": "8f9617d2648ab261",
|
"x-spec-enum-id": "8f9617d2648ab261",
|
||||||
"nullable": true
|
"nullable": true
|
||||||
},
|
},
|
||||||
|
"color": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[0-9a-f]{6}$",
|
||||||
|
"maxLength": 6
|
||||||
|
},
|
||||||
"power_port": {
|
"power_port": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
@ -264451,6 +264633,11 @@
|
|||||||
"x-spec-enum-id": "8f9617d2648ab261",
|
"x-spec-enum-id": "8f9617d2648ab261",
|
||||||
"nullable": true
|
"nullable": true
|
||||||
},
|
},
|
||||||
|
"color": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[0-9a-f]{6}$",
|
||||||
|
"maxLength": 6
|
||||||
|
},
|
||||||
"power_port": {
|
"power_port": {
|
||||||
"oneOf": [
|
"oneOf": [
|
||||||
{
|
{
|
||||||
|
|||||||
@ -53,6 +53,16 @@ Sets content for the top banner in the user interface.
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## COPILOT_ENABLED
|
||||||
|
|
||||||
|
!!! tip "Dynamic Configuration Parameter"
|
||||||
|
|
||||||
|
Default: `True`
|
||||||
|
|
||||||
|
Enables or disables the [NetBox Copilot](https://netboxlabs.com/docs/copilot/) agent globally. When enabled, users can opt to toggle the agent individually.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## CENSUS_REPORTING_ENABLED
|
## CENSUS_REPORTING_ENABLED
|
||||||
|
|
||||||
Default: `True`
|
Default: `True`
|
||||||
|
|||||||
@ -393,6 +393,61 @@ A complete date & time. Returns a `datetime.datetime` object.
|
|||||||
|
|
||||||
Custom scripts can be run via the web UI by navigating to the script, completing any required form data, and clicking the "run script" button. It is possible to schedule a script to be executed at specified time in the future. A scheduled script can be canceled by deleting the associated job result object.
|
Custom scripts can be run via the web UI by navigating to the script, completing any required form data, and clicking the "run script" button. It is possible to schedule a script to be executed at specified time in the future. A scheduled script can be canceled by deleting the associated job result object.
|
||||||
|
|
||||||
|
#### Prefilling variables via URL parameters
|
||||||
|
|
||||||
|
Script form fields can be prefilled by appending query parameters to the script URL. Each parameter name must match the variable name defined on the script class. Prefilled values are treated as initial values and can be edited before execution. Multiple values can be supplied by repeating the same parameter. Query values must be percent‑encoded where required (for example, spaces as `%20`).
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
|
||||||
|
For string and integer variables, when a script defines:
|
||||||
|
|
||||||
|
```python
|
||||||
|
from extras.scripts import Script, StringVar, IntegerVar
|
||||||
|
|
||||||
|
class MyScript(Script):
|
||||||
|
name = StringVar()
|
||||||
|
count = IntegerVar()
|
||||||
|
```
|
||||||
|
|
||||||
|
the following URL prefills the `name` and `count` fields:
|
||||||
|
|
||||||
|
```
|
||||||
|
https://<netbox>/extras/scripts/<script_id>/?name=Branch42&count=3
|
||||||
|
```
|
||||||
|
|
||||||
|
For object variables (`ObjectVar`), supply the object’s primary key (PK):
|
||||||
|
|
||||||
|
```
|
||||||
|
https://<netbox>/extras/scripts/<script_id>/?device=1
|
||||||
|
```
|
||||||
|
|
||||||
|
If an object ID cannot be resolved or the object is not visible to the requesting user, the field remains unpopulated.
|
||||||
|
|
||||||
|
Supported variable types:
|
||||||
|
|
||||||
|
| Variable class | Expected input | Example query string |
|
||||||
|
|--------------------------|---------------------------------|---------------------------------------------|
|
||||||
|
| `StringVar` | string (percent‑encoded) | `?name=Branch42` |
|
||||||
|
| `TextVar` | string (percent‑encoded) | `?notes=Initial%20value` |
|
||||||
|
| `IntegerVar` | integer | `?count=3` |
|
||||||
|
| `DecimalVar` | decimal number | `?ratio=0.75` |
|
||||||
|
| `BooleanVar` | value → `True`; empty → `False` | `?enabled=true` (True), `?enabled=` (False) |
|
||||||
|
| `ChoiceVar` | choice value (not label) | `?role=edge` |
|
||||||
|
| `MultiChoiceVar` | choice values (repeat) | `?roles=edge&roles=core` |
|
||||||
|
| `ObjectVar(Device)` | PK (integer) | `?device=1` |
|
||||||
|
| `MultiObjectVar(Device)` | PKs (repeat) | `?devices=1&devices=2` |
|
||||||
|
| `IPAddressVar` | IP address | `?ip=198.51.100.10` |
|
||||||
|
| `IPAddressWithMaskVar` | IP address with mask | `?addr=192.0.2.1/24` |
|
||||||
|
| `IPNetworkVar` | IP network prefix | `?network=2001:db8::/64` |
|
||||||
|
| `DateVar` | date `YYYY-MM-DD` | `?date=2025-01-05` |
|
||||||
|
| `DateTimeVar` | ISO datetime | `?when=2025-01-05T14:30:00` |
|
||||||
|
| `FileVar` | — (not supported) | — |
|
||||||
|
|
||||||
|
!!! note
|
||||||
|
- The parameter names above are examples; use the actual variable attribute names defined by the script.
|
||||||
|
- For `BooleanVar`, only an empty value (`?enabled=`) unchecks the box; any other value including `false` or `0` checks it.
|
||||||
|
- File uploads (`FileVar`) cannot be prefilled via URL parameters.
|
||||||
|
|
||||||
### Via the API
|
### Via the API
|
||||||
|
|
||||||
To run a script via the REST API, issue a POST request to the script's endpoint specifying the form data and commitment. For example, to run a script named `example.MyReport`, we would make a request such as the following:
|
To run a script via the REST API, issue a POST request to the script's endpoint specifying the form data and commitment. For example, to run a script named `example.MyReport`, we would make a request such as the following:
|
||||||
|
|||||||
@ -6,10 +6,14 @@ For end‑user guidance on resetting saved table layouts, see [Features > User P
|
|||||||
|
|
||||||
## Available Preferences
|
## Available Preferences
|
||||||
|
|
||||||
| Name | Description |
|
| Name | Description |
|
||||||
|--------------------------|---------------------------------------------------------------|
|
|----------------------------|---------------------------------------------------------------|
|
||||||
| data_format | Preferred format when rendering raw data (JSON or YAML) |
|
| `csv_delimiter` | The delimiting character used when exporting CSV data |
|
||||||
| pagination.per_page | The number of items to display per page of a paginated table |
|
| `data_format` | Preferred format when rendering raw data (JSON or YAML) |
|
||||||
| pagination.placement | Where to display the paginator controls relative to the table |
|
| `locale.language` | The language selected for UI translation |
|
||||||
| tables.${table}.columns | The ordered list of columns to display when viewing the table |
|
| `pagination.per_page` | The number of items to display per page of a paginated table |
|
||||||
| tables.${table}.ordering | A list of column names by which the table should be ordered |
|
| `pagination.placement` | Where to display the paginator controls relative to the table |
|
||||||
|
| `tables.${table}.columns` | The ordered list of columns to display when viewing the table |
|
||||||
|
| `tables.${table}.ordering` | A list of column names by which the table should be ordered |
|
||||||
|
| `ui.copilot_enabled` | Toggles the NetBox Copilot AI agent |
|
||||||
|
| `ui.tables.striping` | Toggles visual striping of tables in the UI |
|
||||||
|
|||||||
@ -60,6 +60,13 @@ Four of the standard Python logging levels are supported:
|
|||||||
|
|
||||||
Log entries recorded using the runner's logger will be saved in the job's log in the database in addition to being processed by other [system logging handlers](../../configuration/system.md#logging).
|
Log entries recorded using the runner's logger will be saved in the job's log in the database in addition to being processed by other [system logging handlers](../../configuration/system.md#logging).
|
||||||
|
|
||||||
|
### Jobs running for Model instances
|
||||||
|
|
||||||
|
A Job can be executed for a specific instance of a Model.
|
||||||
|
To enable this functionality, the model must include the `JobsMixin`.
|
||||||
|
|
||||||
|
When enqueuing a Job, you can associate it with a particular instance by passing that instance to the `instance` parameter.
|
||||||
|
|
||||||
### Scheduled Jobs
|
### Scheduled Jobs
|
||||||
|
|
||||||
As described above, jobs can be scheduled for immediate execution or at any later time using the `enqueue()` method. However, for management purposes, the `enqueue_once()` method allows a job to be scheduled exactly once avoiding duplicates. If a job is already scheduled for a particular instance, a second one won't be scheduled, respecting thread safety. An example use case would be to schedule a periodic task that is bound to an instance in general, but not to any event of that instance (such as updates). The parameters of the `enqueue_once()` method are identical to those of `enqueue()`.
|
As described above, jobs can be scheduled for immediate execution or at any later time using the `enqueue()` method. However, for management purposes, the `enqueue_once()` method allows a job to be scheduled exactly once avoiding duplicates. If a job is already scheduled for a particular instance, a second one won't be scheduled, respecting thread safety. An example use case would be to schedule a periodic task that is bound to an instance in general, but not to any event of that instance (such as updates). The parameters of the `enqueue_once()` method are identical to those of `enqueue()`.
|
||||||
@ -73,9 +80,10 @@ As described above, jobs can be scheduled for immediate execution or at any late
|
|||||||
from django.db import models
|
from django.db import models
|
||||||
from core.choices import JobIntervalChoices
|
from core.choices import JobIntervalChoices
|
||||||
from netbox.models import NetBoxModel
|
from netbox.models import NetBoxModel
|
||||||
|
from netbox.models.features import JobsMixin
|
||||||
from .jobs import MyTestJob
|
from .jobs import MyTestJob
|
||||||
|
|
||||||
class MyModel(NetBoxModel):
|
class MyModel(JobsMixin, NetBoxModel):
|
||||||
foo = models.CharField()
|
foo = models.CharField()
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
|
|||||||
@ -55,6 +55,27 @@ class MyModelViewSet(...):
|
|||||||
filterset_class = filtersets.MyModelFilterSet
|
filterset_class = filtersets.MyModelFilterSet
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Implementing Quick Search
|
||||||
|
|
||||||
|
The `ObjectListView` has a field called Quick Search. For Quick Search to work the corresponding FilterSet has to override the `search` method that is implemented in `NetBoxModelFilterSet`. This function takes a queryset and can perform arbitrary operations on it and return it. A common use-case is to search for the given search value in multiple fields:
|
||||||
|
|
||||||
|
```python
|
||||||
|
from django.db.models import Q
|
||||||
|
from netbox.filtersets import NetBoxModelFilterSet
|
||||||
|
|
||||||
|
class MyFilterSet(NetBoxModelFilterSet):
|
||||||
|
...
|
||||||
|
def search(self, queryset, name, value):
|
||||||
|
if not value.strip():
|
||||||
|
return queryset
|
||||||
|
return queryset.filter(
|
||||||
|
Q(name__icontains=value) |
|
||||||
|
Q(description__icontains=value)
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
The `search` method is also used by the `q` filter in `NetBoxModelFilterSet` which in turn is used by the Search field in the filters tab.
|
||||||
|
|
||||||
## Filter Classes
|
## Filter Classes
|
||||||
|
|
||||||
### TagFilter
|
### TagFilter
|
||||||
|
|||||||
@ -1,5 +1,35 @@
|
|||||||
# NetBox v4.4
|
# NetBox v4.4
|
||||||
|
|
||||||
|
## v4.4.5 (2025-10-28)
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
|
||||||
|
* [#19751](https://github.com/netbox-community/netbox/issues/19751) - Disable occupied module bays in form dropdowns when installing a new module
|
||||||
|
* [#20301](https://github.com/netbox-community/netbox/issues/20301) - Add a "dismiss all" option to the notifications dropdown
|
||||||
|
* [#20399](https://github.com/netbox-community/netbox/issues/20399) - Add `assigned` and `primary` boolean filters for MAC addresses
|
||||||
|
* [#20567](https://github.com/netbox-community/netbox/issues/20567) - Add contacts column to services table
|
||||||
|
* [#20675](https://github.com/netbox-community/netbox/issues/20675) - Enable [NetBox Copilot](https://netboxlabs.com/products/netbox-copilot/) integration
|
||||||
|
* [#20692](https://github.com/netbox-community/netbox/issues/20692) - Add contacts column to IP addresses table
|
||||||
|
* [#20700](https://github.com/netbox-community/netbox/issues/20700) - Add contacts table column for various additional models
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* [#19872](https://github.com/netbox-community/netbox/issues/19872) - Ensure custom script validation failures display error messages
|
||||||
|
* [#20389](https://github.com/netbox-community/netbox/issues/20389) - Fix "select all" behavior for bulk rename views
|
||||||
|
* [#20422](https://github.com/netbox-community/netbox/issues/20422) - Enable filtering of aggregates and prefixes by family in GraphQL API
|
||||||
|
* [#20459](https://github.com/netbox-community/netbox/issues/20459) - Fix validation of `is_oob` & `is_primary` fields under IP address bulk import
|
||||||
|
* [#20466](https://github.com/netbox-community/netbox/issues/20466) - Fix querying of devices with a primary IP assigned in GraphQL API
|
||||||
|
* [#20498](https://github.com/netbox-community/netbox/issues/20498) - Enforce the validation regex (if set) for custom URL fields
|
||||||
|
* [#20524](https://github.com/netbox-community/netbox/issues/20524) - Raise a validation error when attempting to schedule a custom script for a past date/time
|
||||||
|
* [#20541](https://github.com/netbox-community/netbox/issues/20541) - Fix resolution of GraphQL object fields which rely on custom filters
|
||||||
|
* [#20551](https://github.com/netbox-community/netbox/issues/20551) - Fix automatic slug generation in quick-add UI form
|
||||||
|
* [#20606](https://github.com/netbox-community/netbox/issues/20606) - Enable copying of values from table columns rendered as badges
|
||||||
|
* [#20641](https://github.com/netbox-community/netbox/issues/20641) - Fix `AttributeError` exception raised by the object changes REST API endpoint
|
||||||
|
* [#20646](https://github.com/netbox-community/netbox/issues/20646) - Prevent cables from connecting to objects marked as connected
|
||||||
|
* [#20655](https://github.com/netbox-community/netbox/issues/20655) - Fix `FieldError` exception when attempting to sort permissions list by actions
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## v4.4.4 (2025-10-15)
|
## v4.4.4 (2025-10-15)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
@ -78,6 +78,7 @@ class ProviderBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Provider, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Provider, 'bulk_rename', path='rename', detail=False)
|
||||||
class ProviderBulkRenameView(generic.BulkRenameView):
|
class ProviderBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Provider.objects.all()
|
queryset = Provider.objects.all()
|
||||||
|
filterset = filtersets.ProviderFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Provider, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Provider, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -145,6 +146,7 @@ class ProviderAccountBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ProviderAccount, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ProviderAccount, 'bulk_rename', path='rename', detail=False)
|
||||||
class ProviderAccountBulkRenameView(generic.BulkRenameView):
|
class ProviderAccountBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ProviderAccount.objects.all()
|
queryset = ProviderAccount.objects.all()
|
||||||
|
filterset = filtersets.ProviderAccountFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ProviderAccount, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ProviderAccount, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -221,6 +223,7 @@ class ProviderNetworkBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ProviderNetwork, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ProviderNetwork, 'bulk_rename', path='rename', detail=False)
|
||||||
class ProviderNetworkBulkRenameView(generic.BulkRenameView):
|
class ProviderNetworkBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ProviderNetwork.objects.all()
|
queryset = ProviderNetwork.objects.all()
|
||||||
|
filterset = filtersets.ProviderNetworkFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ProviderNetwork, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ProviderNetwork, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -285,6 +288,7 @@ class CircuitTypeBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(CircuitType, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(CircuitType, 'bulk_rename', path='rename', detail=False)
|
||||||
class CircuitTypeBulkRenameView(generic.BulkRenameView):
|
class CircuitTypeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = CircuitType.objects.all()
|
queryset = CircuitType.objects.all()
|
||||||
|
filterset = filtersets.CircuitTypeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(CircuitType, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(CircuitType, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -357,6 +361,7 @@ class CircuitBulkEditView(generic.BulkEditView):
|
|||||||
class CircuitBulkRenameView(generic.BulkRenameView):
|
class CircuitBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Circuit.objects.all()
|
queryset = Circuit.objects.all()
|
||||||
field_name = 'cid'
|
field_name = 'cid'
|
||||||
|
filterset = filtersets.CircuitFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Circuit, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Circuit, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -476,6 +481,7 @@ class CircuitGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(CircuitGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(CircuitGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class CircuitGroupBulkRenameView(generic.BulkRenameView):
|
class CircuitGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = CircuitGroup.objects.all()
|
queryset = CircuitGroup.objects.all()
|
||||||
|
filterset = filtersets.CircuitGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(CircuitGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(CircuitGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -591,6 +597,7 @@ class VirtualCircuitTypeBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VirtualCircuitType, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VirtualCircuitType, 'bulk_rename', path='rename', detail=False)
|
||||||
class VirtualCircuitTypeBulkRenameView(generic.BulkRenameView):
|
class VirtualCircuitTypeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VirtualCircuitType.objects.all()
|
queryset = VirtualCircuitType.objects.all()
|
||||||
|
filterset = filtersets.VirtualCircuitTypeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VirtualCircuitType, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VirtualCircuitType, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -663,6 +670,7 @@ class VirtualCircuitBulkEditView(generic.BulkEditView):
|
|||||||
class VirtualCircuitBulkRenameView(generic.BulkRenameView):
|
class VirtualCircuitBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VirtualCircuit.objects.all()
|
queryset = VirtualCircuit.objects.all()
|
||||||
field_name = 'cid'
|
field_name = 'cid'
|
||||||
|
filterset = filtersets.VirtualCircuitFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VirtualCircuit, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VirtualCircuit, 'bulk_delete', path='delete', detail=False)
|
||||||
|
|||||||
@ -166,8 +166,8 @@ class ConfigRevisionForm(forms.ModelForm, metaclass=ConfigFormMetaclass):
|
|||||||
FieldSet('CUSTOM_VALIDATORS', 'PROTECTION_RULES', name=_('Validation')),
|
FieldSet('CUSTOM_VALIDATORS', 'PROTECTION_RULES', name=_('Validation')),
|
||||||
FieldSet('DEFAULT_USER_PREFERENCES', name=_('User Preferences')),
|
FieldSet('DEFAULT_USER_PREFERENCES', name=_('User Preferences')),
|
||||||
FieldSet(
|
FieldSet(
|
||||||
'MAINTENANCE_MODE', 'GRAPHQL_ENABLED', 'CHANGELOG_RETENTION', 'JOB_RETENTION', 'MAPS_URL',
|
'MAINTENANCE_MODE', 'COPILOT_ENABLED', 'GRAPHQL_ENABLED', 'CHANGELOG_RETENTION', 'JOB_RETENTION',
|
||||||
name=_('Miscellaneous')
|
'MAPS_URL', name=_('Miscellaneous'),
|
||||||
),
|
),
|
||||||
FieldSet('comment', name=_('Config Revision'))
|
FieldSet('comment', name=_('Config Revision'))
|
||||||
)
|
)
|
||||||
|
|||||||
@ -125,6 +125,7 @@ class DataSourceBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(DataSource, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(DataSource, 'bulk_rename', path='rename', detail=False)
|
||||||
class DataSourceBulkRenameView(generic.BulkRenameView):
|
class DataSourceBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = DataSource.objects.all()
|
queryset = DataSource.objects.all()
|
||||||
|
filterset = filtersets.DataSourceFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(DataSource, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(DataSource, 'bulk_delete', path='delete', detail=False)
|
||||||
|
|||||||
@ -14,7 +14,7 @@ from netbox.filtersets import (
|
|||||||
AttributeFiltersMixin, BaseFilterSet, ChangeLoggedModelFilterSet, NestedGroupModelFilterSet,
|
AttributeFiltersMixin, BaseFilterSet, ChangeLoggedModelFilterSet, NestedGroupModelFilterSet,
|
||||||
OrganizationalModelFilterSet, PrimaryModelFilterSet, NetBoxModelFilterSet,
|
OrganizationalModelFilterSet, PrimaryModelFilterSet, NetBoxModelFilterSet,
|
||||||
)
|
)
|
||||||
from tenancy.filtersets import TenancyFilterSet, ContactModelFilterSet
|
from tenancy.filtersets import ContactModelFilterSet, TenancyFilterSet
|
||||||
from tenancy.models import *
|
from tenancy.models import *
|
||||||
from users.filterset_mixins import OwnerFilterMixin
|
from users.filterset_mixins import OwnerFilterMixin
|
||||||
from users.models import User
|
from users.models import User
|
||||||
@ -22,9 +22,9 @@ from utilities.filters import (
|
|||||||
ContentTypeFilter, MultiValueCharFilter, MultiValueMACAddressFilter, MultiValueNumberFilter, MultiValueWWNFilter,
|
ContentTypeFilter, MultiValueCharFilter, MultiValueMACAddressFilter, MultiValueNumberFilter, MultiValueWWNFilter,
|
||||||
NumericArrayFilter, TreeNodeMultipleChoiceFilter,
|
NumericArrayFilter, TreeNodeMultipleChoiceFilter,
|
||||||
)
|
)
|
||||||
from virtualization.models import Cluster, ClusterGroup, VMInterface, VirtualMachine
|
from virtualization.models import Cluster, ClusterGroup, VirtualMachine, VMInterface
|
||||||
from vpn.models import L2VPN
|
from vpn.models import L2VPN
|
||||||
from wireless.choices import WirelessRoleChoices, WirelessChannelChoices
|
from wireless.choices import WirelessChannelChoices, WirelessRoleChoices
|
||||||
from wireless.models import WirelessLAN, WirelessLink
|
from wireless.models import WirelessLAN, WirelessLink
|
||||||
from .choices import *
|
from .choices import *
|
||||||
from .constants import *
|
from .constants import *
|
||||||
@ -1289,7 +1289,6 @@ class DeviceFilterSet(
|
|||||||
Q(name__icontains=value) |
|
Q(name__icontains=value) |
|
||||||
Q(virtual_chassis__name__icontains=value) |
|
Q(virtual_chassis__name__icontains=value) |
|
||||||
Q(serial__icontains=value.strip()) |
|
Q(serial__icontains=value.strip()) |
|
||||||
Q(inventoryitems__serial__icontains=value.strip()) |
|
|
||||||
Q(asset_tag__icontains=value.strip()) |
|
Q(asset_tag__icontains=value.strip()) |
|
||||||
Q(description__icontains=value.strip()) |
|
Q(description__icontains=value.strip()) |
|
||||||
Q(comments__icontains=value) |
|
Q(comments__icontains=value) |
|
||||||
@ -1788,6 +1787,14 @@ class MACAddressFilterSet(PrimaryModelFilterSet):
|
|||||||
queryset=VMInterface.objects.all(),
|
queryset=VMInterface.objects.all(),
|
||||||
label=_('VM interface (ID)'),
|
label=_('VM interface (ID)'),
|
||||||
)
|
)
|
||||||
|
assigned = django_filters.BooleanFilter(
|
||||||
|
method='filter_assigned',
|
||||||
|
label=_('Is assigned'),
|
||||||
|
)
|
||||||
|
primary = django_filters.BooleanFilter(
|
||||||
|
method='filter_primary',
|
||||||
|
label=_('Is primary'),
|
||||||
|
)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = MACAddress
|
model = MACAddress
|
||||||
@ -1824,6 +1831,29 @@ class MACAddressFilterSet(PrimaryModelFilterSet):
|
|||||||
vminterface__in=interface_ids
|
vminterface__in=interface_ids
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def filter_assigned(self, queryset, name, value):
|
||||||
|
params = {
|
||||||
|
'assigned_object_type__isnull': True,
|
||||||
|
'assigned_object_id__isnull': True,
|
||||||
|
}
|
||||||
|
if value:
|
||||||
|
return queryset.exclude(**params)
|
||||||
|
else:
|
||||||
|
return queryset.filter(**params)
|
||||||
|
|
||||||
|
def filter_primary(self, queryset, name, value):
|
||||||
|
interface_mac_ids = Interface.objects.filter(primary_mac_address_id__isnull=False).values_list(
|
||||||
|
'primary_mac_address_id', flat=True
|
||||||
|
)
|
||||||
|
vminterface_mac_ids = VMInterface.objects.filter(primary_mac_address_id__isnull=False).values_list(
|
||||||
|
'primary_mac_address_id', flat=True
|
||||||
|
)
|
||||||
|
query = Q(pk__in=interface_mac_ids) | Q(pk__in=vminterface_mac_ids)
|
||||||
|
if value:
|
||||||
|
return queryset.filter(query)
|
||||||
|
else:
|
||||||
|
return queryset.exclude(query)
|
||||||
|
|
||||||
|
|
||||||
class CommonInterfaceFilterSet(django_filters.FilterSet):
|
class CommonInterfaceFilterSet(django_filters.FilterSet):
|
||||||
mode = django_filters.MultipleChoiceFilter(
|
mode = django_filters.MultipleChoiceFilter(
|
||||||
|
|||||||
@ -1697,12 +1697,13 @@ class MACAddressFilterForm(PrimaryModelFilterSetForm):
|
|||||||
model = MACAddress
|
model = MACAddress
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
FieldSet('q', 'filter_id', 'tag', 'owner_id'),
|
FieldSet('q', 'filter_id', 'tag', 'owner_id'),
|
||||||
FieldSet('mac_address', 'device_id', 'virtual_machine_id', name=_('MAC address')),
|
FieldSet('mac_address', name=_('Attributes')),
|
||||||
|
FieldSet('device_id', 'virtual_machine_id', 'assigned', 'primary', name=_('Assignments')),
|
||||||
)
|
)
|
||||||
selector_fields = ('filter_id', 'q', 'device_id', 'virtual_machine_id')
|
selector_fields = ('filter_id', 'q', 'device_id', 'virtual_machine_id')
|
||||||
mac_address = forms.CharField(
|
mac_address = forms.CharField(
|
||||||
required=False,
|
required=False,
|
||||||
label=_('MAC address')
|
label=_('MAC address'),
|
||||||
)
|
)
|
||||||
device_id = DynamicModelMultipleChoiceField(
|
device_id = DynamicModelMultipleChoiceField(
|
||||||
queryset=Device.objects.all(),
|
queryset=Device.objects.all(),
|
||||||
@ -1714,6 +1715,20 @@ class MACAddressFilterForm(PrimaryModelFilterSetForm):
|
|||||||
required=False,
|
required=False,
|
||||||
label=_('Assigned VM'),
|
label=_('Assigned VM'),
|
||||||
)
|
)
|
||||||
|
assigned = forms.NullBooleanField(
|
||||||
|
required=False,
|
||||||
|
label=_('Assigned to an interface'),
|
||||||
|
widget=forms.Select(
|
||||||
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
||||||
|
),
|
||||||
|
)
|
||||||
|
primary = forms.NullBooleanField(
|
||||||
|
required=False,
|
||||||
|
label=_('Primary MAC of an interface'),
|
||||||
|
widget=forms.Select(
|
||||||
|
choices=BOOLEAN_WITH_BLANK_CHOICES
|
||||||
|
),
|
||||||
|
)
|
||||||
tag = TagFilterField(model)
|
tag = TagFilterField(model)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@ -734,7 +734,10 @@ class ModuleForm(ModuleCommonForm, PrimaryModelForm):
|
|||||||
queryset=ModuleBay.objects.all(),
|
queryset=ModuleBay.objects.all(),
|
||||||
query_params={
|
query_params={
|
||||||
'device_id': '$device'
|
'device_id': '$device'
|
||||||
}
|
},
|
||||||
|
context={
|
||||||
|
'disabled': 'installed_module',
|
||||||
|
},
|
||||||
)
|
)
|
||||||
module_type = DynamicModelChoiceField(
|
module_type = DynamicModelChoiceField(
|
||||||
label=_('Module type'),
|
label=_('Module type'),
|
||||||
|
|||||||
@ -18,7 +18,9 @@ from netbox.graphql.filter_mixins import (
|
|||||||
ImageAttachmentFilterMixin,
|
ImageAttachmentFilterMixin,
|
||||||
WeightFilterMixin,
|
WeightFilterMixin,
|
||||||
)
|
)
|
||||||
from tenancy.graphql.filter_mixins import TenancyFilterMixin, ContactFilterMixin
|
from tenancy.graphql.filter_mixins import ContactFilterMixin, TenancyFilterMixin
|
||||||
|
from virtualization.models import VMInterface
|
||||||
|
|
||||||
from .filter_mixins import (
|
from .filter_mixins import (
|
||||||
CabledObjectModelFilterMixin,
|
CabledObjectModelFilterMixin,
|
||||||
ComponentModelFilterMixin,
|
ComponentModelFilterMixin,
|
||||||
@ -419,6 +421,24 @@ class MACAddressFilter(PrimaryModelFilterMixin):
|
|||||||
)
|
)
|
||||||
assigned_object_id: ID | None = strawberry_django.filter_field()
|
assigned_object_id: ID | None = strawberry_django.filter_field()
|
||||||
|
|
||||||
|
@strawberry_django.filter_field()
|
||||||
|
def assigned(self, value: bool, prefix) -> Q:
|
||||||
|
return Q(**{f'{prefix}assigned_object_id__isnull': (not value)})
|
||||||
|
|
||||||
|
@strawberry_django.filter_field()
|
||||||
|
def primary(self, value: bool, prefix) -> Q:
|
||||||
|
interface_mac_ids = models.Interface.objects.filter(primary_mac_address_id__isnull=False).values_list(
|
||||||
|
'primary_mac_address_id', flat=True
|
||||||
|
)
|
||||||
|
vminterface_mac_ids = VMInterface.objects.filter(primary_mac_address_id__isnull=False).values_list(
|
||||||
|
'primary_mac_address_id', flat=True
|
||||||
|
)
|
||||||
|
query = Q(**{f'{prefix}pk__in': interface_mac_ids}) | Q(**{f'{prefix}pk__in': vminterface_mac_ids})
|
||||||
|
if value:
|
||||||
|
return Q(query)
|
||||||
|
else:
|
||||||
|
return ~Q(query)
|
||||||
|
|
||||||
|
|
||||||
@strawberry_django.filter_type(models.Interface, lookups=True)
|
@strawberry_django.filter_type(models.Interface, lookups=True)
|
||||||
class InterfaceFilter(ModularComponentModelFilterMixin, InterfaceBaseFilterMixin, CabledObjectModelFilterMixin):
|
class InterfaceFilter(ModularComponentModelFilterMixin, InterfaceBaseFilterMixin, CabledObjectModelFilterMixin):
|
||||||
|
|||||||
@ -393,6 +393,17 @@ class CableTermination(ChangeLoggedModel):
|
|||||||
def clean(self):
|
def clean(self):
|
||||||
super().clean()
|
super().clean()
|
||||||
|
|
||||||
|
# Disallow connecting a cable to any termination object that is
|
||||||
|
# explicitly flagged as "mark connected".
|
||||||
|
termination = getattr(self, 'termination', None)
|
||||||
|
if termination is not None and getattr(termination, "mark_connected", False):
|
||||||
|
raise ValidationError(
|
||||||
|
_("Cannot connect a cable to {obj_parent} > {obj} because it is marked as connected.").format(
|
||||||
|
obj_parent=termination.parent_object,
|
||||||
|
obj=termination,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
# Check for existing termination
|
# Check for existing termination
|
||||||
qs = CableTermination.objects.filter(
|
qs = CableTermination.objects.filter(
|
||||||
termination_type=self.termination_type,
|
termination_type=self.termination_type,
|
||||||
@ -404,14 +415,14 @@ class CableTermination(ChangeLoggedModel):
|
|||||||
existing_termination = qs.first()
|
existing_termination = qs.first()
|
||||||
if existing_termination is not None:
|
if existing_termination is not None:
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
_("Duplicate termination found for {app_label}.{model} {termination_id}: cable {cable_pk}".format(
|
_("Duplicate termination found for {app_label}.{model} {termination_id}: cable {cable_pk}").format(
|
||||||
app_label=self.termination_type.app_label,
|
app_label=self.termination_type.app_label,
|
||||||
model=self.termination_type.model,
|
model=self.termination_type.model,
|
||||||
termination_id=self.termination_id,
|
termination_id=self.termination_id,
|
||||||
cable_pk=existing_termination.cable.pk
|
cable_pk=existing_termination.cable.pk
|
||||||
))
|
)
|
||||||
)
|
)
|
||||||
# Validate interface type (if applicable)
|
# Validate the interface type (if applicable)
|
||||||
if self.termination_type.model == 'interface' and self.termination.type in NONCONNECTABLE_IFACE_TYPES:
|
if self.termination_type.model == 'interface' and self.termination.type in NONCONNECTABLE_IFACE_TYPES:
|
||||||
raise ValidationError(
|
raise ValidationError(
|
||||||
_("Cables cannot be terminated to {type_display} interfaces").format(
|
_("Cables cannot be terminated to {type_display} interfaces").format(
|
||||||
|
|||||||
@ -1151,6 +1151,9 @@ class MACAddressTable(PrimaryModelTable):
|
|||||||
orderable=False,
|
orderable=False,
|
||||||
verbose_name=_('Parent')
|
verbose_name=_('Parent')
|
||||||
)
|
)
|
||||||
|
is_primary = columns.BooleanColumn(
|
||||||
|
verbose_name=_('Primary')
|
||||||
|
)
|
||||||
tags = columns.TagColumn(
|
tags = columns.TagColumn(
|
||||||
url_name='dcim:macaddress_list'
|
url_name='dcim:macaddress_list'
|
||||||
)
|
)
|
||||||
@ -1161,7 +1164,7 @@ class MACAddressTable(PrimaryModelTable):
|
|||||||
class Meta(PrimaryModelTable.Meta):
|
class Meta(PrimaryModelTable.Meta):
|
||||||
model = models.MACAddress
|
model = models.MACAddress
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'mac_address', 'assigned_object_parent', 'assigned_object', 'description', 'comments', 'tags',
|
'pk', 'id', 'mac_address', 'assigned_object_parent', 'assigned_object', 'description', 'is_primary',
|
||||||
'created', 'last_updated',
|
'comments', 'tags', 'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = ('pk', 'mac_address', 'assigned_object_parent', 'assigned_object', 'description')
|
default_columns = ('pk', 'mac_address', 'assigned_object_parent', 'assigned_object', 'description')
|
||||||
|
|||||||
@ -10,7 +10,7 @@ from netbox.choices import ColorChoices, WeightUnitChoices
|
|||||||
from tenancy.models import Tenant, TenantGroup
|
from tenancy.models import Tenant, TenantGroup
|
||||||
from users.models import User
|
from users.models import User
|
||||||
from utilities.testing import ChangeLoggedFilterSetTests, create_test_device, create_test_virtualmachine
|
from utilities.testing import ChangeLoggedFilterSetTests, create_test_device, create_test_virtualmachine
|
||||||
from virtualization.models import Cluster, ClusterType, ClusterGroup, VMInterface, VirtualMachine
|
from virtualization.models import Cluster, ClusterGroup, ClusterType, VirtualMachine, VMInterface
|
||||||
from wireless.choices import WirelessChannelChoices, WirelessRoleChoices
|
from wireless.choices import WirelessChannelChoices, WirelessRoleChoices
|
||||||
from wireless.models import WirelessLink
|
from wireless.models import WirelessLink
|
||||||
|
|
||||||
@ -7171,9 +7171,20 @@ class MACAddressTestCase(TestCase, ChangeLoggedFilterSetTests):
|
|||||||
MACAddress(mac_address='00-00-00-05-01-01', assigned_object=vm_interfaces[1]),
|
MACAddress(mac_address='00-00-00-05-01-01', assigned_object=vm_interfaces[1]),
|
||||||
MACAddress(mac_address='00-00-00-06-01-01', assigned_object=vm_interfaces[2]),
|
MACAddress(mac_address='00-00-00-06-01-01', assigned_object=vm_interfaces[2]),
|
||||||
MACAddress(mac_address='00-00-00-06-01-02', assigned_object=vm_interfaces[2]),
|
MACAddress(mac_address='00-00-00-06-01-02', assigned_object=vm_interfaces[2]),
|
||||||
|
# unassigned
|
||||||
|
MACAddress(mac_address='00-00-00-07-01-01'),
|
||||||
)
|
)
|
||||||
MACAddress.objects.bulk_create(mac_addresses)
|
MACAddress.objects.bulk_create(mac_addresses)
|
||||||
|
|
||||||
|
# Set MAC addresses as primary
|
||||||
|
for idx, interface in enumerate(interfaces):
|
||||||
|
interface.primary_mac_address = mac_addresses[idx]
|
||||||
|
interface.save()
|
||||||
|
for idx, vm_interface in enumerate(vm_interfaces):
|
||||||
|
# Offset by 4 for device MACs
|
||||||
|
vm_interface.primary_mac_address = mac_addresses[idx + 4]
|
||||||
|
vm_interface.save()
|
||||||
|
|
||||||
def test_mac_address(self):
|
def test_mac_address(self):
|
||||||
params = {'mac_address': ['00-00-00-01-01-01', '00-00-00-02-01-01']}
|
params = {'mac_address': ['00-00-00-01-01-01', '00-00-00-02-01-01']}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
@ -7205,3 +7216,15 @@ class MACAddressTestCase(TestCase, ChangeLoggedFilterSetTests):
|
|||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
params = {'vminterface': [vm_interfaces[0].name, vm_interfaces[1].name]}
|
params = {'vminterface': [vm_interfaces[0].name, vm_interfaces[1].name]}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||||
|
|
||||||
|
def test_assigned(self):
|
||||||
|
params = {'assigned': True}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 8)
|
||||||
|
params = {'assigned': False}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 1)
|
||||||
|
|
||||||
|
def test_primary(self):
|
||||||
|
params = {'primary': True}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 6)
|
||||||
|
params = {'primary': False}
|
||||||
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 3)
|
||||||
|
|||||||
@ -967,6 +967,18 @@ class CableTestCase(TestCase):
|
|||||||
with self.assertRaises(ValidationError):
|
with self.assertRaises(ValidationError):
|
||||||
cable.clean()
|
cable.clean()
|
||||||
|
|
||||||
|
def test_cannot_cable_to_mark_connected(self):
|
||||||
|
"""
|
||||||
|
Test that a cable cannot be connected to an interface marked as connected.
|
||||||
|
"""
|
||||||
|
device1 = Device.objects.get(name='TestDevice1')
|
||||||
|
interface1 = Interface.objects.get(device__name='TestDevice2', name='eth1')
|
||||||
|
|
||||||
|
mark_connected_interface = Interface(device=device1, name='mark_connected1', mark_connected=True)
|
||||||
|
cable = Cable(a_terminations=[mark_connected_interface], b_terminations=[interface1])
|
||||||
|
with self.assertRaises(ValidationError):
|
||||||
|
cable.clean()
|
||||||
|
|
||||||
|
|
||||||
class VirtualDeviceContextTestCase(TestCase):
|
class VirtualDeviceContextTestCase(TestCase):
|
||||||
|
|
||||||
|
|||||||
@ -2885,6 +2885,43 @@ class InterfaceTestCase(ViewTestCases.DeviceComponentViewTestCase):
|
|||||||
self.client.post(self._get_url('bulk_delete'), data)
|
self.client.post(self._get_url('bulk_delete'), data)
|
||||||
self.assertEqual(device.interfaces.count(), 4) # Child & parent were both deleted
|
self.assertEqual(device.interfaces.count(), 4) # Child & parent were both deleted
|
||||||
|
|
||||||
|
def test_rename_select_all_spans_pages(self):
|
||||||
|
"""
|
||||||
|
Tests the bulk rename functionality for interfaces spanning multiple pages in the UI.
|
||||||
|
"""
|
||||||
|
device_name = 'DeviceRename'
|
||||||
|
device = create_test_device(device_name)
|
||||||
|
# Create > default page size (25) so selection spans multiple pages
|
||||||
|
for i in range(37):
|
||||||
|
Interface.objects.create(device=device, name=f'eth{i}')
|
||||||
|
|
||||||
|
self.add_permissions('dcim.change_interface')
|
||||||
|
|
||||||
|
# Filter to this device's interfaces to simulate a real list filter
|
||||||
|
get_qs = {'device_id': Device.objects.get(name=device_name).pk}
|
||||||
|
post_url = f'{self._get_url("bulk_rename")}?device_id={get_qs["device_id"]}'
|
||||||
|
|
||||||
|
# Preview step: ensure 37 selected (not just one page)
|
||||||
|
data = {'_preview': '1', '_all': '1', 'find': 'eth', 'replace': 'xe'}
|
||||||
|
response = self.client.post(post_url, data=data)
|
||||||
|
self.assertHttpStatus(response, 200)
|
||||||
|
self.assertEqual(len(response.context['selected_objects']), 37)
|
||||||
|
|
||||||
|
# Extract pk[] just like the browser would submit on Apply
|
||||||
|
# (either from the form's initial, or from selected_objects)
|
||||||
|
pk_list = response.context['form'].initial.get('pk')
|
||||||
|
if not pk_list:
|
||||||
|
pk_list = [obj.pk for obj in response.context['selected_objects']]
|
||||||
|
pk_list = [str(pk) for pk in pk_list]
|
||||||
|
|
||||||
|
# Apply step: include pk[] in the POST
|
||||||
|
apply_data = {'_apply': '1', '_all': '1', 'find': 'eth', 'replace': 'xe', 'pk': pk_list}
|
||||||
|
response = self.client.post(post_url, data=apply_data)
|
||||||
|
|
||||||
|
# On success the view redirects back to the return URL
|
||||||
|
self.assertHttpStatus(response, 302)
|
||||||
|
self.assertEqual(Interface.objects.filter(device=device, name__startswith='xe').count(), 37)
|
||||||
|
|
||||||
|
|
||||||
class FrontPortTestCase(ViewTestCases.DeviceComponentViewTestCase):
|
class FrontPortTestCase(ViewTestCases.DeviceComponentViewTestCase):
|
||||||
model = FrontPort
|
model = FrontPort
|
||||||
|
|||||||
@ -295,6 +295,7 @@ class RegionBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Region, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Region, 'bulk_rename', path='rename', detail=False)
|
||||||
class RegionBulkRenameView(generic.BulkRenameView):
|
class RegionBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Region.objects.all()
|
queryset = Region.objects.all()
|
||||||
|
filterset = filtersets.RegionFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Region, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Region, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -426,6 +427,7 @@ class SiteGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(SiteGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(SiteGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class SiteGroupBulkRenameView(generic.BulkRenameView):
|
class SiteGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = SiteGroup.objects.all()
|
queryset = SiteGroup.objects.all()
|
||||||
|
filterset = filtersets.SiteGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(SiteGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(SiteGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -516,6 +518,7 @@ class SiteBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Site, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Site, 'bulk_rename', path='rename', detail=False)
|
||||||
class SiteBulkRenameView(generic.BulkRenameView):
|
class SiteBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Site.objects.all()
|
queryset = Site.objects.all()
|
||||||
|
filterset = filtersets.SiteFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Site, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Site, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -625,6 +628,7 @@ class LocationBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Location, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Location, 'bulk_rename', path='rename', detail=False)
|
||||||
class LocationBulkRenameView(generic.BulkRenameView):
|
class LocationBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Location.objects.all()
|
queryset = Location.objects.all()
|
||||||
|
filterset = filtersets.LocationFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Location, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Location, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -695,6 +699,7 @@ class RackRoleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(RackRole, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(RackRole, 'bulk_rename', path='rename', detail=False)
|
||||||
class RackRoleBulkRenameView(generic.BulkRenameView):
|
class RackRoleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = RackRole.objects.all()
|
queryset = RackRole.objects.all()
|
||||||
|
filterset = filtersets.RackRoleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(RackRole, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(RackRole, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -760,6 +765,7 @@ class RackTypeBulkEditView(generic.BulkEditView):
|
|||||||
class RackTypeBulkRenameView(generic.BulkRenameView):
|
class RackTypeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = RackType.objects.all()
|
queryset = RackType.objects.all()
|
||||||
field_name = 'model'
|
field_name = 'model'
|
||||||
|
filterset = filtersets.RackTypeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(RackType, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(RackType, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -944,6 +950,7 @@ class RackBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Rack, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Rack, 'bulk_rename', path='rename', detail=False)
|
||||||
class RackBulkRenameView(generic.BulkRenameView):
|
class RackBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Rack.objects.all()
|
queryset = Rack.objects.all()
|
||||||
|
filterset = filtersets.RackFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Rack, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Rack, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1083,6 +1090,7 @@ class ManufacturerBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Manufacturer, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Manufacturer, 'bulk_rename', path='rename', detail=False)
|
||||||
class ManufacturerBulkRenameView(generic.BulkRenameView):
|
class ManufacturerBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Manufacturer.objects.all()
|
queryset = Manufacturer.objects.all()
|
||||||
|
filterset = filtersets.ManufacturerFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Manufacturer, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Manufacturer, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1336,6 +1344,7 @@ class DeviceTypeBulkEditView(generic.BulkEditView):
|
|||||||
class DeviceTypeBulkRenameView(generic.BulkRenameView):
|
class DeviceTypeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = DeviceType.objects.all()
|
queryset = DeviceType.objects.all()
|
||||||
field_name = 'model'
|
field_name = 'model'
|
||||||
|
filterset = filtersets.DeviceTypeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(DeviceType, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(DeviceType, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1397,6 +1406,7 @@ class ModuleTypeProfileBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ModuleTypeProfile, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ModuleTypeProfile, 'bulk_rename', path='rename', detail=False)
|
||||||
class ModuleTypeProfileBulkRenameView(generic.BulkRenameView):
|
class ModuleTypeProfileBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ModuleTypeProfile.objects.all()
|
queryset = ModuleTypeProfile.objects.all()
|
||||||
|
filterset = filtersets.ModuleTypeProfileFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ModuleTypeProfile, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ModuleTypeProfile, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1612,6 +1622,7 @@ class ModuleTypeBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ModuleType, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ModuleType, 'bulk_rename', path='rename', detail=False)
|
||||||
class ModuleTypeBulkRenameView(generic.BulkRenameView):
|
class ModuleTypeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ModuleType.objects.all()
|
queryset = ModuleType.objects.all()
|
||||||
|
filterset = filtersets.ModuleTypeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ModuleType, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ModuleType, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -2100,6 +2111,7 @@ class DeviceRoleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(DeviceRole, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(DeviceRole, 'bulk_rename', path='rename', detail=False)
|
||||||
class DeviceRoleBulkRenameView(generic.BulkRenameView):
|
class DeviceRoleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = DeviceRole.objects.all()
|
queryset = DeviceRole.objects.all()
|
||||||
|
filterset = filtersets.DeviceRoleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(DeviceRole, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(DeviceRole, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -2175,6 +2187,7 @@ class PlatformBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Platform, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Platform, 'bulk_rename', path='rename', detail=False)
|
||||||
class PlatformBulkRenameView(generic.BulkRenameView):
|
class PlatformBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Platform.objects.all()
|
queryset = Platform.objects.all()
|
||||||
|
filterset = filtersets.PlatformFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Platform, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Platform, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -2582,6 +2595,7 @@ class ConsolePortBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ConsolePort, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ConsolePort, 'bulk_rename', path='rename', detail=False)
|
||||||
class ConsolePortBulkRenameView(generic.BulkRenameView):
|
class ConsolePortBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ConsolePort.objects.all()
|
queryset = ConsolePort.objects.all()
|
||||||
|
filterset = filtersets.ConsolePortFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ConsolePort, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(ConsolePort, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -2652,6 +2666,7 @@ class ConsoleServerPortBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ConsoleServerPort, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ConsoleServerPort, 'bulk_rename', path='rename', detail=False)
|
||||||
class ConsoleServerPortBulkRenameView(generic.BulkRenameView):
|
class ConsoleServerPortBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ConsoleServerPort.objects.all()
|
queryset = ConsoleServerPort.objects.all()
|
||||||
|
filterset = filtersets.ConsoleServerPortFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ConsoleServerPort, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(ConsoleServerPort, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -2722,6 +2737,7 @@ class PowerPortBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(PowerPort, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(PowerPort, 'bulk_rename', path='rename', detail=False)
|
||||||
class PowerPortBulkRenameView(generic.BulkRenameView):
|
class PowerPortBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = PowerPort.objects.all()
|
queryset = PowerPort.objects.all()
|
||||||
|
filterset = filtersets.PowerPortFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(PowerPort, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(PowerPort, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -2792,6 +2808,7 @@ class PowerOutletBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(PowerOutlet, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(PowerOutlet, 'bulk_rename', path='rename', detail=False)
|
||||||
class PowerOutletBulkRenameView(generic.BulkRenameView):
|
class PowerOutletBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = PowerOutlet.objects.all()
|
queryset = PowerOutlet.objects.all()
|
||||||
|
filterset = filtersets.PowerOutletFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(PowerOutlet, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(PowerOutlet, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -2934,6 +2951,7 @@ class InterfaceBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Interface, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Interface, 'bulk_rename', path='rename', detail=False)
|
||||||
class InterfaceBulkRenameView(generic.BulkRenameView):
|
class InterfaceBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Interface.objects.all()
|
queryset = Interface.objects.all()
|
||||||
|
filterset = filtersets.InterfaceFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Interface, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(Interface, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -3005,6 +3023,7 @@ class FrontPortBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(FrontPort, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(FrontPort, 'bulk_rename', path='rename', detail=False)
|
||||||
class FrontPortBulkRenameView(generic.BulkRenameView):
|
class FrontPortBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = FrontPort.objects.all()
|
queryset = FrontPort.objects.all()
|
||||||
|
filterset = filtersets.FrontPortFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(FrontPort, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(FrontPort, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -3080,6 +3099,7 @@ class RearPortBulkRenameView(generic.BulkRenameView):
|
|||||||
@register_model_view(RearPort, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(RearPort, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
class RearPortBulkDisconnectView(BulkDisconnectView):
|
class RearPortBulkDisconnectView(BulkDisconnectView):
|
||||||
queryset = RearPort.objects.all()
|
queryset = RearPort.objects.all()
|
||||||
|
filterset = filtersets.RearPortFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(RearPort, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(RearPort, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3145,6 +3165,7 @@ class ModuleBayBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ModuleBay, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ModuleBay, 'bulk_rename', path='rename', detail=False)
|
||||||
class ModuleBayBulkRenameView(generic.BulkRenameView):
|
class ModuleBayBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ModuleBay.objects.all()
|
queryset = ModuleBay.objects.all()
|
||||||
|
filterset = filtersets.ModuleBayFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ModuleBay, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ModuleBay, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3287,6 +3308,7 @@ class DeviceBayBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(DeviceBay, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(DeviceBay, 'bulk_rename', path='rename', detail=False)
|
||||||
class DeviceBayBulkRenameView(generic.BulkRenameView):
|
class DeviceBayBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = DeviceBay.objects.all()
|
queryset = DeviceBay.objects.all()
|
||||||
|
filterset = filtersets.DeviceBayFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(DeviceBay, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(DeviceBay, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3348,6 +3370,7 @@ class InventoryItemBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(InventoryItem, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(InventoryItem, 'bulk_rename', path='rename', detail=False)
|
||||||
class InventoryItemBulkRenameView(generic.BulkRenameView):
|
class InventoryItemBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = InventoryItem.objects.all()
|
queryset = InventoryItem.objects.all()
|
||||||
|
filterset = filtersets.InventoryItemFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(InventoryItem, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(InventoryItem, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3431,6 +3454,7 @@ class InventoryItemRoleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(InventoryItemRole, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(InventoryItemRole, 'bulk_rename', path='rename', detail=False)
|
||||||
class InventoryItemRoleBulkRenameView(generic.BulkRenameView):
|
class InventoryItemRoleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = InventoryItemRole.objects.all()
|
queryset = InventoryItemRole.objects.all()
|
||||||
|
filterset = filtersets.InventoryItemRoleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(InventoryItemRole, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(InventoryItemRole, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3634,6 +3658,7 @@ class CableBulkEditView(generic.BulkEditView):
|
|||||||
class CableBulkRenameView(generic.BulkRenameView):
|
class CableBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Cable.objects.all()
|
queryset = Cable.objects.all()
|
||||||
field_name = 'label'
|
field_name = 'label'
|
||||||
|
filterset = filtersets.CableFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Cable, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Cable, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3931,6 +3956,7 @@ class VirtualChassisBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VirtualChassis, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VirtualChassis, 'bulk_rename', path='rename', detail=False)
|
||||||
class VirtualChassisBulkRenameView(generic.BulkRenameView):
|
class VirtualChassisBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VirtualChassis.objects.all()
|
queryset = VirtualChassis.objects.all()
|
||||||
|
filterset = filtersets.VirtualChassisFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VirtualChassis, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VirtualChassis, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -3993,6 +4019,7 @@ class PowerPanelBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(PowerPanel, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(PowerPanel, 'bulk_rename', path='rename', detail=False)
|
||||||
class PowerPanelBulkRenameView(generic.BulkRenameView):
|
class PowerPanelBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = PowerPanel.objects.all()
|
queryset = PowerPanel.objects.all()
|
||||||
|
filterset = filtersets.PowerPanelFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(PowerPanel, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(PowerPanel, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -4050,6 +4077,7 @@ class PowerFeedBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(PowerFeed, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(PowerFeed, 'bulk_rename', path='rename', detail=False)
|
||||||
class PowerFeedBulkRenameView(generic.BulkRenameView):
|
class PowerFeedBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = PowerFeed.objects.all()
|
queryset = PowerFeed.objects.all()
|
||||||
|
filterset = filtersets.PowerFeedFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(PowerFeed, 'bulk_disconnect', path='disconnect', detail=False)
|
@register_model_view(PowerFeed, 'bulk_disconnect', path='disconnect', detail=False)
|
||||||
@ -4128,6 +4156,7 @@ class VirtualDeviceContextBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VirtualDeviceContext, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VirtualDeviceContext, 'bulk_rename', path='rename', detail=False)
|
||||||
class VirtualDeviceContextBulkRenameView(generic.BulkRenameView):
|
class VirtualDeviceContextBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VirtualDeviceContext.objects.all()
|
queryset = VirtualDeviceContext.objects.all()
|
||||||
|
filterset = filtersets.VirtualDeviceContextFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VirtualDeviceContext, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VirtualDeviceContext, 'bulk_delete', path='delete', detail=False)
|
||||||
|
|||||||
@ -5,6 +5,7 @@ from rest_framework import serializers
|
|||||||
from core.api.serializers_.jobs import JobSerializer
|
from core.api.serializers_.jobs import JobSerializer
|
||||||
from extras.models import Script
|
from extras.models import Script
|
||||||
from netbox.api.serializers import ValidatedModelSerializer
|
from netbox.api.serializers import ValidatedModelSerializer
|
||||||
|
from utilities.datetime import local_now
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
'ScriptDetailSerializer',
|
'ScriptDetailSerializer',
|
||||||
@ -66,11 +67,31 @@ class ScriptInputSerializer(serializers.Serializer):
|
|||||||
interval = serializers.IntegerField(required=False, allow_null=True)
|
interval = serializers.IntegerField(required=False, allow_null=True)
|
||||||
|
|
||||||
def validate_schedule_at(self, value):
|
def validate_schedule_at(self, value):
|
||||||
if value and not self.context['script'].python_class.scheduling_enabled:
|
"""
|
||||||
raise serializers.ValidationError(_("Scheduling is not enabled for this script."))
|
Validates the specified schedule time for a script execution.
|
||||||
|
"""
|
||||||
|
if value:
|
||||||
|
if not self.context['script'].python_class.scheduling_enabled:
|
||||||
|
raise serializers.ValidationError(_('Scheduling is not enabled for this script.'))
|
||||||
|
if value < local_now():
|
||||||
|
raise serializers.ValidationError(_('Scheduled time must be in the future.'))
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def validate_interval(self, value):
|
def validate_interval(self, value):
|
||||||
|
"""
|
||||||
|
Validates the provided interval based on the script's scheduling configuration.
|
||||||
|
"""
|
||||||
if value and not self.context['script'].python_class.scheduling_enabled:
|
if value and not self.context['script'].python_class.scheduling_enabled:
|
||||||
raise serializers.ValidationError(_("Scheduling is not enabled for this script."))
|
raise serializers.ValidationError(_('Scheduling is not enabled for this script.'))
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
def validate(self, data):
|
||||||
|
"""
|
||||||
|
Validates the given data and ensures the necessary fields are populated.
|
||||||
|
"""
|
||||||
|
# Set the schedule_at time to now if only an interval is provided
|
||||||
|
# while handling the case where schedule_at is null.
|
||||||
|
if data.get('interval') and not data.get('schedule_at'):
|
||||||
|
data['schedule_at'] = local_now()
|
||||||
|
|
||||||
|
return super().validate(data)
|
||||||
|
|||||||
@ -536,6 +536,15 @@ class CustomField(CloningMixin, ExportTemplatesMixin, OwnerMixin, ChangeLoggedMo
|
|||||||
# URL
|
# URL
|
||||||
elif self.type == CustomFieldTypeChoices.TYPE_URL:
|
elif self.type == CustomFieldTypeChoices.TYPE_URL:
|
||||||
field = LaxURLField(assume_scheme='https', required=required, initial=initial)
|
field = LaxURLField(assume_scheme='https', required=required, initial=initial)
|
||||||
|
if self.validation_regex:
|
||||||
|
field.validators = [
|
||||||
|
RegexValidator(
|
||||||
|
regex=self.validation_regex,
|
||||||
|
message=mark_safe(_("Values must match this regex: <code>{regex}</code>").format(
|
||||||
|
regex=escape(self.validation_regex)
|
||||||
|
))
|
||||||
|
)
|
||||||
|
]
|
||||||
|
|
||||||
# JSON
|
# JSON
|
||||||
elif self.type == CustomFieldTypeChoices.TYPE_JSON:
|
elif self.type == CustomFieldTypeChoices.TYPE_JSON:
|
||||||
@ -685,6 +694,13 @@ class CustomField(CloningMixin, ExportTemplatesMixin, OwnerMixin, ChangeLoggedMo
|
|||||||
if self.validation_regex and not re.match(self.validation_regex, value):
|
if self.validation_regex and not re.match(self.validation_regex, value):
|
||||||
raise ValidationError(_("Value must match regex '{regex}'").format(regex=self.validation_regex))
|
raise ValidationError(_("Value must match regex '{regex}'").format(regex=self.validation_regex))
|
||||||
|
|
||||||
|
# Validate URL field
|
||||||
|
elif self.type == CustomFieldTypeChoices.TYPE_URL:
|
||||||
|
if type(value) is not str:
|
||||||
|
raise ValidationError(_("Value must be a string."))
|
||||||
|
if self.validation_regex and not re.match(self.validation_regex, value):
|
||||||
|
raise ValidationError(_("Value must match regex '{regex}'").format(regex=self.validation_regex))
|
||||||
|
|
||||||
# Validate integer
|
# Validate integer
|
||||||
elif self.type == CustomFieldTypeChoices.TYPE_INTEGER:
|
elif self.type == CustomFieldTypeChoices.TYPE_INTEGER:
|
||||||
if type(value) is not int:
|
if type(value) is not int:
|
||||||
|
|||||||
@ -901,16 +901,16 @@ class ConfigTemplateTest(APIViewTestCases.APIViewTestCase):
|
|||||||
class ScriptTest(APITestCase):
|
class ScriptTest(APITestCase):
|
||||||
|
|
||||||
class TestScriptClass(PythonClass):
|
class TestScriptClass(PythonClass):
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
name = "Test script"
|
name = 'Test script'
|
||||||
|
commit = True
|
||||||
|
scheduling_enabled = True
|
||||||
|
|
||||||
var1 = StringVar()
|
var1 = StringVar()
|
||||||
var2 = IntegerVar()
|
var2 = IntegerVar()
|
||||||
var3 = BooleanVar()
|
var3 = BooleanVar()
|
||||||
|
|
||||||
def run(self, data, commit=True):
|
def run(self, data, commit=True):
|
||||||
|
|
||||||
self.log_info(data['var1'])
|
self.log_info(data['var1'])
|
||||||
self.log_success(data['var2'])
|
self.log_success(data['var2'])
|
||||||
self.log_failure(data['var3'])
|
self.log_failure(data['var3'])
|
||||||
@ -921,14 +921,16 @@ class ScriptTest(APITestCase):
|
|||||||
def setUpTestData(cls):
|
def setUpTestData(cls):
|
||||||
module = ScriptModule.objects.create(
|
module = ScriptModule.objects.create(
|
||||||
file_root=ManagedFileRootPathChoices.SCRIPTS,
|
file_root=ManagedFileRootPathChoices.SCRIPTS,
|
||||||
file_path='/var/tmp/script.py'
|
file_path='script.py',
|
||||||
)
|
)
|
||||||
Script.objects.create(
|
script = Script.objects.create(
|
||||||
module=module,
|
module=module,
|
||||||
name="Test script",
|
name='Test script',
|
||||||
is_executable=True,
|
is_executable=True,
|
||||||
)
|
)
|
||||||
|
cls.url = reverse('extras-api:script-detail', kwargs={'pk': script.pk})
|
||||||
|
|
||||||
|
@property
|
||||||
def python_class(self):
|
def python_class(self):
|
||||||
return self.TestScriptClass
|
return self.TestScriptClass
|
||||||
|
|
||||||
@ -941,7 +943,7 @@ class ScriptTest(APITestCase):
|
|||||||
def test_get_script(self):
|
def test_get_script(self):
|
||||||
module = ScriptModule.objects.get(
|
module = ScriptModule.objects.get(
|
||||||
file_root=ManagedFileRootPathChoices.SCRIPTS,
|
file_root=ManagedFileRootPathChoices.SCRIPTS,
|
||||||
file_path='/var/tmp/script.py'
|
file_path='script.py',
|
||||||
)
|
)
|
||||||
script = module.scripts.all().first()
|
script = module.scripts.all().first()
|
||||||
url = reverse('extras-api:script-detail', kwargs={'pk': script.pk})
|
url = reverse('extras-api:script-detail', kwargs={'pk': script.pk})
|
||||||
@ -952,6 +954,76 @@ class ScriptTest(APITestCase):
|
|||||||
self.assertEqual(response.data['vars']['var2'], 'IntegerVar')
|
self.assertEqual(response.data['vars']['var2'], 'IntegerVar')
|
||||||
self.assertEqual(response.data['vars']['var3'], 'BooleanVar')
|
self.assertEqual(response.data['vars']['var3'], 'BooleanVar')
|
||||||
|
|
||||||
|
def test_schedule_script_past_time_rejected(self):
|
||||||
|
"""
|
||||||
|
Scheduling with past schedule_at should fail.
|
||||||
|
"""
|
||||||
|
self.add_permissions('extras.run_script')
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
'data': {'var1': 'hello', 'var2': 1, 'var3': False},
|
||||||
|
'commit': True,
|
||||||
|
'schedule_at': now() - datetime.timedelta(hours=1),
|
||||||
|
}
|
||||||
|
response = self.client.post(self.url, payload, format='json', **self.header)
|
||||||
|
|
||||||
|
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
||||||
|
self.assertIn('schedule_at', response.data)
|
||||||
|
# Be tolerant of exact wording but ensure we failed on schedule_at being in the past
|
||||||
|
self.assertIn('future', str(response.data['schedule_at']).lower())
|
||||||
|
|
||||||
|
def test_schedule_script_interval_only(self):
|
||||||
|
"""
|
||||||
|
Interval without schedule_at should auto-set schedule_at now.
|
||||||
|
"""
|
||||||
|
self.add_permissions('extras.run_script')
|
||||||
|
|
||||||
|
payload = {
|
||||||
|
'data': {'var1': 'hello', 'var2': 1, 'var3': False},
|
||||||
|
'commit': True,
|
||||||
|
'interval': 60,
|
||||||
|
}
|
||||||
|
response = self.client.post(self.url, payload, format='json', **self.header)
|
||||||
|
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
# The latest job is returned in the script detail serializer under "result"
|
||||||
|
self.assertIn('result', response.data)
|
||||||
|
self.assertEqual(response.data['result']['interval'], 60)
|
||||||
|
# Ensure a start time was autopopulated
|
||||||
|
self.assertIsNotNone(response.data['result']['scheduled'])
|
||||||
|
|
||||||
|
def test_schedule_script_when_disabled(self):
|
||||||
|
"""
|
||||||
|
Scheduling should fail when script.scheduling_enabled=False.
|
||||||
|
"""
|
||||||
|
self.add_permissions('extras.run_script')
|
||||||
|
|
||||||
|
# Temporarily disable scheduling on the in-test Python class
|
||||||
|
original = getattr(self.TestScriptClass.Meta, 'scheduling_enabled', True)
|
||||||
|
self.TestScriptClass.Meta.scheduling_enabled = False
|
||||||
|
base = {
|
||||||
|
'data': {'var1': 'hello', 'var2': 1, 'var3': False},
|
||||||
|
'commit': True,
|
||||||
|
}
|
||||||
|
# Check both schedule_at and interval paths
|
||||||
|
cases = [
|
||||||
|
{**base, 'schedule_at': now() + datetime.timedelta(minutes=5)},
|
||||||
|
{**base, 'interval': 60},
|
||||||
|
]
|
||||||
|
try:
|
||||||
|
for case in cases:
|
||||||
|
with self.subTest(case=list(case.keys())):
|
||||||
|
response = self.client.post(self.url, case, format='json', **self.header)
|
||||||
|
|
||||||
|
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
||||||
|
# Error should be attached to whichever field we used
|
||||||
|
key = 'schedule_at' if 'schedule_at' in case else 'interval'
|
||||||
|
self.assertIn(key, response.data)
|
||||||
|
self.assertIn('scheduling is not enabled', str(response.data[key]).lower())
|
||||||
|
finally:
|
||||||
|
# Restore the original setting for other tests
|
||||||
|
self.TestScriptClass.Meta.scheduling_enabled = original
|
||||||
|
|
||||||
|
|
||||||
class CreatedUpdatedFilterTest(APITestCase):
|
class CreatedUpdatedFilterTest(APITestCase):
|
||||||
|
|
||||||
|
|||||||
@ -1300,6 +1300,28 @@ class CustomFieldAPITest(APITestCase):
|
|||||||
response = self.client.patch(url, data, format='json', **self.header)
|
response = self.client.patch(url, data, format='json', **self.header)
|
||||||
self.assertHttpStatus(response, status.HTTP_200_OK)
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
|
||||||
|
def test_url_regex_validation(self):
|
||||||
|
"""
|
||||||
|
Test that validation_regex is applied to URL custom fields (fixes #20498).
|
||||||
|
"""
|
||||||
|
site2 = Site.objects.get(name='Site 2')
|
||||||
|
url = reverse('dcim-api:site-detail', kwargs={'pk': site2.pk})
|
||||||
|
self.add_permissions('dcim.change_site')
|
||||||
|
|
||||||
|
cf_url = CustomField.objects.get(name='url_field')
|
||||||
|
cf_url.validation_regex = r'^https://' # Require HTTPS
|
||||||
|
cf_url.save()
|
||||||
|
|
||||||
|
# Test invalid URL (http instead of https)
|
||||||
|
data = {'custom_fields': {'url_field': 'http://example.com'}}
|
||||||
|
response = self.client.patch(url, data, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
||||||
|
|
||||||
|
# Test valid URL (https)
|
||||||
|
data = {'custom_fields': {'url_field': 'https://example.com'}}
|
||||||
|
response = self.client.patch(url, data, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
|
||||||
def test_uniqueness_validation(self):
|
def test_uniqueness_validation(self):
|
||||||
# Create a unique custom field
|
# Create a unique custom field
|
||||||
cf_text = CustomField.objects.get(name='text_field')
|
cf_text = CustomField.objects.get(name='text_field')
|
||||||
|
|||||||
@ -1,11 +1,14 @@
|
|||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
from django.test import tag
|
||||||
|
|
||||||
|
from core.choices import ManagedFileRootPathChoices
|
||||||
from core.events import *
|
from core.events import *
|
||||||
from core.models import ObjectType
|
from core.models import ObjectType
|
||||||
from dcim.models import DeviceType, Manufacturer, Site
|
from dcim.models import DeviceType, Manufacturer, Site
|
||||||
from extras.choices import *
|
from extras.choices import *
|
||||||
from extras.models import *
|
from extras.models import *
|
||||||
|
from extras.scripts import Script as PythonClass, IntegerVar, BooleanVar
|
||||||
from users.models import Group, User
|
from users.models import Group, User
|
||||||
from utilities.testing import ViewTestCases, TestCase
|
from utilities.testing import ViewTestCases, TestCase
|
||||||
|
|
||||||
@ -897,3 +900,70 @@ class ScriptListViewTest(TestCase):
|
|||||||
response = self.client.get(url, {'embedded': 'true'})
|
response = self.client.get(url, {'embedded': 'true'})
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
self.assertTemplateUsed(response, 'extras/inc/script_list_content.html')
|
self.assertTemplateUsed(response, 'extras/inc/script_list_content.html')
|
||||||
|
|
||||||
|
|
||||||
|
class ScriptValidationErrorTest(TestCase):
|
||||||
|
user_permissions = ['extras.view_script', 'extras.run_script']
|
||||||
|
|
||||||
|
class TestScriptMixin:
|
||||||
|
bar = IntegerVar(min_value=0, max_value=30, default=30)
|
||||||
|
|
||||||
|
class TestScriptClass(TestScriptMixin, PythonClass):
|
||||||
|
class Meta:
|
||||||
|
name = 'Test script'
|
||||||
|
commit_default = False
|
||||||
|
fieldsets = (("Logging", ("debug_mode",)),)
|
||||||
|
|
||||||
|
debug_mode = BooleanVar(default=False)
|
||||||
|
|
||||||
|
def run(self, data, commit):
|
||||||
|
return "Complete"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpTestData(cls):
|
||||||
|
module = ScriptModule.objects.create(file_root=ManagedFileRootPathChoices.SCRIPTS, file_path='test_script.py')
|
||||||
|
cls.script = Script.objects.create(module=module, name='Test script', is_executable=True)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
Script.python_class = property(lambda self: ScriptValidationErrorTest.TestScriptClass)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_script_validation_error_displays_message(self):
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
url = reverse('extras:script', kwargs={'pk': self.script.pk})
|
||||||
|
|
||||||
|
with patch('extras.views.get_workers_for_queue', return_value=['worker']):
|
||||||
|
response = self.client.post(url, {'debug_mode': 'true', '_commit': 'true'})
|
||||||
|
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
messages = list(response.context['messages'])
|
||||||
|
self.assertEqual(len(messages), 1)
|
||||||
|
self.assertEqual(str(messages[0]), "bar: This field is required.")
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_script_validation_error_no_toast_for_fieldset_fields(self):
|
||||||
|
from unittest.mock import patch, PropertyMock
|
||||||
|
|
||||||
|
class FieldsetScript(PythonClass):
|
||||||
|
class Meta:
|
||||||
|
name = 'Fieldset test'
|
||||||
|
commit_default = False
|
||||||
|
fieldsets = (("Fields", ("required_field",)),)
|
||||||
|
|
||||||
|
required_field = IntegerVar(min_value=10)
|
||||||
|
|
||||||
|
def run(self, data, commit):
|
||||||
|
return "Complete"
|
||||||
|
|
||||||
|
url = reverse('extras:script', kwargs={'pk': self.script.pk})
|
||||||
|
|
||||||
|
with patch.object(Script, 'python_class', new_callable=PropertyMock) as mock_python_class:
|
||||||
|
mock_python_class.return_value = FieldsetScript
|
||||||
|
with patch('extras.views.get_workers_for_queue', return_value=['worker']):
|
||||||
|
response = self.client.post(url, {'required_field': '5', '_commit': 'true'})
|
||||||
|
|
||||||
|
self.assertEqual(response.status_code, 200)
|
||||||
|
messages = list(response.context['messages'])
|
||||||
|
self.assertEqual(len(messages), 0)
|
||||||
|
|||||||
@ -4,7 +4,7 @@ from django.contrib.auth.mixins import LoginRequiredMixin
|
|||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django.core.paginator import EmptyPage
|
from django.core.paginator import EmptyPage
|
||||||
from django.db.models import Count, Q
|
from django.db.models import Count, Q
|
||||||
from django.http import HttpResponseBadRequest, HttpResponseForbidden, HttpResponse
|
from django.http import HttpResponseBadRequest, HttpResponseForbidden, HttpResponse, Http404
|
||||||
from django.shortcuts import get_object_or_404, redirect, render
|
from django.shortcuts import get_object_or_404, redirect, render
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
@ -25,7 +25,7 @@ from netbox.object_actions import *
|
|||||||
from netbox.views import generic
|
from netbox.views import generic
|
||||||
from netbox.views.generic.mixins import TableMixin
|
from netbox.views.generic.mixins import TableMixin
|
||||||
from utilities.forms import ConfirmationForm, get_field_value
|
from utilities.forms import ConfirmationForm, get_field_value
|
||||||
from utilities.htmx import htmx_partial
|
from utilities.htmx import htmx_partial, htmx_maybe_redirect_current_page
|
||||||
from utilities.paginator import EnhancedPaginator, get_paginate_count
|
from utilities.paginator import EnhancedPaginator, get_paginate_count
|
||||||
from utilities.query import count_related
|
from utilities.query import count_related
|
||||||
from utilities.querydict import normalize_querydict
|
from utilities.querydict import normalize_querydict
|
||||||
@ -101,6 +101,7 @@ class CustomFieldBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(CustomField, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(CustomField, 'bulk_rename', path='rename', detail=False)
|
||||||
class CustomFieldBulkRenameView(generic.BulkRenameView):
|
class CustomFieldBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = CustomField.objects.all()
|
queryset = CustomField.objects.all()
|
||||||
|
filterset = filtersets.CustomFieldFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(CustomField, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(CustomField, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -175,6 +176,7 @@ class CustomFieldChoiceSetBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(CustomFieldChoiceSet, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(CustomFieldChoiceSet, 'bulk_rename', path='rename', detail=False)
|
||||||
class CustomFieldChoiceSetBulkRenameView(generic.BulkRenameView):
|
class CustomFieldChoiceSetBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = CustomFieldChoiceSet.objects.all()
|
queryset = CustomFieldChoiceSet.objects.all()
|
||||||
|
filterset = filtersets.CustomFieldChoiceSetFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(CustomFieldChoiceSet, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(CustomFieldChoiceSet, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -230,6 +232,7 @@ class CustomLinkBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(CustomLink, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(CustomLink, 'bulk_rename', path='rename', detail=False)
|
||||||
class CustomLinkBulkRenameView(generic.BulkRenameView):
|
class CustomLinkBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = CustomLink.objects.all()
|
queryset = CustomLink.objects.all()
|
||||||
|
filterset = filtersets.CustomLinkFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(CustomLink, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(CustomLink, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -286,6 +289,7 @@ class ExportTemplateBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ExportTemplate, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ExportTemplate, 'bulk_rename', path='rename', detail=False)
|
||||||
class ExportTemplateBulkRenameView(generic.BulkRenameView):
|
class ExportTemplateBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ExportTemplate.objects.all()
|
queryset = ExportTemplate.objects.all()
|
||||||
|
filterset = filtersets.ExportTemplateFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ExportTemplate, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ExportTemplate, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -351,6 +355,7 @@ class SavedFilterBulkEditView(SharedObjectViewMixin, generic.BulkEditView):
|
|||||||
@register_model_view(SavedFilter, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(SavedFilter, 'bulk_rename', path='rename', detail=False)
|
||||||
class SavedFilterBulkRenameView(generic.BulkRenameView):
|
class SavedFilterBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = SavedFilter.objects.all()
|
queryset = SavedFilter.objects.all()
|
||||||
|
filterset = filtersets.SavedFilterFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(SavedFilter, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(SavedFilter, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -413,6 +418,7 @@ class TableConfigBulkEditView(SharedObjectViewMixin, generic.BulkEditView):
|
|||||||
@register_model_view(TableConfig, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(TableConfig, 'bulk_rename', path='rename', detail=False)
|
||||||
class TableConfigBulkRenameView(generic.BulkRenameView):
|
class TableConfigBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = TableConfig.objects.all()
|
queryset = TableConfig.objects.all()
|
||||||
|
filterset = filtersets.TableConfigFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(TableConfig, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(TableConfig, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -499,6 +505,7 @@ class NotificationGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(NotificationGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(NotificationGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class NotificationGroupBulkRenameView(generic.BulkRenameView):
|
class NotificationGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = NotificationGroup.objects.all()
|
queryset = NotificationGroup.objects.all()
|
||||||
|
filterset = filtersets.NotificationGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(NotificationGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(NotificationGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -518,8 +525,9 @@ class NotificationsView(LoginRequiredMixin, View):
|
|||||||
"""
|
"""
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
return render(request, 'htmx/notifications.html', {
|
return render(request, 'htmx/notifications.html', {
|
||||||
'notifications': request.user.notifications.unread(),
|
'notifications': request.user.notifications.unread()[:10],
|
||||||
'total_count': request.user.notifications.count(),
|
'total_count': request.user.notifications.count(),
|
||||||
|
'unread_count': request.user.notifications.unread().count(),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
@ -528,6 +536,7 @@ class NotificationReadView(LoginRequiredMixin, View):
|
|||||||
"""
|
"""
|
||||||
Mark the Notification read and redirect the user to its attached object.
|
Mark the Notification read and redirect the user to its attached object.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def get(self, request, pk):
|
def get(self, request, pk):
|
||||||
# Mark the Notification as read
|
# Mark the Notification as read
|
||||||
notification = get_object_or_404(request.user.notifications, pk=pk)
|
notification = get_object_or_404(request.user.notifications, pk=pk)
|
||||||
@ -541,18 +550,48 @@ class NotificationReadView(LoginRequiredMixin, View):
|
|||||||
return redirect('account:notifications')
|
return redirect('account:notifications')
|
||||||
|
|
||||||
|
|
||||||
|
@register_model_view(Notification, name='dismiss_all', path='dismiss-all', detail=False)
|
||||||
|
class NotificationDismissAllView(LoginRequiredMixin, View):
|
||||||
|
"""
|
||||||
|
Convenience view to clear all *unread* notifications for the current user.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get(self, request):
|
||||||
|
request.user.notifications.unread().delete()
|
||||||
|
if htmx_partial(request):
|
||||||
|
# If a user is currently on the notification page, redirect there (full repaint)
|
||||||
|
redirect_resp = htmx_maybe_redirect_current_page(request, 'account:notifications', preserve_query=True)
|
||||||
|
if redirect_resp:
|
||||||
|
return redirect_resp
|
||||||
|
|
||||||
|
return render(request, 'htmx/notifications.html', {
|
||||||
|
'notifications': request.user.notifications.unread()[:10],
|
||||||
|
'total_count': request.user.notifications.count(),
|
||||||
|
'unread_count': request.user.notifications.unread().count(),
|
||||||
|
})
|
||||||
|
return redirect('account:notifications')
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Notification, 'dismiss')
|
@register_model_view(Notification, 'dismiss')
|
||||||
class NotificationDismissView(LoginRequiredMixin, View):
|
class NotificationDismissView(LoginRequiredMixin, View):
|
||||||
"""
|
"""
|
||||||
A convenience view which allows deleting notifications with one click.
|
A convenience view which allows deleting notifications with one click.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def get(self, request, pk):
|
def get(self, request, pk):
|
||||||
notification = get_object_or_404(request.user.notifications, pk=pk)
|
notification = get_object_or_404(request.user.notifications, pk=pk)
|
||||||
notification.delete()
|
notification.delete()
|
||||||
|
|
||||||
if htmx_partial(request):
|
if htmx_partial(request):
|
||||||
|
# If a user is currently on the notification page, redirect there (full repaint)
|
||||||
|
redirect_resp = htmx_maybe_redirect_current_page(request, 'account:notifications', preserve_query=True)
|
||||||
|
if redirect_resp:
|
||||||
|
return redirect_resp
|
||||||
|
|
||||||
return render(request, 'htmx/notifications.html', {
|
return render(request, 'htmx/notifications.html', {
|
||||||
'notifications': request.user.notifications.unread()[:10],
|
'notifications': request.user.notifications.unread()[:10],
|
||||||
|
'total_count': request.user.notifications.count(),
|
||||||
|
'unread_count': request.user.notifications.unread().count(),
|
||||||
})
|
})
|
||||||
|
|
||||||
return redirect('account:notifications')
|
return redirect('account:notifications')
|
||||||
@ -650,6 +689,7 @@ class WebhookBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Webhook, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Webhook, 'bulk_rename', path='rename', detail=False)
|
||||||
class WebhookBulkRenameView(generic.BulkRenameView):
|
class WebhookBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Webhook.objects.all()
|
queryset = Webhook.objects.all()
|
||||||
|
filterset = filtersets.WebhookFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Webhook, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Webhook, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -705,6 +745,7 @@ class EventRuleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(EventRule, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(EventRule, 'bulk_rename', path='rename', detail=False)
|
||||||
class EventRuleBulkRenameView(generic.BulkRenameView):
|
class EventRuleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = EventRule.objects.all()
|
queryset = EventRule.objects.all()
|
||||||
|
filterset = filtersets.EventRuleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(EventRule, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(EventRule, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -841,6 +882,7 @@ class ConfigContextProfileBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ConfigContextProfile, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ConfigContextProfile, 'bulk_rename', path='rename', detail=False)
|
||||||
class ConfigContextProfileBulkRenameView(generic.BulkRenameView):
|
class ConfigContextProfileBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ConfigContextProfile.objects.all()
|
queryset = ConfigContextProfile.objects.all()
|
||||||
|
filterset = filtersets.ConfigContextProfileFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ConfigContextProfile, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ConfigContextProfile, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -929,6 +971,7 @@ class ConfigContextBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ConfigContext, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ConfigContext, 'bulk_rename', path='rename', detail=False)
|
||||||
class ConfigContextBulkRenameView(generic.BulkRenameView):
|
class ConfigContextBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ConfigContext.objects.all()
|
queryset = ConfigContext.objects.all()
|
||||||
|
filterset = filtersets.ConfigContextFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ConfigContext, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ConfigContext, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1020,6 +1063,7 @@ class ConfigTemplateBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ConfigTemplate, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ConfigTemplate, 'bulk_rename', path='rename', detail=False)
|
||||||
class ConfigTemplateBulkRenameView(generic.BulkRenameView):
|
class ConfigTemplateBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ConfigTemplate.objects.all()
|
queryset = ConfigTemplate.objects.all()
|
||||||
|
filterset = filtersets.ConfigTemplateFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ConfigTemplate, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ConfigTemplate, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1143,6 +1187,7 @@ class ImageAttachmentBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ImageAttachment, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ImageAttachment, 'bulk_rename', path='rename', detail=False)
|
||||||
class ImageAttachmentBulkRenameView(generic.BulkRenameView):
|
class ImageAttachmentBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ImageAttachment.objects.all()
|
queryset = ImageAttachment.objects.all()
|
||||||
|
filterset = filtersets.ImageAttachmentFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ImageAttachment, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ImageAttachment, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1485,6 +1530,15 @@ class ScriptView(BaseScriptView):
|
|||||||
)
|
)
|
||||||
|
|
||||||
return redirect('extras:script_result', job_pk=job.pk)
|
return redirect('extras:script_result', job_pk=job.pk)
|
||||||
|
else:
|
||||||
|
fieldset_fields = {field for _, fields in script_class.get_fieldsets() for field in fields}
|
||||||
|
hidden_errors = {
|
||||||
|
field: errors for field, errors in form.errors.items()
|
||||||
|
if field not in fieldset_fields
|
||||||
|
}
|
||||||
|
if hidden_errors:
|
||||||
|
error_msg = '; '.join(f"{field}: {', '.join(errors)}" for field, errors in hidden_errors.items())
|
||||||
|
messages.error(request, error_msg)
|
||||||
|
|
||||||
return render(request, 'extras/script.html', {
|
return render(request, 'extras/script.html', {
|
||||||
'object': script,
|
'object': script,
|
||||||
|
|||||||
@ -368,6 +368,20 @@ class IPAddressImportForm(PrimaryModelImportForm):
|
|||||||
**{f"virtual_machine__{self.fields['virtual_machine'].to_field_name}": data['virtual_machine']}
|
**{f"virtual_machine__{self.fields['virtual_machine'].to_field_name}": data['virtual_machine']}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def clean_is_primary(self):
|
||||||
|
# Make sure is_primary is None when it's not included in the uploaded data
|
||||||
|
if 'is_primary' not in self.data:
|
||||||
|
return None
|
||||||
|
else:
|
||||||
|
return self.cleaned_data['is_primary']
|
||||||
|
|
||||||
|
def clean_is_oob(self):
|
||||||
|
# Make sure is_oob is None when it's not included in the uploaded data
|
||||||
|
if 'is_oob' not in self.data:
|
||||||
|
return None
|
||||||
|
else:
|
||||||
|
return self.cleaned_data['is_oob']
|
||||||
|
|
||||||
def clean(self):
|
def clean(self):
|
||||||
super().clean()
|
super().clean()
|
||||||
|
|
||||||
@ -410,18 +424,18 @@ class IPAddressImportForm(PrimaryModelImportForm):
|
|||||||
ipaddress = super().save(*args, **kwargs)
|
ipaddress = super().save(*args, **kwargs)
|
||||||
|
|
||||||
# Set as primary for device/VM
|
# Set as primary for device/VM
|
||||||
if self.cleaned_data.get('is_primary'):
|
if self.cleaned_data.get('is_primary') is not None:
|
||||||
parent = self.cleaned_data.get('device') or self.cleaned_data.get('virtual_machine')
|
parent = self.cleaned_data.get('device') or self.cleaned_data.get('virtual_machine')
|
||||||
if self.instance.address.version == 4:
|
if self.instance.address.version == 4:
|
||||||
parent.primary_ip4 = ipaddress
|
parent.primary_ip4 = ipaddress if self.cleaned_data.get('is_primary') else None
|
||||||
elif self.instance.address.version == 6:
|
elif self.instance.address.version == 6:
|
||||||
parent.primary_ip6 = ipaddress
|
parent.primary_ip6 = ipaddress if self.cleaned_data.get('is_primary') else None
|
||||||
parent.save()
|
parent.save()
|
||||||
|
|
||||||
# Set as OOB for device
|
# Set as OOB for device
|
||||||
if self.cleaned_data.get('is_oob'):
|
if self.cleaned_data.get('is_oob') is not None:
|
||||||
parent = self.cleaned_data.get('device')
|
parent = self.cleaned_data.get('device')
|
||||||
parent.oob_ip = ipaddress
|
parent.oob_ip = ipaddress if self.cleaned_data.get('is_oob') else None
|
||||||
parent.save()
|
parent.save()
|
||||||
|
|
||||||
return ipaddress
|
return ipaddress
|
||||||
|
|||||||
@ -79,12 +79,36 @@ class ASNRangeFilter(TenancyFilterMixin, OrganizationalModelFilterMixin):
|
|||||||
|
|
||||||
@strawberry_django.filter_type(models.Aggregate, lookups=True)
|
@strawberry_django.filter_type(models.Aggregate, lookups=True)
|
||||||
class AggregateFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilterMixin):
|
class AggregateFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilterMixin):
|
||||||
prefix: Annotated['PrefixFilter', strawberry.lazy('ipam.graphql.filters')] | None = strawberry_django.filter_field()
|
prefix: FilterLookup[str] | None = strawberry_django.filter_field()
|
||||||
prefix_id: ID | None = strawberry_django.filter_field()
|
|
||||||
rir: Annotated['RIRFilter', strawberry.lazy('ipam.graphql.filters')] | None = strawberry_django.filter_field()
|
rir: Annotated['RIRFilter', strawberry.lazy('ipam.graphql.filters')] | None = strawberry_django.filter_field()
|
||||||
rir_id: ID | None = strawberry_django.filter_field()
|
rir_id: ID | None = strawberry_django.filter_field()
|
||||||
date_added: DateFilterLookup[date] | None = strawberry_django.filter_field()
|
date_added: DateFilterLookup[date] | None = strawberry_django.filter_field()
|
||||||
|
|
||||||
|
@strawberry_django.filter_field()
|
||||||
|
def contains(self, value: list[str], prefix) -> Q:
|
||||||
|
"""
|
||||||
|
Return aggregates whose `prefix` contains any of the supplied networks.
|
||||||
|
Mirrors PrefixFilter.contains but operates on the Aggregate.prefix field itself.
|
||||||
|
"""
|
||||||
|
if not value:
|
||||||
|
return Q()
|
||||||
|
q = Q()
|
||||||
|
for subnet in value:
|
||||||
|
try:
|
||||||
|
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
||||||
|
except (AddrFormatError, ValueError):
|
||||||
|
continue
|
||||||
|
q |= Q(**{f"{prefix}prefix__net_contains": query})
|
||||||
|
return q
|
||||||
|
|
||||||
|
@strawberry_django.filter_field()
|
||||||
|
def family(
|
||||||
|
self,
|
||||||
|
value: Annotated['IPAddressFamilyEnum', strawberry.lazy('ipam.graphql.enums')],
|
||||||
|
prefix,
|
||||||
|
) -> Q:
|
||||||
|
return Q(**{f"{prefix}prefix__family": value.value})
|
||||||
|
|
||||||
|
|
||||||
@strawberry_django.filter_type(models.FHRPGroup, lookups=True)
|
@strawberry_django.filter_type(models.FHRPGroup, lookups=True)
|
||||||
class FHRPGroupFilter(PrimaryModelFilterMixin):
|
class FHRPGroupFilter(PrimaryModelFilterMixin):
|
||||||
@ -119,28 +143,28 @@ class FHRPGroupAssignmentFilter(BaseObjectTypeFilterMixin, ChangeLogFilterMixin)
|
|||||||
)
|
)
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def device_id(self, queryset, value: list[str], prefix) -> Q:
|
def device_id(self, value: list[str], prefix) -> Q:
|
||||||
return self.filter_device('id', value)
|
return self.filter_device('id', value, prefix)
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def device(self, value: list[str], prefix) -> Q:
|
def device(self, value: list[str], prefix) -> Q:
|
||||||
return self.filter_device('name', value)
|
return self.filter_device('name', value, prefix)
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def virtual_machine_id(self, value: list[str], prefix) -> Q:
|
def virtual_machine_id(self, value: list[str], prefix) -> Q:
|
||||||
return Q(interface_id__in=VMInterface.objects.filter(virtual_machine_id__in=value))
|
return Q(**{f"{prefix}interface_id__in": VMInterface.objects.filter(virtual_machine_id__in=value)})
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def virtual_machine(self, value: list[str], prefix) -> Q:
|
def virtual_machine(self, value: list[str], prefix) -> Q:
|
||||||
return Q(interface_id__in=VMInterface.objects.filter(virtual_machine__name__in=value))
|
return Q(**{f"{prefix}interface_id__in": VMInterface.objects.filter(virtual_machine__name__in=value)})
|
||||||
|
|
||||||
def filter_device(self, field, value) -> Q:
|
def filter_device(self, field, value, prefix) -> Q:
|
||||||
"""Helper to standardize logic for device and device_id filters"""
|
"""Helper to standardize logic for device and device_id filters"""
|
||||||
devices = Device.objects.filter(**{f'{field}__in': value})
|
devices = Device.objects.filter(**{f'{field}__in': value})
|
||||||
interface_ids = []
|
interface_ids = []
|
||||||
for device in devices:
|
for device in devices:
|
||||||
interface_ids.extend(device.vc_interfaces().values_list('id', flat=True))
|
interface_ids.extend(device.vc_interfaces().values_list('id', flat=True))
|
||||||
return Q(interface_id__in=interface_ids)
|
return Q(**{f"{prefix}interface_id__in": interface_ids})
|
||||||
|
|
||||||
|
|
||||||
@strawberry_django.filter_type(models.IPAddress, lookups=True)
|
@strawberry_django.filter_type(models.IPAddress, lookups=True)
|
||||||
@ -170,7 +194,7 @@ class IPAddressFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilter
|
|||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def assigned(self, value: bool, prefix) -> Q:
|
def assigned(self, value: bool, prefix) -> Q:
|
||||||
return Q(assigned_object_id__isnull=(not value))
|
return Q(**{f"{prefix}assigned_object_id__isnull": not value})
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
def parent(self, value: list[str], prefix) -> Q:
|
def parent(self, value: list[str], prefix) -> Q:
|
||||||
@ -180,9 +204,9 @@ class IPAddressFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilter
|
|||||||
for subnet in value:
|
for subnet in value:
|
||||||
try:
|
try:
|
||||||
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
||||||
q |= Q(address__net_host_contained=query)
|
|
||||||
except (AddrFormatError, ValueError):
|
except (AddrFormatError, ValueError):
|
||||||
return Q()
|
continue
|
||||||
|
q |= Q(**{f"{prefix}address__net_host_contained": query})
|
||||||
return q
|
return q
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
@ -217,9 +241,14 @@ class IPRangeFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilterMi
|
|||||||
for subnet in value:
|
for subnet in value:
|
||||||
try:
|
try:
|
||||||
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
||||||
q |= Q(start_address__net_host_contained=query, end_address__net_host_contained=query)
|
|
||||||
except (AddrFormatError, ValueError):
|
except (AddrFormatError, ValueError):
|
||||||
return Q()
|
continue
|
||||||
|
q |= Q(
|
||||||
|
**{
|
||||||
|
f"{prefix}start_address__net_host_contained": query,
|
||||||
|
f"{prefix}end_address__net_host_contained": query,
|
||||||
|
}
|
||||||
|
)
|
||||||
return q
|
return q
|
||||||
|
|
||||||
@strawberry_django.filter_field()
|
@strawberry_django.filter_field()
|
||||||
@ -228,10 +257,17 @@ class IPRangeFilter(ContactFilterMixin, TenancyFilterMixin, PrimaryModelFilterMi
|
|||||||
return Q()
|
return Q()
|
||||||
q = Q()
|
q = Q()
|
||||||
for subnet in value:
|
for subnet in value:
|
||||||
net = netaddr.IPNetwork(subnet.strip())
|
try:
|
||||||
|
net = netaddr.IPNetwork(subnet.strip())
|
||||||
|
query_start = str(netaddr.IPAddress(net.first))
|
||||||
|
query_end = str(netaddr.IPAddress(net.last))
|
||||||
|
except (AddrFormatError, ValueError):
|
||||||
|
continue
|
||||||
q |= Q(
|
q |= Q(
|
||||||
start_address__host__inet__lte=str(netaddr.IPAddress(net.first)),
|
**{
|
||||||
end_address__host__inet__gte=str(netaddr.IPAddress(net.last)),
|
f"{prefix}start_address__host__inet__lte": query_start,
|
||||||
|
f"{prefix}end_address__host__inet__gte": query_end,
|
||||||
|
}
|
||||||
)
|
)
|
||||||
return q
|
return q
|
||||||
|
|
||||||
@ -257,10 +293,21 @@ class PrefixFilter(ContactFilterMixin, ScopedFilterMixin, TenancyFilterMixin, Pr
|
|||||||
return Q()
|
return Q()
|
||||||
q = Q()
|
q = Q()
|
||||||
for subnet in value:
|
for subnet in value:
|
||||||
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
try:
|
||||||
q |= Q(prefix__net_contains=query)
|
query = str(netaddr.IPNetwork(subnet.strip()).cidr)
|
||||||
|
except (AddrFormatError, ValueError):
|
||||||
|
continue
|
||||||
|
q |= Q(**{f"{prefix}prefix__net_contains": query})
|
||||||
return q
|
return q
|
||||||
|
|
||||||
|
@strawberry_django.filter_field()
|
||||||
|
def family(
|
||||||
|
self,
|
||||||
|
value: Annotated['IPAddressFamilyEnum', strawberry.lazy('ipam.graphql.enums')],
|
||||||
|
prefix,
|
||||||
|
) -> Q:
|
||||||
|
return Q(**{f"{prefix}prefix__family": value.value})
|
||||||
|
|
||||||
|
|
||||||
@strawberry_django.filter_type(models.RIR, lookups=True)
|
@strawberry_django.filter_type(models.RIR, lookups=True)
|
||||||
class RIRFilter(OrganizationalModelFilterMixin):
|
class RIRFilter(OrganizationalModelFilterMixin):
|
||||||
|
|||||||
@ -0,0 +1,27 @@
|
|||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
def populate_vlangroup_total_vlan_ids(apps, schema_editor):
|
||||||
|
VLANGroup = apps.get_model('ipam', 'VLANGroup')
|
||||||
|
db_alias = schema_editor.connection.alias
|
||||||
|
|
||||||
|
vlan_groups = VLANGroup.objects.using(db_alias).only('id', 'vid_ranges')
|
||||||
|
for group in vlan_groups:
|
||||||
|
total_vlan_ids = 0
|
||||||
|
if group.vid_ranges:
|
||||||
|
for r in group.vid_ranges:
|
||||||
|
# Half-open [lo, hi): length is (hi - lo).
|
||||||
|
if r is not None and r.lower is not None and r.upper is not None:
|
||||||
|
total_vlan_ids += r.upper - r.lower
|
||||||
|
group._total_vlan_ids = total_vlan_ids
|
||||||
|
VLANGroup.objects.using(db_alias).bulk_update(vlan_groups, ['_total_vlan_ids'], batch_size=100)
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
('ipam', '0082_add_prefix_network_containment_indexes'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RunPython(populate_vlangroup_total_vlan_ids, migrations.RunPython.noop),
|
||||||
|
]
|
||||||
@ -4,7 +4,7 @@ from django.db import migrations, models
|
|||||||
|
|
||||||
class Migration(migrations.Migration):
|
class Migration(migrations.Migration):
|
||||||
dependencies = [
|
dependencies = [
|
||||||
('ipam', '0082_add_prefix_network_containment_indexes'),
|
('ipam', '0083_vlangroup_populate_total_vlan_ids'),
|
||||||
('users', '0015_owner'),
|
('users', '0015_owner'),
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -132,7 +132,8 @@ class VLANGroup(OrganizationalModel):
|
|||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
self._total_vlan_ids = 0
|
self._total_vlan_ids = 0
|
||||||
for vid_range in self.vid_ranges:
|
for vid_range in self.vid_ranges:
|
||||||
self._total_vlan_ids += vid_range.upper - vid_range.lower + 1
|
# VID range is inclusive on lower-bound, exclusive on upper-bound
|
||||||
|
self._total_vlan_ids += vid_range.upper - vid_range.lower
|
||||||
|
|
||||||
super().save(*args, **kwargs)
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
|||||||
@ -3,7 +3,7 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
|
|
||||||
from ipam.models import *
|
from ipam.models import *
|
||||||
from netbox.tables import OrganizationalModelTable, PrimaryModelTable, columns
|
from netbox.tables import OrganizationalModelTable, PrimaryModelTable, columns
|
||||||
from tenancy.tables import TenancyColumnsMixin
|
from tenancy.tables import ContactsColumnMixin, TenancyColumnsMixin
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
'ASNTable',
|
'ASNTable',
|
||||||
@ -36,7 +36,7 @@ class ASNRangeTable(TenancyColumnsMixin, OrganizationalModelTable):
|
|||||||
default_columns = ('pk', 'name', 'rir', 'start', 'end', 'tenant', 'asn_count', 'description')
|
default_columns = ('pk', 'name', 'rir', 'start', 'end', 'tenant', 'asn_count', 'description')
|
||||||
|
|
||||||
|
|
||||||
class ASNTable(TenancyColumnsMixin, PrimaryModelTable):
|
class ASNTable(TenancyColumnsMixin, ContactsColumnMixin, PrimaryModelTable):
|
||||||
asn = tables.Column(
|
asn = tables.Column(
|
||||||
verbose_name=_('ASN'),
|
verbose_name=_('ASN'),
|
||||||
linkify=True
|
linkify=True
|
||||||
@ -73,7 +73,7 @@ class ASNTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
model = ASN
|
model = ASN
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'asn', 'asn_asdot', 'rir', 'site_count', 'provider_count', 'tenant', 'tenant_group', 'description',
|
'pk', 'asn', 'asn_asdot', 'rir', 'site_count', 'provider_count', 'tenant', 'tenant_group', 'description',
|
||||||
'comments', 'sites', 'tags', 'created', 'last_updated', 'actions',
|
'contacts', 'comments', 'sites', 'tags', 'created', 'last_updated', 'actions',
|
||||||
)
|
)
|
||||||
default_columns = (
|
default_columns = (
|
||||||
'pk', 'asn', 'rir', 'site_count', 'provider_count', 'sites', 'description', 'tenant',
|
'pk', 'asn', 'rir', 'site_count', 'provider_count', 'sites', 'description', 'tenant',
|
||||||
|
|||||||
@ -5,7 +5,7 @@ from django_tables2.utils import Accessor
|
|||||||
|
|
||||||
from ipam.models import *
|
from ipam.models import *
|
||||||
from netbox.tables import NetBoxTable, OrganizationalModelTable, PrimaryModelTable, columns
|
from netbox.tables import NetBoxTable, OrganizationalModelTable, PrimaryModelTable, columns
|
||||||
from tenancy.tables import TenancyColumnsMixin, TenantColumn
|
from tenancy.tables import ContactsColumnMixin, TenancyColumnsMixin, TenantColumn
|
||||||
from .template_code import *
|
from .template_code import *
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
@ -58,7 +58,7 @@ class RIRTable(OrganizationalModelTable):
|
|||||||
# Aggregates
|
# Aggregates
|
||||||
#
|
#
|
||||||
|
|
||||||
class AggregateTable(TenancyColumnsMixin, PrimaryModelTable):
|
class AggregateTable(TenancyColumnsMixin, ContactsColumnMixin, PrimaryModelTable):
|
||||||
prefix = tables.Column(
|
prefix = tables.Column(
|
||||||
linkify=True,
|
linkify=True,
|
||||||
verbose_name=_('Aggregate'),
|
verbose_name=_('Aggregate'),
|
||||||
@ -90,7 +90,7 @@ class AggregateTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
model = Aggregate
|
model = Aggregate
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'prefix', 'rir', 'tenant', 'tenant_group', 'child_count', 'utilization', 'date_added',
|
'pk', 'id', 'prefix', 'rir', 'tenant', 'tenant_group', 'child_count', 'utilization', 'date_added',
|
||||||
'description', 'comments', 'tags', 'created', 'last_updated',
|
'description', 'contacts', 'comments', 'tags', 'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = ('pk', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description')
|
default_columns = ('pk', 'prefix', 'rir', 'tenant', 'child_count', 'utilization', 'date_added', 'description')
|
||||||
|
|
||||||
@ -151,7 +151,7 @@ class PrefixUtilizationColumn(columns.UtilizationColumn):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
class PrefixTable(TenancyColumnsMixin, PrimaryModelTable):
|
class PrefixTable(TenancyColumnsMixin, ContactsColumnMixin, PrimaryModelTable):
|
||||||
prefix = columns.TemplateColumn(
|
prefix = columns.TemplateColumn(
|
||||||
verbose_name=_('Prefix'),
|
verbose_name=_('Prefix'),
|
||||||
template_code=PREFIX_LINK_WITH_DEPTH,
|
template_code=PREFIX_LINK_WITH_DEPTH,
|
||||||
@ -231,8 +231,8 @@ class PrefixTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
model = Prefix
|
model = Prefix
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'prefix', 'prefix_flat', 'status', 'children', 'vrf', 'utilization', 'tenant', 'tenant_group',
|
'pk', 'id', 'prefix', 'prefix_flat', 'status', 'children', 'vrf', 'utilization', 'tenant', 'tenant_group',
|
||||||
'scope', 'scope_type', 'vlan_group', 'vlan', 'role', 'is_pool', 'mark_utilized', 'description', 'comments',
|
'scope', 'scope_type', 'vlan_group', 'vlan', 'role', 'is_pool', 'mark_utilized', 'description', 'contacts',
|
||||||
'tags', 'created', 'last_updated',
|
'comments', 'tags', 'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = (
|
default_columns = (
|
||||||
'pk', 'prefix', 'status', 'children', 'vrf', 'utilization', 'tenant', 'scope', 'vlan', 'role',
|
'pk', 'prefix', 'status', 'children', 'vrf', 'utilization', 'tenant', 'scope', 'vlan', 'role',
|
||||||
@ -246,7 +246,8 @@ class PrefixTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
#
|
#
|
||||||
# IP ranges
|
# IP ranges
|
||||||
#
|
#
|
||||||
class IPRangeTable(TenancyColumnsMixin, PrimaryModelTable):
|
|
||||||
|
class IPRangeTable(TenancyColumnsMixin, ContactsColumnMixin, PrimaryModelTable):
|
||||||
start_address = tables.Column(
|
start_address = tables.Column(
|
||||||
verbose_name=_('Start address'),
|
verbose_name=_('Start address'),
|
||||||
linkify=True
|
linkify=True
|
||||||
@ -284,8 +285,8 @@ class IPRangeTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
model = IPRange
|
model = IPRange
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'tenant_group',
|
'pk', 'id', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'tenant_group',
|
||||||
'mark_populated', 'mark_utilized', 'utilization', 'description', 'comments', 'tags', 'created',
|
'mark_populated', 'mark_utilized', 'utilization', 'description', 'contacts', 'comments', 'tags',
|
||||||
'last_updated',
|
'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = (
|
default_columns = (
|
||||||
'pk', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
|
'pk', 'start_address', 'end_address', 'size', 'vrf', 'status', 'role', 'tenant', 'description',
|
||||||
@ -296,10 +297,10 @@ class IPRangeTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# IPAddresses
|
# IP Addresses
|
||||||
#
|
#
|
||||||
|
|
||||||
class IPAddressTable(TenancyColumnsMixin, PrimaryModelTable):
|
class IPAddressTable(TenancyColumnsMixin, ContactsColumnMixin, PrimaryModelTable):
|
||||||
address = tables.TemplateColumn(
|
address = tables.TemplateColumn(
|
||||||
template_code=IPADDRESS_LINK,
|
template_code=IPADDRESS_LINK,
|
||||||
verbose_name=_('IP Address')
|
verbose_name=_('IP Address')
|
||||||
@ -353,7 +354,7 @@ class IPAddressTable(TenancyColumnsMixin, PrimaryModelTable):
|
|||||||
model = IPAddress
|
model = IPAddress
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'address', 'vrf', 'status', 'role', 'tenant', 'tenant_group', 'nat_inside', 'nat_outside',
|
'pk', 'id', 'address', 'vrf', 'status', 'role', 'tenant', 'tenant_group', 'nat_inside', 'nat_outside',
|
||||||
'assigned', 'dns_name', 'description', 'comments', 'tags', 'created', 'last_updated',
|
'assigned', 'dns_name', 'description', 'comments', 'contacts', 'tags', 'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = (
|
default_columns = (
|
||||||
'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'assigned', 'dns_name', 'description',
|
'pk', 'address', 'vrf', 'status', 'role', 'tenant', 'assigned', 'dns_name', 'description',
|
||||||
|
|||||||
@ -3,6 +3,7 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
|
|
||||||
from ipam.models import *
|
from ipam.models import *
|
||||||
from netbox.tables import PrimaryModelTable, columns
|
from netbox.tables import PrimaryModelTable, columns
|
||||||
|
from tenancy.tables import ContactsColumnMixin
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
'ServiceTable',
|
'ServiceTable',
|
||||||
@ -32,7 +33,7 @@ class ServiceTemplateTable(PrimaryModelTable):
|
|||||||
default_columns = ('pk', 'name', 'protocol', 'ports', 'description')
|
default_columns = ('pk', 'name', 'protocol', 'ports', 'description')
|
||||||
|
|
||||||
|
|
||||||
class ServiceTable(PrimaryModelTable):
|
class ServiceTable(ContactsColumnMixin, PrimaryModelTable):
|
||||||
name = tables.Column(
|
name = tables.Column(
|
||||||
verbose_name=_('Name'),
|
verbose_name=_('Name'),
|
||||||
linkify=True
|
linkify=True
|
||||||
@ -54,7 +55,7 @@ class ServiceTable(PrimaryModelTable):
|
|||||||
class Meta(PrimaryModelTable.Meta):
|
class Meta(PrimaryModelTable.Meta):
|
||||||
model = Service
|
model = Service
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'name', 'parent', 'protocol', 'ports', 'ipaddresses', 'description', 'comments', 'tags',
|
'pk', 'id', 'name', 'parent', 'protocol', 'ports', 'ipaddresses', 'description', 'contacts', 'comments',
|
||||||
'created', 'last_updated',
|
'tags', 'created', 'last_updated',
|
||||||
)
|
)
|
||||||
default_columns = ('pk', 'name', 'parent', 'protocol', 'ports', 'description')
|
default_columns = ('pk', 'name', 'parent', 'protocol', 'ports', 'description')
|
||||||
|
|||||||
@ -323,6 +323,55 @@ class AggregateTest(APIViewTestCases.APIViewTestCase):
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_aggregate_prefix_exact(self):
|
||||||
|
"""
|
||||||
|
Test case to verify aggregate prefix equality via field lookup in GraphQL API.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('ipam.view_aggregate', 'ipam.view_rir')
|
||||||
|
|
||||||
|
rir = RIR.objects.create(name='RFC6598', slug='rfc6598', is_private=True)
|
||||||
|
aggregate1 = Aggregate.objects.create(prefix='100.64.0.0/10', rir=rir)
|
||||||
|
Aggregate.objects.create(prefix='203.0.113.0/24', rir=rir)
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
aggregate_list(filters: { prefix: { exact: "100.64.0.0/10" } }) { prefix }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
|
||||||
|
prefixes = {row['prefix'] for row in data['data']['aggregate_list']}
|
||||||
|
self.assertIn(str(aggregate1.prefix), prefixes)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_aggregate_contains_skips_invalid(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Aggregate `contains` filter skips invalid input.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('ipam.view_aggregate', 'ipam.view_rir')
|
||||||
|
|
||||||
|
rir = RIR.objects.create(name='RIR 3', slug='rir-3', is_private=False)
|
||||||
|
aggregate1 = Aggregate.objects.create(prefix='100.64.0.0/10', rir=rir)
|
||||||
|
Aggregate.objects.create(prefix='203.0.113.0/24', rir=rir)
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
aggregate_list(filters: { contains: ["100.64.16.0/24", "not-a-cidr", ""] }) { prefix }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
|
||||||
|
prefixes = {row['prefix'] for row in data['data']['aggregate_list']}
|
||||||
|
self.assertIn(str(aggregate1.prefix), prefixes)
|
||||||
|
# No exception occurred; invalid entries were ignored
|
||||||
|
|
||||||
|
|
||||||
class RoleTest(APIViewTestCases.APIViewTestCase):
|
class RoleTest(APIViewTestCases.APIViewTestCase):
|
||||||
model = Role
|
model = Role
|
||||||
@ -546,6 +595,30 @@ class PrefixTest(APIViewTestCases.APIViewTestCase):
|
|||||||
self.assertHttpStatus(response, status.HTTP_201_CREATED)
|
self.assertHttpStatus(response, status.HTTP_201_CREATED)
|
||||||
self.assertEqual(len(response.data), 8)
|
self.assertEqual(len(response.data), 8)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_tenant_prefixes_contains_nested_skips_invalid(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Tenant nested Prefix `contains` filter skips invalid input.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('ipam.view_prefix', 'ipam.view_vrf', 'tenancy.view_tenant')
|
||||||
|
|
||||||
|
tenant = Tenant.objects.create(name='Tenant 1', slug='tenant-1')
|
||||||
|
vrf = VRF.objects.create(name='Test VRF 1', rd='64512:1')
|
||||||
|
Prefix.objects.create(prefix='10.20.0.0/16', vrf=vrf, tenant=tenant)
|
||||||
|
Prefix.objects.create(prefix='198.51.100.0/24', vrf=vrf) # non-tenant
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
tenant_list(filters: { prefixes: { contains: ["10.20.1.0/24", "not-a-cidr"] } }) { id }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
|
||||||
|
self.assertTrue(data['data']['tenant_list']) # tenant returned
|
||||||
|
|
||||||
|
|
||||||
class IPRangeTest(APIViewTestCases.APIViewTestCase):
|
class IPRangeTest(APIViewTestCases.APIViewTestCase):
|
||||||
model = IPRange
|
model = IPRange
|
||||||
@ -645,6 +718,65 @@ class IPRangeTest(APIViewTestCases.APIViewTestCase):
|
|||||||
self.assertHttpStatus(response, status.HTTP_201_CREATED)
|
self.assertHttpStatus(response, status.HTTP_201_CREATED)
|
||||||
self.assertEqual(len(response.data), 8)
|
self.assertEqual(len(response.data), 8)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_tenant_ip_ranges_parent_nested_skips_invalid(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Tenant nested IP Range `parent` filter skips invalid input.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('tenancy.view_tenant', 'ipam.view_iprange', 'ipam.view_vrf')
|
||||||
|
|
||||||
|
tenant = Tenant.objects.create(name='Tenant 1', slug='tenant-1')
|
||||||
|
vrf = VRF.objects.create(name='Test VRF 1', rd='64512:1')
|
||||||
|
IPRange.objects.create(
|
||||||
|
start_address=IPNetwork('10.30.0.1/24'), end_address=IPNetwork('10.30.0.255/24'), vrf=vrf, tenant=tenant
|
||||||
|
)
|
||||||
|
IPRange.objects.create(
|
||||||
|
start_address=IPNetwork('10.31.0.1/24'), end_address=IPNetwork('10.31.0.255/24'), vrf=vrf, tenant=tenant
|
||||||
|
)
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
tenant_list(filters: {
|
||||||
|
name: { exact: "Tenant 1" }
|
||||||
|
ip_ranges: { parent: ["10.30.0.0/24", "bogus"] }
|
||||||
|
}) { id }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
self.assertTrue(data['data']['tenant_list']) # tenant returned
|
||||||
|
# No exception occurred; invalid entries were ignored
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_tenant_ip_ranges_contains_nested_skips_invalid(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Tenant nested IP Range `contains` filter skips invalid input.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('tenancy.view_tenant', 'ipam.view_iprange', 'ipam.view_vrf')
|
||||||
|
|
||||||
|
tenant = Tenant.objects.create(name='Tenant 2', slug='tenant-2')
|
||||||
|
vrf = VRF.objects.create(name='Test VRF 1', rd='64512:2')
|
||||||
|
IPRange.objects.create(
|
||||||
|
start_address=IPNetwork('10.40.0.1/24'), end_address=IPNetwork('10.40.0.255/24'), vrf=vrf, tenant=tenant
|
||||||
|
)
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
tenant_list(filters: {
|
||||||
|
name: { exact: "Tenant 2" }
|
||||||
|
ip_ranges: { contains: ["10.40.0.128/25", "###"] }
|
||||||
|
}) { id }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
self.assertTrue(data['data']['tenant_list']) # tenant returned
|
||||||
|
# No exception occurred; invalid entries were ignored
|
||||||
|
|
||||||
|
|
||||||
class IPAddressTest(APIViewTestCases.APIViewTestCase):
|
class IPAddressTest(APIViewTestCases.APIViewTestCase):
|
||||||
model = IPAddress
|
model = IPAddress
|
||||||
@ -731,6 +863,75 @@ class IPAddressTest(APIViewTestCases.APIViewTestCase):
|
|||||||
response = self.client.patch(url, data, format='json', **self.header)
|
response = self.client.patch(url, data, format='json', **self.header)
|
||||||
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
self.assertHttpStatus(response, status.HTTP_400_BAD_REQUEST)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_device_primary_ip4_assigned_nested(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Device nested IP Address `primary_ip4` filter.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('dcim.view_device', 'dcim.view_interface', 'ipam.view_ipaddress')
|
||||||
|
|
||||||
|
site = Site.objects.create(name='Site 1')
|
||||||
|
manufacturer = Manufacturer.objects.create(name='Manufacturer 1')
|
||||||
|
device_type = DeviceType.objects.create(model='Device Type 1', manufacturer=manufacturer)
|
||||||
|
role = DeviceRole.objects.create(name='Switch')
|
||||||
|
|
||||||
|
device1 = Device.objects.create(name='Device 1', site=site, device_type=device_type, role=role, status='active')
|
||||||
|
interface1 = Interface.objects.create(name='Interface 1', device=device1, type='1000baset')
|
||||||
|
ip1 = IPAddress.objects.create(address='10.0.0.1/24')
|
||||||
|
ip1.assigned_object = interface1
|
||||||
|
ip1.save()
|
||||||
|
device1.primary_ip4 = ip1
|
||||||
|
device1.save()
|
||||||
|
|
||||||
|
device2 = Device.objects.create(name='Device 2', site=site, device_type=device_type, role=role, status='active')
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
device_list(filters: { primary_ip4: { assigned: true } }) { id name }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
|
||||||
|
ids = {row['id'] for row in data['data']['device_list']}
|
||||||
|
self.assertIn(str(device1.pk), ids)
|
||||||
|
self.assertNotIn(str(device2.pk), ids)
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_graphql_device_primary_ip4_parent_nested_skips_invalid(self):
|
||||||
|
"""
|
||||||
|
Test the GraphQL API Device nested IP Address `parent` filter skips invalid input.
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.add_permissions('dcim.view_device', 'dcim.view_interface', 'ipam.view_ipaddress')
|
||||||
|
|
||||||
|
site = Site.objects.create(name='Site 1')
|
||||||
|
manufacturer = Manufacturer.objects.create(name='Manufacturer 1')
|
||||||
|
device_type = DeviceType.objects.create(model='Device Type 1', manufacturer=manufacturer)
|
||||||
|
role = DeviceRole.objects.create(name='Switch')
|
||||||
|
|
||||||
|
device1 = Device.objects.create(name='Device 1', site=site, device_type=device_type, role=role, status='active')
|
||||||
|
interface1 = Interface.objects.create(name='Interface 1', device=device1, type='1000baset')
|
||||||
|
ip1 = IPAddress.objects.create(address='192.0.2.10/24')
|
||||||
|
ip1.assigned_object = interface1
|
||||||
|
ip1.save()
|
||||||
|
device1.primary_ip4 = ip1
|
||||||
|
device1.save()
|
||||||
|
|
||||||
|
url = reverse('graphql')
|
||||||
|
query = """{
|
||||||
|
device_list(filters: { primary_ip4: { parent: ["192.0.2.0/24", "bad-cidr"] } }) { id }
|
||||||
|
}"""
|
||||||
|
response = self.client.post(url, data={'query': query}, format='json', **self.header)
|
||||||
|
self.assertHttpStatus(response, status.HTTP_200_OK)
|
||||||
|
data = response.json()
|
||||||
|
self.assertNotIn('errors', data)
|
||||||
|
|
||||||
|
ids = {row['id'] for row in data['data']['device_list']}
|
||||||
|
self.assertIn(str(device1.pk), ids)
|
||||||
|
|
||||||
|
|
||||||
class FHRPGroupTest(APIViewTestCases.APIViewTestCase):
|
class FHRPGroupTest(APIViewTestCases.APIViewTestCase):
|
||||||
model = FHRPGroup
|
model = FHRPGroup
|
||||||
|
|||||||
@ -661,6 +661,10 @@ class TestVLANGroup(TestCase):
|
|||||||
vlangroup.full_clean()
|
vlangroup.full_clean()
|
||||||
vlangroup.save()
|
vlangroup.save()
|
||||||
|
|
||||||
|
def test_total_vlan_ids(self):
|
||||||
|
vlangroup = VLANGroup.objects.first()
|
||||||
|
self.assertEqual(vlangroup._total_vlan_ids, 100)
|
||||||
|
|
||||||
|
|
||||||
class TestVLAN(TestCase):
|
class TestVLAN(TestCase):
|
||||||
|
|
||||||
|
|||||||
@ -108,6 +108,7 @@ class VRFBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VRF, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VRF, 'bulk_rename', path='rename', detail=False)
|
||||||
class VRFBulkRenameView(generic.BulkRenameView):
|
class VRFBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VRF.objects.all()
|
queryset = VRF.objects.all()
|
||||||
|
filterset = filtersets.VRFFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VRF, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VRF, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -163,6 +164,7 @@ class RouteTargetBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(RouteTarget, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(RouteTarget, 'bulk_rename', path='rename', detail=False)
|
||||||
class RouteTargetBulkRenameView(generic.BulkRenameView):
|
class RouteTargetBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = RouteTarget.objects.all()
|
queryset = RouteTarget.objects.all()
|
||||||
|
filterset = filtersets.RouteTargetFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(RouteTarget, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(RouteTarget, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -227,6 +229,7 @@ class RIRBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(RIR, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(RIR, 'bulk_rename', path='rename', detail=False)
|
||||||
class RIRBulkRenameView(generic.BulkRenameView):
|
class RIRBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = RIR.objects.all()
|
queryset = RIR.objects.all()
|
||||||
|
filterset = filtersets.RIRFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(RIR, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(RIR, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -305,6 +308,7 @@ class ASNRangeBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ASNRange, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ASNRange, 'bulk_rename', path='rename', detail=False)
|
||||||
class ASNRangeBulkRenameView(generic.BulkRenameView):
|
class ASNRangeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ASNRange.objects.all()
|
queryset = ASNRange.objects.all()
|
||||||
|
filterset = filtersets.ASNRangeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ASNRange, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ASNRange, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -377,6 +381,7 @@ class ASNBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ASN, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ASN, 'bulk_rename', path='rename', detail=False)
|
||||||
class ASNBulkRenameView(generic.BulkRenameView):
|
class ASNBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ASN.objects.all()
|
queryset = ASN.objects.all()
|
||||||
|
filterset = filtersets.ASNFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ASN, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ASN, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -536,6 +541,7 @@ class RoleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Role, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Role, 'bulk_rename', path='rename', detail=False)
|
||||||
class RoleBulkRenameView(generic.BulkRenameView):
|
class RoleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Role.objects.all()
|
queryset = Role.objects.all()
|
||||||
|
filterset = filtersets.RoleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Role, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Role, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -820,6 +826,7 @@ class IPRangeBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(IPRange, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(IPRange, 'bulk_rename', path='rename', detail=False)
|
||||||
class IPRangeBulkRenameView(generic.BulkRenameView):
|
class IPRangeBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = IPRange.objects.all()
|
queryset = IPRange.objects.all()
|
||||||
|
filterset = filtersets.IPRangeFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(IPRange, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(IPRange, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1066,6 +1073,7 @@ class VLANGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VLANGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VLANGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class VLANGroupBulkRenameView(generic.BulkRenameView):
|
class VLANGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VLANGroup.objects.all()
|
queryset = VLANGroup.objects.all()
|
||||||
|
filterset = filtersets.VLANGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VLANGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VLANGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1160,6 +1168,7 @@ class VLANTranslationPolicyBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VLANTranslationPolicy, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VLANTranslationPolicy, 'bulk_rename', path='rename', detail=False)
|
||||||
class VLANTranslationPolicyBulkRenameView(generic.BulkRenameView):
|
class VLANTranslationPolicyBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VLANTranslationPolicy.objects.all()
|
queryset = VLANTranslationPolicy.objects.all()
|
||||||
|
filterset = filtersets.VLANTranslationPolicyFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VLANTranslationPolicy, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VLANTranslationPolicy, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1315,6 +1324,7 @@ class FHRPGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(FHRPGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(FHRPGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class FHRPGroupBulkRenameView(generic.BulkRenameView):
|
class FHRPGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = FHRPGroup.objects.all()
|
queryset = FHRPGroup.objects.all()
|
||||||
|
filterset = filtersets.FHRPGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(FHRPGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(FHRPGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1447,6 +1457,7 @@ class VLANBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(VLAN, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(VLAN, 'bulk_rename', path='rename', detail=False)
|
||||||
class VLANBulkRenameView(generic.BulkRenameView):
|
class VLANBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = VLAN.objects.all()
|
queryset = VLAN.objects.all()
|
||||||
|
filterset = filtersets.VLANFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(VLAN, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(VLAN, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1502,6 +1513,7 @@ class ServiceTemplateBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ServiceTemplate, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ServiceTemplate, 'bulk_rename', path='rename', detail=False)
|
||||||
class ServiceTemplateBulkRenameView(generic.BulkRenameView):
|
class ServiceTemplateBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ServiceTemplate.objects.all()
|
queryset = ServiceTemplate.objects.all()
|
||||||
|
filterset = filtersets.ServiceTemplateFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ServiceTemplate, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ServiceTemplate, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -1574,6 +1586,7 @@ class ServiceBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Service, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Service, 'bulk_rename', path='rename', detail=False)
|
||||||
class ServiceBulkRenameView(generic.BulkRenameView):
|
class ServiceBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Service.objects.all()
|
queryset = Service.objects.all()
|
||||||
|
filterset = filtersets.ServiceFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Service, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Service, 'bulk_delete', path='delete', detail=False)
|
||||||
|
|||||||
@ -183,6 +183,15 @@ PARAMS = (
|
|||||||
description=_("Enable maintenance mode"),
|
description=_("Enable maintenance mode"),
|
||||||
field=forms.BooleanField
|
field=forms.BooleanField
|
||||||
),
|
),
|
||||||
|
ConfigParam(
|
||||||
|
name='COPILOT_ENABLED',
|
||||||
|
label=_('NetBox Copilot enabled'),
|
||||||
|
default=True,
|
||||||
|
description=_(
|
||||||
|
"Enable the NetBox Copilot AI agent globally. If enabled, users can toggle the agent individually."
|
||||||
|
),
|
||||||
|
field=forms.BooleanField
|
||||||
|
),
|
||||||
ConfigParam(
|
ConfigParam(
|
||||||
name='GRAPHQL_ENABLED',
|
name='GRAPHQL_ENABLED',
|
||||||
label=_('GraphQL enabled'),
|
label=_('GraphQL enabled'),
|
||||||
|
|||||||
@ -25,9 +25,14 @@ def preferences(request):
|
|||||||
Adds preferences for the current user (if authenticated) to the template context.
|
Adds preferences for the current user (if authenticated) to the template context.
|
||||||
Example: {{ preferences|get_key:"pagination.placement" }}
|
Example: {{ preferences|get_key:"pagination.placement" }}
|
||||||
"""
|
"""
|
||||||
|
config = get_config()
|
||||||
user_preferences = request.user.config if request.user.is_authenticated else {}
|
user_preferences = request.user.config if request.user.is_authenticated else {}
|
||||||
return {
|
return {
|
||||||
'preferences': user_preferences,
|
'preferences': user_preferences,
|
||||||
|
'copilot_enabled': (
|
||||||
|
config.COPILOT_ENABLED and not django_settings.ISOLATED_DEPLOYMENT and
|
||||||
|
user_preferences.get('ui.copilot_enabled', False) == 'true'
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@ -39,6 +39,15 @@ PREFERENCES = {
|
|||||||
else ''
|
else ''
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
|
'ui.copilot_enabled': UserPreference(
|
||||||
|
label=_('NetBox Copilot'),
|
||||||
|
choices=(
|
||||||
|
('', _('Disabled')),
|
||||||
|
('true', _('Enabled')),
|
||||||
|
),
|
||||||
|
description=_('Enable the NetBox Copilot AI agent'),
|
||||||
|
default=False,
|
||||||
|
),
|
||||||
'pagination.per_page': UserPreference(
|
'pagination.per_page': UserPreference(
|
||||||
label=_('Page length'),
|
label=_('Page length'),
|
||||||
choices=get_page_lengths(),
|
choices=get_page_lengths(),
|
||||||
|
|||||||
@ -659,6 +659,13 @@ DEPLOYMENT_ID = hashlib.sha256(SECRET_KEY.encode('utf-8')).hexdigest()[:16]
|
|||||||
CENSUS_URL = 'https://census.netbox.oss.netboxlabs.com/api/v1/'
|
CENSUS_URL = 'https://census.netbox.oss.netboxlabs.com/api/v1/'
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# NetBox Copilot
|
||||||
|
#
|
||||||
|
|
||||||
|
NETBOX_COPILOT_URL = 'https://static.copilot.netboxlabs.ai/load.js'
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# Django social auth
|
# Django social auth
|
||||||
#
|
#
|
||||||
|
|||||||
@ -799,6 +799,9 @@ class BulkRenameView(GetReturnURLMixin, BaseMultiObjectView):
|
|||||||
"""
|
"""
|
||||||
field_name = 'name'
|
field_name = 'name'
|
||||||
template_name = 'generic/bulk_rename.html'
|
template_name = 'generic/bulk_rename.html'
|
||||||
|
# Match BulkEditView/BulkDeleteView behavior: allow passing a FilterSet
|
||||||
|
# so "Select all N matching query" can expand across the full queryset.
|
||||||
|
filterset = None
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super().__init__(*args, **kwargs)
|
super().__init__(*args, **kwargs)
|
||||||
@ -840,9 +843,16 @@ class BulkRenameView(GetReturnURLMixin, BaseMultiObjectView):
|
|||||||
def post(self, request):
|
def post(self, request):
|
||||||
logger = logging.getLogger('netbox.views.BulkRenameView')
|
logger = logging.getLogger('netbox.views.BulkRenameView')
|
||||||
|
|
||||||
|
# If we are editing *all* objects in the queryset, replace the PK list with all matched objects.
|
||||||
|
if request.POST.get('_all') and self.filterset is not None:
|
||||||
|
pk_list = self.filterset(request.GET, self.queryset.values_list('pk', flat=True), request=request).qs
|
||||||
|
else:
|
||||||
|
pk_list = request.POST.getlist('pk')
|
||||||
|
|
||||||
|
selected_objects = self.queryset.filter(pk__in=pk_list)
|
||||||
|
|
||||||
if '_preview' in request.POST or '_apply' in request.POST:
|
if '_preview' in request.POST or '_apply' in request.POST:
|
||||||
form = self.form(request.POST, initial={'pk': request.POST.getlist('pk')})
|
form = self.form(request.POST, initial={'pk': pk_list})
|
||||||
selected_objects = self.queryset.filter(pk__in=form.initial['pk'])
|
|
||||||
|
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
try:
|
try:
|
||||||
@ -877,8 +887,7 @@ class BulkRenameView(GetReturnURLMixin, BaseMultiObjectView):
|
|||||||
clear_events.send(sender=self)
|
clear_events.send(sender=self)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
form = self.form(initial={'pk': request.POST.getlist('pk')})
|
form = self.form(initial={'pk': pk_list})
|
||||||
selected_objects = self.queryset.filter(pk__in=form.initial['pk'])
|
|
||||||
|
|
||||||
return render(request, self.template_name, {
|
return render(request, self.template_name, {
|
||||||
'field_name': self.field_name,
|
'field_name': self.field_name,
|
||||||
|
|||||||
BIN
netbox/project-static/dist/netbox.css
vendored
BIN
netbox/project-static/dist/netbox.css
vendored
Binary file not shown.
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.
@ -28,7 +28,7 @@
|
|||||||
"clipboard": "2.0.11",
|
"clipboard": "2.0.11",
|
||||||
"flatpickr": "4.6.13",
|
"flatpickr": "4.6.13",
|
||||||
"gridstack": "12.3.3",
|
"gridstack": "12.3.3",
|
||||||
"htmx.org": "2.0.7",
|
"htmx.org": "2.0.8",
|
||||||
"query-string": "9.3.1",
|
"query-string": "9.3.1",
|
||||||
"sass": "1.93.2",
|
"sass": "1.93.2",
|
||||||
"tom-select": "2.4.3",
|
"tom-select": "2.4.3",
|
||||||
|
|||||||
@ -20,11 +20,13 @@ function slugify(slug: string, chars: number): string {
|
|||||||
* For any slug fields, add event listeners to handle automatically generating slug values.
|
* For any slug fields, add event listeners to handle automatically generating slug values.
|
||||||
*/
|
*/
|
||||||
export function initReslug(): void {
|
export function initReslug(): void {
|
||||||
for (const slugButton of getElements<HTMLButtonElement>('button#reslug')) {
|
for (const slugButton of getElements<HTMLButtonElement>('button.reslug')) {
|
||||||
const form = slugButton.form;
|
const form = slugButton.form;
|
||||||
if (form == null) continue;
|
if (form == null) continue;
|
||||||
const slugField = form.querySelector('#id_slug') as HTMLInputElement;
|
|
||||||
|
const slugField = form.querySelector('input.slug-field') as HTMLInputElement;
|
||||||
if (slugField == null) continue;
|
if (slugField == null) continue;
|
||||||
|
|
||||||
const sourceId = slugField.getAttribute('slug-source');
|
const sourceId = slugField.getAttribute('slug-source');
|
||||||
const sourceField = form.querySelector(`#id_${sourceId}`) as HTMLInputElement;
|
const sourceField = form.querySelector(`#id_${sourceId}`) as HTMLInputElement;
|
||||||
|
|
||||||
|
|||||||
@ -16,6 +16,11 @@ pre {
|
|||||||
background: var(--#{$prefix}bg-surface);
|
background: var(--#{$prefix}bg-surface);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Permit copying of badge text
|
||||||
|
.badge {
|
||||||
|
user-select: text;
|
||||||
|
}
|
||||||
|
|
||||||
// Button adjustments
|
// Button adjustments
|
||||||
.btn {
|
.btn {
|
||||||
// Tabler sets display: flex
|
// Tabler sets display: flex
|
||||||
|
|||||||
@ -2241,10 +2241,10 @@ hey-listen@^1.0.8:
|
|||||||
resolved "https://registry.yarnpkg.com/hey-listen/-/hey-listen-1.0.8.tgz#8e59561ff724908de1aa924ed6ecc84a56a9aa68"
|
resolved "https://registry.yarnpkg.com/hey-listen/-/hey-listen-1.0.8.tgz#8e59561ff724908de1aa924ed6ecc84a56a9aa68"
|
||||||
integrity sha512-COpmrF2NOg4TBWUJ5UVyaCU2A88wEMkUPK4hNqyCkqHbxT92BbvfjoSozkAIIm6XhicGlJHhFdullInrdhwU8Q==
|
integrity sha512-COpmrF2NOg4TBWUJ5UVyaCU2A88wEMkUPK4hNqyCkqHbxT92BbvfjoSozkAIIm6XhicGlJHhFdullInrdhwU8Q==
|
||||||
|
|
||||||
htmx.org@2.0.7:
|
htmx.org@2.0.8:
|
||||||
version "2.0.7"
|
version "2.0.8"
|
||||||
resolved "https://registry.yarnpkg.com/htmx.org/-/htmx.org-2.0.7.tgz#991571e009a2ea4cb60e7af8bb4c1c8c0de32ecd"
|
resolved "https://registry.yarnpkg.com/htmx.org/-/htmx.org-2.0.8.tgz#8ac8ba87c141b7bfda7576117476062eeb4aceda"
|
||||||
integrity sha512-YiJqF3U5KyO28VC5mPfehKJPF+n1Gni+cupK+D69TF0nm7wY6AXn3a4mPWIikfAXtl1u1F1+ZhSCS7KT8pVmqA==
|
integrity sha512-fm297iru0iWsNJlBrjvtN7V9zjaxd+69Oqjh4F/Vq9Wwi2kFisLcrLCiv5oBX0KLfOX/zG8AUo9ROMU5XUB44Q==
|
||||||
|
|
||||||
ignore@^5.2.0:
|
ignore@^5.2.0:
|
||||||
version "5.3.2"
|
version "5.3.2"
|
||||||
|
|||||||
@ -1,3 +1,3 @@
|
|||||||
version: "4.4.4"
|
version: "4.4.5"
|
||||||
edition: "Community"
|
edition: "Community"
|
||||||
published: "2025-10-15"
|
published: "2025-10-28"
|
||||||
|
|||||||
@ -69,6 +69,9 @@
|
|||||||
{% block layout %}{% endblock %}
|
{% block layout %}{% endblock %}
|
||||||
|
|
||||||
{# Additional Javascript #}
|
{# Additional Javascript #}
|
||||||
|
{% if copilot_enabled and request.user.is_authenticated %}
|
||||||
|
<script src="{{ settings.NETBOX_COPILOT_URL }}" defer></script>
|
||||||
|
{% endif %}
|
||||||
{% block javascript %}{% endblock %}
|
{% block javascript %}{% endblock %}
|
||||||
|
|
||||||
{# User messages #}
|
{# User messages #}
|
||||||
|
|||||||
@ -129,6 +129,10 @@
|
|||||||
<th scope="row" class="ps-3">{% trans "Maintenance mode" %}</th>
|
<th scope="row" class="ps-3">{% trans "Maintenance mode" %}</th>
|
||||||
<td>{% checkmark config.MAINTENANCE_MODE %}</td>
|
<td>{% checkmark config.MAINTENANCE_MODE %}</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<th scope="row" class="ps-3">{% trans "NetBox Copilot enabled" %}</th>
|
||||||
|
<td>{% checkmark config.COPILOT_ENABLED %}</td>
|
||||||
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row" class="ps-3">{% trans "GraphQL enabled" %}</th>
|
<th scope="row" class="ps-3">{% trans "GraphQL enabled" %}</th>
|
||||||
<td>{% checkmark config.GRAPHQL_ENABLED %}</td>
|
<td>{% checkmark config.GRAPHQL_ENABLED %}</td>
|
||||||
|
|||||||
@ -1,4 +1,15 @@
|
|||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
<div class="card-header px-2 py-1">
|
||||||
|
<h3 class="card-title flex-fill">Notifications</h3>
|
||||||
|
{% if notifications %}
|
||||||
|
<a href="#" hx-get="{% url 'extras:notification_dismiss_all' %}" hx-target="closest .notifications"
|
||||||
|
hx-confirm="{% blocktrans trimmed count count=unread_count %}Dismiss {{ count }} unread notification?{% plural %}Dismiss {{ count }} unread notifications?{% endblocktrans %}"
|
||||||
|
class="btn btn-2 text-danger" title="{% trans 'Dismiss all unread notifications' %}">
|
||||||
|
<i class="icon mdi mdi-delete-sweep-outline"></i>
|
||||||
|
{% trans "Dismiss all" %}
|
||||||
|
</a>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
<div class="list-group list-group-flush list-group-hoverable" style="min-width: 300px">
|
<div class="list-group list-group-flush list-group-hoverable" style="min-width: 300px">
|
||||||
{% for notification in notifications %}
|
{% for notification in notifications %}
|
||||||
<div class="list-group-item p-2">
|
<div class="list-group-item p-2">
|
||||||
|
|||||||
@ -35,23 +35,23 @@
|
|||||||
</a>
|
</a>
|
||||||
<div class="dropdown-menu dropdown-menu-end dropdown-menu-arrow">
|
<div class="dropdown-menu dropdown-menu-end dropdown-menu-arrow">
|
||||||
<a href="{% url 'account:profile' %}" class="dropdown-item">
|
<a href="{% url 'account:profile' %}" class="dropdown-item">
|
||||||
<i class="mdi mdi-account"></i> {% trans "Profile" %}
|
<i class="dropdown-item-icon mdi mdi-account"></i> {% trans "Profile" %}
|
||||||
</a>
|
</a>
|
||||||
<a href="{% url 'account:bookmarks' %}" class="dropdown-item">
|
<a href="{% url 'account:bookmarks' %}" class="dropdown-item">
|
||||||
<i class="mdi mdi-bookmark"></i> {% trans "Bookmarks" %}
|
<i class="dropdown-item-icon mdi mdi-bookmark"></i> {% trans "Bookmarks" %}
|
||||||
</a>
|
</a>
|
||||||
<a href="{% url 'account:subscriptions' %}" class="dropdown-item">
|
<a href="{% url 'account:subscriptions' %}" class="dropdown-item">
|
||||||
<i class="mdi mdi-bell"></i> {% trans "Subscriptions" %}
|
<i class="dropdown-item-icon mdi mdi-bell"></i> {% trans "Subscriptions" %}
|
||||||
</a>
|
</a>
|
||||||
<a href="{% url 'account:preferences' %}" class="dropdown-item">
|
<a href="{% url 'account:preferences' %}" class="dropdown-item">
|
||||||
<i class="mdi mdi-wrench"></i> {% trans "Preferences" %}
|
<i class="dropdown-item-icon mdi mdi-wrench"></i> {% trans "Preferences" %}
|
||||||
</a>
|
</a>
|
||||||
<a href="{% url 'account:usertoken_list' %}" class="dropdown-item">
|
<a href="{% url 'account:usertoken_list' %}" class="dropdown-item">
|
||||||
<i class="mdi mdi-key"></i> {% trans "API Tokens" %}
|
<i class="dropdown-item-icon mdi mdi-key"></i> {% trans "API Tokens" %}
|
||||||
</a>
|
</a>
|
||||||
<hr class="dropdown-divider" />
|
<hr class="dropdown-divider" />
|
||||||
<a href="{% url 'logout' %}" hx-disable="true" class="dropdown-item">
|
<a href="{% url 'logout' %}" hx-disable="true" class="dropdown-item">
|
||||||
<i class="mdi mdi-logout-variant"></i> {% trans "Log Out" %}
|
<i class="dropdown-item-icon mdi mdi-logout-variant"></i> {% trans "Log Out" %}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -74,6 +74,7 @@ class TenantGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(TenantGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(TenantGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class TenantGroupBulkRenameView(generic.BulkRenameView):
|
class TenantGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = TenantGroup.objects.all()
|
queryset = TenantGroup.objects.all()
|
||||||
|
filterset = filtersets.TenantGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(TenantGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(TenantGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -140,6 +141,7 @@ class TenantBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Tenant, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Tenant, 'bulk_rename', path='rename', detail=False)
|
||||||
class TenantBulkRenameView(generic.BulkRenameView):
|
class TenantBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Tenant.objects.all()
|
queryset = Tenant.objects.all()
|
||||||
|
filterset = filtersets.TenantFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Tenant, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Tenant, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -220,6 +222,7 @@ class ContactGroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ContactGroup, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ContactGroup, 'bulk_rename', path='rename', detail=False)
|
||||||
class ContactGroupBulkRenameView(generic.BulkRenameView):
|
class ContactGroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ContactGroup.objects.all()
|
queryset = ContactGroup.objects.all()
|
||||||
|
filterset = filtersets.ContactGroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ContactGroup, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ContactGroup, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -286,6 +289,7 @@ class ContactRoleBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(ContactRole, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(ContactRole, 'bulk_rename', path='rename', detail=False)
|
||||||
class ContactRoleBulkRenameView(generic.BulkRenameView):
|
class ContactRoleBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = ContactRole.objects.all()
|
queryset = ContactRole.objects.all()
|
||||||
|
filterset = filtersets.ContactRoleFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ContactRole, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(ContactRole, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -354,6 +358,7 @@ class ContactBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Contact, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Contact, 'bulk_rename', path='rename', detail=False)
|
||||||
class ContactBulkRenameView(generic.BulkRenameView):
|
class ContactBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Contact.objects.all()
|
queryset = Contact.objects.all()
|
||||||
|
filterset = filtersets.ContactFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Contact, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Contact, 'bulk_delete', path='delete', detail=False)
|
||||||
|
|||||||
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
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
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
Binary file not shown.
File diff suppressed because it is too large
Load Diff
@ -10,6 +10,7 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from core.models import ObjectType
|
from core.models import ObjectType
|
||||||
from ipam.formfields import IPNetworkFormField
|
from ipam.formfields import IPNetworkFormField
|
||||||
from ipam.validators import prefix_validator
|
from ipam.validators import prefix_validator
|
||||||
|
from netbox.config import get_config
|
||||||
from netbox.preferences import PREFERENCES
|
from netbox.preferences import PREFERENCES
|
||||||
from users.choices import TokenVersionChoices
|
from users.choices import TokenVersionChoices
|
||||||
from users.constants import *
|
from users.constants import *
|
||||||
@ -63,8 +64,8 @@ class UserConfigFormMetaclass(forms.models.ModelFormMetaclass):
|
|||||||
class UserConfigForm(forms.ModelForm, metaclass=UserConfigFormMetaclass):
|
class UserConfigForm(forms.ModelForm, metaclass=UserConfigFormMetaclass):
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
FieldSet(
|
FieldSet(
|
||||||
'locale.language', 'pagination.per_page', 'pagination.placement', 'ui.tables.striping',
|
'locale.language', 'ui.copilot_enabled', 'pagination.per_page', 'pagination.placement',
|
||||||
name=_('User Interface')
|
'ui.tables.striping', name=_('User Interface')
|
||||||
),
|
),
|
||||||
FieldSet('data_format', 'csv_delimiter', name=_('Miscellaneous')),
|
FieldSet('data_format', 'csv_delimiter', name=_('Miscellaneous')),
|
||||||
)
|
)
|
||||||
@ -81,8 +82,7 @@ class UserConfigForm(forms.ModelForm, metaclass=UserConfigFormMetaclass):
|
|||||||
def __init__(self, *args, instance=None, **kwargs):
|
def __init__(self, *args, instance=None, **kwargs):
|
||||||
|
|
||||||
# Get initial data from UserConfig instance
|
# Get initial data from UserConfig instance
|
||||||
initial_data = flatten_dict(instance.data)
|
kwargs['initial'] = flatten_dict(instance.data)
|
||||||
kwargs['initial'] = initial_data
|
|
||||||
|
|
||||||
super().__init__(*args, instance=instance, **kwargs)
|
super().__init__(*args, instance=instance, **kwargs)
|
||||||
|
|
||||||
@ -91,6 +91,10 @@ class UserConfigForm(forms.ModelForm, metaclass=UserConfigFormMetaclass):
|
|||||||
(f'tables.{table_name}', '') for table_name in instance.data.get('tables', [])
|
(f'tables.{table_name}', '') for table_name in instance.data.get('tables', [])
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Disable Copilot preference if it has been disabled globally
|
||||||
|
if not get_config().COPILOT_ENABLED:
|
||||||
|
self.fields['ui.copilot_enabled'].disabled = True
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
|
|
||||||
# Set UserConfig data
|
# Set UserConfig data
|
||||||
|
|||||||
@ -110,15 +110,19 @@ class ObjectPermissionTable(NetBoxTable):
|
|||||||
)
|
)
|
||||||
can_view = columns.BooleanColumn(
|
can_view = columns.BooleanColumn(
|
||||||
verbose_name=_('Can View'),
|
verbose_name=_('Can View'),
|
||||||
|
orderable=False,
|
||||||
)
|
)
|
||||||
can_add = columns.BooleanColumn(
|
can_add = columns.BooleanColumn(
|
||||||
verbose_name=_('Can Add'),
|
verbose_name=_('Can Add'),
|
||||||
|
orderable=False,
|
||||||
)
|
)
|
||||||
can_change = columns.BooleanColumn(
|
can_change = columns.BooleanColumn(
|
||||||
verbose_name=_('Can Change'),
|
verbose_name=_('Can Change'),
|
||||||
|
orderable=False,
|
||||||
)
|
)
|
||||||
can_delete = columns.BooleanColumn(
|
can_delete = columns.BooleanColumn(
|
||||||
verbose_name=_('Can Delete'),
|
verbose_name=_('Can Delete'),
|
||||||
|
orderable=False,
|
||||||
)
|
)
|
||||||
custom_actions = columns.ArrayColumn(
|
custom_actions = columns.ArrayColumn(
|
||||||
verbose_name=_('Custom Actions'),
|
verbose_name=_('Custom Actions'),
|
||||||
|
|||||||
@ -118,6 +118,7 @@ class UserBulkEditView(generic.BulkEditView):
|
|||||||
class UserBulkRenameView(generic.BulkRenameView):
|
class UserBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = User.objects.all()
|
queryset = User.objects.all()
|
||||||
field_name = 'username'
|
field_name = 'username'
|
||||||
|
filterset = filtersets.UserFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(User, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(User, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -174,6 +175,7 @@ class GroupBulkEditView(generic.BulkEditView):
|
|||||||
@register_model_view(Group, 'bulk_rename', path='rename', detail=False)
|
@register_model_view(Group, 'bulk_rename', path='rename', detail=False)
|
||||||
class GroupBulkRenameView(generic.BulkRenameView):
|
class GroupBulkRenameView(generic.BulkRenameView):
|
||||||
queryset = Group.objects.all()
|
queryset = Group.objects.all()
|
||||||
|
filterset = filtersets.GroupFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(Group, 'bulk_delete', path='delete', detail=False)
|
@register_model_view(Group, 'bulk_delete', path='delete', detail=False)
|
||||||
@ -212,6 +214,7 @@ class ObjectPermissionEditView(generic.ObjectEditView):
|
|||||||
@register_model_view(ObjectPermission, 'delete')
|
@register_model_view(ObjectPermission, 'delete')
|
||||||
class ObjectPermissionDeleteView(generic.ObjectDeleteView):
|
class ObjectPermissionDeleteView(generic.ObjectDeleteView):
|
||||||
queryset = ObjectPermission.objects.all()
|
queryset = ObjectPermission.objects.all()
|
||||||
|
filterset = filtersets.ObjectPermissionFilterSet
|
||||||
|
|
||||||
|
|
||||||
@register_model_view(ObjectPermission, 'bulk_edit', path='edit', detail=False)
|
@register_model_view(ObjectPermission, 'bulk_edit', path='edit', detail=False)
|
||||||
|
|||||||
@ -82,7 +82,7 @@ def get_view_name(view):
|
|||||||
Derive the view name from its associated model, if it has one. Fall back to DRF's built-in `get_view_name()`.
|
Derive the view name from its associated model, if it has one. Fall back to DRF's built-in `get_view_name()`.
|
||||||
This function is provided to DRF as its VIEW_NAME_FUNCTION.
|
This function is provided to DRF as its VIEW_NAME_FUNCTION.
|
||||||
"""
|
"""
|
||||||
if hasattr(view, 'queryset'):
|
if hasattr(view, 'queryset') and view.queryset is not None:
|
||||||
# Derive the model name from the queryset.
|
# Derive the model name from the queryset.
|
||||||
name = title(view.queryset.model._meta.verbose_name)
|
name = title(view.queryset.model._meta.verbose_name)
|
||||||
if suffix := getattr(view, 'suffix', None):
|
if suffix := getattr(view, 'suffix', None):
|
||||||
|
|||||||
@ -53,6 +53,14 @@ class SlugField(forms.SlugField):
|
|||||||
|
|
||||||
self.widget.attrs['slug-source'] = slug_source
|
self.widget.attrs['slug-source'] = slug_source
|
||||||
|
|
||||||
|
def get_bound_field(self, form, field_name):
|
||||||
|
if prefix := form.prefix:
|
||||||
|
slug_source = self.widget.attrs.get('slug-source')
|
||||||
|
if slug_source and not slug_source.startswith(f'{prefix}-'):
|
||||||
|
self.widget.attrs['slug-source'] = f"{prefix}-{slug_source}"
|
||||||
|
|
||||||
|
return super().get_bound_field(form, field_name)
|
||||||
|
|
||||||
|
|
||||||
class ColorField(forms.CharField):
|
class ColorField(forms.CharField):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@ -56,6 +56,14 @@ class SlugWidget(forms.TextInput):
|
|||||||
"""
|
"""
|
||||||
template_name = 'widgets/sluginput.html'
|
template_name = 'widgets/sluginput.html'
|
||||||
|
|
||||||
|
def __init__(self, attrs=None):
|
||||||
|
local_attrs = {} if attrs is None else attrs.copy()
|
||||||
|
if 'class' in local_attrs:
|
||||||
|
local_attrs['class'] = f"{local_attrs['class']} slug-field"
|
||||||
|
else:
|
||||||
|
local_attrs['class'] = 'slug-field'
|
||||||
|
super().__init__(local_attrs)
|
||||||
|
|
||||||
|
|
||||||
class ArrayWidget(forms.Textarea):
|
class ArrayWidget(forms.Textarea):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@ -1,5 +1,11 @@
|
|||||||
|
from django.http import HttpResponse
|
||||||
|
from django.urls import reverse
|
||||||
|
from urllib.parse import urlsplit
|
||||||
|
|
||||||
__all__ = (
|
__all__ = (
|
||||||
|
'htmx_current_url',
|
||||||
'htmx_partial',
|
'htmx_partial',
|
||||||
|
'htmx_maybe_redirect_current_page',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -9,3 +15,45 @@ def htmx_partial(request):
|
|||||||
in response to an HTMX request, based on the target element.
|
in response to an HTMX request, based on the target element.
|
||||||
"""
|
"""
|
||||||
return request.htmx and not request.htmx.boosted
|
return request.htmx and not request.htmx.boosted
|
||||||
|
|
||||||
|
|
||||||
|
def htmx_current_url(request) -> str:
|
||||||
|
"""
|
||||||
|
Extracts the current URL from the HTMX-specific headers in the given request object.
|
||||||
|
|
||||||
|
This function checks for the `HX-Current-URL` header in the request's headers
|
||||||
|
and `HTTP_HX_CURRENT_URL` in the META data of the request. It preferentially
|
||||||
|
chooses the value present in the `HX-Current-URL` header and falls back to the
|
||||||
|
`HTTP_HX_CURRENT_URL` META data if the former is unavailable. If neither value
|
||||||
|
exists, it returns an empty string.
|
||||||
|
"""
|
||||||
|
return request.headers.get('HX-Current-URL') or request.META.get('HTTP_HX_CURRENT_URL', '') or ''
|
||||||
|
|
||||||
|
|
||||||
|
def htmx_maybe_redirect_current_page(
|
||||||
|
request, url_name: str, *, preserve_query: bool = True, status: int = 200
|
||||||
|
) -> HttpResponse | None:
|
||||||
|
"""
|
||||||
|
Redirects the current page in an HTMX request if conditions are met.
|
||||||
|
|
||||||
|
This function checks whether a request is an HTMX partial request and if the
|
||||||
|
current URL matches the provided target URL. If the conditions are met, it
|
||||||
|
returns an HTTP response signaling a redirect to the provided or updated target
|
||||||
|
URL. Otherwise, it returns None.
|
||||||
|
"""
|
||||||
|
if not htmx_partial(request):
|
||||||
|
return None
|
||||||
|
|
||||||
|
current = urlsplit(htmx_current_url(request))
|
||||||
|
target_path = reverse(url_name) # will raise NoReverseMatch if misconfigured
|
||||||
|
|
||||||
|
if current.path.rstrip('/') != target_path.rstrip('/'):
|
||||||
|
return None
|
||||||
|
|
||||||
|
redirect_to = target_path
|
||||||
|
if preserve_query and current.query:
|
||||||
|
redirect_to = f'{target_path}?{current.query}'
|
||||||
|
|
||||||
|
resp = HttpResponse(status=status)
|
||||||
|
resp['HX-Redirect'] = redirect_to
|
||||||
|
return resp
|
||||||
|
|||||||
@ -19,7 +19,7 @@
|
|||||||
{% if field|widget_type == 'slugwidget' %}
|
{% if field|widget_type == 'slugwidget' %}
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
{{ field }}
|
{{ field }}
|
||||||
<button id="reslug" type="button" title="{% trans "Regenerate Slug" %}" class="btn">
|
<button type="button" title="{% trans "Regenerate Slug" %}" class="btn reslug">
|
||||||
<i class="mdi mdi-reload"></i>
|
<i class="mdi mdi-reload"></i>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -1,4 +1,4 @@
|
|||||||
from django.test import Client, TestCase, override_settings
|
from django.test import Client, TestCase, override_settings, tag
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
from drf_spectacular.drainage import GENERATOR_STATS
|
from drf_spectacular.drainage import GENERATOR_STATS
|
||||||
from rest_framework import status
|
from rest_framework import status
|
||||||
@ -9,6 +9,7 @@ from extras.choices import CustomFieldTypeChoices
|
|||||||
from extras.models import CustomField
|
from extras.models import CustomField
|
||||||
from ipam.models import VLAN
|
from ipam.models import VLAN
|
||||||
from netbox.config import get_config
|
from netbox.config import get_config
|
||||||
|
from utilities.api import get_view_name
|
||||||
from utilities.testing import APITestCase, disable_warnings
|
from utilities.testing import APITestCase, disable_warnings
|
||||||
|
|
||||||
|
|
||||||
@ -267,3 +268,19 @@ class APIDocsTestCase(TestCase):
|
|||||||
with GENERATOR_STATS.silence(): # Suppress schema generator warnings
|
with GENERATOR_STATS.silence(): # Suppress schema generator warnings
|
||||||
response = self.client.get(url)
|
response = self.client.get(url)
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
|
|
||||||
|
|
||||||
|
class GetViewNameTestCase(TestCase):
|
||||||
|
|
||||||
|
@tag('regression')
|
||||||
|
def test_get_view_name_with_none_queryset(self):
|
||||||
|
from rest_framework.viewsets import ReadOnlyModelViewSet
|
||||||
|
|
||||||
|
class MockViewSet(ReadOnlyModelViewSet):
|
||||||
|
queryset = None
|
||||||
|
|
||||||
|
view = MockViewSet()
|
||||||
|
view.suffix = 'List'
|
||||||
|
|
||||||
|
name = get_view_name(view)
|
||||||
|
self.assertEqual(name, 'Mock List')
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user