mirror of
https://github.com/netbox-community/netbox.git
synced 2026-03-28 20:32:07 +01:00
Initial implementation of UserConfig model
This commit is contained in:
@@ -2,6 +2,7 @@ import binascii
|
||||
import os
|
||||
|
||||
from django.contrib.auth.models import User
|
||||
from django.contrib.postgres.fields import JSONField
|
||||
from django.core.validators import MinLengthValidator
|
||||
from django.db import models
|
||||
from django.utils import timezone
|
||||
@@ -9,9 +10,111 @@ from django.utils import timezone
|
||||
|
||||
__all__ = (
|
||||
'Token',
|
||||
'UserConfig',
|
||||
)
|
||||
|
||||
|
||||
class UserConfig(models.Model):
|
||||
"""
|
||||
This model stores arbitrary user-specific preferences in a JSON data structure.
|
||||
"""
|
||||
user = models.OneToOneField(
|
||||
to=User,
|
||||
on_delete=models.CASCADE,
|
||||
related_name='config'
|
||||
)
|
||||
data = JSONField(
|
||||
default=dict
|
||||
)
|
||||
|
||||
class Meta:
|
||||
ordering = ['user']
|
||||
|
||||
def get(self, path):
|
||||
"""
|
||||
Retrieve a configuration parameter specified by its dotted path. Example:
|
||||
|
||||
userconfig.get('foo.bar.baz')
|
||||
|
||||
:param path: Dotted path to the configuration key. For example, 'foo.bar' returns self.data['foo']['bar'].
|
||||
"""
|
||||
d = self.data
|
||||
keys = path.split('.')
|
||||
|
||||
# Iterate down the hierarchy, returning None for any invalid keys
|
||||
for key in keys:
|
||||
if type(d) is dict:
|
||||
d = d.get(key)
|
||||
else:
|
||||
return None
|
||||
|
||||
return d
|
||||
|
||||
def set(self, path, value, commit=False):
|
||||
"""
|
||||
Define or overwrite a configuration parameter. Example:
|
||||
|
||||
userconfig.set('foo.bar.baz', 123)
|
||||
|
||||
Leaf nodes (those which are not dictionaries of other nodes) cannot be overwritten as dictionaries. Similarly,
|
||||
branch nodes (dictionaries) cannot be overwritten as single values. (A TypeError exception will be raised.) In
|
||||
both cases, the existing key must first be cleared. This safeguard is in place to help avoid inadvertently
|
||||
overwriting the wrong key.
|
||||
|
||||
:param path: Dotted path to the configuration key. For example, 'foo.bar' sets self.data['foo']['bar'].
|
||||
:param value: The value to be written. This can be any type supported by JSON.
|
||||
:param commit: If true, the UserConfig instance will be saved once the new value has been applied.
|
||||
"""
|
||||
d = self.data
|
||||
keys = path.split('.')
|
||||
|
||||
# Iterate through the hierarchy to find the key we're setting. Raise TypeError if we encounter any
|
||||
# interim leaf nodes (keys which do not contain dictionaries).
|
||||
for i, key in enumerate(keys[:-1]):
|
||||
if key in d and type(d[key]) is dict:
|
||||
d = d[key]
|
||||
elif key in d:
|
||||
err_path = '.'.join(path.split('.')[:i + 1])
|
||||
raise TypeError(f"Key '{err_path}' is a leaf node; cannot assign new keys")
|
||||
else:
|
||||
d = d.setdefault(key, {})
|
||||
|
||||
# Set a key based on the last item in the path. Raise TypeError if attempting to overwrite a non-leaf node.
|
||||
key = keys[-1]
|
||||
if key in d and type(d[key]) is dict:
|
||||
raise TypeError(f"Key '{path}' has child keys; cannot assign a value")
|
||||
else:
|
||||
d[key] = value
|
||||
|
||||
if commit:
|
||||
self.save()
|
||||
|
||||
def clear(self, path, commit=False):
|
||||
"""
|
||||
Delete a configuration parameter specified by its dotted path. The key and any child keys will be deleted.
|
||||
Example:
|
||||
|
||||
userconfig.clear('foo.bar.baz')
|
||||
|
||||
A KeyError is raised in the event any key along the path does not exist.
|
||||
|
||||
:param path: Dotted path to the configuration key. For example, 'foo.bar' deletes self.data['foo']['bar'].
|
||||
:param commit: If true, the UserConfig instance will be saved once the new value has been applied.
|
||||
"""
|
||||
d = self.data
|
||||
keys = path.split('.')
|
||||
|
||||
for key in keys[:-1]:
|
||||
if key in d and type(d[key]) is dict:
|
||||
d = d[key]
|
||||
|
||||
key = keys[-1]
|
||||
del(d[key])
|
||||
|
||||
if commit:
|
||||
self.save()
|
||||
|
||||
|
||||
class Token(models.Model):
|
||||
"""
|
||||
An API token used for user authentication. This extends the stock model to allow each user to have multiple tokens.
|
||||
|
||||
Reference in New Issue
Block a user