From 022bb272e6f3b43a66ac7ec6a65f5ec9fca245b6 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Tue, 8 Aug 2023 16:27:40 -0700 Subject: [PATCH 1/3] Restrict status messages by owner if set --- .../data/websocket-consumer-status-message.ts | 1 + .../services/consumer-status.service.spec.ts | 45 ++++++++++++++++++- .../app/services/consumer-status.service.ts | 14 +++++- src/documents/consumer.py | 3 +- 4 files changed, 60 insertions(+), 3 deletions(-) diff --git a/src-ui/src/app/data/websocket-consumer-status-message.ts b/src-ui/src/app/data/websocket-consumer-status-message.ts index aecdda7c0..d1ac590b1 100644 --- a/src-ui/src/app/data/websocket-consumer-status-message.ts +++ b/src-ui/src/app/data/websocket-consumer-status-message.ts @@ -6,4 +6,5 @@ export interface WebsocketConsumerStatusMessage { status?: string message?: string document_id: number + owner_id?: number } diff --git a/src-ui/src/app/services/consumer-status.service.spec.ts b/src-ui/src/app/services/consumer-status.service.spec.ts index 3725f847d..d3867e889 100644 --- a/src-ui/src/app/services/consumer-status.service.spec.ts +++ b/src-ui/src/app/services/consumer-status.service.spec.ts @@ -12,6 +12,7 @@ import { environment } from 'src/environments/environment' import { DocumentService } from './rest/document.service' import { HttpEventType, HttpResponse } from '@angular/common/http' import WS from 'jest-websocket-mock' +import { SettingsService } from './settings.service' describe('ConsumerStatusService', () => { let httpTestingController: HttpTestingController @@ -24,7 +25,21 @@ describe('ConsumerStatusService', () => { beforeEach(() => { TestBed.configureTestingModule({ - providers: [ConsumerStatusService, DocumentService], + providers: [ + ConsumerStatusService, + DocumentService, + SettingsService, + { + provide: SettingsService, + useValue: { + currentUser: { + id: 1, + username: 'testuser', + is_superuser: false, + }, + }, + }, + ], imports: [HttpClientTestingModule], }) @@ -275,4 +290,32 @@ describe('ConsumerStatusService', () => { 1 ) }) + + it('should not notify current user if document has different expected owner', () => { + consumerStatusService.connect() + server.send({ + task_id: '1234', + filename: 'file1.pdf', + current_progress: 50, + max_progress: 100, + docuement_id: 12, + owner_id: 1, + status: 'WORKING', + }) + + server.send({ + task_id: '5678', + filename: 'file2.pdf', + current_progress: 50, + max_progress: 100, + docuement_id: 13, + owner_id: 2, + status: 'WORKING', + }) + + consumerStatusService.disconnect() + expect(consumerStatusService.getConsumerStatusNotCompleted()).toHaveLength( + 1 + ) + }) }) diff --git a/src-ui/src/app/services/consumer-status.service.ts b/src-ui/src/app/services/consumer-status.service.ts index 394975333..3e21da138 100644 --- a/src-ui/src/app/services/consumer-status.service.ts +++ b/src-ui/src/app/services/consumer-status.service.ts @@ -2,6 +2,7 @@ import { Injectable } from '@angular/core' import { Subject } from 'rxjs' import { environment } from 'src/environments/environment' import { WebsocketConsumerStatusMessage } from '../data/websocket-consumer-status-message' +import { SettingsService } from './settings.service' // see ConsumerFilePhase in src/documents/consumer.py export enum FileStatusPhase { @@ -44,6 +45,8 @@ export class FileStatus { documentId: number + ownerId: number + getProgress(): number { switch (this.phase) { case FileStatusPhase.STARTED: @@ -81,7 +84,7 @@ export class FileStatus { providedIn: 'root', }) export class ConsumerStatusService { - constructor() {} + constructor(private settingsService: SettingsService) {} private statusWebSocket: WebSocket @@ -143,6 +146,15 @@ export class ConsumerStatusService { this.statusWebSocket.onmessage = (ev) => { let statusMessage: WebsocketConsumerStatusMessage = JSON.parse(ev['data']) + // tasks are async so we rely on checking user id + if ( + statusMessage.owner_id && + statusMessage.owner_id !== this.settingsService.currentUser?.id && + !this.settingsService.currentUser?.is_superuser + ) { + return + } + let statusMessageGet = this.get( statusMessage.task_id, statusMessage.filename diff --git a/src/documents/consumer.py b/src/documents/consumer.py index 6fa830101..0ec6090c2 100644 --- a/src/documents/consumer.py +++ b/src/documents/consumer.py @@ -90,6 +90,7 @@ class Consumer(LoggingMixin): "status": status, "message": message, "document_id": document_id, + "owner_id": self.override_owner_id if self.override_owner_id else None, } async_to_sync(self.channel_layer.group_send)( "status_updates", @@ -118,7 +119,7 @@ class Consumer(LoggingMixin): self.override_document_type_id = None self.override_asn = None self.task_id = None - self.owner_id = None + self.override_owner_id = None self.channel_layer = get_channel_layer() From 9291c98189963d69098359c6bb105762c8e3d9dd Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Tue, 8 Aug 2023 23:59:13 -0700 Subject: [PATCH 2/3] Improve 404 navigation and styling --- src-ui/messages.xlf | 83 ++++++++++--------- src-ui/src/app/app.module.ts | 2 + .../common/logo/logo.component.html | 18 ++++ .../common/logo/logo.component.scss | 0 .../common/logo/logo.component.spec.ts | 36 ++++++++ .../components/common/logo/logo.component.ts | 18 ++++ .../dashboard/dashboard.component.html | 19 +---- .../dashboard/dashboard.component.spec.ts | 2 + .../document-asn.component.spec.ts | 2 +- .../document-asn/document-asn.component.ts | 4 +- .../document-detail.component.spec.ts | 4 +- .../document-detail.component.ts | 8 +- .../document-list.component.spec.ts | 2 +- .../document-list/document-list.component.ts | 4 +- .../not-found/not-found.component.html | 25 ++++-- .../not-found/not-found.component.spec.ts | 7 +- 16 files changed, 160 insertions(+), 74 deletions(-) create mode 100644 src-ui/src/app/components/common/logo/logo.component.html create mode 100644 src-ui/src/app/components/common/logo/logo.component.scss create mode 100644 src-ui/src/app/components/common/logo/logo.component.spec.ts create mode 100644 src-ui/src/app/components/common/logo/logo.component.ts diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index e7dd14aab..28f4b06cb 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -1247,7 +1247,7 @@ src/app/components/dashboard/dashboard.component.html - 27 + 10 src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html @@ -2808,43 +2808,43 @@ Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 395 + 397 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 417 + 419 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 529 + 533 src/app/components/document-detail/document-detail.component.ts - 537 + 541 Error saving document src/app/components/document-detail/document-detail.component.ts - 542 + 546 src/app/components/document-detail/document-detail.component.ts - 587 + 591 Confirm delete src/app/components/document-detail/document-detail.component.ts - 616 + 620 src/app/components/manage/management-list/management-list.component.ts @@ -2855,35 +2855,35 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 617 + 621 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 618 + 622 Delete document src/app/components/document-detail/document-detail.component.ts - 620 + 624 Error deleting document: src/app/components/document-detail/document-detail.component.ts - 640,642 + 644,646 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 663 + 667 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2894,14 +2894,14 @@ This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 664 + 668 This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 665 + 669 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2932,7 +2932,7 @@ Proceed src/app/components/document-detail/document-detail.component.ts - 667 + 671 src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -2959,7 +2959,7 @@ Redo OCR operation will begin in the background. Close and re-open or reload this document after the operation has completed to see new content. src/app/components/document-detail/document-detail.component.ts - 675 + 679 @@ -2968,7 +2968,7 @@ )"/> src/app/components/document-detail/document-detail.component.ts - 686,688 + 690,692 @@ -3629,14 +3629,14 @@ View "" saved successfully. src/app/components/document-list/document-list.component.ts - 205 + 207 View "" created successfully. src/app/components/document-list/document-list.component.ts - 246 + 248 @@ -3907,7 +3907,7 @@ Do you really want to delete the correspondent ""? src/app/components/manage/correspondent-list/correspondent-list.component.ts - 55 + 67 @@ -4903,11 +4903,18 @@ 137 - - 404 Not Found + + Not Found src/app/components/not-found/not-found.component.html - 7 + 6 + + + + Go to Dashboard + + src/app/components/not-found/not-found.component.html + 12 @@ -5105,28 +5112,28 @@ Document already exists. src/app/services/consumer-status.service.ts - 16 + 17 Document with ASN already exists. src/app/services/consumer-status.service.ts - 17 + 18 File not found. src/app/services/consumer-status.service.ts - 18 + 19 Pre-consume script does not exist. src/app/services/consumer-status.service.ts - 19 + 20 Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation @@ -5134,7 +5141,7 @@ Error while executing pre-consume script. src/app/services/consumer-status.service.ts - 20 + 21 Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation @@ -5142,7 +5149,7 @@ Post-consume script does not exist. src/app/services/consumer-status.service.ts - 21 + 22 Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation @@ -5150,7 +5157,7 @@ Error while executing post-consume script. src/app/services/consumer-status.service.ts - 22 + 23 Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation @@ -5158,49 +5165,49 @@ Received new file. src/app/services/consumer-status.service.ts - 23 + 24 File type not supported. src/app/services/consumer-status.service.ts - 24 + 25 Processing document... src/app/services/consumer-status.service.ts - 25 + 26 Generating thumbnail... src/app/services/consumer-status.service.ts - 26 + 27 Retrieving date from document... src/app/services/consumer-status.service.ts - 27 + 28 Saving document... src/app/services/consumer-status.service.ts - 28 + 29 Finished. src/app/services/consumer-status.service.ts - 29 + 30 diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 86b6d29c4..7a94f4316 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -92,6 +92,7 @@ import { PermissionsDialogComponent } from './components/common/permissions-dial import { PermissionsFormComponent } from './components/common/input/permissions/permissions-form/permissions-form.component' import { PermissionsFilterDropdownComponent } from './components/common/permissions-filter-dropdown/permissions-filter-dropdown.component' import { UsernamePipe } from './pipes/username.pipe' +import { LogoComponent } from './components/common/logo/logo.component' import localeAr from '@angular/common/locales/ar' import localeBe from '@angular/common/locales/be' @@ -221,6 +222,7 @@ function initializeApp(settings: SettingsService) { PermissionsFormComponent, PermissionsFilterDropdownComponent, UsernamePipe, + LogoComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/logo/logo.component.html b/src-ui/src/app/components/common/logo/logo.component.html new file mode 100644 index 000000000..6c688902e --- /dev/null +++ b/src-ui/src/app/components/common/logo/logo.component.html @@ -0,0 +1,18 @@ + + + + + + + + + + + + + + + + + + diff --git a/src-ui/src/app/components/common/logo/logo.component.scss b/src-ui/src/app/components/common/logo/logo.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/common/logo/logo.component.spec.ts b/src-ui/src/app/components/common/logo/logo.component.spec.ts new file mode 100644 index 000000000..118e0e40e --- /dev/null +++ b/src-ui/src/app/components/common/logo/logo.component.spec.ts @@ -0,0 +1,36 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' + +import { LogoComponent } from './logo.component' +import { By } from '@angular/platform-browser' + +describe('LogoComponent', () => { + let component: LogoComponent + let fixture: ComponentFixture + + beforeEach(() => { + TestBed.configureTestingModule({ + declarations: [LogoComponent], + }) + fixture = TestBed.createComponent(LogoComponent) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should support extra classes', () => { + expect(fixture.debugElement.queryAll(By.css('.foo'))).toHaveLength(0) + component.extra_classes = 'foo' + fixture.detectChanges() + expect(fixture.debugElement.queryAll(By.css('.foo'))).toHaveLength(1) + }) + + it('should support setting height', () => { + expect(fixture.debugElement.query(By.css('svg')).attributes.height).toEqual( + '6em' + ) + component.height = '10em' + fixture.detectChanges() + expect(fixture.debugElement.query(By.css('svg')).attributes.height).toEqual( + '10em' + ) + }) +}) diff --git a/src-ui/src/app/components/common/logo/logo.component.ts b/src-ui/src/app/components/common/logo/logo.component.ts new file mode 100644 index 000000000..0c5ecefda --- /dev/null +++ b/src-ui/src/app/components/common/logo/logo.component.ts @@ -0,0 +1,18 @@ +import { Component, Input } from '@angular/core' + +@Component({ + selector: 'app-logo', + templateUrl: './logo.component.html', + styleUrls: ['./logo.component.scss'], +}) +export class LogoComponent { + @Input() + extra_classes: string + + @Input() + height = '6em' + + getClasses() { + return ['logo'].concat(this.extra_classes).join(' ') + } +} diff --git a/src-ui/src/app/components/dashboard/dashboard.component.html b/src-ui/src/app/components/dashboard/dashboard.component.html index f2dcf855e..db396a62b 100644 --- a/src-ui/src/app/components/dashboard/dashboard.component.html +++ b/src-ui/src/app/components/dashboard/dashboard.component.html @@ -1,22 +1,5 @@ - +
diff --git a/src-ui/src/app/components/dashboard/dashboard.component.spec.ts b/src-ui/src/app/components/dashboard/dashboard.component.spec.ts index 911565526..6d100510d 100644 --- a/src-ui/src/app/components/dashboard/dashboard.component.spec.ts +++ b/src-ui/src/app/components/dashboard/dashboard.component.spec.ts @@ -16,6 +16,7 @@ import { IfPermissionsDirective } from 'src/app/directives/if-permissions.direct import { NgxFileDropModule } from 'ngx-file-drop' import { RouterTestingModule } from '@angular/router/testing' import { TourNgBootstrapModule, TourService } from 'ngx-ui-tour-ng-bootstrap' +import { LogoComponent } from '../common/logo/logo.component' describe('DashboardComponent', () => { let component: DashboardComponent @@ -33,6 +34,7 @@ describe('DashboardComponent', () => { UploadFileWidgetComponent, IfPermissionsDirective, SavedViewWidgetComponent, + LogoComponent, ], providers: [ PermissionsGuard, diff --git a/src-ui/src/app/components/document-asn/document-asn.component.spec.ts b/src-ui/src/app/components/document-asn/document-asn.component.spec.ts index 62b1113db..c8ad0d13d 100644 --- a/src-ui/src/app/components/document-asn/document-asn.component.spec.ts +++ b/src-ui/src/app/components/document-asn/document-asn.component.spec.ts @@ -53,6 +53,6 @@ describe('DocumentAsnComponent', () => { .mockReturnValue(of(convertToParamMap({ id: '5578' }))) const navigateSpy = jest.spyOn(router, 'navigate') component.ngOnInit() - expect(navigateSpy).toHaveBeenCalledWith(['404']) + expect(navigateSpy).toHaveBeenCalledWith(['404'], { replaceUrl: true }) }) }) diff --git a/src-ui/src/app/components/document-asn/document-asn.component.ts b/src-ui/src/app/components/document-asn/document-asn.component.ts index 4fb9f474a..6003f1621 100644 --- a/src-ui/src/app/components/document-asn/document-asn.component.ts +++ b/src-ui/src/app/components/document-asn/document-asn.component.ts @@ -25,7 +25,9 @@ export class DocumentAsnComponent implements OnInit { if (documentId.length == 1) { this.router.navigate(['documents', documentId[0]]) } else { - this.router.navigate(['404']) + this.router.navigate(['404'], { + replaceUrl: true, + }) } }) }) diff --git a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts index dee1435b7..bbee477cc 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.spec.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.spec.ts @@ -355,14 +355,14 @@ describe('DocumentDetailComponent', () => { .mockReturnValueOnce(throwError(() => new Error('unable to discard'))) component.discard() fixture.detectChanges() - expect(navigateSpy).toHaveBeenCalledWith(['404']) + expect(navigateSpy).toHaveBeenCalledWith(['404'], { replaceUrl: true }) }) it('should 404 on invalid id', () => { jest.spyOn(documentService, 'get').mockReturnValueOnce(of(null)) const navigateSpy = jest.spyOn(router, 'navigate') fixture.detectChanges() - expect(navigateSpy).toHaveBeenCalledWith(['404']) + expect(navigateSpy).toHaveBeenCalledWith(['404'], { replaceUrl: true }) }) it('should support save, close and show success toast', () => { diff --git a/src-ui/src/app/components/document-detail/document-detail.component.ts b/src-ui/src/app/components/document-detail/document-detail.component.ts index 7a385bebe..7337028da 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.ts @@ -341,7 +341,9 @@ export class DocumentDetailComponent this.openDocumentService.setDirty(doc, dirty) }, error: (error) => { - this.router.navigate(['404']) + this.router.navigate(['404'], { + replaceUrl: true, + }) }, }) @@ -513,7 +515,9 @@ export class DocumentDetailComponent this.openDocumentService.setDirty(doc, false) }, error: () => { - this.router.navigate(['404']) + this.router.navigate(['404'], { + replaceUrl: true, + }) }, }) } diff --git a/src-ui/src/app/components/document-list/document-list.component.spec.ts b/src-ui/src/app/components/document-list/document-list.component.spec.ts index 2b14747bf..6e6dd8f6c 100644 --- a/src-ui/src/app/components/document-list/document-list.component.spec.ts +++ b/src-ui/src/app/components/document-list/document-list.component.spec.ts @@ -259,7 +259,7 @@ describe('DocumentListComponent', () => { .mockReturnValue(of(convertToParamMap({ id: '10' }))) const navigateSpy = jest.spyOn(router, 'navigate') fixture.detectChanges() - expect(navigateSpy).toHaveBeenCalledWith(['404']) + expect(navigateSpy).toHaveBeenCalledWith(['404'], { replaceUrl: true }) }) it('should load saved view from query params', () => { diff --git a/src-ui/src/app/components/document-list/document-list.component.ts b/src-ui/src/app/components/document-list/document-list.component.ts index 32431167b..25a95401f 100644 --- a/src-ui/src/app/components/document-list/document-list.component.ts +++ b/src-ui/src/app/components/document-list/document-list.component.ts @@ -153,7 +153,9 @@ export class DocumentListComponent .pipe(takeUntil(this.unsubscribeNotifier)) .subscribe(({ view }) => { if (!view) { - this.router.navigate(['404']) + this.router.navigate(['404'], { + replaceUrl: true, + }) return } this.unmodifiedSavedView = view diff --git a/src-ui/src/app/components/not-found/not-found.component.html b/src-ui/src/app/components/not-found/not-found.component.html index 913132d1b..4b4dc7c09 100644 --- a/src-ui/src/app/components/not-found/not-found.component.html +++ b/src-ui/src/app/components/not-found/not-found.component.html @@ -1,8 +1,17 @@ -
- - - - - -

404 Not Found

-
+
+
+

404

+
+
+

Not Found

+

+ + + + + Go to Dashboard + +

+
+ +
diff --git a/src-ui/src/app/components/not-found/not-found.component.spec.ts b/src-ui/src/app/components/not-found/not-found.component.spec.ts index 8962d833f..2a0ab9d7c 100644 --- a/src-ui/src/app/components/not-found/not-found.component.spec.ts +++ b/src-ui/src/app/components/not-found/not-found.component.spec.ts @@ -1,5 +1,7 @@ import { ComponentFixture, TestBed } from '@angular/core/testing' import { NotFoundComponent } from './not-found.component' +import { By } from '@angular/platform-browser' +import { LogoComponent } from '../common/logo/logo.component' describe('NotFoundComponent', () => { let component: NotFoundComponent @@ -7,7 +9,7 @@ describe('NotFoundComponent', () => { beforeEach(async () => { TestBed.configureTestingModule({ - declarations: [NotFoundComponent], + declarations: [NotFoundComponent, LogoComponent], }).compileComponents() fixture = TestBed.createComponent(NotFoundComponent) @@ -18,6 +20,7 @@ describe('NotFoundComponent', () => { it('should create component', () => { expect(component).toBeTruthy() - expect(fixture.nativeElement.textContent).toContain('404 Not Found') + expect(fixture.nativeElement.textContent).toContain('Not Found') + expect(fixture.debugElement.queryAll(By.css('a'))).toHaveLength(1) }) }) From 3b666fef77c5c70f006c8d282825a3cd7e572af3 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Wed, 9 Aug 2023 08:52:23 -0700 Subject: [PATCH 3/3] Add backend check for ws message ownership --- src-ui/src/app/services/consumer-status.service.ts | 2 +- src/paperless/consumers.py | 13 ++++++++++++- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src-ui/src/app/services/consumer-status.service.ts b/src-ui/src/app/services/consumer-status.service.ts index 3e21da138..2b587fbfd 100644 --- a/src-ui/src/app/services/consumer-status.service.ts +++ b/src-ui/src/app/services/consumer-status.service.ts @@ -146,7 +146,7 @@ export class ConsumerStatusService { this.statusWebSocket.onmessage = (ev) => { let statusMessage: WebsocketConsumerStatusMessage = JSON.parse(ev['data']) - // tasks are async so we rely on checking user id + // fallback if backend didnt restrict message if ( statusMessage.owner_id && statusMessage.owner_id !== this.settingsService.currentUser?.id && diff --git a/src/paperless/consumers.py b/src/paperless/consumers.py index 7c34c8c39..cf1a3b548 100644 --- a/src/paperless/consumers.py +++ b/src/paperless/consumers.py @@ -10,6 +10,16 @@ class StatusConsumer(WebsocketConsumer): def _authenticated(self): return "user" in self.scope and self.scope["user"].is_authenticated + def _is_owner_or_unowned(self, data): + return ( + ( + self.scope["user"].is_superuser + or self.scope["user"].id == data["owner_id"] + ) + if "owner_id" in data and "user" in self.scope + else True + ) + def connect(self): if not self._authenticated(): raise DenyConnection @@ -30,4 +40,5 @@ class StatusConsumer(WebsocketConsumer): if not self._authenticated(): self.close() else: - self.send(json.dumps(event["data"])) + if self._is_owner_or_unowned(event["data"]): + self.send(json.dumps(event["data"]))