mirror of
https://github.com/netbox-community/netbox.git
synced 2024-05-10 07:54:54 +00:00
137 lines
4.8 KiB
Python
137 lines
4.8 KiB
Python
import re
|
|
|
|
from django import forms
|
|
from django.forms.models import fields_for_model
|
|
|
|
from utilities.querysets import RestrictedQuerySet
|
|
from .constants import *
|
|
|
|
__all__ = (
|
|
'add_blank_choice',
|
|
'expand_alphanumeric_pattern',
|
|
'expand_ipaddress_pattern',
|
|
'form_from_model',
|
|
'parse_alphanumeric_range',
|
|
'parse_numeric_range',
|
|
'restrict_form_fields',
|
|
)
|
|
|
|
|
|
def parse_numeric_range(string, base=10):
|
|
"""
|
|
Expand a numeric range (continuous or not) into a decimal or
|
|
hexadecimal list, as specified by the base parameter
|
|
'0-3,5' => [0, 1, 2, 3, 5]
|
|
'2,8-b,d,f' => [2, 8, 9, a, b, d, f]
|
|
"""
|
|
values = list()
|
|
for dash_range in string.split(','):
|
|
try:
|
|
begin, end = dash_range.split('-')
|
|
except ValueError:
|
|
begin, end = dash_range, dash_range
|
|
begin, end = int(begin.strip(), base=base), int(end.strip(), base=base) + 1
|
|
values.extend(range(begin, end))
|
|
return list(set(values))
|
|
|
|
|
|
def parse_alphanumeric_range(string):
|
|
"""
|
|
Expand an alphanumeric range (continuous or not) into a list.
|
|
'a-d,f' => [a, b, c, d, f]
|
|
'0-3,a-d' => [0, 1, 2, 3, a, b, c, d]
|
|
"""
|
|
values = []
|
|
for dash_range in string.split(','):
|
|
try:
|
|
begin, end = dash_range.split('-')
|
|
vals = begin + end
|
|
# Break out of loop if there's an invalid pattern to return an error
|
|
if (not (vals.isdigit() or vals.isalpha())) or (vals.isalpha() and not (vals.isupper() or vals.islower())):
|
|
return []
|
|
except ValueError:
|
|
begin, end = dash_range, dash_range
|
|
if begin.isdigit() and end.isdigit():
|
|
for n in list(range(int(begin), int(end) + 1)):
|
|
values.append(n)
|
|
else:
|
|
# Value-based
|
|
if begin == end:
|
|
values.append(begin)
|
|
# Range-based
|
|
else:
|
|
# Not a valid range (more than a single character)
|
|
if not len(begin) == len(end) == 1:
|
|
raise forms.ValidationError('Range "{}" is invalid.'.format(dash_range))
|
|
for n in list(range(ord(begin), ord(end) + 1)):
|
|
values.append(chr(n))
|
|
return values
|
|
|
|
|
|
def expand_alphanumeric_pattern(string):
|
|
"""
|
|
Expand an alphabetic pattern into a list of strings.
|
|
"""
|
|
lead, pattern, remnant = re.split(ALPHANUMERIC_EXPANSION_PATTERN, string, maxsplit=1)
|
|
parsed_range = parse_alphanumeric_range(pattern)
|
|
for i in parsed_range:
|
|
if re.search(ALPHANUMERIC_EXPANSION_PATTERN, remnant):
|
|
for string in expand_alphanumeric_pattern(remnant):
|
|
yield "{}{}{}".format(lead, i, string)
|
|
else:
|
|
yield "{}{}{}".format(lead, i, remnant)
|
|
|
|
|
|
def expand_ipaddress_pattern(string, family):
|
|
"""
|
|
Expand an IP address pattern into a list of strings. Examples:
|
|
'192.0.2.[1,2,100-250]/24' => ['192.0.2.1/24', '192.0.2.2/24', '192.0.2.100/24' ... '192.0.2.250/24']
|
|
'2001:db8:0:[0,fd-ff]::/64' => ['2001:db8:0:0::/64', '2001:db8:0:fd::/64', ... '2001:db8:0:ff::/64']
|
|
"""
|
|
if family not in [4, 6]:
|
|
raise Exception("Invalid IP address family: {}".format(family))
|
|
if family == 4:
|
|
regex = IP4_EXPANSION_PATTERN
|
|
base = 10
|
|
else:
|
|
regex = IP6_EXPANSION_PATTERN
|
|
base = 16
|
|
lead, pattern, remnant = re.split(regex, string, maxsplit=1)
|
|
parsed_range = parse_numeric_range(pattern, base)
|
|
for i in parsed_range:
|
|
if re.search(regex, remnant):
|
|
for string in expand_ipaddress_pattern(remnant, family):
|
|
yield ''.join([lead, format(i, 'x' if family == 6 else 'd'), string])
|
|
else:
|
|
yield ''.join([lead, format(i, 'x' if family == 6 else 'd'), remnant])
|
|
|
|
|
|
def add_blank_choice(choices):
|
|
"""
|
|
Add a blank choice to the beginning of a choices list.
|
|
"""
|
|
return ((None, '---------'),) + tuple(choices)
|
|
|
|
|
|
def form_from_model(model, fields):
|
|
"""
|
|
Return a Form class with the specified fields derived from a model. This is useful when we need a form to be used
|
|
for creating objects, but want to avoid the model's validation (e.g. for bulk create/edit functions). All fields
|
|
are marked as not required.
|
|
"""
|
|
form_fields = fields_for_model(model, fields=fields)
|
|
for field in form_fields.values():
|
|
field.required = False
|
|
|
|
return type('FormFromModel', (forms.Form,), form_fields)
|
|
|
|
|
|
def restrict_form_fields(form, user, action='view'):
|
|
"""
|
|
Restrict all form fields which reference a RestrictedQuerySet. This ensures that users see only permitted objects
|
|
as available choices.
|
|
"""
|
|
for field in form.fields.values():
|
|
if hasattr(field, 'queryset') and issubclass(field.queryset.__class__, RestrictedQuerySet):
|
|
field.queryset = field.queryset.restrict(user, action)
|