mirror of
https://github.com/netbox-community/netbox.git
synced 2026-01-16 16:52:17 -06:00
Closes #16137: Remove is_staff boolean from User model
This commit is contained in:
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Local Authentication
|
## Local Authentication
|
||||||
|
|
||||||
Local user accounts and groups can be created in NetBox under the "Authentication" section in the "Admin" menu. This section is available only to users with the "staff" permission enabled.
|
Local user accounts and groups can be created in NetBox under the "Authentication" section in the "Admin" menu.
|
||||||
|
|
||||||
At a minimum, each user account must have a username and password set. User accounts may also denote a first name, last name, and email address. [Permissions](../permissions.md) may also be assigned to individual users and/or groups as needed.
|
At a minimum, each user account must have a username and password set. User accounts may also denote a first name, last name, and email address. [Permissions](../permissions.md) may also be assigned to individual users and/or groups as needed.
|
||||||
|
|
||||||
|
|||||||
@@ -127,19 +127,3 @@ The list of groups that promote an remote User to Superuser on Login. If group i
|
|||||||
Default: `[]` (Empty list)
|
Default: `[]` (Empty list)
|
||||||
|
|
||||||
The list of users that get promoted to Superuser on Login. If user isn't present in list on next Login, the Role gets revoked. (Requires `REMOTE_AUTH_ENABLED` and `REMOTE_AUTH_GROUP_SYNC_ENABLED` )
|
The list of users that get promoted to Superuser on Login. If user isn't present in list on next Login, the Role gets revoked. (Requires `REMOTE_AUTH_ENABLED` and `REMOTE_AUTH_GROUP_SYNC_ENABLED` )
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## REMOTE_AUTH_STAFF_GROUPS
|
|
||||||
|
|
||||||
Default: `[]` (Empty list)
|
|
||||||
|
|
||||||
The list of groups that promote an remote User to Staff on Login. If group isn't present on next Login, the Role gets revoked. (Requires `REMOTE_AUTH_ENABLED` and `REMOTE_AUTH_GROUP_SYNC_ENABLED` )
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## REMOTE_AUTH_STAFF_USERS
|
|
||||||
|
|
||||||
Default: `[]` (Empty list)
|
|
||||||
|
|
||||||
The list of users that get promoted to Staff on Login. If user isn't present in list on next Login, the Role gets revoked. (Requires `REMOTE_AUTH_ENABLED` and `REMOTE_AUTH_GROUP_SYNC_ENABLED` )
|
|
||||||
|
|||||||
@@ -121,7 +121,6 @@ AUTH_LDAP_MIRROR_GROUPS = True
|
|||||||
# Define special user types using groups. Exercise great caution when assigning superuser status.
|
# Define special user types using groups. Exercise great caution when assigning superuser status.
|
||||||
AUTH_LDAP_USER_FLAGS_BY_GROUP = {
|
AUTH_LDAP_USER_FLAGS_BY_GROUP = {
|
||||||
"is_active": "cn=active,ou=groups,dc=example,dc=com",
|
"is_active": "cn=active,ou=groups,dc=example,dc=com",
|
||||||
"is_staff": "cn=staff,ou=groups,dc=example,dc=com",
|
|
||||||
"is_superuser": "cn=superuser,ou=groups,dc=example,dc=com"
|
"is_superuser": "cn=superuser,ou=groups,dc=example,dc=com"
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -134,7 +133,6 @@ AUTH_LDAP_CACHE_TIMEOUT = 3600
|
|||||||
```
|
```
|
||||||
|
|
||||||
* `is_active` - All users must be mapped to at least this group to enable authentication. Without this, users cannot log in.
|
* `is_active` - All users must be mapped to at least this group to enable authentication. Without this, users cannot log in.
|
||||||
* `is_staff` - Users mapped to this group are enabled for access to the administration tools; this is the equivalent of checking the "staff status" box on a manually created user. This doesn't grant any specific permissions.
|
|
||||||
* `is_superuser` - Users mapped to this group will be granted superuser status. Superusers are implicitly granted all permissions.
|
* `is_superuser` - Users mapped to this group will be granted superuser status. Superusers are implicitly granted all permissions.
|
||||||
|
|
||||||
!!! warning
|
!!! warning
|
||||||
@@ -248,7 +246,6 @@ AUTH_LDAP_MIRROR_GROUPS = True
|
|||||||
# Define special user types using groups. Exercise great caution when assigning superuser status.
|
# Define special user types using groups. Exercise great caution when assigning superuser status.
|
||||||
AUTH_LDAP_USER_FLAGS_BY_GROUP = {
|
AUTH_LDAP_USER_FLAGS_BY_GROUP = {
|
||||||
"is_active": "cn=active,ou=groups,dc=example,dc=com",
|
"is_active": "cn=active,ou=groups,dc=example,dc=com",
|
||||||
"is_staff": "cn=staff,ou=groups,dc=example,dc=com",
|
|
||||||
"is_superuser": "cn=superuser,ou=groups,dc=example,dc=com"
|
"is_superuser": "cn=superuser,ou=groups,dc=example,dc=com"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -65,14 +65,17 @@ item1 = PluginMenuItem(
|
|||||||
A `PluginMenuItem` has the following attributes:
|
A `PluginMenuItem` has the following attributes:
|
||||||
|
|
||||||
| Attribute | Required | Description |
|
| Attribute | Required | Description |
|
||||||
|-----------------|----------|----------------------------------------------------------------------------------------------------------|
|
|-----------------|----------|------------------------------------------------------|
|
||||||
| `link` | Yes | Name of the URL path to which this menu item links |
|
| `link` | Yes | Name of the URL path to which this menu item links |
|
||||||
| `link_text` | Yes | The text presented to the user |
|
| `link_text` | Yes | The text presented to the user |
|
||||||
| `permissions` | - | A list of permissions required to display this link |
|
| `permissions` | - | A list of permissions required to display this link |
|
||||||
| `auth_required` | - | Display only for authenticated users |
|
| `auth_required` | - | Display only for authenticated users |
|
||||||
| `staff_only` | - | Display only for users who have `is_staff` set to true (any specified permissions will also be required) |
|
| `staff_only` | - | Display only for superusers |
|
||||||
| `buttons` | - | An iterable of PluginMenuButton instances to include |
|
| `buttons` | - | An iterable of PluginMenuButton instances to include |
|
||||||
|
|
||||||
|
!!! note "Changed in NetBox v4.5"
|
||||||
|
In releases prior to NetBox v4.5, `staff_only` restricted display of a menu item to only users with `is_staff` set to True. In NetBox v4.5, the `is_staff` flag was removed from the user model. Menu items with `staff_only` set to True are now displayed only for superusers.
|
||||||
|
|
||||||
## Menu Buttons
|
## Menu Buttons
|
||||||
|
|
||||||
Each menu item can include a set of buttons. These can be handy for providing shortcuts related to the menu item. For instance, most items in NetBox's navigation menu include buttons to create and import new objects.
|
Each menu item can include a set of buttons. These can be handy for providing shortcuts related to the menu item. For instance, most items in NetBox's navigation menu include buttons to create and import new objects.
|
||||||
|
|||||||
@@ -134,10 +134,7 @@ class BackgroundTaskTestCase(TestCase):
|
|||||||
Create a user and token for API calls.
|
Create a user and token for API calls.
|
||||||
"""
|
"""
|
||||||
# Create the test user and assign permissions
|
# Create the test user and assign permissions
|
||||||
self.user = User.objects.create_user(username='testuser')
|
self.user = User.objects.create_user(username='testuser', is_active=True)
|
||||||
self.user.is_staff = True
|
|
||||||
self.user.is_active = True
|
|
||||||
self.user.save()
|
|
||||||
self.token = Token.objects.create(user=self.user)
|
self.token = Token.objects.create(user=self.user)
|
||||||
self.header = {'HTTP_AUTHORIZATION': f'Token {self.token.key}'}
|
self.header = {'HTTP_AUTHORIZATION': f'Token {self.token.key}'}
|
||||||
|
|
||||||
@@ -150,13 +147,13 @@ class BackgroundTaskTestCase(TestCase):
|
|||||||
url = reverse('core-api:rqqueue-list')
|
url = reverse('core-api:rqqueue-list')
|
||||||
|
|
||||||
# Attempt to load view without permission
|
# Attempt to load view without permission
|
||||||
self.user.is_staff = False
|
self.user.is_superuser = False
|
||||||
self.user.save()
|
self.user.save()
|
||||||
response = self.client.get(url, **self.header)
|
response = self.client.get(url, **self.header)
|
||||||
self.assertEqual(response.status_code, 403)
|
self.assertEqual(response.status_code, 403)
|
||||||
|
|
||||||
# Load view with permission
|
# Load view with permission
|
||||||
self.user.is_staff = True
|
self.user.is_superuser = True
|
||||||
self.user.save()
|
self.user.save()
|
||||||
response = self.client.get(url, **self.header)
|
response = self.client.get(url, **self.header)
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
|
|||||||
@@ -158,7 +158,7 @@ class BackgroundTaskTestCase(TestCase):
|
|||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super().setUp()
|
super().setUp()
|
||||||
self.user.is_staff = True
|
self.user.is_superuser = True
|
||||||
self.user.is_active = True
|
self.user.is_active = True
|
||||||
self.user.save()
|
self.user.save()
|
||||||
|
|
||||||
@@ -171,13 +171,13 @@ class BackgroundTaskTestCase(TestCase):
|
|||||||
url = reverse('core:background_queue_list')
|
url = reverse('core:background_queue_list')
|
||||||
|
|
||||||
# Attempt to load view without permission
|
# Attempt to load view without permission
|
||||||
self.user.is_staff = False
|
self.user.is_superuser = False
|
||||||
self.user.save()
|
self.user.save()
|
||||||
response = self.client.get(url)
|
response = self.client.get(url)
|
||||||
self.assertEqual(response.status_code, 403)
|
self.assertEqual(response.status_code, 403)
|
||||||
|
|
||||||
# Load view with permission
|
# Load view with permission
|
||||||
self.user.is_staff = True
|
self.user.is_superuser = True
|
||||||
self.user.save()
|
self.user.save()
|
||||||
response = self.client.get(url)
|
response = self.client.get(url)
|
||||||
self.assertEqual(response.status_code, 200)
|
self.assertEqual(response.status_code, 200)
|
||||||
@@ -356,7 +356,7 @@ class SystemTestCase(TestCase):
|
|||||||
def setUp(self):
|
def setUp(self):
|
||||||
super().setUp()
|
super().setUp()
|
||||||
|
|
||||||
self.user.is_staff = True
|
self.user.is_superuser = True
|
||||||
self.user.save()
|
self.user.save()
|
||||||
|
|
||||||
def test_system_view_default(self):
|
def test_system_view_default(self):
|
||||||
|
|||||||
@@ -366,7 +366,7 @@ class ConfigRevisionRestoreView(ContentTypePermissionRequiredMixin, View):
|
|||||||
class BaseRQView(UserPassesTestMixin, View):
|
class BaseRQView(UserPassesTestMixin, View):
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
return self.request.user.is_staff
|
return self.request.user.is_superuser
|
||||||
|
|
||||||
|
|
||||||
class BackgroundQueueListView(TableMixin, BaseRQView):
|
class BackgroundQueueListView(TableMixin, BaseRQView):
|
||||||
@@ -549,7 +549,7 @@ class WorkerView(BaseRQView):
|
|||||||
class SystemView(UserPassesTestMixin, View):
|
class SystemView(UserPassesTestMixin, View):
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
return self.request.user.is_staff
|
return self.request.user.is_superuser
|
||||||
|
|
||||||
def get(self, request):
|
def get(self, request):
|
||||||
|
|
||||||
@@ -632,7 +632,7 @@ class BasePluginView(UserPassesTestMixin, View):
|
|||||||
CACHE_KEY_CATALOG_ERROR = 'plugins-catalog-error'
|
CACHE_KEY_CATALOG_ERROR = 'plugins-catalog-error'
|
||||||
|
|
||||||
def test_func(self):
|
def test_func(self):
|
||||||
return self.request.user.is_staff
|
return self.request.user.is_superuser
|
||||||
|
|
||||||
def get_cached_plugins(self, request):
|
def get_cached_plugins(self, request):
|
||||||
catalog_plugins = {}
|
catalog_plugins = {}
|
||||||
|
|||||||
@@ -184,14 +184,18 @@ class RemoteUserBackend(_RemoteUserBackend):
|
|||||||
else:
|
else:
|
||||||
user.groups.clear()
|
user.groups.clear()
|
||||||
logger.debug(f"Stripping user {user} from Groups")
|
logger.debug(f"Stripping user {user} from Groups")
|
||||||
|
|
||||||
|
# Evaluate superuser status
|
||||||
user.is_superuser = self._is_superuser(user)
|
user.is_superuser = self._is_superuser(user)
|
||||||
logger.debug(f"User {user} is Superuser: {user.is_superuser}")
|
logger.debug(f"User {user} is Superuser: {user.is_superuser}")
|
||||||
logger.debug(
|
logger.debug(
|
||||||
f"User {user} should be Superuser: {self._is_superuser(user)}")
|
f"User {user} should be Superuser: {self._is_superuser(user)}")
|
||||||
|
|
||||||
user.is_staff = self._is_staff(user)
|
# Set is_staff attribute for compatibility with pre-v4.5
|
||||||
logger.debug(f"User {user} is Staff: {user.is_staff}")
|
user.is_staff = user.is_superuser
|
||||||
logger.debug(f"User {user} should be Staff: {self._is_staff(user)}")
|
if user.is_staff:
|
||||||
|
logger.debug(f"Marked user {user} as staff due to superuser status")
|
||||||
|
|
||||||
user.save()
|
user.save()
|
||||||
return user
|
return user
|
||||||
|
|
||||||
@@ -251,19 +255,8 @@ class RemoteUserBackend(_RemoteUserBackend):
|
|||||||
return bool(result)
|
return bool(result)
|
||||||
|
|
||||||
def _is_staff(self, user):
|
def _is_staff(self, user):
|
||||||
logger = logging.getLogger('netbox.auth.RemoteUserBackend')
|
# Retain for pre-v4.5 compatibility
|
||||||
staff_groups = settings.REMOTE_AUTH_STAFF_GROUPS
|
return user.is_superuser
|
||||||
logger.debug(f"Superuser Groups: {staff_groups}")
|
|
||||||
staff_users = settings.REMOTE_AUTH_STAFF_USERS
|
|
||||||
logger.debug(f"Staff Users :{staff_users}")
|
|
||||||
user_groups = set()
|
|
||||||
for g in user.groups.all():
|
|
||||||
user_groups.add(g.name)
|
|
||||||
logger.debug(f"User {user.username} is in Groups:{user_groups}")
|
|
||||||
result = user.username in staff_users or (
|
|
||||||
set(user_groups) & set(staff_groups))
|
|
||||||
logger.debug(f"User {user.username} in Staff Users :{result}")
|
|
||||||
return bool(result)
|
|
||||||
|
|
||||||
def configure_user(self, request, user):
|
def configure_user(self, request, user):
|
||||||
logger = logging.getLogger('netbox.auth.RemoteUserBackend')
|
logger = logging.getLogger('netbox.auth.RemoteUserBackend')
|
||||||
|
|||||||
@@ -49,7 +49,6 @@ class MenuItem:
|
|||||||
_url: Optional[str] = None
|
_url: Optional[str] = None
|
||||||
permissions: Optional[Sequence[str]] = ()
|
permissions: Optional[Sequence[str]] = ()
|
||||||
auth_required: Optional[bool] = False
|
auth_required: Optional[bool] = False
|
||||||
staff_only: Optional[bool] = False
|
|
||||||
buttons: Optional[Sequence[MenuItemButton]] = ()
|
buttons: Optional[Sequence[MenuItemButton]] = ()
|
||||||
|
|
||||||
def __post_init__(self):
|
def __post_init__(self):
|
||||||
|
|||||||
@@ -161,8 +161,6 @@ REMOTE_AUTH_SUPERUSERS = getattr(configuration, 'REMOTE_AUTH_SUPERUSERS', [])
|
|||||||
REMOTE_AUTH_USER_EMAIL = getattr(configuration, 'REMOTE_AUTH_USER_EMAIL', 'HTTP_REMOTE_USER_EMAIL')
|
REMOTE_AUTH_USER_EMAIL = getattr(configuration, 'REMOTE_AUTH_USER_EMAIL', 'HTTP_REMOTE_USER_EMAIL')
|
||||||
REMOTE_AUTH_USER_FIRST_NAME = getattr(configuration, 'REMOTE_AUTH_USER_FIRST_NAME', 'HTTP_REMOTE_USER_FIRST_NAME')
|
REMOTE_AUTH_USER_FIRST_NAME = getattr(configuration, 'REMOTE_AUTH_USER_FIRST_NAME', 'HTTP_REMOTE_USER_FIRST_NAME')
|
||||||
REMOTE_AUTH_USER_LAST_NAME = getattr(configuration, 'REMOTE_AUTH_USER_LAST_NAME', 'HTTP_REMOTE_USER_LAST_NAME')
|
REMOTE_AUTH_USER_LAST_NAME = getattr(configuration, 'REMOTE_AUTH_USER_LAST_NAME', 'HTTP_REMOTE_USER_LAST_NAME')
|
||||||
REMOTE_AUTH_STAFF_GROUPS = getattr(configuration, 'REMOTE_AUTH_STAFF_GROUPS', [])
|
|
||||||
REMOTE_AUTH_STAFF_USERS = getattr(configuration, 'REMOTE_AUTH_STAFF_USERS', [])
|
|
||||||
# Required by extras/migrations/0109_script_models.py
|
# Required by extras/migrations/0109_script_models.py
|
||||||
REPORTS_ROOT = getattr(configuration, 'REPORTS_ROOT', os.path.join(BASE_DIR, 'reports')).rstrip('/')
|
REPORTS_ROOT = getattr(configuration, 'REPORTS_ROOT', os.path.join(BASE_DIR, 'reports')).rstrip('/')
|
||||||
RQ_DEFAULT_TIMEOUT = getattr(configuration, 'RQ_DEFAULT_TIMEOUT', 300)
|
RQ_DEFAULT_TIMEOUT = getattr(configuration, 'RQ_DEFAULT_TIMEOUT', 300)
|
||||||
|
|||||||
@@ -47,9 +47,9 @@ class HomeView(ConditionalLoginRequiredMixin, View):
|
|||||||
))
|
))
|
||||||
dashboard = get_default_dashboard(config=DEFAULT_DASHBOARD).get_layout()
|
dashboard = get_default_dashboard(config=DEFAULT_DASHBOARD).get_layout()
|
||||||
|
|
||||||
# Check whether a new release is available. (Only for staff/superusers.)
|
# Check whether a new release is available. (Only for superusers.)
|
||||||
new_release = None
|
new_release = None
|
||||||
if request.user.is_staff or request.user.is_superuser:
|
if request.user.is_superuser:
|
||||||
latest_release = cache.get('latest_release')
|
latest_release = cache.get('latest_release')
|
||||||
if latest_release:
|
if latest_release:
|
||||||
release_version, release_url = latest_release
|
release_version, release_url = latest_release
|
||||||
|
|||||||
@@ -39,10 +39,6 @@
|
|||||||
<th scope="row">{% trans "Superuser" %}</th>
|
<th scope="row">{% trans "Superuser" %}</th>
|
||||||
<td>{% checkmark request.user.is_superuser %}</td>
|
<td>{% checkmark request.user.is_superuser %}</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
|
||||||
<th scope="row">{% trans "Staff" %}</th>
|
|
||||||
<td>{% checkmark request.user.is_staff %}</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -27,8 +27,6 @@
|
|||||||
<div class="mt-1 small text-secondary">
|
<div class="mt-1 small text-secondary">
|
||||||
{% if request.user.is_superuser %}
|
{% if request.user.is_superuser %}
|
||||||
{% trans "Admin" %}
|
{% trans "Admin" %}
|
||||||
{% elif request.user.is_staff %}
|
|
||||||
{% trans "Staff" %}
|
|
||||||
{% else %}
|
{% else %}
|
||||||
{% trans "User" %}
|
{% trans "User" %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|||||||
@@ -37,7 +37,7 @@
|
|||||||
path. Refer to <a href="{{ docs_url }}">the installation documentation</a> for further guidance.
|
path. Refer to <a href="{{ docs_url }}">the installation documentation</a> for further guidance.
|
||||||
{% endblocktrans %}
|
{% endblocktrans %}
|
||||||
<ul>
|
<ul>
|
||||||
{% if request.user.is_staff or request.user.is_superuser %}
|
{% if request.user.is_superuser %}
|
||||||
<li><code>STATIC_ROOT: <strong>{{ settings.STATIC_ROOT }}</strong></code></li>
|
<li><code>STATIC_ROOT: <strong>{{ settings.STATIC_ROOT }}</strong></code></li>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<li><code>STATIC_URL: <strong>{{ settings.STATIC_URL }}</strong></code></li>
|
<li><code>STATIC_URL: <strong>{{ settings.STATIC_URL }}</strong></code></li>
|
||||||
|
|||||||
@@ -35,10 +35,6 @@
|
|||||||
<th scope="row">{% trans "Active" %}</th>
|
<th scope="row">{% trans "Active" %}</th>
|
||||||
<td>{% checkmark object.is_active %}</td>
|
<td>{% checkmark object.is_active %}</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
|
||||||
<th scope="row">{% trans "Staff" %}</th>
|
|
||||||
<td>{% checkmark object.is_staff %}</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">{% trans "Superuser" %}</th>
|
<th scope="row">{% trans "Superuser" %}</th>
|
||||||
<td>{% checkmark object.is_superuser %}</td>
|
<td>{% checkmark object.is_superuser %}</td>
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ class UserSerializer(ValidatedModelSerializer):
|
|||||||
model = User
|
model = User
|
||||||
fields = (
|
fields = (
|
||||||
'id', 'url', 'display_url', 'display', 'username', 'password', 'first_name', 'last_name', 'email',
|
'id', 'url', 'display_url', 'display', 'username', 'password', 'first_name', 'last_name', 'email',
|
||||||
'is_staff', 'is_active', 'date_joined', 'last_login', 'groups', 'permissions',
|
'is_active', 'date_joined', 'last_login', 'groups', 'permissions',
|
||||||
)
|
)
|
||||||
brief_fields = ('id', 'url', 'display', 'username')
|
brief_fields = ('id', 'url', 'display', 'username')
|
||||||
extra_kwargs = {
|
extra_kwargs = {
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ class UserFilterSet(BaseFilterSet):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = User
|
model = User
|
||||||
fields = (
|
fields = (
|
||||||
'id', 'username', 'first_name', 'last_name', 'email', 'date_joined', 'last_login', 'is_staff', 'is_active',
|
'id', 'username', 'first_name', 'last_name', 'email', 'date_joined', 'last_login', 'is_active',
|
||||||
'is_superuser',
|
'is_superuser',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -37,11 +37,6 @@ class UserBulkEditForm(BulkEditForm):
|
|||||||
widget=BulkEditNullBooleanSelect,
|
widget=BulkEditNullBooleanSelect,
|
||||||
label=_('Active')
|
label=_('Active')
|
||||||
)
|
)
|
||||||
is_staff = forms.NullBooleanField(
|
|
||||||
required=False,
|
|
||||||
widget=BulkEditNullBooleanSelect,
|
|
||||||
label=_('Staff status')
|
|
||||||
)
|
|
||||||
is_superuser = forms.NullBooleanField(
|
is_superuser = forms.NullBooleanField(
|
||||||
required=False,
|
required=False,
|
||||||
widget=BulkEditNullBooleanSelect,
|
widget=BulkEditNullBooleanSelect,
|
||||||
@@ -50,7 +45,7 @@ class UserBulkEditForm(BulkEditForm):
|
|||||||
|
|
||||||
model = User
|
model = User
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
FieldSet('first_name', 'last_name', 'is_active', 'is_staff', 'is_superuser'),
|
FieldSet('first_name', 'last_name', 'is_active', 'is_superuser'),
|
||||||
)
|
)
|
||||||
nullable_fields = ('first_name', 'last_name')
|
nullable_fields = ('first_name', 'last_name')
|
||||||
|
|
||||||
|
|||||||
@@ -23,8 +23,7 @@ class UserImportForm(CSVModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = User
|
model = User
|
||||||
fields = (
|
fields = (
|
||||||
'username', 'first_name', 'last_name', 'email', 'password', 'is_staff',
|
'username', 'first_name', 'last_name', 'email', 'password', 'is_active', 'is_superuser'
|
||||||
'is_active', 'is_superuser'
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ class UserFilterForm(NetBoxModelFilterSetForm):
|
|||||||
fieldsets = (
|
fieldsets = (
|
||||||
FieldSet('q', 'filter_id',),
|
FieldSet('q', 'filter_id',),
|
||||||
FieldSet('group_id', name=_('Group')),
|
FieldSet('group_id', name=_('Group')),
|
||||||
FieldSet('is_active', 'is_staff', 'is_superuser', name=_('Status')),
|
FieldSet('is_active', 'is_superuser', name=_('Status')),
|
||||||
)
|
)
|
||||||
group_id = DynamicModelMultipleChoiceField(
|
group_id = DynamicModelMultipleChoiceField(
|
||||||
queryset=Group.objects.all(),
|
queryset=Group.objects.all(),
|
||||||
@@ -43,13 +43,6 @@ class UserFilterForm(NetBoxModelFilterSetForm):
|
|||||||
),
|
),
|
||||||
label=_('Is Active'),
|
label=_('Is Active'),
|
||||||
)
|
)
|
||||||
is_staff = forms.NullBooleanField(
|
|
||||||
required=False,
|
|
||||||
widget=forms.Select(
|
|
||||||
choices=BOOLEAN_WITH_BLANK_CHOICES
|
|
||||||
),
|
|
||||||
label=_('Is Staff'),
|
|
||||||
)
|
|
||||||
is_superuser = forms.NullBooleanField(
|
is_superuser = forms.NullBooleanField(
|
||||||
required=False,
|
required=False,
|
||||||
widget=forms.Select(
|
widget=forms.Select(
|
||||||
|
|||||||
@@ -192,7 +192,7 @@ class UserForm(forms.ModelForm):
|
|||||||
fieldsets = (
|
fieldsets = (
|
||||||
FieldSet('username', 'password', 'confirm_password', 'first_name', 'last_name', 'email', name=_('User')),
|
FieldSet('username', 'password', 'confirm_password', 'first_name', 'last_name', 'email', name=_('User')),
|
||||||
FieldSet('groups', name=_('Groups')),
|
FieldSet('groups', name=_('Groups')),
|
||||||
FieldSet('is_active', 'is_staff', 'is_superuser', name=_('Status')),
|
FieldSet('is_active', 'is_superuser', name=_('Status')),
|
||||||
FieldSet('object_permissions', name=_('Permissions')),
|
FieldSet('object_permissions', name=_('Permissions')),
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -200,7 +200,7 @@ class UserForm(forms.ModelForm):
|
|||||||
model = User
|
model = User
|
||||||
fields = [
|
fields = [
|
||||||
'username', 'first_name', 'last_name', 'email', 'groups', 'object_permissions',
|
'username', 'first_name', 'last_name', 'email', 'groups', 'object_permissions',
|
||||||
'is_active', 'is_staff', 'is_superuser',
|
'is_active', 'is_superuser',
|
||||||
]
|
]
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ class UserFilter(BaseObjectTypeFilterMixin):
|
|||||||
last_name: FilterLookup[str] | None = strawberry_django.filter_field()
|
last_name: FilterLookup[str] | None = strawberry_django.filter_field()
|
||||||
email: FilterLookup[str] | None = strawberry_django.filter_field()
|
email: FilterLookup[str] | None = strawberry_django.filter_field()
|
||||||
is_superuser: FilterLookup[bool] | None = strawberry_django.filter_field()
|
is_superuser: FilterLookup[bool] | None = strawberry_django.filter_field()
|
||||||
is_staff: FilterLookup[bool] | None = strawberry_django.filter_field()
|
|
||||||
is_active: FilterLookup[bool] | None = strawberry_django.filter_field()
|
is_active: FilterLookup[bool] | None = strawberry_django.filter_field()
|
||||||
date_joined: DatetimeFilterLookup[datetime] | None = strawberry_django.filter_field()
|
date_joined: DatetimeFilterLookup[datetime] | None = strawberry_django.filter_field()
|
||||||
last_login: DatetimeFilterLookup[datetime] | None = strawberry_django.filter_field()
|
last_login: DatetimeFilterLookup[datetime] | None = strawberry_django.filter_field()
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ class GroupType(BaseObjectType):
|
|||||||
@strawberry_django.type(
|
@strawberry_django.type(
|
||||||
User,
|
User,
|
||||||
fields=[
|
fields=[
|
||||||
'id', 'username', 'first_name', 'last_name', 'email', 'is_staff', 'is_active', 'date_joined', 'groups',
|
'id', 'username', 'first_name', 'last_name', 'email', 'is_active', 'date_joined', 'groups',
|
||||||
],
|
],
|
||||||
filters=UserFilter,
|
filters=UserFilter,
|
||||||
pagination=True
|
pagination=True
|
||||||
|
|||||||
15
netbox/users/migrations/0012_user_remove_is_staff.py
Normal file
15
netbox/users/migrations/0012_user_remove_is_staff.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('users', '0011_concrete_objecttype'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='user',
|
||||||
|
name='is_staff',
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -1,12 +1,16 @@
|
|||||||
|
from django.contrib.auth.base_user import AbstractBaseUser
|
||||||
from django.contrib.auth.models import (
|
from django.contrib.auth.models import (
|
||||||
AbstractUser,
|
|
||||||
GroupManager as DjangoGroupManager,
|
GroupManager as DjangoGroupManager,
|
||||||
Permission,
|
Permission,
|
||||||
|
PermissionsMixin,
|
||||||
UserManager as DjangoUserManager
|
UserManager as DjangoUserManager
|
||||||
)
|
)
|
||||||
|
from django.contrib.auth.validators import UnicodeUsernameValidator
|
||||||
from django.core.exceptions import ValidationError
|
from django.core.exceptions import ValidationError
|
||||||
|
from django.core.mail import send_mail
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from utilities.querysets import RestrictedQuerySet
|
from utilities.querysets import RestrictedQuerySet
|
||||||
@@ -71,7 +75,42 @@ class UserManager(DjangoUserManager.from_queryset(RestrictedQuerySet)):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
class User(AbstractUser):
|
class User(AbstractBaseUser, PermissionsMixin):
|
||||||
|
username = models.CharField(
|
||||||
|
_("username"),
|
||||||
|
max_length=150,
|
||||||
|
unique=True,
|
||||||
|
help_text=_("Required. 150 characters or fewer. Letters, digits and @/./+/-/_ only."),
|
||||||
|
validators=[UnicodeUsernameValidator()],
|
||||||
|
error_messages={
|
||||||
|
"unique": _("A user with that username already exists."),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
first_name = models.CharField(
|
||||||
|
_("first name"),
|
||||||
|
max_length=150,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
last_name = models.CharField(
|
||||||
|
_("last name"),
|
||||||
|
max_length=150,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
email = models.EmailField(
|
||||||
|
_("email address"),
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
is_active = models.BooleanField(
|
||||||
|
_("active"),
|
||||||
|
default=True,
|
||||||
|
help_text=_(
|
||||||
|
"Designates whether this user should be treated as active. Unselect this instead of deleting accounts."
|
||||||
|
),
|
||||||
|
)
|
||||||
|
date_joined = models.DateTimeField(
|
||||||
|
_("date joined"),
|
||||||
|
default=timezone.now,
|
||||||
|
)
|
||||||
groups = models.ManyToManyField(
|
groups = models.ManyToManyField(
|
||||||
to='users.Group',
|
to='users.Group',
|
||||||
verbose_name=_('groups'),
|
verbose_name=_('groups'),
|
||||||
@@ -87,6 +126,11 @@ class User(AbstractUser):
|
|||||||
|
|
||||||
objects = UserManager()
|
objects = UserManager()
|
||||||
|
|
||||||
|
# Ensure compatibility with Django's stock User model
|
||||||
|
EMAIL_FIELD = "email"
|
||||||
|
USERNAME_FIELD = "username"
|
||||||
|
REQUIRED_FIELDS = ["email"]
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('username',)
|
ordering = ('username',)
|
||||||
verbose_name = _('user')
|
verbose_name = _('user')
|
||||||
@@ -98,7 +142,25 @@ class User(AbstractUser):
|
|||||||
def clean(self):
|
def clean(self):
|
||||||
super().clean()
|
super().clean()
|
||||||
|
|
||||||
|
# Normalize email address
|
||||||
|
self.email = self.__class__.objects.normalize_email(self.email)
|
||||||
|
|
||||||
# Check for any existing Users with names that differ only in case
|
# Check for any existing Users with names that differ only in case
|
||||||
model = self._meta.model
|
model = self._meta.model
|
||||||
if model.objects.exclude(pk=self.pk).filter(username__iexact=self.username).exists():
|
if model.objects.exclude(pk=self.pk).filter(username__iexact=self.username).exists():
|
||||||
raise ValidationError(_("A user with this username already exists."))
|
raise ValidationError(_("A user with this username already exists."))
|
||||||
|
|
||||||
|
def get_full_name(self):
|
||||||
|
"""
|
||||||
|
Return the first_name plus the last_name, with a space in between.
|
||||||
|
"""
|
||||||
|
full_name = "%s %s" % (self.first_name, self.last_name)
|
||||||
|
return full_name.strip()
|
||||||
|
|
||||||
|
def get_short_name(self):
|
||||||
|
"""Return the short name for the user."""
|
||||||
|
return self.first_name
|
||||||
|
|
||||||
|
def email_user(self, subject, message, from_email=None, **kwargs):
|
||||||
|
"""Send an email to this user."""
|
||||||
|
send_mail(subject, message, from_email, [self.email], **kwargs)
|
||||||
|
|||||||
@@ -38,9 +38,6 @@ class UserTable(NetBoxTable):
|
|||||||
is_active = columns.BooleanColumn(
|
is_active = columns.BooleanColumn(
|
||||||
verbose_name=_('Is Active'),
|
verbose_name=_('Is Active'),
|
||||||
)
|
)
|
||||||
is_staff = columns.BooleanColumn(
|
|
||||||
verbose_name=_('Is Staff'),
|
|
||||||
)
|
|
||||||
is_superuser = columns.BooleanColumn(
|
is_superuser = columns.BooleanColumn(
|
||||||
verbose_name=_('Is Superuser'),
|
verbose_name=_('Is Superuser'),
|
||||||
)
|
)
|
||||||
@@ -51,8 +48,8 @@ class UserTable(NetBoxTable):
|
|||||||
class Meta(NetBoxTable.Meta):
|
class Meta(NetBoxTable.Meta):
|
||||||
model = User
|
model = User
|
||||||
fields = (
|
fields = (
|
||||||
'pk', 'id', 'username', 'first_name', 'last_name', 'email', 'groups', 'is_active', 'is_staff',
|
'pk', 'id', 'username', 'first_name', 'last_name', 'email', 'groups', 'is_active', 'is_superuser',
|
||||||
'is_superuser', 'last_login',
|
'last_login',
|
||||||
)
|
)
|
||||||
default_columns = ('pk', 'username', 'first_name', 'last_name', 'email', 'is_active')
|
default_columns = ('pk', 'username', 'first_name', 'last_name', 'email', 'is_active')
|
||||||
|
|
||||||
|
|||||||
@@ -30,7 +30,6 @@ class UserTestCase(TestCase, BaseFilterSetTests):
|
|||||||
first_name='Hank',
|
first_name='Hank',
|
||||||
last_name='Hill',
|
last_name='Hill',
|
||||||
email='hank@stricklandpropane.com',
|
email='hank@stricklandpropane.com',
|
||||||
is_staff=True,
|
|
||||||
is_superuser=True
|
is_superuser=True
|
||||||
),
|
),
|
||||||
User(
|
User(
|
||||||
@@ -98,10 +97,6 @@ class UserTestCase(TestCase, BaseFilterSetTests):
|
|||||||
params = {'is_active': True}
|
params = {'is_active': True}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 4)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 4)
|
||||||
|
|
||||||
def test_is_staff(self):
|
|
||||||
params = {'is_staff': True}
|
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 1)
|
|
||||||
|
|
||||||
def test_is_superuser(self):
|
def test_is_superuser(self):
|
||||||
params = {'is_superuser': True}
|
params = {'is_superuser': True}
|
||||||
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 1)
|
self.assertEqual(self.filterset(params, self.queryset).qs.count(), 1)
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ def nav(context):
|
|||||||
continue
|
continue
|
||||||
if not user.has_perms(item.permissions):
|
if not user.has_perms(item.permissions):
|
||||||
continue
|
continue
|
||||||
if item.staff_only and not user.is_staff:
|
if item.staff_only and not user.is_superuser:
|
||||||
continue
|
continue
|
||||||
buttons = [
|
buttons = [
|
||||||
button for button in item.buttons if user.has_perms(button.permissions)
|
button for button in item.buttons if user.has_perms(button.permissions)
|
||||||
|
|||||||
Reference in New Issue
Block a user