mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Fix: remove admin.logentry perm, use admin (staff) status (#6380)
This commit is contained in:
parent
47b4a602a7
commit
f812f2af4d
@ -241,6 +241,11 @@ permissions can be granted to limit access to certain parts of the UI (and corre
|
|||||||
|
|
||||||
Superusers can access all parts of the front and backend application as well as any and all objects.
|
Superusers can access all parts of the front and backend application as well as any and all objects.
|
||||||
|
|
||||||
|
#### Admin Status
|
||||||
|
|
||||||
|
Admin status (Django 'staff status') grants access to viewing the paperless logs and the system status dialog
|
||||||
|
as well as accessing the Django backend.
|
||||||
|
|
||||||
#### Detailed Explanation of Global Permissions {#global-permissions}
|
#### Detailed Explanation of Global Permissions {#global-permissions}
|
||||||
|
|
||||||
Global permissions define what areas of the app and API endpoints the user can access. For example, they
|
Global permissions define what areas of the app and API endpoints the user can access. For example, they
|
||||||
@ -249,7 +254,6 @@ still have "object-level" permissions.
|
|||||||
|
|
||||||
| Type | Details |
|
| Type | Details |
|
||||||
| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
||||||
| Admin | _View_ or higher permissions grants access to the logs view as well as the system status. |
|
|
||||||
| AppConfig | _Change_ or higher permissions grants access to the "Application Configuration" area. |
|
| AppConfig | _Change_ or higher permissions grants access to the "Application Configuration" area. |
|
||||||
| Correspondent | Grants global permissions to add, edit, delete or view Correspondents. |
|
| Correspondent | Grants global permissions to add, edit, delete or view Correspondents. |
|
||||||
| CustomField | Grants global permissions to add, edit, delete or view Custom Fields. |
|
| CustomField | Grants global permissions to add, edit, delete or view Custom Fields. |
|
||||||
|
@ -141,10 +141,7 @@ export const routes: Routes = [
|
|||||||
component: LogsComponent,
|
component: LogsComponent,
|
||||||
canActivate: [PermissionsGuard],
|
canActivate: [PermissionsGuard],
|
||||||
data: {
|
data: {
|
||||||
requiredPermission: {
|
requireAdmin: true,
|
||||||
action: PermissionAction.View,
|
|
||||||
type: PermissionType.Admin,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
// redirect old paths
|
// redirect old paths
|
||||||
|
@ -7,29 +7,30 @@
|
|||||||
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
||||||
<i-bs class="me-1" name="airplane"></i-bs> <ng-container i18n>Start tour</ng-container>
|
<i-bs class="me-1" name="airplane"></i-bs> <ng-container i18n>Start tour</ng-container>
|
||||||
</button>
|
</button>
|
||||||
<button class="btn btn-sm btn-outline-primary position-relative ms-md-5 me-1" (click)="showSystemStatus()"
|
@if (permissionsService.isAdmin()) {
|
||||||
[disabled]="!systemStatus"
|
<button class="btn btn-sm btn-outline-primary position-relative ms-md-5 me-1" (click)="showSystemStatus()"
|
||||||
*pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Admin }">
|
[disabled]="!systemStatus">
|
||||||
@if (!systemStatus) {
|
@if (!systemStatus) {
|
||||||
<div class="spinner-border spinner-border-sm me-1 h-75" role="status"></div>
|
<div class="spinner-border spinner-border-sm me-1 h-75" role="status"></div>
|
||||||
} @else {
|
|
||||||
<i-bs class="me-2" name="card-checklist"></i-bs>
|
|
||||||
@if (systemStatusHasErrors) {
|
|
||||||
<span class="badge bg-body position-absolute top-0 start-100 translate-middle rounded-pill p-0">
|
|
||||||
<i-bs name="exclamation-circle-fill" class="text-danger" width="1.75em" height="1.75em"></i-bs>
|
|
||||||
</span>
|
|
||||||
} @else {
|
} @else {
|
||||||
<span class="badge bg-body position-absolute top-0 start-100 translate-middle rounded-pill p-0">
|
<i-bs class="me-2" name="card-checklist"></i-bs>
|
||||||
<i-bs name="check-circle-fill" class="text-primary" width="1.75em" height="1.75em"></i-bs>
|
@if (systemStatusHasErrors) {
|
||||||
</span>
|
<span class="badge bg-body position-absolute top-0 start-100 translate-middle rounded-pill p-0">
|
||||||
|
<i-bs name="exclamation-circle-fill" class="text-danger" width="1.75em" height="1.75em"></i-bs>
|
||||||
|
</span>
|
||||||
|
} @else {
|
||||||
|
<span class="badge bg-body position-absolute top-0 start-100 translate-middle rounded-pill p-0">
|
||||||
|
<i-bs name="check-circle-fill" class="text-primary" width="1.75em" height="1.75em"></i-bs>
|
||||||
|
</span>
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
<ng-container i18n>System Status</ng-container>
|
||||||
<ng-container i18n>System Status</ng-container>
|
</button>
|
||||||
</button>
|
<a class="btn btn-sm btn-primary" href="admin/" target="_blank">
|
||||||
<a *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Admin }" class="btn btn-sm btn-primary" href="admin/" target="_blank">
|
<ng-container i18n>Open Django Admin</ng-container>
|
||||||
<ng-container i18n>Open Django Admin</ng-container>
|
<i-bs name="arrow-up-right"></i-bs>
|
||||||
<i-bs name="arrow-up-right"></i-bs>
|
</a>
|
||||||
</a>
|
}
|
||||||
</pngx-page-header>
|
</pngx-page-header>
|
||||||
|
|
||||||
<form [formGroup]="settingsForm" (ngSubmit)="saveSettings()">
|
<form [formGroup]="settingsForm" (ngSubmit)="saveSettings()">
|
||||||
|
@ -418,6 +418,7 @@ describe('SettingsComponent', () => {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status))
|
jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status))
|
||||||
|
jest.spyOn(permissionsService, 'isAdmin').mockReturnValue(true)
|
||||||
completeSetup()
|
completeSetup()
|
||||||
expect(component['systemStatus']).toEqual(status) // private
|
expect(component['systemStatus']).toEqual(status) // private
|
||||||
expect(component.systemStatusHasErrors).toBeTruthy()
|
expect(component.systemStatusHasErrors).toBeTruthy()
|
||||||
|
@ -121,7 +121,7 @@ export class SettingsComponent
|
|||||||
users: User[]
|
users: User[]
|
||||||
groups: Group[]
|
groups: Group[]
|
||||||
|
|
||||||
private systemStatus: SystemStatus
|
public systemStatus: SystemStatus
|
||||||
|
|
||||||
get systemStatusHasErrors(): boolean {
|
get systemStatusHasErrors(): boolean {
|
||||||
return (
|
return (
|
||||||
@ -385,12 +385,7 @@ export class SettingsComponent
|
|||||||
this.settingsForm.patchValue(currentFormValue)
|
this.settingsForm.patchValue(currentFormValue)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (this.permissionsService.isAdmin()) {
|
||||||
this.permissionsService.currentUserCan(
|
|
||||||
PermissionAction.View,
|
|
||||||
PermissionType.Admin
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
this.systemStatusService.get().subscribe((status) => {
|
this.systemStatusService.get().subscribe((status) => {
|
||||||
this.systemStatus = status
|
this.systemStatus = status
|
||||||
})
|
})
|
||||||
|
@ -267,13 +267,15 @@
|
|||||||
}
|
}
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
<li class="nav-item app-link" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Admin }">
|
@if (permissionsService.isAdmin()) {
|
||||||
<a class="nav-link" routerLink="logs" routerLinkActive="active" (click)="closeMenu()" ngbPopover="Logs"
|
<li class="nav-item app-link">
|
||||||
i18n-ngbPopover [disablePopover]="!slimSidebarEnabled" placement="end" container="body"
|
<a class="nav-link" routerLink="logs" routerLinkActive="active" (click)="closeMenu()" ngbPopover="Logs"
|
||||||
triggers="mouseenter:mouseleave" popoverClass="popover-slim">
|
i18n-ngbPopover [disablePopover]="!slimSidebarEnabled" placement="end" container="body"
|
||||||
<i-bs class="me-1" name="text-left"></i-bs><span> <ng-container i18n>Logs</ng-container></span>
|
triggers="mouseenter:mouseleave" popoverClass="popover-slim">
|
||||||
</a>
|
<i-bs class="me-1" name="text-left"></i-bs><span> <ng-container i18n>Logs</ng-container></span>
|
||||||
</li>
|
</a>
|
||||||
|
</li>
|
||||||
|
}
|
||||||
<li class="nav-item mt-2" tourAnchor="tour.outro">
|
<li class="nav-item mt-2" tourAnchor="tour.outro">
|
||||||
<a class="px-3 py-2 text-muted small d-flex align-items-center flex-wrap text-decoration-none"
|
<a class="px-3 py-2 text-muted small d-flex align-items-center flex-wrap text-decoration-none"
|
||||||
target="_blank" rel="noopener noreferrer" href="https://docs.paperless-ngx.com" ngbPopover="Documentation"
|
target="_blank" rel="noopener noreferrer" href="https://docs.paperless-ngx.com" ngbPopover="Documentation"
|
||||||
|
@ -16,11 +16,15 @@
|
|||||||
<pngx-input-text i18n-title title="First name" formControlName="first_name" [error]="error?.first_name"></pngx-input-text>
|
<pngx-input-text i18n-title title="First name" formControlName="first_name" [error]="error?.first_name"></pngx-input-text>
|
||||||
<pngx-input-text i18n-title title="Last name" formControlName="last_name" [error]="error?.first_name"></pngx-input-text>
|
<pngx-input-text i18n-title title="Last name" formControlName="last_name" [error]="error?.first_name"></pngx-input-text>
|
||||||
|
|
||||||
<div class="mb-2">
|
<div class="mb-2 d-flex flex-column">
|
||||||
<div class="form-check form-switch form-check-inline">
|
<div class="form-check form-switch form-check-inline">
|
||||||
<input type="checkbox" class="form-check-input" id="is_active" formControlName="is_active">
|
<input type="checkbox" class="form-check-input" id="is_active" formControlName="is_active">
|
||||||
<label class="form-check-label" for="is_active" i18n>Active</label>
|
<label class="form-check-label" for="is_active" i18n>Active</label>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-check form-switch form-check-inline">
|
||||||
|
<input type="checkbox" class="form-check-input" id="is_staff" formControlName="is_staff">
|
||||||
|
<label class="form-check-label" for="is_staff"><ng-container i18n>Admin</ng-container> <small class="form-text text-muted ms-1" i18n>Access logs, Django backend</small></label>
|
||||||
|
</div>
|
||||||
<div class="form-check form-switch form-check-inline">
|
<div class="form-check form-switch form-check-inline">
|
||||||
<input type="checkbox" class="form-check-input" id="is_superuser" formControlName="is_superuser" (change)="onToggleSuperUser()">
|
<input type="checkbox" class="form-check-input" id="is_superuser" formControlName="is_superuser" (change)="onToggleSuperUser()">
|
||||||
<label class="form-check-label" for="is_superuser"><ng-container i18n>Superuser</ng-container> <small class="form-text text-muted ms-1" i18n>(Grants all permissions and can view objects)</small></label>
|
<label class="form-check-label" for="is_superuser"><ng-container i18n>Superuser</ng-container> <small class="form-text text-muted ms-1" i18n>(Grants all permissions and can view objects)</small></label>
|
||||||
|
@ -56,6 +56,7 @@ export class UserEditDialogComponent
|
|||||||
first_name: new FormControl(''),
|
first_name: new FormControl(''),
|
||||||
last_name: new FormControl(''),
|
last_name: new FormControl(''),
|
||||||
is_active: new FormControl(true),
|
is_active: new FormControl(true),
|
||||||
|
is_staff: new FormControl(true),
|
||||||
is_superuser: new FormControl(false),
|
is_superuser: new FormControl(false),
|
||||||
groups: new FormControl([]),
|
groups: new FormControl([]),
|
||||||
user_permissions: new FormControl([]),
|
user_permissions: new FormControl([]),
|
||||||
|
@ -23,10 +23,12 @@ export class PermissionsGuard {
|
|||||||
state: RouterStateSnapshot
|
state: RouterStateSnapshot
|
||||||
): boolean | UrlTree {
|
): boolean | UrlTree {
|
||||||
if (
|
if (
|
||||||
!this.permissionsService.currentUserCan(
|
(route.data.requireAdmin && !this.permissionsService.isAdmin()) ||
|
||||||
route.data.requiredPermission.action,
|
(route.data.requiredPermission &&
|
||||||
route.data.requiredPermission.type
|
!this.permissionsService.currentUserCan(
|
||||||
)
|
route.data.requiredPermission.action,
|
||||||
|
route.data.requiredPermission.type
|
||||||
|
))
|
||||||
) {
|
) {
|
||||||
// Check if tour is running 1 = TourState.ON
|
// Check if tour is running 1 = TourState.ON
|
||||||
if (this.tourService.getStatus() !== 1) {
|
if (this.tourService.getStatus() !== 1) {
|
||||||
|
@ -418,4 +418,25 @@ describe('PermissionsService', () => {
|
|||||||
)
|
)
|
||||||
).toBeTruthy()
|
).toBeTruthy()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('correctly checks admin status', () => {
|
||||||
|
permissionsService.initialize([], {
|
||||||
|
username: 'testuser',
|
||||||
|
last_name: 'User',
|
||||||
|
first_name: 'Test',
|
||||||
|
id: 1,
|
||||||
|
is_staff: true,
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(permissionsService.isAdmin()).toBeTruthy()
|
||||||
|
|
||||||
|
permissionsService.initialize([], {
|
||||||
|
username: 'testuser',
|
||||||
|
last_name: 'User',
|
||||||
|
first_name: 'Test',
|
||||||
|
id: 1,
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(permissionsService.isAdmin()).toBeFalsy()
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
@ -24,7 +24,6 @@ export enum PermissionType {
|
|||||||
MailRule = '%s_mailrule',
|
MailRule = '%s_mailrule',
|
||||||
User = '%s_user',
|
User = '%s_user',
|
||||||
Group = '%s_group',
|
Group = '%s_group',
|
||||||
Admin = '%s_logentry',
|
|
||||||
ShareLink = '%s_sharelink',
|
ShareLink = '%s_sharelink',
|
||||||
CustomField = '%s_customfield',
|
CustomField = '%s_customfield',
|
||||||
Workflow = '%s_workflow',
|
Workflow = '%s_workflow',
|
||||||
@ -52,6 +51,10 @@ export class PermissionsService {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public isAdmin(): boolean {
|
||||||
|
return this.currentUser?.is_staff
|
||||||
|
}
|
||||||
|
|
||||||
public currentUserOwnsObject(object: ObjectWithPermissions): boolean {
|
public currentUserOwnsObject(object: ObjectWithPermissions): boolean {
|
||||||
return (
|
return (
|
||||||
!object ||
|
!object ||
|
||||||
|
@ -40,7 +40,7 @@ class PaperlessObjectPermissions(DjangoObjectPermissions):
|
|||||||
|
|
||||||
class PaperlessAdminPermissions(BasePermission):
|
class PaperlessAdminPermissions(BasePermission):
|
||||||
def has_permission(self, request, view):
|
def has_permission(self, request, view):
|
||||||
return request.user.has_perm("admin.view_logentry")
|
return request.user.is_staff
|
||||||
|
|
||||||
|
|
||||||
def get_groups_with_only_permission(obj, codename):
|
def get_groups_with_only_permission(obj, codename):
|
||||||
|
@ -131,6 +131,7 @@ class TestApiAuth(DirectoriesMixin, APITestCase):
|
|||||||
def test_api_sufficient_permissions(self):
|
def test_api_sufficient_permissions(self):
|
||||||
user = User.objects.create_user(username="test")
|
user = User.objects.create_user(username="test")
|
||||||
user.user_permissions.add(*Permission.objects.all())
|
user.user_permissions.add(*Permission.objects.all())
|
||||||
|
user.is_staff = True
|
||||||
self.client.force_authenticate(user)
|
self.client.force_authenticate(user)
|
||||||
|
|
||||||
Document.objects.create(title="Test")
|
Document.objects.create(title="Test")
|
||||||
|
@ -27,6 +27,7 @@ class TestApiUiSettings(DirectoriesMixin, APITestCase):
|
|||||||
{
|
{
|
||||||
"id": self.test_user.id,
|
"id": self.test_user.id,
|
||||||
"username": self.test_user.username,
|
"username": self.test_user.username,
|
||||||
|
"is_staff": True,
|
||||||
"is_superuser": True,
|
"is_superuser": True,
|
||||||
"groups": [],
|
"groups": [],
|
||||||
"first_name": self.test_user.first_name,
|
"first_name": self.test_user.first_name,
|
||||||
|
@ -1270,6 +1270,7 @@ class UiSettingsView(GenericAPIView):
|
|||||||
user_resp = {
|
user_resp = {
|
||||||
"id": user.id,
|
"id": user.id,
|
||||||
"username": user.username,
|
"username": user.username,
|
||||||
|
"is_staff": user.is_staff,
|
||||||
"is_superuser": user.is_superuser,
|
"is_superuser": user.is_superuser,
|
||||||
"groups": list(user.groups.values_list("id", flat=True)),
|
"groups": list(user.groups.values_list("id", flat=True)),
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user