mirror of
https://github.com/netbox-community/netbox.git
synced 2026-02-05 00:19:32 +01:00
Compare commits
26 Commits
02496-max-
...
v4.4.4
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bee0080917 | ||
|
|
389c44e5d6 | ||
|
|
9cb2c78e34 | ||
|
|
2ae98f0353 | ||
|
|
addda0538f | ||
|
|
c902a1c510 | ||
|
|
f23ee0a46f | ||
|
|
b4acc3fb36 | ||
|
|
a69bbcf651 | ||
|
|
2edfde5753 | ||
|
|
cfbd9632ac | ||
|
|
c9386bc9c3 | ||
|
|
c826c5cdb0 | ||
|
|
a4ab4f885d | ||
|
|
61d77dff14 | ||
|
|
24a83acc34 | ||
|
|
dbc71158ec | ||
|
|
f0523611d1 | ||
|
|
7719b98697 | ||
|
|
f383067ecb | ||
|
|
20de263565 | ||
|
|
5ceb6a60da | ||
|
|
33d4759871 | ||
|
|
2abc5ac69a | ||
|
|
f8c074045f | ||
|
|
d8e4c95bcc |
@@ -15,7 +15,7 @@ body:
|
||||
attributes:
|
||||
label: NetBox version
|
||||
description: What version of NetBox are you currently running?
|
||||
placeholder: v4.4.2
|
||||
placeholder: v4.4.4
|
||||
validations:
|
||||
required: true
|
||||
- type: dropdown
|
||||
|
||||
2
.github/ISSUE_TEMPLATE/02-bug_report.yaml
vendored
2
.github/ISSUE_TEMPLATE/02-bug_report.yaml
vendored
@@ -27,7 +27,7 @@ body:
|
||||
attributes:
|
||||
label: NetBox Version
|
||||
description: What version of NetBox are you currently running?
|
||||
placeholder: v4.4.2
|
||||
placeholder: v4.4.4
|
||||
validations:
|
||||
required: true
|
||||
- type: dropdown
|
||||
|
||||
@@ -12,9 +12,7 @@ django-cors-headers
|
||||
|
||||
# Runtime UI tool for debugging Django
|
||||
# https://github.com/jazzband/django-debug-toolbar/blob/main/docs/changes.rst
|
||||
# django-debug-toolbar v6.0.0 raises "Attribute Error at /: 'function' object has no attribute 'set'"
|
||||
# see https://github.com/netbox-community/netbox/issues/19974
|
||||
django-debug-toolbar==5.2.0
|
||||
django-debug-toolbar
|
||||
|
||||
# Library for writing reusable URL query filters
|
||||
# https://github.com/carltongibson/django-filter/blob/main/CHANGES.rst
|
||||
@@ -71,7 +69,8 @@ django-timezone-field
|
||||
|
||||
# A REST API framework for Django projects
|
||||
# https://www.django-rest-framework.org/community/release-notes/
|
||||
djangorestframework
|
||||
# TODO: Re-evaluate the monkey-patch of get_unique_validators() before upgrading
|
||||
djangorestframework==3.16.1
|
||||
|
||||
# Sane and flexible OpenAPI 3 schema generation for Django REST framework.
|
||||
# https://github.com/tfranzel/drf-spectacular/blob/master/CHANGELOG.rst
|
||||
@@ -167,7 +166,8 @@ strawberry-graphql-django
|
||||
svgwrite
|
||||
|
||||
# Tabular dataset library (for table-based exports)
|
||||
# https://github.com/jazzband/tablib/blob/master/HISTORY.md
|
||||
# Current: https://github.com/jazzband/tablib/releases
|
||||
# Previous: https://github.com/jazzband/tablib/blob/master/HISTORY.md
|
||||
tablib
|
||||
|
||||
# Timezone data (required by django-timezone-field on Python 3.9+)
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
"openapi": "3.0.3",
|
||||
"info": {
|
||||
"title": "NetBox REST API",
|
||||
"version": "4.4.2",
|
||||
"version": "4.4.4",
|
||||
"license": {
|
||||
"name": "Apache v2 License"
|
||||
}
|
||||
@@ -19678,14 +19678,14 @@
|
||||
"in": "query",
|
||||
"name": "object_type",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
"in": "query",
|
||||
"name": "object_type__n",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -20507,14 +20507,14 @@
|
||||
"in": "query",
|
||||
"name": "related_object_type",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
"in": "query",
|
||||
"name": "related_object_type__n",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -60734,14 +60734,14 @@
|
||||
"in": "query",
|
||||
"name": "assigned_object_type",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
"in": "query",
|
||||
"name": "assigned_object_type__n",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -136422,14 +136422,14 @@
|
||||
"in": "query",
|
||||
"name": "assigned_object_type",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
"in": "query",
|
||||
"name": "assigned_object_type__n",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -148274,14 +148274,14 @@
|
||||
"in": "query",
|
||||
"name": "parent_object_type",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
"in": "query",
|
||||
"name": "parent_object_type__n",
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
"type": "string"
|
||||
}
|
||||
},
|
||||
{
|
||||
@@ -228988,6 +228988,7 @@
|
||||
},
|
||||
"key": {
|
||||
"type": "string",
|
||||
"writeOnly": true,
|
||||
"maxLength": 40,
|
||||
"minLength": 40
|
||||
},
|
||||
@@ -245222,11 +245223,6 @@
|
||||
"format": "date-time",
|
||||
"nullable": true
|
||||
},
|
||||
"key": {
|
||||
"type": "string",
|
||||
"maxLength": 40,
|
||||
"minLength": 40
|
||||
},
|
||||
"write_enabled": {
|
||||
"type": "boolean",
|
||||
"description": "Permit create/update/delete operations using this key"
|
||||
@@ -245373,6 +245369,7 @@
|
||||
},
|
||||
"key": {
|
||||
"type": "string",
|
||||
"writeOnly": true,
|
||||
"maxLength": 40,
|
||||
"minLength": 40
|
||||
},
|
||||
|
||||
@@ -1,5 +1,44 @@
|
||||
# NetBox v4.4
|
||||
|
||||
## v4.4.4 (2025-10-15)
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* [#20554](https://github.com/netbox-community/netbox/issues/20554) - Fix generic relation filters to accept `<app>.<model>` format matching POST requests
|
||||
* [#20574](https://github.com/netbox-community/netbox/issues/20574) - Fix excessive storage initialization overhead when listing scripts with remote backends
|
||||
* [#20584](https://github.com/netbox-community/netbox/issues/20584) - Enforce PoE mode requirement on interface templates when PoE type is set
|
||||
* [#20585](https://github.com/netbox-community/netbox/issues/20585) - Fix API schema generation crash for models with single-field UniqueConstraints
|
||||
* [#20587](https://github.com/netbox-community/netbox/issues/20587) - Fix upgrade.sh failure when removing stale content types
|
||||
|
||||
---
|
||||
|
||||
## v4.4.3 (2025-10-14)
|
||||
|
||||
### Enhancements
|
||||
|
||||
* [#20426](https://github.com/netbox-community/netbox/issues/20426) - Add a copy-to-clipboard button for custom script output
|
||||
* [#20516](https://github.com/netbox-community/netbox/issues/20516) - Improve rendering of VLAN ID ranges in VLAN group tables
|
||||
|
||||
### Bug Fixes
|
||||
|
||||
* [#19302](https://github.com/netbox-community/netbox/issues/19302) - Fix uniqueness validation in REST API for nullable fields
|
||||
* [#19615](https://github.com/netbox-community/netbox/issues/19615) - Fix support for static file parameters in templates when external storage is in use
|
||||
* [#19818](https://github.com/netbox-community/netbox/issues/19818) - Hide primary IP assignment fields when creating a new virtual machine in the UI
|
||||
* [#19825](https://github.com/netbox-community/netbox/issues/19825) - Prevent cache for config revisions from being erroneously overwritten when debugging is enabled
|
||||
* [#20140](https://github.com/netbox-community/netbox/issues/20140) - Changing a site's region or group should update any associated circuit terminations
|
||||
* [#20156](https://github.com/netbox-community/netbox/issues/20156) - Fix display of rack elevation labels
|
||||
* [#20290](https://github.com/netbox-community/netbox/issues/20290) - Fix migration error when upgrading to NetBox v4.4 from releases earlier than v4.3
|
||||
* [#20471](https://github.com/netbox-community/netbox/issues/20471) - Saving an unmodified VLAN group should not generate a change record
|
||||
* [#20475](https://github.com/netbox-community/netbox/issues/20475) - Collapse singleton VLAN IDs in VLAN group display
|
||||
* [#20494](https://github.com/netbox-community/netbox/issues/20494) - Correct OpenAPI schema definition for `IntegerRangeSerializer`
|
||||
* [#20496](https://github.com/netbox-community/netbox/issues/20496) - REST API should always honor `MAX_PAGE_SIZE` value
|
||||
* [#20497](https://github.com/netbox-community/netbox/issues/20497) - Fix filtering of VLAN groups by VLAN ID range in GraphQL API
|
||||
* [#20507](https://github.com/netbox-community/netbox/issues/20507) - Fix support for fetching ASN contacts via GraphQL API
|
||||
* [#20523](https://github.com/netbox-community/netbox/issues/20523) - Hide password change form for users authenticated via SSO
|
||||
* [#20542](https://github.com/netbox-community/netbox/issues/20542) - Fix the creation of MAC addresses using the "quick add" form
|
||||
|
||||
---
|
||||
|
||||
## v4.4.2 (2025-09-30)
|
||||
|
||||
### Enhancements
|
||||
|
||||
@@ -80,6 +80,7 @@ class JobFilterSet(BaseFilterSet):
|
||||
method='search',
|
||||
label=_('Search'),
|
||||
)
|
||||
object_type = ContentTypeFilter()
|
||||
created = django_filters.DateTimeFilter()
|
||||
created__before = django_filters.DateTimeFilter(
|
||||
field_name='created',
|
||||
@@ -169,6 +170,7 @@ class ObjectChangeFilterSet(BaseFilterSet):
|
||||
changed_object_type_id = django_filters.ModelMultipleChoiceFilter(
|
||||
queryset=ContentType.objects.all()
|
||||
)
|
||||
related_object_type = ContentTypeFilter()
|
||||
user_id = django_filters.ModelMultipleChoiceFilter(
|
||||
queryset=User.objects.all(),
|
||||
label=_('User (ID)'),
|
||||
|
||||
@@ -3,12 +3,12 @@ from typing import Annotated, List, TYPE_CHECKING
|
||||
import strawberry
|
||||
import strawberry_django
|
||||
from django.contrib.contenttypes.models import ContentType
|
||||
from strawberry.types import Info
|
||||
|
||||
from core.models import ObjectChange
|
||||
|
||||
if TYPE_CHECKING:
|
||||
from core.graphql.types import DataFileType, DataSourceType
|
||||
from netbox.core.graphql.types import ObjectChangeType
|
||||
from core.graphql.types import DataFileType, DataSourceType, ObjectChangeType
|
||||
|
||||
__all__ = (
|
||||
'ChangelogMixin',
|
||||
@@ -20,7 +20,7 @@ __all__ = (
|
||||
class ChangelogMixin:
|
||||
|
||||
@strawberry_django.field
|
||||
def changelog(self, info) -> List[Annotated["ObjectChangeType", strawberry.lazy('.types')]]: # noqa: F821
|
||||
def changelog(self, info: Info) -> List[Annotated['ObjectChangeType', strawberry.lazy('.types')]]: # noqa: F821
|
||||
content_type = ContentType.objects.get_for_model(self)
|
||||
object_changes = ObjectChange.objects.filter(
|
||||
changed_object_type=content_type,
|
||||
@@ -31,5 +31,5 @@ class ChangelogMixin:
|
||||
|
||||
@strawberry.type
|
||||
class SyncedDataMixin:
|
||||
data_source: Annotated["DataSourceType", strawberry.lazy('core.graphql.types')] | None
|
||||
data_file: Annotated["DataFileType", strawberry.lazy('core.graphql.types')] | None
|
||||
data_source: Annotated['DataSourceType', strawberry.lazy('core.graphql.types')] | None
|
||||
data_file: Annotated['DataFileType', strawberry.lazy('core.graphql.types')] | None
|
||||
|
||||
48
netbox/core/migrations/0019_configrevision_active.py
Normal file
48
netbox/core/migrations/0019_configrevision_active.py
Normal file
@@ -0,0 +1,48 @@
|
||||
# Generated by Django 5.2.5 on 2025-09-09 16:48
|
||||
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
def get_active(apps, schema_editor):
|
||||
from django.core.cache import cache
|
||||
ConfigRevision = apps.get_model('core', 'ConfigRevision')
|
||||
version = None
|
||||
revision = None
|
||||
|
||||
# Try and get the latest version from cache
|
||||
try:
|
||||
version = cache.get('config_version')
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
# If there is a version in cache, attempt to set revision to the current version from cache
|
||||
# If the version in cache does not exist or there is no version, try the lastest revision in the database
|
||||
if not version or (version and not (revision := ConfigRevision.objects.filter(pk=version).first())):
|
||||
revision = ConfigRevision.objects.order_by('-created').first()
|
||||
|
||||
# If there is a revision set, set the active revision
|
||||
if revision:
|
||||
revision.active = True
|
||||
revision.save()
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('core', '0018_concrete_objecttype'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='configrevision',
|
||||
name='active',
|
||||
field=models.BooleanField(default=False),
|
||||
),
|
||||
migrations.RunPython(code=get_active, reverse_code=migrations.RunPython.noop),
|
||||
migrations.AddConstraint(
|
||||
model_name='configrevision',
|
||||
constraint=models.UniqueConstraint(
|
||||
condition=models.Q(('active', True)), fields=('active',), name='unique_active_config_revision'
|
||||
),
|
||||
),
|
||||
]
|
||||
@@ -14,6 +14,9 @@ class ConfigRevision(models.Model):
|
||||
"""
|
||||
An atomic revision of NetBox's configuration.
|
||||
"""
|
||||
active = models.BooleanField(
|
||||
default=False
|
||||
)
|
||||
created = models.DateTimeField(
|
||||
verbose_name=_('created'),
|
||||
auto_now_add=True
|
||||
@@ -35,6 +38,13 @@ class ConfigRevision(models.Model):
|
||||
ordering = ['-created']
|
||||
verbose_name = _('config revision')
|
||||
verbose_name_plural = _('config revisions')
|
||||
constraints = [
|
||||
models.UniqueConstraint(
|
||||
fields=('active',),
|
||||
condition=models.Q(active=True),
|
||||
name='unique_active_config_revision',
|
||||
)
|
||||
]
|
||||
|
||||
def __str__(self):
|
||||
if not self.pk:
|
||||
@@ -59,8 +69,13 @@ class ConfigRevision(models.Model):
|
||||
"""
|
||||
cache.set('config', self.data, None)
|
||||
cache.set('config_version', self.pk, None)
|
||||
|
||||
# Set all instances of ConfigRevision to false and set this instance to true
|
||||
ConfigRevision.objects.all().update(active=False)
|
||||
ConfigRevision.objects.filter(pk=self.pk).update(active=True)
|
||||
|
||||
activate.alters_data = True
|
||||
|
||||
@property
|
||||
def is_active(self):
|
||||
return cache.get('config_version') == self.pk
|
||||
return self.active
|
||||
|
||||
@@ -5,7 +5,7 @@ from django.contrib.contenttypes.models import ContentType
|
||||
from django.contrib.postgres.fields import ArrayField
|
||||
from django.contrib.postgres.indexes import GinIndex
|
||||
from django.core.exceptions import ObjectDoesNotExist
|
||||
from django.db import models
|
||||
from django.db import connection, models
|
||||
from django.db.models import Q
|
||||
from django.utils.translation import gettext as _
|
||||
|
||||
@@ -66,6 +66,14 @@ class ObjectTypeManager(models.Manager):
|
||||
"""
|
||||
from netbox.models.features import get_model_features, model_is_public
|
||||
|
||||
# TODO: Remove this in NetBox v5.0
|
||||
# If the ObjectType table has not yet been provisioned (e.g. because we're in a pre-v4.4 migration),
|
||||
# fall back to ContentType.
|
||||
if 'core_objecttype' not in connection.introspection.table_names():
|
||||
ct = ContentType.objects.get_for_model(model, for_concrete_model=for_concrete_model)
|
||||
ct.features = get_model_features(ct.model_class())
|
||||
return ct
|
||||
|
||||
if not inspect.isclass(model):
|
||||
model = model.__class__
|
||||
opts = self._get_opts(model, for_concrete_model)
|
||||
|
||||
@@ -1764,6 +1764,7 @@ class PowerOutletFilterSet(
|
||||
|
||||
class MACAddressFilterSet(NetBoxModelFilterSet):
|
||||
mac_address = MultiValueMACAddressFilter()
|
||||
assigned_object_type = ContentTypeFilter()
|
||||
device = MultiValueCharFilter(
|
||||
method='filter_device',
|
||||
field_name='name',
|
||||
|
||||
@@ -1,3 +1,5 @@
|
||||
from strawberry.types import Info
|
||||
|
||||
from circuits.graphql.types import CircuitTerminationType, ProviderNetworkType
|
||||
from circuits.models import CircuitTermination, ProviderNetwork
|
||||
from dcim.graphql.types import (
|
||||
@@ -49,7 +51,7 @@ class InventoryItemTemplateComponentType:
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def resolve_type(cls, instance, info):
|
||||
def resolve_type(cls, instance, info: Info):
|
||||
if type(instance) is ConsolePortTemplate:
|
||||
return ConsolePortTemplateType
|
||||
if type(instance) is ConsoleServerPortTemplate:
|
||||
@@ -79,7 +81,7 @@ class InventoryItemComponentType:
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def resolve_type(cls, instance, info):
|
||||
def resolve_type(cls, instance, info: Info):
|
||||
if type(instance) is ConsolePort:
|
||||
return ConsolePortType
|
||||
if type(instance) is ConsoleServerPort:
|
||||
@@ -112,7 +114,7 @@ class ConnectedEndpointType:
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def resolve_type(cls, instance, info):
|
||||
def resolve_type(cls, instance, info: Info):
|
||||
if type(instance) is CircuitTermination:
|
||||
return CircuitTerminationType
|
||||
if type(instance) is ConsolePortType:
|
||||
|
||||
@@ -7,6 +7,7 @@ from mptt.models import MPTTModel, TreeForeignKey
|
||||
|
||||
from dcim.choices import *
|
||||
from dcim.constants import *
|
||||
from dcim.models.mixins import InterfaceValidationMixin
|
||||
from netbox.models import ChangeLoggedModel
|
||||
from utilities.fields import ColorField, NaturalOrderingField
|
||||
from utilities.mptt import TreeManager
|
||||
@@ -405,7 +406,7 @@ class PowerOutletTemplate(ModularComponentTemplateModel):
|
||||
}
|
||||
|
||||
|
||||
class InterfaceTemplate(ModularComponentTemplateModel):
|
||||
class InterfaceTemplate(InterfaceValidationMixin, ModularComponentTemplateModel):
|
||||
"""
|
||||
A template for a physical data interface on a new Device.
|
||||
"""
|
||||
@@ -469,8 +470,6 @@ class InterfaceTemplate(ModularComponentTemplateModel):
|
||||
super().clean()
|
||||
|
||||
if self.bridge:
|
||||
if self.pk and self.bridge_id == self.pk:
|
||||
raise ValidationError({'bridge': _("An interface cannot be bridged to itself.")})
|
||||
if self.device_type and self.device_type != self.bridge.device_type:
|
||||
raise ValidationError({
|
||||
'bridge': _(
|
||||
@@ -484,11 +483,6 @@ class InterfaceTemplate(ModularComponentTemplateModel):
|
||||
).format(bridge=self.bridge)
|
||||
})
|
||||
|
||||
if self.rf_role and self.type not in WIRELESS_IFACE_TYPES:
|
||||
raise ValidationError({
|
||||
'rf_role': "Wireless role may be set only on wireless interfaces."
|
||||
})
|
||||
|
||||
def instantiate(self, **kwargs):
|
||||
return self.component_model(
|
||||
name=self.resolve_name(kwargs.get('module')),
|
||||
|
||||
@@ -11,6 +11,7 @@ from mptt.models import MPTTModel, TreeForeignKey
|
||||
from dcim.choices import *
|
||||
from dcim.constants import *
|
||||
from dcim.fields import WWNField
|
||||
from dcim.models.mixins import InterfaceValidationMixin
|
||||
from netbox.choices import ColorChoices
|
||||
from netbox.models import OrganizationalModel, NetBoxModel
|
||||
from utilities.fields import ColorField, NaturalOrderingField
|
||||
@@ -676,7 +677,14 @@ class BaseInterface(models.Model):
|
||||
return self.primary_mac_address.mac_address
|
||||
|
||||
|
||||
class Interface(ModularComponentModel, BaseInterface, CabledObjectModel, PathEndpoint, TrackingModelMixin):
|
||||
class Interface(
|
||||
InterfaceValidationMixin,
|
||||
ModularComponentModel,
|
||||
BaseInterface,
|
||||
CabledObjectModel,
|
||||
PathEndpoint,
|
||||
TrackingModelMixin,
|
||||
):
|
||||
"""
|
||||
A network interface within a Device. A physical Interface can connect to exactly one other Interface.
|
||||
"""
|
||||
@@ -893,10 +901,6 @@ class Interface(ModularComponentModel, BaseInterface, CabledObjectModel, PathEnd
|
||||
|
||||
# Bridge validation
|
||||
|
||||
# An interface cannot be bridged to itself
|
||||
if self.pk and self.bridge_id == self.pk:
|
||||
raise ValidationError({'bridge': _("An interface cannot be bridged to itself.")})
|
||||
|
||||
# A bridged interface belongs to the same device or virtual chassis
|
||||
if self.bridge and self.bridge.device != self.device:
|
||||
if self.device.virtual_chassis is None:
|
||||
@@ -942,29 +946,9 @@ class Interface(ModularComponentModel, BaseInterface, CabledObjectModel, PathEnd
|
||||
)
|
||||
})
|
||||
|
||||
# PoE validation
|
||||
|
||||
# Only physical interfaces may have a PoE mode/type assigned
|
||||
if self.poe_mode and self.is_virtual:
|
||||
raise ValidationError({
|
||||
'poe_mode': _("Virtual interfaces cannot have a PoE mode.")
|
||||
})
|
||||
if self.poe_type and self.is_virtual:
|
||||
raise ValidationError({
|
||||
'poe_type': _("Virtual interfaces cannot have a PoE type.")
|
||||
})
|
||||
|
||||
# An interface with a PoE type set must also specify a mode
|
||||
if self.poe_type and not self.poe_mode:
|
||||
raise ValidationError({
|
||||
'poe_type': _("Must specify PoE mode when designating a PoE type.")
|
||||
})
|
||||
|
||||
# Wireless validation
|
||||
|
||||
# RF role & channel may only be set for wireless interfaces
|
||||
if self.rf_role and not self.is_wireless:
|
||||
raise ValidationError({'rf_role': _("Wireless role may be set only on wireless interfaces.")})
|
||||
# RF channel may only be set for wireless interfaces
|
||||
if self.rf_channel and not self.is_wireless:
|
||||
raise ValidationError({'rf_channel': _("Channel may be set only on wireless interfaces.")})
|
||||
|
||||
|
||||
@@ -4,8 +4,11 @@ from django.core.exceptions import ValidationError
|
||||
from django.db import models
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
|
||||
from dcim.constants import VIRTUAL_IFACE_TYPES, WIRELESS_IFACE_TYPES
|
||||
|
||||
__all__ = (
|
||||
'CachedScopeMixin',
|
||||
'InterfaceValidationMixin',
|
||||
'RenderConfigMixin',
|
||||
)
|
||||
|
||||
@@ -116,3 +119,33 @@ class CachedScopeMixin(models.Model):
|
||||
self._site = self.scope.site
|
||||
self._location = self.scope
|
||||
cache_related_objects.alters_data = True
|
||||
|
||||
|
||||
class InterfaceValidationMixin:
|
||||
|
||||
def clean(self):
|
||||
super().clean()
|
||||
|
||||
# An interface cannot be bridged to itself
|
||||
if self.pk and self.bridge_id == self.pk:
|
||||
raise ValidationError({'bridge': _("An interface cannot be bridged to itself.")})
|
||||
|
||||
# Only physical interfaces may have a PoE mode/type assigned
|
||||
if self.poe_mode and self.type in VIRTUAL_IFACE_TYPES:
|
||||
raise ValidationError({
|
||||
'poe_mode': _("Virtual interfaces cannot have a PoE mode.")
|
||||
})
|
||||
if self.poe_type and self.type in VIRTUAL_IFACE_TYPES:
|
||||
raise ValidationError({
|
||||
'poe_type': _("Virtual interfaces cannot have a PoE type.")
|
||||
})
|
||||
|
||||
# An interface with a PoE type set must also specify a mode
|
||||
if self.poe_type and not self.poe_mode:
|
||||
raise ValidationError({
|
||||
'poe_type': _("Must specify PoE mode when designating a PoE type.")
|
||||
})
|
||||
|
||||
# RF role may be set only for wireless interfaces
|
||||
if self.rf_role and self.type not in WIRELESS_IFACE_TYPES:
|
||||
raise ValidationError({'rf_role': _("Wireless role may be set only on wireless interfaces.")})
|
||||
|
||||
@@ -196,7 +196,7 @@ class DeviceTable(TenancyColumnsMixin, ContactsColumnMixin, NetBoxTable):
|
||||
verbose_name=_('Type')
|
||||
)
|
||||
u_height = columns.TemplateColumn(
|
||||
accessor=tables.A('device_type.u_height'),
|
||||
accessor=tables.A('device_type__u_height'),
|
||||
verbose_name=_('U Height'),
|
||||
template_code='{{ value|floatformat }}'
|
||||
)
|
||||
|
||||
@@ -7,13 +7,14 @@ from django.test import override_settings, tag
|
||||
from django.urls import reverse
|
||||
from netaddr import EUI
|
||||
|
||||
from core.models import ObjectType
|
||||
from dcim.choices import *
|
||||
from dcim.constants import *
|
||||
from dcim.models import *
|
||||
from ipam.models import ASN, RIR, VLAN, VRF
|
||||
from netbox.choices import CSVDelimiterChoices, ImportFormatChoices, WeightUnitChoices
|
||||
from tenancy.models import Tenant
|
||||
from users.models import User
|
||||
from users.models import ObjectPermission, User
|
||||
from utilities.testing import ViewTestCases, create_tags, create_test_device, post_data
|
||||
from wireless.models import WirelessLAN
|
||||
|
||||
@@ -3728,3 +3729,29 @@ class MACAddressTestCase(ViewTestCases.PrimaryObjectViewTestCase):
|
||||
cls.bulk_edit_data = {
|
||||
'description': 'New description',
|
||||
}
|
||||
|
||||
@tag('regression') # Issue #20542
|
||||
@override_settings(EXEMPT_VIEW_PERMISSIONS=['*'], EXEMPT_EXCLUDE_MODELS=[])
|
||||
def test_create_macaddress_via_quickadd(self):
|
||||
"""
|
||||
Test creating a MAC address via quick-add modal (e.g., from Interface form).
|
||||
Regression test for issue #20542 where form prefix was missing in POST handler.
|
||||
"""
|
||||
obj_perm = ObjectPermission(name='Test permission', actions=['add'])
|
||||
obj_perm.save()
|
||||
obj_perm.users.add(self.user)
|
||||
obj_perm.object_types.add(ObjectType.objects.get_for_model(self.model))
|
||||
|
||||
# Simulate quick-add form submission with 'quickadd-' prefix
|
||||
formatted_data = post_data(self.form_data)
|
||||
quickadd_data = {f'quickadd-{k}': v for k, v in formatted_data.items()}
|
||||
quickadd_data['_quickadd'] = 'True'
|
||||
|
||||
initial_count = self._get_queryset().count()
|
||||
url = f"{self._get_url('add')}?_quickadd=True&target=id_primary_mac_address"
|
||||
response = self.client.post(url, data=quickadd_data)
|
||||
|
||||
# Should successfully create the MAC address and return the quick_add_created template
|
||||
self.assertHttpStatus(response, 200)
|
||||
self.assertIn(b'quick-add-object', response.content)
|
||||
self.assertEqual(initial_count + 1, self._get_queryset().count())
|
||||
|
||||
@@ -2,6 +2,7 @@ from typing import TYPE_CHECKING, Annotated, List
|
||||
|
||||
import strawberry
|
||||
import strawberry_django
|
||||
from strawberry.types import Info
|
||||
|
||||
__all__ = (
|
||||
'ConfigContextMixin',
|
||||
@@ -37,7 +38,7 @@ class CustomFieldsMixin:
|
||||
class ImageAttachmentsMixin:
|
||||
|
||||
@strawberry_django.field
|
||||
def image_attachments(self, info) -> List[Annotated["ImageAttachmentType", strawberry.lazy('.types')]]:
|
||||
def image_attachments(self, info: Info) -> List[Annotated['ImageAttachmentType', strawberry.lazy('.types')]]:
|
||||
return self.images.restrict(info.context.request.user, 'view')
|
||||
|
||||
|
||||
@@ -45,17 +46,17 @@ class ImageAttachmentsMixin:
|
||||
class JournalEntriesMixin:
|
||||
|
||||
@strawberry_django.field
|
||||
def journal_entries(self, info) -> List[Annotated["JournalEntryType", strawberry.lazy('.types')]]:
|
||||
def journal_entries(self, info: Info) -> List[Annotated['JournalEntryType', strawberry.lazy('.types')]]:
|
||||
return self.journal_entries.all()
|
||||
|
||||
|
||||
@strawberry.type
|
||||
class TagsMixin:
|
||||
|
||||
tags: List[Annotated["TagType", strawberry.lazy('.types')]]
|
||||
tags: List[Annotated['TagType', strawberry.lazy('.types')]]
|
||||
|
||||
|
||||
@strawberry.type
|
||||
class ContactsMixin:
|
||||
|
||||
contacts: List[Annotated["ContactAssignmentType", strawberry.lazy('tenancy.graphql.types')]]
|
||||
contacts: List[Annotated['ContactAssignmentType', strawberry.lazy('tenancy.graphql.types')]]
|
||||
|
||||
@@ -1,9 +1,39 @@
|
||||
from django.contrib.postgres.fields import ArrayField
|
||||
from django.contrib.postgres.fields.ranges import RangeField
|
||||
from django.db.models import CharField, JSONField, Lookup
|
||||
from django.db.models.fields.json import KeyTextTransform
|
||||
|
||||
from .fields import CachedValueField
|
||||
|
||||
|
||||
class RangeContains(Lookup):
|
||||
"""
|
||||
Filter ArrayField(RangeField) columns where ANY element-range contains the scalar RHS.
|
||||
|
||||
Usage (ORM):
|
||||
Model.objects.filter(<range_array_field>__range_contains=<scalar>)
|
||||
|
||||
Works with int4range[], int8range[], daterange[], tstzrange[], etc.
|
||||
"""
|
||||
|
||||
lookup_name = 'range_contains'
|
||||
|
||||
def as_sql(self, compiler, connection):
|
||||
# Compile LHS (the array-of-ranges column/expression) and RHS (scalar)
|
||||
lhs, lhs_params = self.process_lhs(compiler, connection)
|
||||
rhs, rhs_params = self.process_rhs(compiler, connection)
|
||||
|
||||
# Guard: only allow ArrayField whose base_field is a PostgreSQL RangeField
|
||||
field = getattr(self.lhs, 'output_field', None)
|
||||
if not (isinstance(field, ArrayField) and isinstance(field.base_field, RangeField)):
|
||||
raise TypeError('range_contains is only valid for ArrayField(RangeField) columns')
|
||||
|
||||
# Range-contains-element using EXISTS + UNNEST keeps the range on the LHS: r @> value
|
||||
sql = f"EXISTS (SELECT 1 FROM unnest({lhs}) AS r WHERE r @> {rhs})"
|
||||
params = lhs_params + rhs_params
|
||||
return sql, params
|
||||
|
||||
|
||||
class Empty(Lookup):
|
||||
"""
|
||||
Filter on whether a string is empty.
|
||||
@@ -25,7 +55,7 @@ class JSONEmpty(Lookup):
|
||||
|
||||
A key is considered empty if it is "", null, or does not exist.
|
||||
"""
|
||||
lookup_name = "empty"
|
||||
lookup_name = 'empty'
|
||||
|
||||
def as_sql(self, compiler, connection):
|
||||
# self.lhs.lhs is the parent expression (could be a JSONField or another KeyTransform)
|
||||
@@ -69,6 +99,7 @@ class NetContainsOrEquals(Lookup):
|
||||
return 'CAST(%s AS INET) >>= %s' % (lhs, rhs), params
|
||||
|
||||
|
||||
ArrayField.register_lookup(RangeContains)
|
||||
CharField.register_lookup(Empty)
|
||||
JSONField.register_lookup(JSONEmpty)
|
||||
CachedValueField.register_lookup(NetHost)
|
||||
|
||||
@@ -90,7 +90,7 @@ class ConfigContextModelQuerySet(RestrictedQuerySet):
|
||||
ConfigContext.objects.filter(
|
||||
self._get_config_context_filters()
|
||||
).annotate(
|
||||
_data=EmptyGroupByJSONBAgg('data', ordering=['weight', 'name'])
|
||||
_data=EmptyGroupByJSONBAgg('data', order_by=['weight', 'name'])
|
||||
).values("_data").order_by()
|
||||
)
|
||||
)
|
||||
|
||||
@@ -329,6 +329,9 @@ class BaseScript:
|
||||
# Declare the placeholder for the current request
|
||||
self.request = None
|
||||
|
||||
# Initiate the storage backend (local, S3, etc) as a class attr
|
||||
self.storage = storages.create_storage(storages.backends["scripts"])
|
||||
|
||||
# Compile test methods and initialize results skeleton
|
||||
for method in dir(self):
|
||||
if method.startswith('test_') and callable(getattr(self, method)):
|
||||
@@ -394,8 +397,7 @@ class BaseScript:
|
||||
return inspect.getfile(self.__class__)
|
||||
|
||||
def findsource(self, object):
|
||||
storage = storages.create_storage(storages.backends["scripts"])
|
||||
with storage.open(os.path.basename(self.filename), 'r') as f:
|
||||
with self.storage.open(os.path.basename(self.filename), 'r') as f:
|
||||
data = f.read()
|
||||
|
||||
# Break the source code into lines
|
||||
|
||||
@@ -595,6 +595,7 @@ class IPAddressFilterSet(NetBoxModelFilterSet, TenancyFilterSet, ContactModelFil
|
||||
to_field_name='rd',
|
||||
label=_('VRF (RD)'),
|
||||
)
|
||||
assigned_object_type = ContentTypeFilter()
|
||||
device = MultiValueCharFilter(
|
||||
method='filter_device',
|
||||
field_name='name',
|
||||
@@ -908,7 +909,8 @@ class VLANGroupFilterSet(OrganizationalModelFilterSet, TenancyFilterSet):
|
||||
method='filter_scope'
|
||||
)
|
||||
contains_vid = django_filters.NumberFilter(
|
||||
method='filter_contains_vid'
|
||||
field_name='vid_ranges',
|
||||
lookup_expr='range_contains',
|
||||
)
|
||||
|
||||
class Meta:
|
||||
@@ -931,21 +933,6 @@ class VLANGroupFilterSet(OrganizationalModelFilterSet, TenancyFilterSet):
|
||||
scope_id=value
|
||||
)
|
||||
|
||||
def filter_contains_vid(self, queryset, name, value):
|
||||
"""
|
||||
Return all VLANGroups which contain the given VLAN ID.
|
||||
"""
|
||||
table_name = VLANGroup._meta.db_table
|
||||
# TODO: See if this can be optimized without compromising queryset integrity
|
||||
# Expand VLAN ID ranges to query by integer
|
||||
groups = VLANGroup.objects.raw(
|
||||
f'SELECT id FROM {table_name}, unnest(vid_ranges) vid_range WHERE %s <@ vid_range',
|
||||
params=(value,)
|
||||
)
|
||||
return queryset.filter(
|
||||
pk__in=[g.id for g in groups]
|
||||
)
|
||||
|
||||
|
||||
class VLANFilterSet(NetBoxModelFilterSet, TenancyFilterSet):
|
||||
region_id = TreeNodeMultipleChoiceFilter(
|
||||
@@ -1166,6 +1153,7 @@ class ServiceTemplateFilterSet(NetBoxModelFilterSet):
|
||||
|
||||
|
||||
class ServiceFilterSet(ContactModelFilterSet, NetBoxModelFilterSet):
|
||||
parent_object_type = ContentTypeFilter()
|
||||
device = MultiValueCharFilter(
|
||||
method='filter_device',
|
||||
field_name='name',
|
||||
|
||||
@@ -19,7 +19,7 @@ from tenancy.graphql.filter_mixins import ContactFilterMixin, TenancyFilterMixin
|
||||
from virtualization.models import VMInterface
|
||||
|
||||
if TYPE_CHECKING:
|
||||
from netbox.graphql.filter_lookups import IntegerArrayLookup, IntegerLookup
|
||||
from netbox.graphql.filter_lookups import IntegerLookup, IntegerRangeArrayLookup
|
||||
from circuits.graphql.filters import ProviderFilter
|
||||
from core.graphql.filters import ContentTypeFilter
|
||||
from dcim.graphql.filters import SiteFilter
|
||||
@@ -340,7 +340,7 @@ class VLANFilter(TenancyFilterMixin, PrimaryModelFilterMixin):
|
||||
|
||||
@strawberry_django.filter_type(models.VLANGroup, lookups=True)
|
||||
class VLANGroupFilter(ScopedFilterMixin, OrganizationalModelFilterMixin):
|
||||
vid_ranges: Annotated['IntegerArrayLookup', strawberry.lazy('netbox.graphql.filter_lookups')] | None = (
|
||||
vid_ranges: Annotated['IntegerRangeArrayLookup', strawberry.lazy('netbox.graphql.filter_lookups')] | None = (
|
||||
strawberry_django.filter_field()
|
||||
)
|
||||
|
||||
|
||||
@@ -10,9 +10,9 @@ from django.utils.translation import gettext_lazy as _
|
||||
from dcim.models import Interface, Site, SiteGroup
|
||||
from ipam.choices import *
|
||||
from ipam.constants import *
|
||||
from ipam.querysets import VLANQuerySet, VLANGroupQuerySet
|
||||
from ipam.querysets import VLANGroupQuerySet, VLANQuerySet
|
||||
from netbox.models import OrganizationalModel, PrimaryModel, NetBoxModel
|
||||
from utilities.data import check_ranges_overlap, ranges_to_string
|
||||
from utilities.data import check_ranges_overlap, ranges_to_string, ranges_to_string_list
|
||||
from virtualization.models import VMInterface
|
||||
|
||||
__all__ = (
|
||||
@@ -164,8 +164,18 @@ class VLANGroup(OrganizationalModel):
|
||||
"""
|
||||
return VLAN.objects.filter(group=self).order_by('vid')
|
||||
|
||||
@property
|
||||
def vid_ranges_items(self):
|
||||
"""
|
||||
Property that converts VID ranges to a list of string representations.
|
||||
"""
|
||||
return ranges_to_string_list(self.vid_ranges)
|
||||
|
||||
@property
|
||||
def vid_ranges_list(self):
|
||||
"""
|
||||
Property that converts VID ranges into a string representation.
|
||||
"""
|
||||
return ranges_to_string(self.vid_ranges)
|
||||
|
||||
|
||||
|
||||
@@ -41,7 +41,8 @@ class VLANGroupTable(TenancyColumnsMixin, NetBoxTable):
|
||||
linkify=True,
|
||||
orderable=False
|
||||
)
|
||||
vid_ranges_list = tables.Column(
|
||||
vid_ranges_list = columns.ArrayColumn(
|
||||
accessor='vid_ranges_items',
|
||||
verbose_name=_('VID Ranges'),
|
||||
orderable=False
|
||||
)
|
||||
|
||||
@@ -1723,6 +1723,10 @@ class VLANGroupTestCase(TestCase, ChangeLoggedFilterSetTests):
|
||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 2)
|
||||
params = {'contains_vid': 1}
|
||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 8)
|
||||
params = {'contains_vid': 12} # 11 is NOT in [1,11)
|
||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 1)
|
||||
params = {'contains_vid': 4095}
|
||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 0)
|
||||
|
||||
def test_region(self):
|
||||
params = {'region': Region.objects.first().pk}
|
||||
|
||||
66
netbox/ipam/tests/test_lookups.py
Normal file
66
netbox/ipam/tests/test_lookups.py
Normal file
@@ -0,0 +1,66 @@
|
||||
from django.test import TestCase
|
||||
from django.db.backends.postgresql.psycopg_any import NumericRange
|
||||
from ipam.models import VLANGroup
|
||||
|
||||
|
||||
class VLANGroupRangeContainsLookupTests(TestCase):
|
||||
@classmethod
|
||||
def setUpTestData(cls):
|
||||
# Two ranges: [1,11) and [20,31)
|
||||
cls.g1 = VLANGroup.objects.create(
|
||||
name='VlanGroup-A',
|
||||
slug='VlanGroup-A',
|
||||
vid_ranges=[NumericRange(1, 11), NumericRange(20, 31)],
|
||||
)
|
||||
# One range: [100,201)
|
||||
cls.g2 = VLANGroup.objects.create(
|
||||
name='VlanGroup-B',
|
||||
slug='VlanGroup-B',
|
||||
vid_ranges=[NumericRange(100, 201)],
|
||||
)
|
||||
cls.g_empty = VLANGroup.objects.create(
|
||||
name='VlanGroup-empty',
|
||||
slug='VlanGroup-empty',
|
||||
vid_ranges=[],
|
||||
)
|
||||
|
||||
def test_contains_value_in_first_range(self):
|
||||
"""
|
||||
Tests whether a specific value is contained within the first range in a queried
|
||||
set of VLANGroup objects.
|
||||
"""
|
||||
names = list(
|
||||
VLANGroup.objects.filter(vid_ranges__range_contains=10).values_list('name', flat=True).order_by('name')
|
||||
)
|
||||
self.assertEqual(names, ['VlanGroup-A'])
|
||||
|
||||
def test_contains_value_in_second_range(self):
|
||||
"""
|
||||
Tests if a value exists in the second range of VLANGroup objects and
|
||||
validates the result against the expected list of names.
|
||||
"""
|
||||
names = list(
|
||||
VLANGroup.objects.filter(vid_ranges__range_contains=25).values_list('name', flat=True).order_by('name')
|
||||
)
|
||||
self.assertEqual(names, ['VlanGroup-A'])
|
||||
|
||||
def test_upper_bound_is_exclusive(self):
|
||||
"""
|
||||
Tests if the upper bound of the range is exclusive in the filter method.
|
||||
"""
|
||||
# 11 is NOT in [1,11)
|
||||
self.assertFalse(VLANGroup.objects.filter(vid_ranges__range_contains=11).exists())
|
||||
|
||||
def test_no_match_far_outside(self):
|
||||
"""
|
||||
Tests that no VLANGroup contains a VID within a specified range far outside
|
||||
common VID bounds and returns `False`.
|
||||
"""
|
||||
self.assertFalse(VLANGroup.objects.filter(vid_ranges__range_contains=4095).exists())
|
||||
|
||||
def test_empty_array_never_matches(self):
|
||||
"""
|
||||
Tests the behavior of VLANGroup objects when an empty array is used to match a
|
||||
specific condition.
|
||||
"""
|
||||
self.assertFalse(VLANGroup.objects.filter(pk=self.g_empty.pk, vid_ranges__range_contains=1).exists())
|
||||
@@ -78,11 +78,16 @@ class Config:
|
||||
from core.models import ConfigRevision
|
||||
|
||||
try:
|
||||
revision = ConfigRevision.objects.last()
|
||||
# Enforce the creation date as the ordering parameter
|
||||
revision = ConfigRevision.objects.get(active=True)
|
||||
logger.debug(f"Loaded active configuration revision #{revision.pk}")
|
||||
except (ConfigRevision.DoesNotExist, ConfigRevision.MultipleObjectsReturned):
|
||||
logger.warning("No active configuration revision found - falling back to most recent")
|
||||
revision = ConfigRevision.objects.order_by('-created').first()
|
||||
if revision is None:
|
||||
logger.debug("No previous configuration found in database; proceeding with default values")
|
||||
return
|
||||
logger.debug("Loaded configuration data from database")
|
||||
logger.debug(f"Using fallback configuration revision #{revision.pk}")
|
||||
except DatabaseError:
|
||||
# The database may not be available yet (e.g. when running a management command)
|
||||
logger.warning("Skipping config initialization (database unavailable)")
|
||||
|
||||
@@ -7,6 +7,7 @@ from django.core.exceptions import FieldDoesNotExist
|
||||
from django.db.models import Q, QuerySet
|
||||
from django.db.models.fields.related import ForeignKey, ManyToManyField, ManyToManyRel, ManyToOneRel
|
||||
from strawberry import ID
|
||||
from strawberry.directive import DirectiveValue
|
||||
from strawberry.types import Info
|
||||
from strawberry_django import (
|
||||
ComparisonFilterLookup,
|
||||
@@ -24,6 +25,7 @@ __all__ = (
|
||||
'FloatLookup',
|
||||
'IntegerArrayLookup',
|
||||
'IntegerLookup',
|
||||
'IntegerRangeArrayLookup',
|
||||
'JSONFilter',
|
||||
'StringArrayLookup',
|
||||
'TreeNodeFilter',
|
||||
@@ -67,7 +69,7 @@ class IntegerLookup:
|
||||
return None
|
||||
|
||||
@strawberry_django.filter_field
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: str = '') -> Tuple[QuerySet, Q]:
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: DirectiveValue[str] = '') -> Tuple[QuerySet, Q]:
|
||||
filters = self.get_filter()
|
||||
|
||||
if not filters:
|
||||
@@ -90,7 +92,7 @@ class FloatLookup:
|
||||
return None
|
||||
|
||||
@strawberry_django.filter_field
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: str = '') -> Tuple[QuerySet, Q]:
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: DirectiveValue[str] = '') -> Tuple[QuerySet, Q]:
|
||||
filters = self.get_filter()
|
||||
|
||||
if not filters:
|
||||
@@ -109,7 +111,7 @@ class JSONFilter:
|
||||
lookup: JSONLookup
|
||||
|
||||
@strawberry_django.filter_field
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: str = '') -> Tuple[QuerySet, Q]:
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: DirectiveValue[str] = '') -> Tuple[QuerySet, Q]:
|
||||
filters = self.lookup.get_filter()
|
||||
|
||||
if not filters:
|
||||
@@ -136,7 +138,7 @@ class TreeNodeFilter:
|
||||
match_type: TreeNodeMatch
|
||||
|
||||
@strawberry_django.filter_field
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: str = '') -> Tuple[QuerySet, Q]:
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: DirectiveValue[str] = '') -> Tuple[QuerySet, Q]:
|
||||
model_field_name = prefix.removesuffix('__').removesuffix('_id')
|
||||
model_field = None
|
||||
try:
|
||||
@@ -217,3 +219,30 @@ class FloatArrayLookup(ArrayLookup[float]):
|
||||
@strawberry.input(one_of=True, description='Lookup for Array fields. Only one of the lookup fields can be set.')
|
||||
class StringArrayLookup(ArrayLookup[str]):
|
||||
pass
|
||||
|
||||
|
||||
@strawberry.input(one_of=True, description='Lookups for an ArrayField(RangeField). Only one may be set.')
|
||||
class RangeArrayValueLookup(Generic[T]):
|
||||
"""
|
||||
class for Array field of Range fields lookups
|
||||
"""
|
||||
|
||||
contains: T | None = strawberry.field(
|
||||
default=strawberry.UNSET, description='Return rows where any stored range contains this value.'
|
||||
)
|
||||
|
||||
@strawberry_django.filter_field
|
||||
def filter(self, info: Info, queryset: QuerySet, prefix: str = '') -> Tuple[QuerySet, Q]:
|
||||
"""
|
||||
Map GraphQL: { <field>: { contains: <T> } } To Django ORM: <field>__range_contains=<T>
|
||||
"""
|
||||
if self.contains is strawberry.UNSET or self.contains is None:
|
||||
return queryset, Q()
|
||||
|
||||
# Build '<prefix>range_contains' so it works for nested paths too
|
||||
return queryset, Q(**{f'{prefix}range_contains': self.contains})
|
||||
|
||||
|
||||
@strawberry.input(one_of=True, description='Lookups for an ArrayField(IntegerRangeField). Only one may be set.')
|
||||
class IntegerRangeArrayLookup(RangeArrayValueLookup[int]):
|
||||
pass
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import strawberry
|
||||
import strawberry_django
|
||||
from strawberry.types import Info
|
||||
from django.contrib.contenttypes.models import ContentType
|
||||
|
||||
from core.graphql.mixins import ChangelogMixin
|
||||
@@ -26,7 +27,7 @@ class BaseObjectType:
|
||||
"""
|
||||
|
||||
@classmethod
|
||||
def get_queryset(cls, queryset, info, **kwargs):
|
||||
def get_queryset(cls, queryset, info: Info, **kwargs):
|
||||
# Enforce object permissions on the queryset
|
||||
if hasattr(queryset, 'restrict'):
|
||||
return queryset.restrict(info.context.request.user, 'view')
|
||||
|
||||
@@ -673,10 +673,17 @@ def has_feature(model_or_ct, feature):
|
||||
# If an ObjectType was passed, we can use it directly
|
||||
if type(model_or_ct) is ObjectType:
|
||||
ot = model_or_ct
|
||||
# If a ContentType was passed, resolve its model class
|
||||
# If a ContentType was passed, resolve its model class and run the associated feature test
|
||||
elif type(model_or_ct) is ContentType:
|
||||
model_class = model_or_ct.model_class()
|
||||
ot = ObjectType.objects.get_for_model(model_class) if model_class else None
|
||||
model = model_or_ct.model_class()
|
||||
if model is None: # Stale content type
|
||||
return False
|
||||
try:
|
||||
test_func = registry['model_features'][feature]
|
||||
except KeyError:
|
||||
# Unknown feature
|
||||
return False
|
||||
return test_func(model)
|
||||
# For anything else, look up the ObjectType
|
||||
else:
|
||||
ot = ObjectType.objects.get_for_model(model_or_ct)
|
||||
|
||||
39
netbox/netbox/monkey.py
Normal file
39
netbox/netbox/monkey.py
Normal file
@@ -0,0 +1,39 @@
|
||||
from django.db.models import UniqueConstraint
|
||||
from rest_framework.utils.field_mapping import get_unique_error_message
|
||||
from rest_framework.validators import UniqueValidator
|
||||
|
||||
__all__ = (
|
||||
'get_unique_validators',
|
||||
)
|
||||
|
||||
|
||||
def get_unique_validators(field_name, model_field):
|
||||
"""
|
||||
Extend Django REST Framework's get_unique_validators() function to attach a UniqueValidator to a field *only* if the
|
||||
associated UniqueConstraint does NOT have a condition which references another field. See bug #19302.
|
||||
"""
|
||||
field_set = {field_name}
|
||||
conditions = {
|
||||
c.condition
|
||||
for c in model_field.model._meta.constraints
|
||||
if isinstance(c, UniqueConstraint) and set(c.fields) == field_set
|
||||
}
|
||||
|
||||
# START custom logic
|
||||
conditions = {
|
||||
cond for cond in conditions
|
||||
if cond is None or cond.referenced_base_fields == field_set
|
||||
}
|
||||
# END custom logic
|
||||
|
||||
if getattr(model_field, 'unique', False):
|
||||
conditions.add(None)
|
||||
if not conditions:
|
||||
return
|
||||
unique_error_message = get_unique_error_message(model_field)
|
||||
queryset = model_field.model._default_manager
|
||||
for condition in conditions:
|
||||
yield UniqueValidator(
|
||||
queryset=queryset if condition is None else queryset.filter(condition),
|
||||
message=unique_error_message
|
||||
)
|
||||
@@ -11,6 +11,7 @@ from django.core.exceptions import ImproperlyConfigured, ValidationError
|
||||
from django.core.validators import URLValidator
|
||||
from django.utils.module_loading import import_string
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from rest_framework.utils import field_mapping
|
||||
|
||||
from core.exceptions import IncompatiblePluginError
|
||||
from netbox.config import PARAMS as CONFIG_PARAMS
|
||||
@@ -20,6 +21,17 @@ from netbox.registry import registry
|
||||
import storages.utils # type: ignore
|
||||
from utilities.release import load_release_data
|
||||
from utilities.string import trailing_slash
|
||||
from .monkey import get_unique_validators
|
||||
|
||||
|
||||
#
|
||||
# Monkey-patching
|
||||
#
|
||||
|
||||
# TODO: Remove this once #20547 has been implemented
|
||||
# Override DRF's get_unique_validators() function with our own (see bug #19302)
|
||||
field_mapping.get_unique_validators = get_unique_validators
|
||||
|
||||
|
||||
#
|
||||
# Environment setup
|
||||
|
||||
@@ -281,7 +281,8 @@ class ObjectEditView(GetReturnURLMixin, BaseObjectView):
|
||||
|
||||
obj = self.alter_object(obj, request, args, kwargs)
|
||||
|
||||
form = self.form(data=request.POST, files=request.FILES, instance=obj)
|
||||
form_prefix = 'quickadd' if request.GET.get('_quickadd') else None
|
||||
form = self.form(data=request.POST, files=request.FILES, instance=obj, prefix=form_prefix)
|
||||
restrict_form_fields(form, request.user)
|
||||
|
||||
if form.is_valid():
|
||||
|
||||
2
netbox/project-static/dist/netbox.css
vendored
2
netbox/project-static/dist/netbox.css
vendored
File diff suppressed because one or more lines are too long
6
netbox/project-static/dist/netbox.js
vendored
6
netbox/project-static/dist/netbox.js
vendored
File diff suppressed because one or more lines are too long
4
netbox/project-static/dist/netbox.js.map
vendored
4
netbox/project-static/dist/netbox.js.map
vendored
File diff suppressed because one or more lines are too long
@@ -41,7 +41,7 @@
|
||||
"@types/node": "^22.3.0",
|
||||
"@typescript-eslint/eslint-plugin": "^8.37.0",
|
||||
"@typescript-eslint/parser": "^8.37.0",
|
||||
"esbuild": "^0.25.6",
|
||||
"esbuild": "^0.25.11",
|
||||
"esbuild-sass-plugin": "^3.3.1",
|
||||
"eslint": "<9.0",
|
||||
"eslint-config-prettier": "^9.1.0",
|
||||
|
||||
@@ -83,7 +83,7 @@ export function initRackElevation(): void {
|
||||
}
|
||||
|
||||
for (const element of getElements<HTMLObjectElement>('.rack_elevation')) {
|
||||
element.addEventListener('load', () => {
|
||||
element.addEventListener('htmx:afterSettle', () => {
|
||||
setRackView(initialView, element);
|
||||
});
|
||||
}
|
||||
|
||||
@@ -19,135 +19,135 @@
|
||||
resolved "https://registry.yarnpkg.com/@emotion/memoize/-/memoize-0.7.4.tgz#19bf0f5af19149111c40d98bb0cf82119f5d9eeb"
|
||||
integrity sha512-Ja/Vfqe3HpuzRsG1oBtWTHk2PGZ7GR+2Vz5iYGelAw8dx32K0y7PjVuxK6z1nMpZOqAFsRUPCkK1YjJ56qJlgw==
|
||||
|
||||
"@esbuild/aix-ppc64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.25.8.tgz#a1414903bb38027382f85f03dda6065056757727"
|
||||
integrity sha512-urAvrUedIqEiFR3FYSLTWQgLu5tb+m0qZw0NBEasUeo6wuqatkMDaRT+1uABiGXEu5vqgPd7FGE1BhsAIy9QVA==
|
||||
"@esbuild/aix-ppc64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.25.11.tgz#2ae33300598132cc4cf580dbbb28d30fed3c5c49"
|
||||
integrity sha512-Xt1dOL13m8u0WE8iplx9Ibbm+hFAO0GsU2P34UNoDGvZYkY8ifSiy6Zuc1lYxfG7svWE2fzqCUmFp5HCn51gJg==
|
||||
|
||||
"@esbuild/android-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.25.8.tgz#c859994089e9767224269884061f89dae6fb51c6"
|
||||
integrity sha512-OD3p7LYzWpLhZEyATcTSJ67qB5D+20vbtr6vHlHWSQYhKtzUYrETuWThmzFpZtFsBIxRvhO07+UgVA9m0i/O1w==
|
||||
"@esbuild/android-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.25.11.tgz#927708b3db5d739d6cb7709136924cc81bec9b03"
|
||||
integrity sha512-9slpyFBc4FPPz48+f6jyiXOx/Y4v34TUeDDXJpZqAWQn/08lKGeD8aDp9TMn9jDz2CiEuHwfhRmGBvpnd/PWIQ==
|
||||
|
||||
"@esbuild/android-arm@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.25.8.tgz#96a8f2ca91c6cd29ea90b1af79d83761c8ba0059"
|
||||
integrity sha512-RONsAvGCz5oWyePVnLdZY/HHwA++nxYWIX1atInlaW6SEkwq6XkP3+cb825EUcRs5Vss/lGh/2YxAb5xqc07Uw==
|
||||
"@esbuild/android-arm@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.25.11.tgz#571f94e7f4068957ec4c2cfb907deae3d01b55ae"
|
||||
integrity sha512-uoa7dU+Dt3HYsethkJ1k6Z9YdcHjTrSb5NUy66ZfZaSV8hEYGD5ZHbEMXnqLFlbBflLsl89Zke7CAdDJ4JI+Gg==
|
||||
|
||||
"@esbuild/android-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.25.8.tgz#a3a626c4fec4a024a9fa8c7679c39996e92916f0"
|
||||
integrity sha512-yJAVPklM5+4+9dTeKwHOaA+LQkmrKFX96BM0A/2zQrbS6ENCmxc4OVoBs5dPkCCak2roAD+jKCdnmOqKszPkjA==
|
||||
"@esbuild/android-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.25.11.tgz#8a3bf5cae6c560c7ececa3150b2bde76e0fb81e6"
|
||||
integrity sha512-Sgiab4xBjPU1QoPEIqS3Xx+R2lezu0LKIEcYe6pftr56PqPygbB7+szVnzoShbx64MUupqoE0KyRlN7gezbl8g==
|
||||
|
||||
"@esbuild/darwin-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.25.8.tgz#a5e1252ca2983d566af1c0ea39aded65736fc66d"
|
||||
integrity sha512-Jw0mxgIaYX6R8ODrdkLLPwBqHTtYHJSmzzd+QeytSugzQ0Vg4c5rDky5VgkoowbZQahCbsv1rT1KW72MPIkevw==
|
||||
"@esbuild/darwin-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.25.11.tgz#0a678c4ac4bf8717e67481e1a797e6c152f93c84"
|
||||
integrity sha512-VekY0PBCukppoQrycFxUqkCojnTQhdec0vevUL/EDOCnXd9LKWqD/bHwMPzigIJXPhC59Vd1WFIL57SKs2mg4w==
|
||||
|
||||
"@esbuild/darwin-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.25.8.tgz#5271b0df2bb12ce8df886704bfdd1c7cc01385d2"
|
||||
integrity sha512-Vh2gLxxHnuoQ+GjPNvDSDRpoBCUzY4Pu0kBqMBDlK4fuWbKgGtmDIeEC081xi26PPjn+1tct+Bh8FjyLlw1Zlg==
|
||||
"@esbuild/darwin-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.25.11.tgz#70f5e925a30c8309f1294d407a5e5e002e0315fe"
|
||||
integrity sha512-+hfp3yfBalNEpTGp9loYgbknjR695HkqtY3d3/JjSRUyPg/xd6q+mQqIb5qdywnDxRZykIHs3axEqU6l1+oWEQ==
|
||||
|
||||
"@esbuild/freebsd-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.8.tgz#d0a0e7fdf19733b8bb1566b81df1aa0bb7e46ada"
|
||||
integrity sha512-YPJ7hDQ9DnNe5vxOm6jaie9QsTwcKedPvizTVlqWG9GBSq+BuyWEDazlGaDTC5NGU4QJd666V0yqCBL2oWKPfA==
|
||||
"@esbuild/freebsd-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.11.tgz#4ec1db687c5b2b78b44148025da9632397553e8a"
|
||||
integrity sha512-CmKjrnayyTJF2eVuO//uSjl/K3KsMIeYeyN7FyDBjsR3lnSJHaXlVoAK8DZa7lXWChbuOk7NjAc7ygAwrnPBhA==
|
||||
|
||||
"@esbuild/freebsd-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.25.8.tgz#2de8b2e0899d08f1cb1ef3128e159616e7e85343"
|
||||
integrity sha512-MmaEXxQRdXNFsRN/KcIimLnSJrk2r5H8v+WVafRWz5xdSVmWLoITZQXcgehI2ZE6gioE6HirAEToM/RvFBeuhw==
|
||||
"@esbuild/freebsd-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.25.11.tgz#4c81abd1b142f1e9acfef8c5153d438ca53f44bb"
|
||||
integrity sha512-Dyq+5oscTJvMaYPvW3x3FLpi2+gSZTCE/1ffdwuM6G1ARang/mb3jvjxs0mw6n3Lsw84ocfo9CrNMqc5lTfGOw==
|
||||
|
||||
"@esbuild/linux-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.25.8.tgz#a4209efadc0c2975716458484a4e90c237c48ae9"
|
||||
integrity sha512-WIgg00ARWv/uYLU7lsuDK00d/hHSfES5BzdWAdAig1ioV5kaFNrtK8EqGcUBJhYqotlUByUKz5Qo6u8tt7iD/w==
|
||||
"@esbuild/linux-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.25.11.tgz#69517a111acfc2b93aa0fb5eaeb834c0202ccda5"
|
||||
integrity sha512-Qr8AzcplUhGvdyUF08A1kHU3Vr2O88xxP0Tm8GcdVOUm25XYcMPp2YqSVHbLuXzYQMf9Bh/iKx7YPqECs6ffLA==
|
||||
|
||||
"@esbuild/linux-arm@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.25.8.tgz#ccd9e291c24cd8d9142d819d463e2e7200d25b19"
|
||||
integrity sha512-FuzEP9BixzZohl1kLf76KEVOsxtIBFwCaLupVuk4eFVnOZfU+Wsn+x5Ryam7nILV2pkq2TqQM9EZPsOBuMC+kg==
|
||||
"@esbuild/linux-arm@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.25.11.tgz#58dac26eae2dba0fac5405052b9002dac088d38f"
|
||||
integrity sha512-TBMv6B4kCfrGJ8cUPo7vd6NECZH/8hPpBHHlYI3qzoYFvWu2AdTvZNuU/7hsbKWqu/COU7NIK12dHAAqBLLXgw==
|
||||
|
||||
"@esbuild/linux-ia32@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.25.8.tgz#006ad1536d0c2b28fb3a1cf0b53bcb85aaf92c4d"
|
||||
integrity sha512-A1D9YzRX1i+1AJZuFFUMP1E9fMaYY+GnSQil9Tlw05utlE86EKTUA7RjwHDkEitmLYiFsRd9HwKBPEftNdBfjg==
|
||||
"@esbuild/linux-ia32@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.25.11.tgz#b89d4efe9bdad46ba944f0f3b8ddd40834268c2b"
|
||||
integrity sha512-TmnJg8BMGPehs5JKrCLqyWTVAvielc615jbkOirATQvWWB1NMXY77oLMzsUjRLa0+ngecEmDGqt5jiDC6bfvOw==
|
||||
|
||||
"@esbuild/linux-loong64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.25.8.tgz#127b3fbfb2c2e08b1397e985932f718f09a8f5c4"
|
||||
integrity sha512-O7k1J/dwHkY1RMVvglFHl1HzutGEFFZ3kNiDMSOyUrB7WcoHGf96Sh+64nTRT26l3GMbCW01Ekh/ThKM5iI7hQ==
|
||||
"@esbuild/linux-loong64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.25.11.tgz#11f603cb60ad14392c3f5c94d64b3cc8b630fbeb"
|
||||
integrity sha512-DIGXL2+gvDaXlaq8xruNXUJdT5tF+SBbJQKbWy/0J7OhU8gOHOzKmGIlfTTl6nHaCOoipxQbuJi7O++ldrxgMw==
|
||||
|
||||
"@esbuild/linux-mips64el@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.25.8.tgz#837d1449517791e3fa7d82675a2d06d9f56cb340"
|
||||
integrity sha512-uv+dqfRazte3BzfMp8PAQXmdGHQt2oC/y2ovwpTteqrMx2lwaksiFZ/bdkXJC19ttTvNXBuWH53zy/aTj1FgGw==
|
||||
"@esbuild/linux-mips64el@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.25.11.tgz#b7d447ff0676b8ab247d69dac40a5cf08e5eeaf5"
|
||||
integrity sha512-Osx1nALUJu4pU43o9OyjSCXokFkFbyzjXb6VhGIJZQ5JZi8ylCQ9/LFagolPsHtgw6himDSyb5ETSfmp4rpiKQ==
|
||||
|
||||
"@esbuild/linux-ppc64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.25.8.tgz#aa2e3bd93ab8df084212f1895ca4b03c42d9e0fe"
|
||||
integrity sha512-GyG0KcMi1GBavP5JgAkkstMGyMholMDybAf8wF5A70CALlDM2p/f7YFE7H92eDeH/VBtFJA5MT4nRPDGg4JuzQ==
|
||||
"@esbuild/linux-ppc64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.25.11.tgz#b3a28ed7cc252a61b07ff7c8fd8a984ffd3a2f74"
|
||||
integrity sha512-nbLFgsQQEsBa8XSgSTSlrnBSrpoWh7ioFDUmwo158gIm5NNP+17IYmNWzaIzWmgCxq56vfr34xGkOcZ7jX6CPw==
|
||||
|
||||
"@esbuild/linux-riscv64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.25.8.tgz#a340620e31093fef72767dd28ab04214b3442083"
|
||||
integrity sha512-rAqDYFv3yzMrq7GIcen3XP7TUEG/4LK86LUPMIz6RT8A6pRIDn0sDcvjudVZBiiTcZCY9y2SgYX2lgK3AF+1eg==
|
||||
"@esbuild/linux-riscv64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.25.11.tgz#ce75b08f7d871a75edcf4d2125f50b21dc9dc273"
|
||||
integrity sha512-HfyAmqZi9uBAbgKYP1yGuI7tSREXwIb438q0nqvlpxAOs3XnZ8RsisRfmVsgV486NdjD7Mw2UrFSw51lzUk1ww==
|
||||
|
||||
"@esbuild/linux-s390x@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.25.8.tgz#ddfed266c8c13f5efb3105a0cd47f6dcd0e79e71"
|
||||
integrity sha512-Xutvh6VjlbcHpsIIbwY8GVRbwoviWT19tFhgdA7DlenLGC/mbc3lBoVb7jxj9Z+eyGqvcnSyIltYUrkKzWqSvg==
|
||||
"@esbuild/linux-s390x@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.25.11.tgz#cd08f6c73b6b6ff9ccdaabbd3ff6ad3dca99c263"
|
||||
integrity sha512-HjLqVgSSYnVXRisyfmzsH6mXqyvj0SA7pG5g+9W7ESgwA70AXYNpfKBqh1KbTxmQVaYxpzA/SvlB9oclGPbApw==
|
||||
|
||||
"@esbuild/linux-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.25.8.tgz#9a4f78c75c051e8c060183ebb39a269ba936a2ac"
|
||||
integrity sha512-ASFQhgY4ElXh3nDcOMTkQero4b1lgubskNlhIfJrsH5OKZXDpUAKBlNS0Kx81jwOBp+HCeZqmoJuihTv57/jvQ==
|
||||
"@esbuild/linux-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.25.11.tgz#3c3718af31a95d8946ebd3c32bb1e699bdf74910"
|
||||
integrity sha512-HSFAT4+WYjIhrHxKBwGmOOSpphjYkcswF449j6EjsjbinTZbp8PJtjsVK1XFJStdzXdy/jaddAep2FGY+wyFAQ==
|
||||
|
||||
"@esbuild/netbsd-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.8.tgz#902c80e1d678047926387230bc037e63e00697d0"
|
||||
integrity sha512-d1KfruIeohqAi6SA+gENMuObDbEjn22olAR7egqnkCD9DGBG0wsEARotkLgXDu6c4ncgWTZJtN5vcgxzWRMzcw==
|
||||
"@esbuild/netbsd-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.11.tgz#b4c767082401e3a4e8595fe53c47cd7f097c8077"
|
||||
integrity sha512-hr9Oxj1Fa4r04dNpWr3P8QKVVsjQhqrMSUzZzf+LZcYjZNqhA3IAfPQdEh1FLVUJSiu6sgAwp3OmwBfbFgG2Xg==
|
||||
|
||||
"@esbuild/netbsd-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.25.8.tgz#2d9eb4692add2681ff05a14ce99de54fbed7079c"
|
||||
integrity sha512-nVDCkrvx2ua+XQNyfrujIG38+YGyuy2Ru9kKVNyh5jAys6n+l44tTtToqHjino2My8VAY6Lw9H7RI73XFi66Cg==
|
||||
"@esbuild/netbsd-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.25.11.tgz#f2a930458ed2941d1f11ebc34b9c7d61f7a4d034"
|
||||
integrity sha512-u7tKA+qbzBydyj0vgpu+5h5AeudxOAGncb8N6C9Kh1N4n7wU1Xw1JDApsRjpShRpXRQlJLb9wY28ELpwdPcZ7A==
|
||||
|
||||
"@esbuild/openbsd-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.8.tgz#89c3b998c6de739db38ab7fb71a8a76b3fa84a45"
|
||||
integrity sha512-j8HgrDuSJFAujkivSMSfPQSAa5Fxbvk4rgNAS5i3K+r8s1X0p1uOO2Hl2xNsGFppOeHOLAVgYwDVlmxhq5h+SQ==
|
||||
"@esbuild/openbsd-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.11.tgz#b4ae93c75aec48bc1e8a0154957a05f0641f2dad"
|
||||
integrity sha512-Qq6YHhayieor3DxFOoYM1q0q1uMFYb7cSpLD2qzDSvK1NAvqFi8Xgivv0cFC6J+hWVw2teCYltyy9/m/14ryHg==
|
||||
|
||||
"@esbuild/openbsd-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.25.8.tgz#2f01615cf472b0e48c077045cfd96b5c149365cc"
|
||||
integrity sha512-1h8MUAwa0VhNCDp6Af0HToI2TJFAn1uqT9Al6DJVzdIBAd21m/G0Yfc77KDM3uF3T/YaOgQq3qTJHPbTOInaIQ==
|
||||
"@esbuild/openbsd-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.25.11.tgz#b42863959c8dcf9b01581522e40012d2c70045e2"
|
||||
integrity sha512-CN+7c++kkbrckTOz5hrehxWN7uIhFFlmS/hqziSFVWpAzpWrQoAG4chH+nN3Be+Kzv/uuo7zhX716x3Sn2Jduw==
|
||||
|
||||
"@esbuild/openharmony-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.8.tgz#a201f720cd2c3ebf9a6033fcc3feb069a54b509a"
|
||||
integrity sha512-r2nVa5SIK9tSWd0kJd9HCffnDHKchTGikb//9c7HX+r+wHYCpQrSgxhlY6KWV1nFo1l4KFbsMlHk+L6fekLsUg==
|
||||
"@esbuild/openharmony-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/openharmony-arm64/-/openharmony-arm64-0.25.11.tgz#b2e717141c8fdf6bddd4010f0912e6b39e1640f1"
|
||||
integrity sha512-rOREuNIQgaiR+9QuNkbkxubbp8MSO9rONmwP5nKncnWJ9v5jQ4JxFnLu4zDSRPf3x4u+2VN4pM4RdyIzDty/wQ==
|
||||
|
||||
"@esbuild/sunos-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.25.8.tgz#07046c977985a3334667f19e6ab3a01a80862afb"
|
||||
integrity sha512-zUlaP2S12YhQ2UzUfcCuMDHQFJyKABkAjvO5YSndMiIkMimPmxA+BYSBikWgsRpvyxuRnow4nS5NPnf9fpv41w==
|
||||
"@esbuild/sunos-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.25.11.tgz#9fbea1febe8778927804828883ec0f6dd80eb244"
|
||||
integrity sha512-nq2xdYaWxyg9DcIyXkZhcYulC6pQ2FuCgem3LI92IwMgIZ69KHeY8T4Y88pcwoLIjbed8n36CyKoYRDygNSGhA==
|
||||
|
||||
"@esbuild/win32-arm64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.25.8.tgz#4a5470caf0d16127c05d4833d4934213c69392d1"
|
||||
integrity sha512-YEGFFWESlPva8hGL+zvj2z/SaK+pH0SwOM0Nc/d+rVnW7GSTFlLBGzZkuSU9kFIGIo8q9X3ucpZhu8PDN5A2sQ==
|
||||
"@esbuild/win32-arm64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.25.11.tgz#501539cedb24468336073383989a7323005a8935"
|
||||
integrity sha512-3XxECOWJq1qMZ3MN8srCJ/QfoLpL+VaxD/WfNRm1O3B4+AZ/BnLVgFbUV3eiRYDMXetciH16dwPbbHqwe1uU0Q==
|
||||
|
||||
"@esbuild/win32-ia32@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.25.8.tgz#3de3e8470b7b328d99dbc3e9ec1eace207e5bbc4"
|
||||
integrity sha512-hiGgGC6KZ5LZz58OL/+qVVoZiuZlUYlYHNAmczOm7bs2oE1XriPFi5ZHHrS8ACpV5EjySrnoCKmcbQMN+ojnHg==
|
||||
"@esbuild/win32-ia32@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.25.11.tgz#8ac7229aa82cef8f16ffb58f1176a973a7a15343"
|
||||
integrity sha512-3ukss6gb9XZ8TlRyJlgLn17ecsK4NSQTmdIXRASVsiS2sQ6zPPZklNJT5GR5tE/MUarymmy8kCEf5xPCNCqVOA==
|
||||
|
||||
"@esbuild/win32-x64@0.25.8":
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.25.8.tgz#610d7ea539d2fcdbe39237b5cc175eb2c4451f9c"
|
||||
integrity sha512-cn3Yr7+OaaZq1c+2pe+8yxC8E144SReCQjN6/2ynubzYjvyqZjTXfQJpAcQpsdJq3My7XADANiYGHoFC69pLQw==
|
||||
"@esbuild/win32-x64@0.25.11":
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.25.11.tgz#5ecda6f3fe138b7e456f4e429edde33c823f392f"
|
||||
integrity sha512-D7Hpz6A2L4hzsRpPaCYkQnGOotdUpDzSGRIv9I+1ITdHROSFUWW95ZPZWQmGka1Fg7W3zFJowyn9WGwMJ0+KPA==
|
||||
|
||||
"@eslint-community/eslint-utils@^4.2.0":
|
||||
version "4.4.0"
|
||||
@@ -1642,37 +1642,37 @@ esbuild-sass-plugin@^3.3.1:
|
||||
safe-identifier "^0.4.2"
|
||||
sass "^1.71.1"
|
||||
|
||||
esbuild@^0.25.6:
|
||||
version "0.25.8"
|
||||
resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.25.8.tgz#482d42198b427c9c2f3a81b63d7663aecb1dda07"
|
||||
integrity sha512-vVC0USHGtMi8+R4Kz8rt6JhEWLxsv9Rnu/lGYbPR8u47B+DCBksq9JarW0zOO7bs37hyOK1l2/oqtbciutL5+Q==
|
||||
esbuild@^0.25.11:
|
||||
version "0.25.11"
|
||||
resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.25.11.tgz#0f31b82f335652580f75ef6897bba81962d9ae3d"
|
||||
integrity sha512-KohQwyzrKTQmhXDW1PjCv3Tyspn9n5GcY2RTDqeORIdIJY8yKIF7sTSopFmn/wpMPW4rdPXI0UE5LJLuq3bx0Q==
|
||||
optionalDependencies:
|
||||
"@esbuild/aix-ppc64" "0.25.8"
|
||||
"@esbuild/android-arm" "0.25.8"
|
||||
"@esbuild/android-arm64" "0.25.8"
|
||||
"@esbuild/android-x64" "0.25.8"
|
||||
"@esbuild/darwin-arm64" "0.25.8"
|
||||
"@esbuild/darwin-x64" "0.25.8"
|
||||
"@esbuild/freebsd-arm64" "0.25.8"
|
||||
"@esbuild/freebsd-x64" "0.25.8"
|
||||
"@esbuild/linux-arm" "0.25.8"
|
||||
"@esbuild/linux-arm64" "0.25.8"
|
||||
"@esbuild/linux-ia32" "0.25.8"
|
||||
"@esbuild/linux-loong64" "0.25.8"
|
||||
"@esbuild/linux-mips64el" "0.25.8"
|
||||
"@esbuild/linux-ppc64" "0.25.8"
|
||||
"@esbuild/linux-riscv64" "0.25.8"
|
||||
"@esbuild/linux-s390x" "0.25.8"
|
||||
"@esbuild/linux-x64" "0.25.8"
|
||||
"@esbuild/netbsd-arm64" "0.25.8"
|
||||
"@esbuild/netbsd-x64" "0.25.8"
|
||||
"@esbuild/openbsd-arm64" "0.25.8"
|
||||
"@esbuild/openbsd-x64" "0.25.8"
|
||||
"@esbuild/openharmony-arm64" "0.25.8"
|
||||
"@esbuild/sunos-x64" "0.25.8"
|
||||
"@esbuild/win32-arm64" "0.25.8"
|
||||
"@esbuild/win32-ia32" "0.25.8"
|
||||
"@esbuild/win32-x64" "0.25.8"
|
||||
"@esbuild/aix-ppc64" "0.25.11"
|
||||
"@esbuild/android-arm" "0.25.11"
|
||||
"@esbuild/android-arm64" "0.25.11"
|
||||
"@esbuild/android-x64" "0.25.11"
|
||||
"@esbuild/darwin-arm64" "0.25.11"
|
||||
"@esbuild/darwin-x64" "0.25.11"
|
||||
"@esbuild/freebsd-arm64" "0.25.11"
|
||||
"@esbuild/freebsd-x64" "0.25.11"
|
||||
"@esbuild/linux-arm" "0.25.11"
|
||||
"@esbuild/linux-arm64" "0.25.11"
|
||||
"@esbuild/linux-ia32" "0.25.11"
|
||||
"@esbuild/linux-loong64" "0.25.11"
|
||||
"@esbuild/linux-mips64el" "0.25.11"
|
||||
"@esbuild/linux-ppc64" "0.25.11"
|
||||
"@esbuild/linux-riscv64" "0.25.11"
|
||||
"@esbuild/linux-s390x" "0.25.11"
|
||||
"@esbuild/linux-x64" "0.25.11"
|
||||
"@esbuild/netbsd-arm64" "0.25.11"
|
||||
"@esbuild/netbsd-x64" "0.25.11"
|
||||
"@esbuild/openbsd-arm64" "0.25.11"
|
||||
"@esbuild/openbsd-x64" "0.25.11"
|
||||
"@esbuild/openharmony-arm64" "0.25.11"
|
||||
"@esbuild/sunos-x64" "0.25.11"
|
||||
"@esbuild/win32-arm64" "0.25.11"
|
||||
"@esbuild/win32-ia32" "0.25.11"
|
||||
"@esbuild/win32-x64" "0.25.11"
|
||||
|
||||
escape-string-regexp@^4.0.0:
|
||||
version "4.0.0"
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
version: "4.4.2"
|
||||
version: "4.4.4"
|
||||
edition: "Community"
|
||||
published: "2025-09-30"
|
||||
published: "2025-10-15"
|
||||
|
||||
@@ -18,7 +18,7 @@
|
||||
<li class="nav-item">
|
||||
<a class="nav-link{% if active_tab == 'preferences' %} active{% endif %}" href="{% url 'account:preferences' %}">{% trans "Preferences" %}</a>
|
||||
</li>
|
||||
{% if not request.user.ldap_username %}
|
||||
{% if request.user.has_usable_password %}
|
||||
<li class="nav-item">
|
||||
<a class="nav-link{% if active_tab == 'password' %} active{% endif %}" href="{% url 'account:change_password' %}">{% trans "Password" %}</a>
|
||||
</li>
|
||||
|
||||
@@ -26,7 +26,7 @@
|
||||
{# Initialize color mode #}
|
||||
<script
|
||||
type="text/javascript"
|
||||
src="{% static 'setmode.js' %}?v={{ settings.RELEASE.version }}"
|
||||
src="{% static_with_params 'setmode.js' v=settings.RELEASE.version %}"
|
||||
onerror="window.location='{% url 'media_failure' %}?filename=setmode.js'">
|
||||
</script>
|
||||
<script type="text/javascript">
|
||||
@@ -39,12 +39,12 @@
|
||||
{# Static resources #}
|
||||
<link
|
||||
rel="stylesheet"
|
||||
href="{% static 'netbox-external.css'%}?v={{ settings.RELEASE.version }}"
|
||||
href="{% static_with_params 'netbox-external.css' v=settings.RELEASE.version %}"
|
||||
onerror="window.location='{% url 'media_failure' %}?filename=netbox-external.css'"
|
||||
/>
|
||||
<link
|
||||
rel="stylesheet"
|
||||
href="{% static 'netbox.css'%}?v={{ settings.RELEASE.version }}"
|
||||
href="{% static_with_params 'netbox.css' v=settings.RELEASE.version %}"
|
||||
onerror="window.location='{% url 'media_failure' %}?filename=netbox.css'"
|
||||
/>
|
||||
<link rel="icon" type="image/png" href="{% static 'netbox.ico' %}" />
|
||||
@@ -53,7 +53,7 @@
|
||||
{# Javascript #}
|
||||
<script
|
||||
type="text/javascript"
|
||||
src="{% static 'netbox.js' %}?v={{ settings.RELEASE.version }}"
|
||||
src="{% static_with_params 'netbox.js' v=settings.RELEASE.version %}"
|
||||
onerror="window.location='{% url 'media_failure' %}?filename=netbox.js'">
|
||||
</script>
|
||||
{% django_htmx_script %}
|
||||
|
||||
@@ -40,7 +40,7 @@
|
||||
</tr>
|
||||
<tr>
|
||||
<th scope="row">{% trans "VLAN IDs" %}</th>
|
||||
<td>{{ object.vid_ranges_list }}</td>
|
||||
<td>{{ object.vid_ranges_items|join:", " }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th scope="row">Utilization</th>
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
from django import forms
|
||||
from django.contrib.contenttypes.models import ContentType
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
|
||||
@@ -25,7 +26,7 @@ class TenantGroupImportForm(NetBoxModelImportForm):
|
||||
queryset=TenantGroup.objects.all(),
|
||||
required=False,
|
||||
to_field_name='name',
|
||||
help_text=_('Parent group')
|
||||
help_text=_('Parent group'),
|
||||
)
|
||||
slug = SlugField()
|
||||
|
||||
@@ -41,7 +42,7 @@ class TenantImportForm(NetBoxModelImportForm):
|
||||
queryset=TenantGroup.objects.all(),
|
||||
required=False,
|
||||
to_field_name='name',
|
||||
help_text=_('Assigned group')
|
||||
help_text=_('Assigned group'),
|
||||
)
|
||||
|
||||
class Meta:
|
||||
@@ -59,7 +60,7 @@ class ContactGroupImportForm(NetBoxModelImportForm):
|
||||
queryset=ContactGroup.objects.all(),
|
||||
required=False,
|
||||
to_field_name='name',
|
||||
help_text=_('Parent group')
|
||||
help_text=_('Parent group'),
|
||||
)
|
||||
slug = SlugField()
|
||||
|
||||
@@ -81,7 +82,12 @@ class ContactImportForm(NetBoxModelImportForm):
|
||||
queryset=ContactGroup.objects.all(),
|
||||
required=False,
|
||||
to_field_name='name',
|
||||
help_text=_('Group names separated by commas, encased with double quotes (e.g. "Group 1,Group 2")')
|
||||
help_text=_('Group names separated by commas, encased with double quotes (e.g. "Group 1,Group 2")'),
|
||||
)
|
||||
link = forms.URLField(
|
||||
label=_('Link'),
|
||||
assume_scheme='https',
|
||||
required=False,
|
||||
)
|
||||
|
||||
class Meta:
|
||||
|
||||
@@ -100,6 +100,11 @@ class ContactForm(NetBoxModelForm):
|
||||
queryset=ContactGroup.objects.all(),
|
||||
required=False
|
||||
)
|
||||
link = forms.URLField(
|
||||
label=_('Link'),
|
||||
assume_scheme='https',
|
||||
required=False,
|
||||
)
|
||||
comments = CommentField()
|
||||
|
||||
fieldsets = (
|
||||
|
||||
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
@@ -8,7 +8,7 @@ msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: PACKAGE VERSION\n"
|
||||
"Report-Msgid-Bugs-To: \n"
|
||||
"POT-Creation-Date: 2025-10-07 05:02+0000\n"
|
||||
"POT-Creation-Date: 2025-10-15 05:03+0000\n"
|
||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||
@@ -167,7 +167,7 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:467 netbox/dcim/filtersets.py:1108
|
||||
#: netbox/dcim/filtersets.py:1430 netbox/dcim/filtersets.py:1528
|
||||
#: netbox/dcim/filtersets.py:2221 netbox/dcim/filtersets.py:2464
|
||||
#: netbox/dcim/filtersets.py:2522 netbox/ipam/filtersets.py:955
|
||||
#: netbox/dcim/filtersets.py:2522 netbox/ipam/filtersets.py:941
|
||||
#: netbox/virtualization/filtersets.py:139 netbox/vpn/filtersets.py:361
|
||||
msgid "Region (ID)"
|
||||
msgstr ""
|
||||
@@ -180,7 +180,7 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:1437 netbox/dcim/filtersets.py:1535
|
||||
#: netbox/dcim/filtersets.py:2228 netbox/dcim/filtersets.py:2471
|
||||
#: netbox/dcim/filtersets.py:2529 netbox/extras/filtersets.py:646
|
||||
#: netbox/ipam/filtersets.py:962 netbox/virtualization/filtersets.py:146
|
||||
#: netbox/ipam/filtersets.py:948 netbox/virtualization/filtersets.py:146
|
||||
#: netbox/vpn/filtersets.py:356
|
||||
msgid "Region (slug)"
|
||||
msgstr ""
|
||||
@@ -192,7 +192,7 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:1121 netbox/dcim/filtersets.py:1443
|
||||
#: netbox/dcim/filtersets.py:1541 netbox/dcim/filtersets.py:2234
|
||||
#: netbox/dcim/filtersets.py:2477 netbox/dcim/filtersets.py:2535
|
||||
#: netbox/ipam/filtersets.py:239 netbox/ipam/filtersets.py:968
|
||||
#: netbox/ipam/filtersets.py:239 netbox/ipam/filtersets.py:954
|
||||
#: netbox/virtualization/filtersets.py:152
|
||||
msgid "Site group (ID)"
|
||||
msgstr ""
|
||||
@@ -205,7 +205,7 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:1548 netbox/dcim/filtersets.py:2241
|
||||
#: netbox/dcim/filtersets.py:2484 netbox/dcim/filtersets.py:2542
|
||||
#: netbox/extras/filtersets.py:652 netbox/ipam/filtersets.py:246
|
||||
#: netbox/ipam/filtersets.py:975 netbox/virtualization/filtersets.py:159
|
||||
#: netbox/ipam/filtersets.py:961 netbox/virtualization/filtersets.py:159
|
||||
msgid "Site group (slug)"
|
||||
msgstr ""
|
||||
|
||||
@@ -234,7 +234,7 @@ msgstr ""
|
||||
#: netbox/ipam/forms/bulk_import.py:475 netbox/ipam/forms/filtersets.py:161
|
||||
#: netbox/ipam/forms/filtersets.py:236 netbox/ipam/forms/filtersets.py:457
|
||||
#: netbox/ipam/forms/filtersets.py:552 netbox/ipam/forms/model_forms.py:673
|
||||
#: netbox/ipam/tables/vlans.py:89 netbox/ipam/tables/vlans.py:199
|
||||
#: netbox/ipam/tables/vlans.py:90 netbox/ipam/tables/vlans.py:200
|
||||
#: netbox/templates/dcim/device.html:22
|
||||
#: netbox/templates/dcim/inc/cable_termination.html:8
|
||||
#: netbox/templates/dcim/inc/cable_termination.html:36
|
||||
@@ -262,7 +262,7 @@ msgstr ""
|
||||
#: netbox/circuits/filtersets.py:315 netbox/dcim/base_filtersets.py:53
|
||||
#: netbox/dcim/filtersets.py:245 netbox/dcim/filtersets.py:366
|
||||
#: netbox/dcim/filtersets.py:461 netbox/extras/filtersets.py:668
|
||||
#: netbox/ipam/filtersets.py:257 netbox/ipam/filtersets.py:985
|
||||
#: netbox/ipam/filtersets.py:257 netbox/ipam/filtersets.py:971
|
||||
#: netbox/virtualization/filtersets.py:169 netbox/vpn/filtersets.py:366
|
||||
msgid "Site (slug)"
|
||||
msgstr ""
|
||||
@@ -321,7 +321,7 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:1132 netbox/dcim/filtersets.py:1455
|
||||
#: netbox/dcim/filtersets.py:1553 netbox/dcim/filtersets.py:2246
|
||||
#: netbox/dcim/filtersets.py:2488 netbox/dcim/filtersets.py:2547
|
||||
#: netbox/ipam/filtersets.py:251 netbox/ipam/filtersets.py:979
|
||||
#: netbox/ipam/filtersets.py:251 netbox/ipam/filtersets.py:965
|
||||
#: netbox/virtualization/filtersets.py:163 netbox/vpn/filtersets.py:371
|
||||
msgid "Site (ID)"
|
||||
msgstr ""
|
||||
@@ -498,7 +498,7 @@ msgstr ""
|
||||
#: netbox/ipam/forms/bulk_edit.py:358 netbox/ipam/forms/bulk_edit.py:401
|
||||
#: netbox/ipam/forms/bulk_edit.py:417 netbox/ipam/forms/bulk_edit.py:511
|
||||
#: netbox/ipam/forms/bulk_edit.py:543 netbox/ipam/forms/bulk_edit.py:586
|
||||
#: netbox/ipam/tables/vlans.py:242 netbox/ipam/tables/vlans.py:269
|
||||
#: netbox/ipam/tables/vlans.py:243 netbox/ipam/tables/vlans.py:270
|
||||
#: netbox/templates/account/token.html:35
|
||||
#: netbox/templates/circuits/circuit.html:69
|
||||
#: netbox/templates/circuits/circuitgroup.html:32
|
||||
@@ -791,7 +791,7 @@ msgstr ""
|
||||
#: netbox/ipam/forms/model_forms.py:512 netbox/ipam/tables/ip.py:184
|
||||
#: netbox/ipam/tables/ip.py:265 netbox/ipam/tables/ip.py:321
|
||||
#: netbox/ipam/tables/ip.py:394 netbox/ipam/tables/ip.py:421
|
||||
#: netbox/ipam/tables/vlans.py:97 netbox/ipam/tables/vlans.py:210
|
||||
#: netbox/ipam/tables/vlans.py:98 netbox/ipam/tables/vlans.py:211
|
||||
#: netbox/templates/circuits/circuit.html:34
|
||||
#: netbox/templates/circuits/virtualcircuit.html:43
|
||||
#: netbox/templates/core/datasource.html:46 netbox/templates/core/job.html:21
|
||||
@@ -873,7 +873,7 @@ msgstr ""
|
||||
#: netbox/ipam/forms/filtersets.py:182 netbox/ipam/forms/filtersets.py:282
|
||||
#: netbox/ipam/forms/filtersets.py:333 netbox/ipam/forms/filtersets.py:441
|
||||
#: netbox/ipam/forms/filtersets.py:532 netbox/ipam/tables/ip.py:424
|
||||
#: netbox/ipam/tables/vlans.py:207 netbox/templates/circuits/circuit.html:48
|
||||
#: netbox/ipam/tables/vlans.py:208 netbox/templates/circuits/circuit.html:48
|
||||
#: netbox/templates/circuits/circuitgroup.html:36
|
||||
#: netbox/templates/circuits/virtualcircuit.html:47
|
||||
#: netbox/templates/dcim/cable.html:23 netbox/templates/dcim/device.html:85
|
||||
@@ -1111,7 +1111,7 @@ msgstr ""
|
||||
#: netbox/ipam/forms/model_forms.py:221 netbox/ipam/forms/model_forms.py:260
|
||||
#: netbox/ipam/forms/model_forms.py:688 netbox/ipam/tables/ip.py:210
|
||||
#: netbox/ipam/tables/ip.py:269 netbox/ipam/tables/ip.py:325
|
||||
#: netbox/ipam/tables/vlans.py:101 netbox/ipam/tables/vlans.py:213
|
||||
#: netbox/ipam/tables/vlans.py:102 netbox/ipam/tables/vlans.py:214
|
||||
#: netbox/templates/circuits/virtualcircuittermination.html:42
|
||||
#: netbox/templates/dcim/device.html:188
|
||||
#: netbox/templates/dcim/inc/panels/inventory_items.html:20
|
||||
@@ -1125,7 +1125,7 @@ msgstr ""
|
||||
#: netbox/templates/vpn/tunneltermination.html:17
|
||||
#: netbox/templates/wireless/inc/wirelesslink_interface.html:20
|
||||
#: netbox/tenancy/forms/bulk_edit.py:159 netbox/tenancy/forms/filtersets.py:107
|
||||
#: netbox/tenancy/forms/model_forms.py:139
|
||||
#: netbox/tenancy/forms/model_forms.py:144
|
||||
#: netbox/tenancy/tables/contacts.py:110
|
||||
#: netbox/virtualization/forms/bulk_edit.py:127
|
||||
#: netbox/virtualization/forms/bulk_import.py:112
|
||||
@@ -1224,7 +1224,7 @@ msgstr ""
|
||||
#: netbox/dcim/tables/connections.py:65 netbox/dcim/tables/devices.py:1169
|
||||
#: netbox/ipam/forms/bulk_import.py:324 netbox/ipam/forms/model_forms.py:291
|
||||
#: netbox/ipam/forms/model_forms.py:300 netbox/ipam/tables/fhrp.py:64
|
||||
#: netbox/ipam/tables/ip.py:330 netbox/ipam/tables/vlans.py:147
|
||||
#: netbox/ipam/tables/ip.py:330 netbox/ipam/tables/vlans.py:148
|
||||
#: netbox/templates/circuits/inc/circuit_termination_fields.html:52
|
||||
#: netbox/templates/circuits/virtualcircuittermination.html:53
|
||||
#: netbox/templates/circuits/virtualcircuittermination.html:60
|
||||
@@ -1238,7 +1238,7 @@ msgstr ""
|
||||
#: netbox/templates/wireless/inc/wirelesslink_interface.html:10
|
||||
#: netbox/templates/wireless/wirelesslink.html:10
|
||||
#: netbox/templates/wireless/wirelesslink.html:55
|
||||
#: netbox/virtualization/forms/model_forms.py:377
|
||||
#: netbox/virtualization/forms/model_forms.py:375
|
||||
#: netbox/vpn/forms/bulk_import.py:302 netbox/vpn/forms/model_forms.py:439
|
||||
#: netbox/vpn/forms/model_forms.py:448 netbox/wireless/forms/model_forms.py:118
|
||||
#: netbox/wireless/forms/model_forms.py:160
|
||||
@@ -1386,10 +1386,10 @@ msgstr ""
|
||||
#: netbox/circuits/tables/circuits.py:191 netbox/dcim/forms/bulk_edit.py:127
|
||||
#: netbox/dcim/forms/bulk_import.py:103 netbox/dcim/forms/model_forms.py:126
|
||||
#: netbox/dcim/tables/sites.py:103 netbox/extras/forms/filtersets.py:572
|
||||
#: netbox/ipam/filtersets.py:995 netbox/ipam/forms/bulk_edit.py:488
|
||||
#: netbox/ipam/filtersets.py:981 netbox/ipam/forms/bulk_edit.py:488
|
||||
#: netbox/ipam/forms/bulk_import.py:482 netbox/ipam/forms/model_forms.py:571
|
||||
#: netbox/ipam/tables/fhrp.py:67 netbox/ipam/tables/vlans.py:93
|
||||
#: netbox/ipam/tables/vlans.py:204
|
||||
#: netbox/ipam/tables/fhrp.py:67 netbox/ipam/tables/vlans.py:94
|
||||
#: netbox/ipam/tables/vlans.py:205
|
||||
#: netbox/templates/circuits/circuitgroupassignment.html:22
|
||||
#: netbox/templates/dcim/interface.html:341 netbox/templates/dcim/site.html:37
|
||||
#: netbox/templates/ipam/inc/panels/fhrp_groups.html:23
|
||||
@@ -1398,10 +1398,10 @@ msgstr ""
|
||||
#: netbox/templates/virtualization/cluster.html:29
|
||||
#: netbox/templates/vpn/tunnel.html:29
|
||||
#: netbox/templates/wireless/wirelesslan.html:18
|
||||
#: netbox/tenancy/forms/bulk_edit.py:44 netbox/tenancy/forms/bulk_import.py:40
|
||||
#: netbox/tenancy/forms/bulk_edit.py:44 netbox/tenancy/forms/bulk_import.py:41
|
||||
#: netbox/tenancy/forms/filtersets.py:48 netbox/tenancy/forms/filtersets.py:97
|
||||
#: netbox/tenancy/forms/model_forms.py:46
|
||||
#: netbox/tenancy/forms/model_forms.py:124 netbox/tenancy/tables/tenants.py:50
|
||||
#: netbox/tenancy/forms/model_forms.py:129 netbox/tenancy/tables/tenants.py:50
|
||||
#: netbox/users/filtersets.py:62 netbox/users/filtersets.py:185
|
||||
#: netbox/users/forms/filtersets.py:31 netbox/users/forms/filtersets.py:37
|
||||
#: netbox/users/forms/filtersets.py:79
|
||||
@@ -1473,7 +1473,7 @@ msgstr ""
|
||||
#: netbox/dcim/models/racks.py:294 netbox/dcim/models/racks.py:677
|
||||
#: netbox/dcim/models/sites.py:154 netbox/dcim/models/sites.py:270
|
||||
#: netbox/ipam/models/ip.py:243 netbox/ipam/models/ip.py:529
|
||||
#: netbox/ipam/models/ip.py:758 netbox/ipam/models/vlans.py:217
|
||||
#: netbox/ipam/models/ip.py:758 netbox/ipam/models/vlans.py:227
|
||||
#: netbox/virtualization/models/clusters.py:70
|
||||
#: netbox/virtualization/models/virtualmachines.py:79
|
||||
#: netbox/vpn/models/l2vpn.py:36 netbox/vpn/models/tunnels.py:38
|
||||
@@ -1576,7 +1576,7 @@ msgstr ""
|
||||
#: netbox/extras/models/models.py:408 netbox/extras/models/models.py:479
|
||||
#: netbox/extras/models/models.py:558 netbox/extras/models/models.py:684
|
||||
#: netbox/extras/models/notifications.py:131 netbox/extras/models/tags.py:33
|
||||
#: netbox/ipam/models/vlans.py:373 netbox/netbox/models/__init__.py:115
|
||||
#: netbox/ipam/models/vlans.py:383 netbox/netbox/models/__init__.py:115
|
||||
#: netbox/netbox/models/__init__.py:150 netbox/netbox/models/__init__.py:200
|
||||
#: netbox/users/models/permissions.py:24 netbox/users/models/tokens.py:57
|
||||
#: netbox/users/models/users.py:33
|
||||
@@ -1614,8 +1614,8 @@ msgstr ""
|
||||
#: netbox/extras/models/notifications.py:126 netbox/extras/models/scripts.py:30
|
||||
#: netbox/ipam/models/asns.py:18 netbox/ipam/models/fhrp.py:24
|
||||
#: netbox/ipam/models/services.py:51 netbox/ipam/models/services.py:80
|
||||
#: netbox/ipam/models/vlans.py:38 netbox/ipam/models/vlans.py:206
|
||||
#: netbox/ipam/models/vlans.py:352 netbox/ipam/models/vrfs.py:20
|
||||
#: netbox/ipam/models/vlans.py:38 netbox/ipam/models/vlans.py:216
|
||||
#: netbox/ipam/models/vlans.py:362 netbox/ipam/models/vrfs.py:20
|
||||
#: netbox/ipam/models/vrfs.py:75 netbox/netbox/models/__init__.py:142
|
||||
#: netbox/netbox/models/__init__.py:190 netbox/tenancy/models/contacts.py:56
|
||||
#: netbox/tenancy/models/tenants.py:19 netbox/tenancy/models/tenants.py:42
|
||||
@@ -1740,8 +1740,8 @@ msgstr ""
|
||||
#: netbox/ipam/forms/filtersets.py:496 netbox/ipam/tables/asn.py:16
|
||||
#: netbox/ipam/tables/ip.py:32 netbox/ipam/tables/ip.py:107
|
||||
#: netbox/ipam/tables/services.py:15 netbox/ipam/tables/services.py:40
|
||||
#: netbox/ipam/tables/vlans.py:33 netbox/ipam/tables/vlans.py:85
|
||||
#: netbox/ipam/tables/vlans.py:233 netbox/ipam/tables/vrfs.py:26
|
||||
#: netbox/ipam/tables/vlans.py:33 netbox/ipam/tables/vlans.py:86
|
||||
#: netbox/ipam/tables/vlans.py:234 netbox/ipam/tables/vrfs.py:26
|
||||
#: netbox/ipam/tables/vrfs.py:68 netbox/templates/circuits/circuitgroup.html:28
|
||||
#: netbox/templates/circuits/circuittype.html:22
|
||||
#: netbox/templates/circuits/provideraccount.html:28
|
||||
@@ -1883,7 +1883,7 @@ msgstr ""
|
||||
#: netbox/ipam/tables/fhrp.py:34 netbox/ipam/tables/ip.py:83
|
||||
#: netbox/ipam/tables/ip.py:227 netbox/ipam/tables/ip.py:286
|
||||
#: netbox/ipam/tables/ip.py:355 netbox/ipam/tables/services.py:24
|
||||
#: netbox/ipam/tables/services.py:54 netbox/ipam/tables/vlans.py:123
|
||||
#: netbox/ipam/tables/services.py:54 netbox/ipam/tables/vlans.py:124
|
||||
#: netbox/ipam/tables/vrfs.py:47 netbox/ipam/tables/vrfs.py:72
|
||||
#: netbox/templates/dcim/htmx/cable_edit.html:90
|
||||
#: netbox/templates/generic/bulk_edit.html:86
|
||||
@@ -1983,7 +1983,7 @@ msgstr ""
|
||||
#: netbox/dcim/tables/devices.py:989 netbox/dcim/tables/devices.py:1118
|
||||
#: netbox/dcim/tables/modules.py:87 netbox/extras/forms/filtersets.py:386
|
||||
#: netbox/ipam/forms/bulk_import.py:310 netbox/ipam/forms/filtersets.py:626
|
||||
#: netbox/ipam/forms/model_forms.py:334 netbox/ipam/tables/vlans.py:158
|
||||
#: netbox/ipam/forms/model_forms.py:334 netbox/ipam/tables/vlans.py:159
|
||||
#: netbox/templates/circuits/virtualcircuittermination.html:56
|
||||
#: netbox/templates/dcim/consoleport.html:20
|
||||
#: netbox/templates/dcim/consoleserverport.html:20
|
||||
@@ -2547,7 +2547,7 @@ msgstr ""
|
||||
msgid "Change logging is not supported for this object type ({type})."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:18 netbox/core/models/data.py:269
|
||||
#: netbox/core/models/config.py:21 netbox/core/models/data.py:269
|
||||
#: netbox/core/models/files.py:30 netbox/core/models/jobs.py:60
|
||||
#: netbox/extras/models/models.py:839 netbox/extras/models/notifications.py:39
|
||||
#: netbox/extras/models/notifications.py:195
|
||||
@@ -2555,31 +2555,31 @@ msgstr ""
|
||||
msgid "created"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:22
|
||||
#: netbox/core/models/config.py:25
|
||||
msgid "comment"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:29
|
||||
#: netbox/core/models/config.py:32
|
||||
msgid "configuration data"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:36
|
||||
#: netbox/core/models/config.py:39
|
||||
msgid "config revision"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:37
|
||||
#: netbox/core/models/config.py:40
|
||||
msgid "config revisions"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:41
|
||||
#: netbox/core/models/config.py:51
|
||||
msgid "Default configuration"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:43
|
||||
#: netbox/core/models/config.py:53
|
||||
msgid "Current configuration"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/config.py:44
|
||||
#: netbox/core/models/config.py:54
|
||||
#, python-brace-format
|
||||
msgid "Config revision #{id}"
|
||||
msgstr ""
|
||||
@@ -2792,11 +2792,11 @@ msgid ""
|
||||
"enqueue() cannot be called with values for both schedule_at and immediate."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/object_types.py:180
|
||||
#: netbox/core/models/object_types.py:188
|
||||
msgid "object type"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/core/models/object_types.py:181 netbox/extras/models/models.py:56
|
||||
#: netbox/core/models/object_types.py:189 netbox/extras/models/models.py:56
|
||||
msgid "object types"
|
||||
msgstr ""
|
||||
|
||||
@@ -3196,8 +3196,8 @@ msgstr ""
|
||||
#: netbox/templates/virtualization/vminterface.html:39
|
||||
#: netbox/templates/wireless/wirelesslangroup.html:37
|
||||
#: netbox/tenancy/forms/bulk_edit.py:27 netbox/tenancy/forms/bulk_edit.py:67
|
||||
#: netbox/tenancy/forms/bulk_import.py:24
|
||||
#: netbox/tenancy/forms/bulk_import.py:58
|
||||
#: netbox/tenancy/forms/bulk_import.py:25
|
||||
#: netbox/tenancy/forms/bulk_import.py:59
|
||||
#: netbox/tenancy/forms/model_forms.py:25
|
||||
#: netbox/tenancy/forms/model_forms.py:69 netbox/tenancy/tables/contacts.py:23
|
||||
#: netbox/tenancy/tables/tenants.py:20
|
||||
@@ -3425,8 +3425,8 @@ msgstr ""
|
||||
msgid "Access"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/choices.py:1518 netbox/ipam/tables/vlans.py:150
|
||||
#: netbox/ipam/tables/vlans.py:195
|
||||
#: netbox/dcim/choices.py:1518 netbox/ipam/tables/vlans.py:151
|
||||
#: netbox/ipam/tables/vlans.py:196
|
||||
#: netbox/templates/dcim/inc/interface_vlans_table.html:7
|
||||
msgid "Tagged"
|
||||
msgstr ""
|
||||
@@ -3571,7 +3571,7 @@ msgid "Parent site group (slug)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:167 netbox/extras/filtersets.py:422
|
||||
#: netbox/ipam/filtersets.py:837 netbox/ipam/filtersets.py:989
|
||||
#: netbox/ipam/filtersets.py:837 netbox/ipam/filtersets.py:975
|
||||
msgid "Group (ID)"
|
||||
msgstr ""
|
||||
|
||||
@@ -3618,14 +3618,14 @@ msgstr ""
|
||||
#: netbox/dcim/filtersets.py:414 netbox/dcim/filtersets.py:928
|
||||
#: netbox/dcim/filtersets.py:1077 netbox/dcim/filtersets.py:2164
|
||||
#: netbox/ipam/filtersets.py:376 netbox/ipam/filtersets.py:488
|
||||
#: netbox/ipam/filtersets.py:999 netbox/virtualization/filtersets.py:177
|
||||
#: netbox/ipam/filtersets.py:985 netbox/virtualization/filtersets.py:177
|
||||
msgid "Role (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:420 netbox/dcim/filtersets.py:934
|
||||
#: netbox/dcim/filtersets.py:1084 netbox/dcim/filtersets.py:2170
|
||||
#: netbox/extras/filtersets.py:695 netbox/ipam/filtersets.py:382
|
||||
#: netbox/ipam/filtersets.py:494 netbox/ipam/filtersets.py:1005
|
||||
#: netbox/ipam/filtersets.py:494 netbox/ipam/filtersets.py:991
|
||||
#: netbox/virtualization/filtersets.py:184
|
||||
msgid "Role (slug)"
|
||||
msgstr ""
|
||||
@@ -3871,14 +3871,14 @@ msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1487 netbox/dcim/filtersets.py:1585
|
||||
#: netbox/dcim/filtersets.py:1775 netbox/ipam/filtersets.py:606
|
||||
#: netbox/ipam/filtersets.py:847 netbox/ipam/filtersets.py:1177
|
||||
#: netbox/ipam/filtersets.py:847 netbox/ipam/filtersets.py:1163
|
||||
#: netbox/virtualization/filtersets.py:127 netbox/vpn/filtersets.py:382
|
||||
msgid "Device (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1493 netbox/dcim/filtersets.py:1591
|
||||
#: netbox/dcim/filtersets.py:1770 netbox/ipam/filtersets.py:601
|
||||
#: netbox/ipam/filtersets.py:842 netbox/ipam/filtersets.py:1172
|
||||
#: netbox/ipam/filtersets.py:842 netbox/ipam/filtersets.py:1158
|
||||
#: netbox/vpn/filtersets.py:377
|
||||
msgid "Device (name)"
|
||||
msgstr ""
|
||||
@@ -3918,13 +3918,13 @@ msgid "Cable (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1780 netbox/ipam/filtersets.py:611
|
||||
#: netbox/ipam/filtersets.py:852 netbox/ipam/filtersets.py:1182
|
||||
#: netbox/ipam/filtersets.py:852 netbox/ipam/filtersets.py:1168
|
||||
#: netbox/vpn/filtersets.py:388
|
||||
msgid "Virtual machine (name)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1785 netbox/ipam/filtersets.py:616
|
||||
#: netbox/ipam/filtersets.py:857 netbox/ipam/filtersets.py:1187
|
||||
#: netbox/ipam/filtersets.py:857 netbox/ipam/filtersets.py:1173
|
||||
#: netbox/virtualization/filtersets.py:253
|
||||
#: netbox/virtualization/filtersets.py:304 netbox/vpn/filtersets.py:393
|
||||
msgid "Virtual machine (ID)"
|
||||
@@ -3947,7 +3947,7 @@ msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1849 netbox/templates/dcim/interface.html:81
|
||||
#: netbox/templates/virtualization/vminterface.html:55
|
||||
#: netbox/virtualization/forms/model_forms.py:395
|
||||
#: netbox/virtualization/forms/model_forms.py:393
|
||||
msgid "802.1Q Mode"
|
||||
msgstr ""
|
||||
|
||||
@@ -3987,7 +3987,7 @@ msgstr ""
|
||||
#: netbox/virtualization/forms/bulk_edit.py:243
|
||||
#: netbox/virtualization/forms/bulk_import.py:177
|
||||
#: netbox/virtualization/forms/filtersets.py:236
|
||||
#: netbox/virtualization/forms/model_forms.py:368
|
||||
#: netbox/virtualization/forms/model_forms.py:366
|
||||
#: netbox/virtualization/models/virtualmachines.py:336
|
||||
#: netbox/virtualization/tables/virtualmachines.py:113
|
||||
msgid "VRF"
|
||||
@@ -3999,14 +3999,14 @@ msgstr ""
|
||||
msgid "VRF (RD)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1873 netbox/ipam/filtersets.py:1037
|
||||
#: netbox/dcim/filtersets.py:1873 netbox/ipam/filtersets.py:1023
|
||||
#: netbox/vpn/filtersets.py:345
|
||||
msgid "L2VPN (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1879 netbox/dcim/forms/filtersets.py:1531
|
||||
#: netbox/dcim/tables/devices.py:613 netbox/ipam/filtersets.py:1043
|
||||
#: netbox/ipam/forms/filtersets.py:592 netbox/ipam/tables/vlans.py:115
|
||||
#: netbox/dcim/tables/devices.py:613 netbox/ipam/filtersets.py:1029
|
||||
#: netbox/ipam/forms/filtersets.py:592 netbox/ipam/tables/vlans.py:116
|
||||
#: netbox/templates/dcim/interface.html:99 netbox/templates/ipam/vlan.html:82
|
||||
#: netbox/templates/vpn/l2vpntermination.html:12
|
||||
#: netbox/virtualization/forms/filtersets.py:241
|
||||
@@ -4016,7 +4016,7 @@ msgstr ""
|
||||
msgid "L2VPN"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1884 netbox/ipam/filtersets.py:1120
|
||||
#: netbox/dcim/filtersets.py:1884 netbox/ipam/filtersets.py:1106
|
||||
msgid "VLAN Translation Policy (ID)"
|
||||
msgstr ""
|
||||
|
||||
@@ -4027,7 +4027,7 @@ msgstr ""
|
||||
#: netbox/templates/ipam/vlantranslationpolicy.html:11
|
||||
#: netbox/virtualization/forms/bulk_edit.py:248
|
||||
#: netbox/virtualization/forms/filtersets.py:251
|
||||
#: netbox/virtualization/forms/model_forms.py:373
|
||||
#: netbox/virtualization/forms/model_forms.py:371
|
||||
msgid "VLAN Translation Policy"
|
||||
msgstr ""
|
||||
|
||||
@@ -4077,7 +4077,7 @@ msgstr ""
|
||||
|
||||
#: netbox/dcim/filtersets.py:1977 netbox/dcim/forms/model_forms.py:1549
|
||||
#: netbox/virtualization/filtersets.py:284
|
||||
#: netbox/virtualization/forms/model_forms.py:311
|
||||
#: netbox/virtualization/forms/model_forms.py:309
|
||||
msgid "Primary MAC address"
|
||||
msgstr ""
|
||||
|
||||
@@ -4724,21 +4724,21 @@ msgstr ""
|
||||
#: netbox/dcim/forms/bulk_edit.py:1567 netbox/dcim/forms/model_forms.py:1511
|
||||
#: netbox/ipam/forms/bulk_import.py:174 netbox/ipam/forms/filtersets.py:561
|
||||
#: netbox/ipam/models/vlans.py:93 netbox/virtualization/forms/bulk_edit.py:222
|
||||
#: netbox/virtualization/forms/model_forms.py:335
|
||||
#: netbox/virtualization/forms/model_forms.py:333
|
||||
msgid "VLAN group"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_edit.py:1576 netbox/dcim/forms/model_forms.py:1517
|
||||
#: netbox/dcim/tables/devices.py:622
|
||||
#: netbox/virtualization/forms/bulk_edit.py:230
|
||||
#: netbox/virtualization/forms/model_forms.py:340
|
||||
#: netbox/virtualization/forms/model_forms.py:338
|
||||
msgid "Untagged VLAN"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_edit.py:1585 netbox/dcim/forms/model_forms.py:1526
|
||||
#: netbox/dcim/tables/devices.py:628
|
||||
#: netbox/virtualization/forms/bulk_edit.py:238
|
||||
#: netbox/virtualization/forms/model_forms.py:349
|
||||
#: netbox/virtualization/forms/model_forms.py:347
|
||||
msgid "Tagged VLANs"
|
||||
msgstr ""
|
||||
|
||||
@@ -4751,7 +4751,7 @@ msgid "Remove tagged VLANs"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_edit.py:1608 netbox/dcim/forms/model_forms.py:1535
|
||||
#: netbox/virtualization/forms/model_forms.py:358
|
||||
#: netbox/virtualization/forms/model_forms.py:356
|
||||
msgid "Q-in-Q Service VLAN"
|
||||
msgstr ""
|
||||
|
||||
@@ -4774,13 +4774,13 @@ msgstr ""
|
||||
#: netbox/templates/ipam/prefix.html:91
|
||||
#: netbox/templates/virtualization/vminterface.html:76
|
||||
#: netbox/virtualization/forms/filtersets.py:205
|
||||
#: netbox/virtualization/forms/model_forms.py:378
|
||||
#: netbox/virtualization/forms/model_forms.py:376
|
||||
msgid "Addressing"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_edit.py:1638 netbox/dcim/forms/filtersets.py:750
|
||||
#: netbox/dcim/forms/model_forms.py:1570
|
||||
#: netbox/virtualization/forms/model_forms.py:379
|
||||
#: netbox/virtualization/forms/model_forms.py:377
|
||||
msgid "Operation"
|
||||
msgstr ""
|
||||
|
||||
@@ -4792,7 +4792,7 @@ msgstr ""
|
||||
#: netbox/dcim/forms/bulk_edit.py:1640 netbox/dcim/forms/model_forms.py:1571
|
||||
#: netbox/templates/dcim/interface.html:105
|
||||
#: netbox/virtualization/forms/bulk_edit.py:254
|
||||
#: netbox/virtualization/forms/model_forms.py:380
|
||||
#: netbox/virtualization/forms/model_forms.py:378
|
||||
msgid "Related Interfaces"
|
||||
msgstr ""
|
||||
|
||||
@@ -4800,7 +4800,7 @@ msgstr ""
|
||||
#: netbox/dcim/forms/model_forms.py:1575
|
||||
#: netbox/virtualization/forms/bulk_edit.py:257
|
||||
#: netbox/virtualization/forms/filtersets.py:206
|
||||
#: netbox/virtualization/forms/model_forms.py:383
|
||||
#: netbox/virtualization/forms/model_forms.py:381
|
||||
msgid "802.1Q Switching"
|
||||
msgstr ""
|
||||
|
||||
@@ -4828,7 +4828,7 @@ msgstr ""
|
||||
msgid "Assigned region"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_import.py:107 netbox/tenancy/forms/bulk_import.py:44
|
||||
#: netbox/dcim/forms/bulk_import.py:107 netbox/tenancy/forms/bulk_import.py:45
|
||||
#: netbox/wireless/forms/bulk_import.py:42
|
||||
msgid "Assigned group"
|
||||
msgstr ""
|
||||
@@ -4961,7 +4961,7 @@ msgid "Limit platform assignments to this manufacturer"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_import.py:549 netbox/dcim/forms/bulk_import.py:1674
|
||||
#: netbox/tenancy/forms/bulk_import.py:105
|
||||
#: netbox/tenancy/forms/bulk_import.py:111
|
||||
msgid "Assigned role"
|
||||
msgstr ""
|
||||
|
||||
@@ -5072,13 +5072,13 @@ msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_import.py:919 netbox/dcim/forms/model_forms.py:1473
|
||||
#: netbox/virtualization/forms/bulk_import.py:161
|
||||
#: netbox/virtualization/forms/model_forms.py:319
|
||||
#: netbox/virtualization/forms/model_forms.py:317
|
||||
msgid "Parent interface"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_import.py:926 netbox/dcim/forms/model_forms.py:1481
|
||||
#: netbox/virtualization/forms/bulk_import.py:168
|
||||
#: netbox/virtualization/forms/model_forms.py:327
|
||||
#: netbox/virtualization/forms/model_forms.py:325
|
||||
msgid "Bridged interface"
|
||||
msgstr ""
|
||||
|
||||
@@ -5212,7 +5212,7 @@ msgstr ""
|
||||
#: netbox/virtualization/forms/bulk_import.py:213
|
||||
#: netbox/virtualization/forms/filtersets.py:220
|
||||
#: netbox/virtualization/forms/filtersets.py:266
|
||||
#: netbox/virtualization/forms/model_forms.py:295
|
||||
#: netbox/virtualization/forms/model_forms.py:293
|
||||
#: netbox/vpn/forms/bulk_import.py:93 netbox/vpn/forms/bulk_import.py:295
|
||||
msgid "Virtual machine"
|
||||
msgstr ""
|
||||
@@ -5221,7 +5221,7 @@ msgstr ""
|
||||
msgid "Parent VM of assigned interface (if any)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/bulk_import.py:1293 netbox/ipam/filtersets.py:1048
|
||||
#: netbox/dcim/forms/bulk_import.py:1293 netbox/ipam/filtersets.py:1034
|
||||
#: netbox/ipam/forms/bulk_import.py:328
|
||||
msgid "Assigned interface"
|
||||
msgstr ""
|
||||
@@ -5427,8 +5427,8 @@ msgstr ""
|
||||
msgid "Parent region"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/filtersets.py:165 netbox/tenancy/forms/bulk_import.py:28
|
||||
#: netbox/tenancy/forms/bulk_import.py:62 netbox/tenancy/forms/filtersets.py:33
|
||||
#: netbox/dcim/forms/filtersets.py:165 netbox/tenancy/forms/bulk_import.py:29
|
||||
#: netbox/tenancy/forms/bulk_import.py:63 netbox/tenancy/forms/filtersets.py:33
|
||||
#: netbox/tenancy/forms/filtersets.py:62
|
||||
#: netbox/wireless/forms/bulk_import.py:27
|
||||
#: netbox/wireless/forms/filtersets.py:27
|
||||
@@ -5813,7 +5813,7 @@ msgid "VM Interface"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/dcim/forms/model_forms.py:1924 netbox/ipam/forms/filtersets.py:631
|
||||
#: netbox/ipam/forms/model_forms.py:335 netbox/ipam/tables/vlans.py:173
|
||||
#: netbox/ipam/forms/model_forms.py:335 netbox/ipam/tables/vlans.py:174
|
||||
#: netbox/templates/virtualization/virtualdisk.html:21
|
||||
#: netbox/templates/virtualization/virtualmachine.html:12
|
||||
#: netbox/templates/virtualization/vminterface.html:21
|
||||
@@ -6352,7 +6352,7 @@ msgstr ""
|
||||
#: netbox/dcim/models/device_components.py:604
|
||||
#: netbox/dcim/tables/devices.py:631 netbox/ipam/forms/bulk_edit.py:521
|
||||
#: netbox/ipam/forms/bulk_import.py:514 netbox/ipam/forms/filtersets.py:587
|
||||
#: netbox/ipam/forms/model_forms.py:694 netbox/ipam/tables/vlans.py:108
|
||||
#: netbox/ipam/forms/model_forms.py:694 netbox/ipam/tables/vlans.py:109
|
||||
#: netbox/templates/dcim/interface.html:86 netbox/templates/ipam/vlan.html:77
|
||||
#: netbox/templates/virtualization/vminterface.html:60
|
||||
msgid "Q-in-Q SVLAN"
|
||||
@@ -8045,7 +8045,8 @@ msgid "No"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/extras/choices.py:108 netbox/templates/tenancy/contact.html:67
|
||||
#: netbox/tenancy/forms/bulk_edit.py:130
|
||||
#: netbox/tenancy/forms/bulk_edit.py:130 netbox/tenancy/forms/bulk_import.py:88
|
||||
#: netbox/tenancy/forms/model_forms.py:104
|
||||
#: netbox/wireless/forms/model_forms.py:173
|
||||
msgid "Link"
|
||||
msgstr ""
|
||||
@@ -8477,7 +8478,7 @@ msgstr ""
|
||||
#: netbox/extras/forms/bulk_import.py:176
|
||||
#: netbox/extras/forms/bulk_import.py:200
|
||||
#: netbox/extras/forms/bulk_import.py:254
|
||||
#: netbox/tenancy/forms/bulk_import.py:95
|
||||
#: netbox/tenancy/forms/bulk_import.py:101
|
||||
msgid "One or more assigned object types"
|
||||
msgstr ""
|
||||
|
||||
@@ -10166,51 +10167,51 @@ msgstr ""
|
||||
msgid "NAT inside IP address (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1028
|
||||
#: netbox/ipam/filtersets.py:1014
|
||||
msgid "Q-in-Q SVLAN (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1032
|
||||
#: netbox/ipam/filtersets.py:1018
|
||||
msgid "Q-in-Q SVLAN number (1-4094)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1053
|
||||
#: netbox/ipam/filtersets.py:1039
|
||||
msgid "Assigned VM interface"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1126
|
||||
#: netbox/ipam/filtersets.py:1112
|
||||
msgid "VLAN Translation Policy (name)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1192
|
||||
#: netbox/ipam/filtersets.py:1178
|
||||
msgid "FHRP Group (name)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1197
|
||||
#: netbox/ipam/filtersets.py:1183
|
||||
msgid "FHRP Group (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1202
|
||||
#: netbox/ipam/filtersets.py:1188
|
||||
msgid "IP address (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1208 netbox/ipam/models/ip.py:816
|
||||
#: netbox/ipam/filtersets.py:1194 netbox/ipam/models/ip.py:816
|
||||
msgid "IP address"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1260
|
||||
#: netbox/ipam/filtersets.py:1246
|
||||
msgid "Primary IPv4 (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1266
|
||||
#: netbox/ipam/filtersets.py:1252
|
||||
msgid "Primary IPv4 (address)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1271
|
||||
#: netbox/ipam/filtersets.py:1257
|
||||
msgid "Primary IPv6 (ID)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/filtersets.py:1277
|
||||
#: netbox/ipam/filtersets.py:1263
|
||||
msgid "Primary IPv6 (address)"
|
||||
msgstr ""
|
||||
|
||||
@@ -10279,7 +10280,7 @@ msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/bulk_edit.py:218 netbox/ipam/forms/bulk_import.py:188
|
||||
#: netbox/ipam/forms/filtersets.py:271 netbox/ipam/forms/model_forms.py:218
|
||||
#: netbox/ipam/models/vlans.py:279 netbox/ipam/tables/ip.py:207
|
||||
#: netbox/ipam/models/vlans.py:289 netbox/ipam/tables/ip.py:207
|
||||
#: netbox/templates/ipam/prefix.html:56 netbox/templates/ipam/vlan.html:12
|
||||
#: netbox/templates/ipam/vlan/base.html:6
|
||||
#: netbox/templates/ipam/vlan_edit.html:14
|
||||
@@ -10367,8 +10368,8 @@ msgid "VLAN ID ranges"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/bulk_edit.py:516 netbox/ipam/forms/bulk_import.py:508
|
||||
#: netbox/ipam/forms/filtersets.py:579 netbox/ipam/models/vlans.py:239
|
||||
#: netbox/ipam/tables/vlans.py:105
|
||||
#: netbox/ipam/forms/filtersets.py:579 netbox/ipam/models/vlans.py:249
|
||||
#: netbox/ipam/tables/vlans.py:106
|
||||
msgid "Q-in-Q role"
|
||||
msgstr ""
|
||||
|
||||
@@ -10381,7 +10382,7 @@ msgid "Site & Group"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/bulk_edit.py:557 netbox/ipam/forms/bulk_import.py:538
|
||||
#: netbox/ipam/forms/model_forms.py:726 netbox/ipam/tables/vlans.py:258
|
||||
#: netbox/ipam/forms/model_forms.py:726 netbox/ipam/tables/vlans.py:259
|
||||
#: netbox/templates/ipam/vlantranslationrule.html:14
|
||||
#: netbox/vpn/forms/model_forms.py:322 netbox/vpn/forms/model_forms.py:359
|
||||
msgid "Policy"
|
||||
@@ -10479,7 +10480,7 @@ msgstr ""
|
||||
msgid "Service VLAN (for Q-in-Q/802.1ad customer VLANs)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/bulk_import.py:541 netbox/ipam/models/vlans.py:358
|
||||
#: netbox/ipam/forms/bulk_import.py:541 netbox/ipam/models/vlans.py:368
|
||||
msgid "VLAN translation policy"
|
||||
msgstr ""
|
||||
|
||||
@@ -10578,8 +10579,8 @@ msgstr ""
|
||||
msgid "DNS Name"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/filtersets.py:440 netbox/ipam/models/vlans.py:280
|
||||
#: netbox/ipam/tables/ip.py:123 netbox/ipam/tables/vlans.py:51
|
||||
#: netbox/ipam/forms/filtersets.py:440 netbox/ipam/models/vlans.py:290
|
||||
#: netbox/ipam/tables/ip.py:123 netbox/ipam/tables/vlans.py:52
|
||||
#: netbox/ipam/views.py:1086 netbox/netbox/navigation/menu.py:200
|
||||
#: netbox/netbox/navigation/menu.py:202
|
||||
msgid "VLANs"
|
||||
@@ -10589,11 +10590,11 @@ msgstr ""
|
||||
msgid "Contains VLAN ID"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/filtersets.py:516 netbox/ipam/models/vlans.py:378
|
||||
#: netbox/ipam/forms/filtersets.py:516 netbox/ipam/models/vlans.py:388
|
||||
msgid "Local VLAN ID"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/filtersets.py:521 netbox/ipam/models/vlans.py:386
|
||||
#: netbox/ipam/forms/filtersets.py:521 netbox/ipam/models/vlans.py:396
|
||||
msgid "Remote VLAN ID"
|
||||
msgstr ""
|
||||
|
||||
@@ -10601,7 +10602,7 @@ msgstr ""
|
||||
msgid "Q-in-Q/802.1ad"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/forms/filtersets.py:576 netbox/ipam/models/vlans.py:198
|
||||
#: netbox/ipam/forms/filtersets.py:576 netbox/ipam/models/vlans.py:208
|
||||
#: netbox/templates/ipam/vlan.html:31
|
||||
msgid "VLAN ID"
|
||||
msgstr ""
|
||||
@@ -11074,63 +11075,63 @@ msgstr ""
|
||||
msgid "Ranges cannot overlap."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:187
|
||||
#: netbox/ipam/models/vlans.py:197
|
||||
msgid "The specific site to which this VLAN is assigned (if any)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:195
|
||||
#: netbox/ipam/models/vlans.py:205
|
||||
msgid "VLAN group (optional)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:203 netbox/ipam/models/vlans.py:383
|
||||
#: netbox/ipam/models/vlans.py:391
|
||||
#: netbox/ipam/models/vlans.py:213 netbox/ipam/models/vlans.py:393
|
||||
#: netbox/ipam/models/vlans.py:401
|
||||
msgid "Numeric VLAN ID (1-4094)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:221
|
||||
#: netbox/ipam/models/vlans.py:231
|
||||
msgid "Operational status of this VLAN"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:229
|
||||
#: netbox/ipam/models/vlans.py:239
|
||||
msgid "The primary function of this VLAN"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:244
|
||||
#: netbox/ipam/models/vlans.py:254
|
||||
msgid "Customer/service VLAN designation (for Q-in-Q/IEEE 802.1ad)"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:293
|
||||
#: netbox/ipam/models/vlans.py:303
|
||||
#, python-brace-format
|
||||
msgid ""
|
||||
"VLAN is assigned to group {group} (scope: {scope}); cannot also assign to "
|
||||
"site {site}."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:300
|
||||
#: netbox/ipam/models/vlans.py:310
|
||||
#, python-brace-format
|
||||
msgid ""
|
||||
"The assigned site {site} is not a member of the assigned group {group} "
|
||||
"(scope: {scope})."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:309
|
||||
#: netbox/ipam/models/vlans.py:319
|
||||
#, python-brace-format
|
||||
msgid "VID must be in ranges {ranges} for VLANs in group {group}"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:316
|
||||
#: netbox/ipam/models/vlans.py:326
|
||||
msgid "Only Q-in-Q customer VLANs maybe assigned to a service VLAN."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:322
|
||||
#: netbox/ipam/models/vlans.py:332
|
||||
msgid "A Q-in-Q customer VLAN must be assigned to a service VLAN."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:359
|
||||
#: netbox/ipam/models/vlans.py:369
|
||||
msgid "VLAN translation policies"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/models/vlans.py:400
|
||||
#: netbox/ipam/models/vlans.py:410
|
||||
msgid "VLAN translation rule"
|
||||
msgstr ""
|
||||
|
||||
@@ -11189,14 +11190,14 @@ msgid "Added"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/ip.py:75 netbox/ipam/tables/ip.py:113
|
||||
#: netbox/ipam/tables/vlans.py:120 netbox/ipam/views.py:420
|
||||
#: netbox/ipam/tables/vlans.py:121 netbox/ipam/views.py:420
|
||||
#: netbox/netbox/navigation/menu.py:172 netbox/netbox/navigation/menu.py:174
|
||||
#: netbox/templates/ipam/vlan.html:100
|
||||
msgid "Prefixes"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/ip.py:78 netbox/ipam/tables/ip.py:222
|
||||
#: netbox/ipam/tables/ip.py:281 netbox/ipam/tables/vlans.py:55
|
||||
#: netbox/ipam/tables/ip.py:281 netbox/ipam/tables/vlans.py:56
|
||||
#: netbox/templates/dcim/device.html:266
|
||||
#: netbox/templates/ipam/aggregate.html:24
|
||||
#: netbox/templates/ipam/iprange.html:37 netbox/templates/ipam/prefix.html:102
|
||||
@@ -11255,26 +11256,26 @@ msgstr ""
|
||||
msgid "Assigned Object"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/vlans.py:45
|
||||
#: netbox/ipam/tables/vlans.py:46
|
||||
msgid "VID Ranges"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/vlans.py:82 netbox/ipam/tables/vlans.py:192
|
||||
#: netbox/ipam/tables/vlans.py:83 netbox/ipam/tables/vlans.py:193
|
||||
#: netbox/templates/dcim/inc/interface_vlans_table.html:4
|
||||
msgid "VID"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/vlans.py:239
|
||||
#: netbox/ipam/tables/vlans.py:240
|
||||
#: netbox/templates/ipam/vlantranslationpolicy.html:22
|
||||
msgid "Rules"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/vlans.py:262
|
||||
#: netbox/ipam/tables/vlans.py:263
|
||||
#: netbox/templates/ipam/vlantranslationrule.html:18
|
||||
msgid "Local VID"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/ipam/tables/vlans.py:266
|
||||
#: netbox/ipam/tables/vlans.py:267
|
||||
#: netbox/templates/ipam/vlantranslationrule.html:22
|
||||
msgid "Remote VID"
|
||||
msgstr ""
|
||||
@@ -12377,63 +12378,63 @@ msgstr ""
|
||||
msgid "Cannot delete stores from registry"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:800
|
||||
#: netbox/netbox/settings.py:812
|
||||
msgid "Czech"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:801
|
||||
#: netbox/netbox/settings.py:813
|
||||
msgid "Danish"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:802
|
||||
#: netbox/netbox/settings.py:814
|
||||
msgid "German"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:803
|
||||
#: netbox/netbox/settings.py:815
|
||||
msgid "English"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:804
|
||||
#: netbox/netbox/settings.py:816
|
||||
msgid "Spanish"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:805
|
||||
#: netbox/netbox/settings.py:817
|
||||
msgid "French"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:806
|
||||
#: netbox/netbox/settings.py:818
|
||||
msgid "Italian"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:807
|
||||
#: netbox/netbox/settings.py:819
|
||||
msgid "Japanese"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:808
|
||||
#: netbox/netbox/settings.py:820
|
||||
msgid "Dutch"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:809
|
||||
#: netbox/netbox/settings.py:821
|
||||
msgid "Polish"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:810
|
||||
#: netbox/netbox/settings.py:822
|
||||
msgid "Portuguese"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:811
|
||||
#: netbox/netbox/settings.py:823
|
||||
msgid "Russian"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:812
|
||||
#: netbox/netbox/settings.py:824
|
||||
msgid "Turkish"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:813
|
||||
#: netbox/netbox/settings.py:825
|
||||
msgid "Ukrainian"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/netbox/settings.py:814
|
||||
#: netbox/netbox/settings.py:826
|
||||
msgid "Chinese"
|
||||
msgstr ""
|
||||
|
||||
@@ -15137,8 +15138,8 @@ msgstr ""
|
||||
|
||||
#: netbox/templates/tenancy/contact.html:18 netbox/tenancy/filtersets.py:152
|
||||
#: netbox/tenancy/forms/bulk_edit.py:154 netbox/tenancy/forms/filtersets.py:102
|
||||
#: netbox/tenancy/forms/forms.py:57 netbox/tenancy/forms/model_forms.py:108
|
||||
#: netbox/tenancy/forms/model_forms.py:132
|
||||
#: netbox/tenancy/forms/forms.py:57 netbox/tenancy/forms/model_forms.py:113
|
||||
#: netbox/tenancy/forms/model_forms.py:137
|
||||
#: netbox/tenancy/tables/contacts.py:106
|
||||
msgid "Contact"
|
||||
msgstr ""
|
||||
@@ -15522,13 +15523,13 @@ msgstr ""
|
||||
msgid "Remove groups"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/tenancy/forms/bulk_import.py:84
|
||||
#: netbox/tenancy/forms/bulk_import.py:85
|
||||
msgid ""
|
||||
"Group names separated by commas, encased with double quotes (e.g. \"Group 1,"
|
||||
"Group 2\")"
|
||||
msgstr ""
|
||||
|
||||
#: netbox/tenancy/forms/bulk_import.py:100
|
||||
#: netbox/tenancy/forms/bulk_import.py:106
|
||||
msgid "Assigned contact"
|
||||
msgstr ""
|
||||
|
||||
@@ -16354,7 +16355,7 @@ msgstr ""
|
||||
msgid "Disk size is managed via the attachment of virtual disks."
|
||||
msgstr ""
|
||||
|
||||
#: netbox/virtualization/forms/model_forms.py:405
|
||||
#: netbox/virtualization/forms/model_forms.py:403
|
||||
#: netbox/virtualization/tables/virtualmachines.py:81
|
||||
msgid "Disk"
|
||||
msgstr ""
|
||||
|
||||
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
@@ -1,7 +1,8 @@
|
||||
import decimal
|
||||
from django.db.backends.postgresql.psycopg_any import NumericRange
|
||||
from itertools import count, groupby
|
||||
|
||||
from django.db.backends.postgresql.psycopg_any import NumericRange
|
||||
|
||||
__all__ = (
|
||||
'array_to_ranges',
|
||||
'array_to_string',
|
||||
@@ -10,6 +11,7 @@ __all__ = (
|
||||
'drange',
|
||||
'flatten_dict',
|
||||
'ranges_to_string',
|
||||
'ranges_to_string_list',
|
||||
'shallow_compare_dict',
|
||||
'string_to_ranges',
|
||||
)
|
||||
@@ -73,8 +75,10 @@ def shallow_compare_dict(source_dict, destination_dict, exclude=tuple()):
|
||||
def array_to_ranges(array):
|
||||
"""
|
||||
Convert an arbitrary array of integers to a list of consecutive values. Nonconsecutive values are returned as
|
||||
single-item tuples. For example:
|
||||
[0, 1, 2, 10, 14, 15, 16] => [(0, 2), (10,), (14, 16)]"
|
||||
single-item tuples.
|
||||
|
||||
Example:
|
||||
[0, 1, 2, 10, 14, 15, 16] => [(0, 2), (10,), (14, 16)]
|
||||
"""
|
||||
group = (
|
||||
list(x) for _, x in groupby(sorted(array), lambda x, c=count(): next(c) - x)
|
||||
@@ -87,7 +91,8 @@ def array_to_ranges(array):
|
||||
def array_to_string(array):
|
||||
"""
|
||||
Generate an efficient, human-friendly string from a set of integers. Intended for use with ArrayField.
|
||||
For example:
|
||||
|
||||
Example:
|
||||
[0, 1, 2, 10, 14, 15, 16] => "0-2, 10, 14-16"
|
||||
"""
|
||||
ret = []
|
||||
@@ -135,6 +140,29 @@ def check_ranges_overlap(ranges):
|
||||
return False
|
||||
|
||||
|
||||
def ranges_to_string_list(ranges):
|
||||
"""
|
||||
Convert numeric ranges to a list of display strings.
|
||||
|
||||
Each range is rendered as "lower-upper" or "lower" (for singletons).
|
||||
Bounds are normalized to inclusive values using ``lower_inc``/``upper_inc``.
|
||||
This underpins ``ranges_to_string()``, which joins the result with commas.
|
||||
|
||||
Example:
|
||||
[NumericRange(1, 6), NumericRange(8, 9), NumericRange(10, 13)] => ["1-5", "8", "10-12"]
|
||||
"""
|
||||
if not ranges:
|
||||
return []
|
||||
|
||||
output: list[str] = []
|
||||
for r in ranges:
|
||||
# Compute inclusive bounds regardless of how the DB range is stored.
|
||||
lower = r.lower if r.lower_inc else r.lower + 1
|
||||
upper = r.upper if r.upper_inc else r.upper - 1
|
||||
output.append(f"{lower}-{upper}" if lower != upper else str(lower))
|
||||
return output
|
||||
|
||||
|
||||
def ranges_to_string(ranges):
|
||||
"""
|
||||
Converts a list of ranges into a string representation.
|
||||
@@ -151,12 +179,7 @@ def ranges_to_string(ranges):
|
||||
"""
|
||||
if not ranges:
|
||||
return ''
|
||||
output = []
|
||||
for r in ranges:
|
||||
lower = r.lower if r.lower_inc else r.lower + 1
|
||||
upper = r.upper if r.upper_inc else r.upper - 1
|
||||
output.append(f"{lower}-{upper}" if lower != upper else str(lower))
|
||||
return ','.join(output)
|
||||
return ','.join(ranges_to_string_list(ranges))
|
||||
|
||||
|
||||
def string_to_ranges(value):
|
||||
|
||||
@@ -1,5 +1,9 @@
|
||||
import logging
|
||||
|
||||
from django import template
|
||||
from django.templatetags.static import static
|
||||
from django.utils.safestring import mark_safe
|
||||
from urllib.parse import urlparse, urlunparse, parse_qs, urlencode
|
||||
|
||||
from extras.choices import CustomFieldTypeChoices
|
||||
from utilities.querydict import dict_to_querydict
|
||||
@@ -11,6 +15,7 @@ __all__ = (
|
||||
'customfield_value',
|
||||
'htmx_table',
|
||||
'formaction',
|
||||
'static_with_params',
|
||||
'tag',
|
||||
)
|
||||
|
||||
@@ -127,3 +132,53 @@ def formaction(context):
|
||||
if context.get('htmx_navigation', False):
|
||||
return mark_safe('hx-push-url="true" hx-post')
|
||||
return 'formaction'
|
||||
|
||||
|
||||
@register.simple_tag
|
||||
def static_with_params(path, **params):
|
||||
"""
|
||||
Generate a static URL with properly appended query parameters.
|
||||
|
||||
The original Django static tag doesn't properly handle appending new parameters to URLs
|
||||
that already contain query parameters, which can result in malformed URLs with double
|
||||
question marks. This template tag handles the case where static files are served from
|
||||
AWS S3 or other CDNs that automatically append query parameters to URLs.
|
||||
|
||||
This implementation correctly appends new parameters to existing URLs and checks for
|
||||
parameter conflicts. A warning will be logged if any of the provided parameters
|
||||
conflict with existing parameters in the URL.
|
||||
|
||||
Args:
|
||||
path: The static file path (e.g., 'setmode.js')
|
||||
**params: Query parameters to append (e.g., v='4.3.1')
|
||||
|
||||
Returns:
|
||||
A properly formatted URL with query parameters.
|
||||
|
||||
Note:
|
||||
If any provided parameters conflict with existing URL parameters, a warning
|
||||
will be logged and the new parameter value will override the existing one.
|
||||
"""
|
||||
# Get the base static URL
|
||||
static_url = static(path)
|
||||
|
||||
# Parse the URL to extract existing query parameters
|
||||
parsed = urlparse(static_url)
|
||||
existing_params = parse_qs(parsed.query)
|
||||
|
||||
# Check for duplicate parameters and log warnings
|
||||
logger = logging.getLogger('netbox.utilities.templatetags.tags')
|
||||
for key, value in params.items():
|
||||
if key in existing_params:
|
||||
logger.warning(
|
||||
f"Parameter '{key}' already exists in static URL '{static_url}' "
|
||||
f"with value(s) {existing_params[key]}, overwriting with '{value}'"
|
||||
)
|
||||
existing_params[key] = [str(value)]
|
||||
|
||||
# Rebuild the query string
|
||||
new_query = urlencode(existing_params, doseq=True)
|
||||
|
||||
# Reconstruct the URL with the new query string
|
||||
new_parsed = parsed._replace(query=new_query)
|
||||
return urlunparse(new_parsed)
|
||||
|
||||
@@ -1,7 +1,11 @@
|
||||
from django.db.backends.postgresql.psycopg_any import NumericRange
|
||||
from django.test import TestCase
|
||||
|
||||
from utilities.data import check_ranges_overlap, ranges_to_string, string_to_ranges
|
||||
from utilities.data import (
|
||||
check_ranges_overlap,
|
||||
ranges_to_string,
|
||||
ranges_to_string_list,
|
||||
string_to_ranges,
|
||||
)
|
||||
|
||||
|
||||
class RangeFunctionsTestCase(TestCase):
|
||||
@@ -47,14 +51,26 @@ class RangeFunctionsTestCase(TestCase):
|
||||
])
|
||||
)
|
||||
|
||||
def test_ranges_to_string_list(self):
|
||||
self.assertEqual(
|
||||
ranges_to_string_list([
|
||||
NumericRange(10, 20), # 10-19
|
||||
NumericRange(30, 40), # 30-39
|
||||
NumericRange(50, 51), # 50-50
|
||||
NumericRange(100, 200), # 100-199
|
||||
]),
|
||||
['10-19', '30-39', '50', '100-199']
|
||||
)
|
||||
|
||||
def test_ranges_to_string(self):
|
||||
self.assertEqual(
|
||||
ranges_to_string([
|
||||
NumericRange(10, 20), # 10-19
|
||||
NumericRange(30, 40), # 30-39
|
||||
NumericRange(50, 51), # 50-50
|
||||
NumericRange(100, 200), # 100-199
|
||||
]),
|
||||
'10-19,30-39,100-199'
|
||||
'10-19,30-39,50,100-199'
|
||||
)
|
||||
|
||||
def test_string_to_ranges(self):
|
||||
|
||||
48
netbox/utilities/tests/test_templatetags.py
Normal file
48
netbox/utilities/tests/test_templatetags.py
Normal file
@@ -0,0 +1,48 @@
|
||||
from unittest.mock import patch
|
||||
|
||||
from django.test import TestCase, override_settings
|
||||
|
||||
from utilities.templatetags.builtins.tags import static_with_params
|
||||
|
||||
|
||||
class StaticWithParamsTest(TestCase):
|
||||
"""
|
||||
Test the static_with_params template tag functionality.
|
||||
"""
|
||||
|
||||
def test_static_with_params_basic(self):
|
||||
"""Test basic parameter appending to static URL."""
|
||||
result = static_with_params('test.js', v='1.0.0')
|
||||
self.assertIn('test.js', result)
|
||||
self.assertIn('v=1.0.0', result)
|
||||
|
||||
@override_settings(STATIC_URL='https://cdn.example.com/static/')
|
||||
def test_static_with_params_existing_query_params(self):
|
||||
"""Test appending parameters to URL that already has query parameters."""
|
||||
# Mock the static() function to return a URL with existing query parameters
|
||||
with patch('utilities.templatetags.builtins.tags.static') as mock_static:
|
||||
mock_static.return_value = 'https://cdn.example.com/static/test.js?existing=param'
|
||||
|
||||
result = static_with_params('test.js', v='1.0.0')
|
||||
|
||||
# Should contain both existing and new parameters
|
||||
self.assertIn('existing=param', result)
|
||||
self.assertIn('v=1.0.0', result)
|
||||
# Should not have double question marks
|
||||
self.assertEqual(result.count('?'), 1)
|
||||
|
||||
@override_settings(STATIC_URL='https://cdn.example.com/static/')
|
||||
def test_static_with_params_duplicate_parameter_warning(self):
|
||||
"""Test that a warning is logged when parameters conflict."""
|
||||
with patch('utilities.templatetags.builtins.tags.static') as mock_static:
|
||||
mock_static.return_value = 'https://cdn.example.com/static/test.js?v=old_version'
|
||||
|
||||
with self.assertLogs('netbox.utilities.templatetags.tags', level='WARNING') as cm:
|
||||
result = static_with_params('test.js', v='new_version')
|
||||
|
||||
# Check that warning was logged
|
||||
self.assertIn("Parameter 'v' already exists", cm.output[0])
|
||||
|
||||
# Check that new parameter value is used
|
||||
self.assertIn('v=new_version', result)
|
||||
self.assertNotIn('v=old_version', result)
|
||||
@@ -280,10 +280,8 @@ class VirtualMachineForm(TenancyForm, NetBoxModelForm):
|
||||
else:
|
||||
|
||||
# An object that doesn't exist yet can't have any IPs assigned to it
|
||||
self.fields['primary_ip4'].choices = []
|
||||
self.fields['primary_ip4'].widget.attrs['readonly'] = True
|
||||
self.fields['primary_ip6'].choices = []
|
||||
self.fields['primary_ip6'].widget.attrs['readonly'] = True
|
||||
self.fields.pop('primary_ip4')
|
||||
self.fields.pop('primary_ip6')
|
||||
|
||||
|
||||
#
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
|
||||
[project]
|
||||
name = "netbox"
|
||||
version = "4.4.2"
|
||||
version = "4.4.4"
|
||||
requires-python = ">=3.10"
|
||||
description = "The premier source of truth powering network automation."
|
||||
readme = "README.md"
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
colorama==0.4.6
|
||||
Django==5.2.6
|
||||
Django==5.2.7
|
||||
django-cors-headers==4.9.0
|
||||
django-debug-toolbar==5.2.0
|
||||
django-filter==25.1
|
||||
django-debug-toolbar==6.0.0
|
||||
django-filter==25.2
|
||||
django-graphiql-debug-toolbar==0.2.0
|
||||
django-htmx==1.26.0
|
||||
django-mptt==0.17.0
|
||||
@@ -17,27 +17,27 @@ django-taggit==6.1.0
|
||||
django-timezone-field==7.1
|
||||
djangorestframework==3.16.1
|
||||
drf-spectacular==0.28.0
|
||||
drf-spectacular-sidecar==2025.9.1
|
||||
drf-spectacular-sidecar==2025.10.1
|
||||
feedparser==6.0.12
|
||||
gunicorn==23.0.0
|
||||
Jinja2==3.1.6
|
||||
jsonschema==4.25.1
|
||||
Markdown==3.9
|
||||
mkdocs-material==9.6.20
|
||||
mkdocs-material==9.6.22
|
||||
mkdocstrings==0.30.1
|
||||
mkdocstrings-python==1.18.2
|
||||
netaddr==1.3.0
|
||||
nh3==0.3.0
|
||||
nh3==0.3.1
|
||||
Pillow==11.3.0
|
||||
psycopg[c,pool]==3.2.10
|
||||
PyYAML==6.0.3
|
||||
requests==2.32.5
|
||||
rq==2.6.0
|
||||
social-auth-app-django==5.5.1
|
||||
social-auth-core==4.7.0
|
||||
social-auth-app-django==5.6.0
|
||||
social-auth-core==4.8.1
|
||||
sorl-thumbnail==12.11.0
|
||||
strawberry-graphql==0.282.0
|
||||
strawberry-graphql-django==0.65.1
|
||||
strawberry-graphql==0.283.3
|
||||
strawberry-graphql-django==0.66.2
|
||||
svgwrite==1.4.3
|
||||
tablib==3.8.0
|
||||
tablib==3.9.0
|
||||
tzdata==2025.2
|
||||
|
||||
Reference in New Issue
Block a user