mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			7 Commits
		
	
	
		
			d609b386fe
			...
			feature-di
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 44d25f72b7 | ||
|   | 495159f0b2 | ||
|   | 33fd8a6579 | ||
|   | e08e34fb90 | ||
|   | 6164bac66e | ||
|   | df86882e8e | ||
|   | 79b30fbade | 
| @@ -1636,7 +1636,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">44</context> | ||||
|           <context context-type="linenumber">45</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/trash/trash.component.html</context> | ||||
| @@ -1862,7 +1862,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">153</context> | ||||
|           <context context-type="linenumber">155</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2134950584701094962" datatype="html"> | ||||
| @@ -1918,63 +1918,77 @@ | ||||
|         <source>Result</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">45</context> | ||||
|           <context context-type="linenumber">46</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5404910960991552159" datatype="html"> | ||||
|         <source>Dismiss selected</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">108</context> | ||||
|           <context context-type="linenumber">110</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8829078752502782653" datatype="html"> | ||||
|         <source>Dismiss all</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">109</context> | ||||
|           <context context-type="linenumber">111</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="1323591410517879795" datatype="html"> | ||||
|         <source>Confirm Dismiss All</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">150</context> | ||||
|           <context context-type="linenumber">152</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4157200209636243740" datatype="html"> | ||||
|         <source>Dismiss all <x id="PH" equiv-text="tasks.size"/> tasks?</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">151</context> | ||||
|           <context context-type="linenumber">153</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3597309129998924778" datatype="html"> | ||||
|         <source>Error dismissing tasks</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">161</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2132179171926568807" datatype="html"> | ||||
|         <source>Error dismissing task</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">170</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9011556615675272238" datatype="html"> | ||||
|         <source>queued</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">236</context> | ||||
|           <context context-type="linenumber">246</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6415892379431855826" datatype="html"> | ||||
|         <source>started</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">238</context> | ||||
|           <context context-type="linenumber">248</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7510279840486540181" datatype="html"> | ||||
|         <source>completed</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">240</context> | ||||
|           <context context-type="linenumber">250</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4083337005045748464" datatype="html"> | ||||
|         <source>failed</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">242</context> | ||||
|           <context context-type="linenumber">252</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3418677553313974490" datatype="html"> | ||||
| @@ -2560,11 +2574,11 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1025</context> | ||||
|           <context context-type="linenumber">1028</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1390</context> | ||||
|           <context context-type="linenumber">1393</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -3176,7 +3190,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">978</context> | ||||
|           <context context-type="linenumber">981</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -6654,7 +6668,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1389</context> | ||||
|           <context context-type="linenumber">1392</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6490688569532630280" datatype="html"> | ||||
| @@ -7018,53 +7032,53 @@ | ||||
|         <source>Error retrieving metadata</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">666</context> | ||||
|           <context context-type="linenumber">669</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3456881259945295697" datatype="html"> | ||||
|         <source>Error retrieving suggestions.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">695</context> | ||||
|           <context context-type="linenumber">698</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2194092841814123758" datatype="html"> | ||||
|         <source>Document "<x id="PH" equiv-text="newValues.title"/>" saved successfully.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">867</context> | ||||
|           <context context-type="linenumber">870</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">891</context> | ||||
|           <context context-type="linenumber">894</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6626387786259219838" datatype="html"> | ||||
|         <source>Error saving document "<x id="PH" equiv-text="this.document.title"/>"</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">897</context> | ||||
|           <context context-type="linenumber">900</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="448882439049417053" datatype="html"> | ||||
|         <source>Error saving document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">947</context> | ||||
|           <context context-type="linenumber">950</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8410796510716511826" datatype="html"> | ||||
|         <source>Do you really want to move the document "<x id="PH" equiv-text="this.document.title"/>" to the trash?</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">979</context> | ||||
|           <context context-type="linenumber">982</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="282586936710748252" datatype="html"> | ||||
|         <source>Documents can be restored prior to permanent deletion.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">980</context> | ||||
|           <context context-type="linenumber">983</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7075,7 +7089,7 @@ | ||||
|         <source>Move to trash</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">982</context> | ||||
|           <context context-type="linenumber">985</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7086,14 +7100,14 @@ | ||||
|         <source>Error deleting document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1001</context> | ||||
|           <context context-type="linenumber">1004</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="619486176823357521" datatype="html"> | ||||
|         <source>Reprocess confirm</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1021</context> | ||||
|           <context context-type="linenumber">1024</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7104,81 +7118,81 @@ | ||||
|         <source>This operation will permanently recreate the archive file for this document.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1022</context> | ||||
|           <context context-type="linenumber">1025</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="302054111564709516" datatype="html"> | ||||
|         <source>The archive file will be re-generated with the current settings.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1023</context> | ||||
|           <context context-type="linenumber">1026</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8251197608401006898" datatype="html"> | ||||
|         <source>Reprocess operation for "<x id="PH" equiv-text="this.document.title"/>" will begin in the background. Close and re-open or reload this document after the operation has completed to see new content.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1033</context> | ||||
|           <context context-type="linenumber">1036</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4409560272830824468" datatype="html"> | ||||
|         <source>Error executing operation</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1044</context> | ||||
|           <context context-type="linenumber">1047</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6030453331794586802" datatype="html"> | ||||
|         <source>Error downloading document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1093</context> | ||||
|           <context context-type="linenumber">1096</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4458954481601077369" datatype="html"> | ||||
|         <source>Page Fit</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1170</context> | ||||
|           <context context-type="linenumber">1173</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4663705961777238777" datatype="html"> | ||||
|         <source>PDF edit operation for "<x id="PH" equiv-text="this.document.title"/>" will begin in the background.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1408</context> | ||||
|           <context context-type="linenumber">1411</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9043972994040261999" datatype="html"> | ||||
|         <source>Error executing PDF edit operation</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1420</context> | ||||
|           <context context-type="linenumber">1423</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3740891324955700797" datatype="html"> | ||||
|         <source>Print failed.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1452</context> | ||||
|           <context context-type="linenumber">1455</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6457245677384603573" datatype="html"> | ||||
|         <source>Error loading document for printing.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1460</context> | ||||
|           <context context-type="linenumber">1463</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6085793215710522488" datatype="html"> | ||||
|         <source>An error occurred loading tiff: <x id="PH" equiv-text="err.toString()"/></source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1525</context> | ||||
|           <context context-type="linenumber">1528</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1529</context> | ||||
|           <context context-type="linenumber">1532</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4958946940233632319" datatype="html"> | ||||
|   | ||||
| @@ -16,6 +16,7 @@ import { | ||||
|   NgbNavItem, | ||||
| } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { throwError } from 'rxjs' | ||||
| import { routes } from 'src/app/app-routing.module' | ||||
| import { | ||||
|   PaperlessTask, | ||||
| @@ -28,6 +29,7 @@ import { PermissionsGuard } from 'src/app/guards/permissions.guard' | ||||
| import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' | ||||
| import { PermissionsService } from 'src/app/services/permissions.service' | ||||
| import { TasksService } from 'src/app/services/tasks.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { environment } from 'src/environments/environment' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { PageHeaderComponent } from '../../common/page-header/page-header.component' | ||||
| @@ -123,6 +125,7 @@ describe('TasksComponent', () => { | ||||
|   let router: Router | ||||
|   let httpTestingController: HttpTestingController | ||||
|   let reloadSpy | ||||
|   let toastService: ToastService | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @@ -157,6 +160,7 @@ describe('TasksComponent', () => { | ||||
|     httpTestingController = TestBed.inject(HttpTestingController) | ||||
|     modalService = TestBed.inject(NgbModal) | ||||
|     router = TestBed.inject(Router) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     fixture = TestBed.createComponent(TasksComponent) | ||||
|     component = fixture.componentInstance | ||||
|     jest.useFakeTimers() | ||||
| @@ -249,6 +253,42 @@ describe('TasksComponent', () => { | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(selected) | ||||
|   }) | ||||
|  | ||||
|   it('should show an error and re-enable modal buttons when dismissing multiple tasks fails', () => { | ||||
|     component.selectedTasks = new Set([tasks[0].id, tasks[1].id]) | ||||
|     const error = new Error('dismiss failed') | ||||
|     const toastSpy = jest.spyOn(toastService, 'showError') | ||||
|     const dismissSpy = jest | ||||
|       .spyOn(tasksService, 'dismissTasks') | ||||
|       .mockReturnValue(throwError(() => error)) | ||||
|  | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
|  | ||||
|     component.dismissTasks() | ||||
|     expect(modal).not.toBeUndefined() | ||||
|  | ||||
|     modal.componentInstance.confirmClicked.emit() | ||||
|  | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(new Set([tasks[0].id, tasks[1].id])) | ||||
|     expect(toastSpy).toHaveBeenCalledWith('Error dismissing tasks', error) | ||||
|     expect(modal.componentInstance.buttonsEnabled).toBe(true) | ||||
|     expect(component.selectedTasks.size).toBe(0) | ||||
|   }) | ||||
|  | ||||
|   it('should show an error when dismissing a single task fails', () => { | ||||
|     const error = new Error('dismiss failed') | ||||
|     const toastSpy = jest.spyOn(toastService, 'showError') | ||||
|     const dismissSpy = jest | ||||
|       .spyOn(tasksService, 'dismissTasks') | ||||
|       .mockReturnValue(throwError(() => error)) | ||||
|  | ||||
|     component.dismissTask(tasks[0]) | ||||
|  | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(new Set([tasks[0].id])) | ||||
|     expect(toastSpy).toHaveBeenCalledWith('Error dismissing task', error) | ||||
|     expect(component.selectedTasks.size).toBe(0) | ||||
|   }) | ||||
|  | ||||
|   it('should support dismiss all tasks', () => { | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
|   | ||||
| @@ -24,6 +24,7 @@ import { PaperlessTask } from 'src/app/data/paperless-task' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' | ||||
| import { TasksService } from 'src/app/services/tasks.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { PageHeaderComponent } from '../../common/page-header/page-header.component' | ||||
| import { LoadingComponentWithPermissions } from '../../loading-component/loading.component' | ||||
| @@ -72,6 +73,7 @@ export class TasksComponent | ||||
|   tasksService = inject(TasksService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private readonly router = inject(Router) | ||||
|   private readonly toastService = inject(ToastService) | ||||
|  | ||||
|   public activeTab: TaskTab | ||||
|   public selectedTasks: Set<number> = new Set() | ||||
| @@ -154,11 +156,19 @@ export class TasksComponent | ||||
|       modal.componentInstance.confirmClicked.pipe(first()).subscribe(() => { | ||||
|         modal.componentInstance.buttonsEnabled = false | ||||
|         modal.close() | ||||
|         this.tasksService.dismissTasks(tasks) | ||||
|         this.tasksService.dismissTasks(tasks).subscribe({ | ||||
|           error: (e) => { | ||||
|             this.toastService.showError($localize`Error dismissing tasks`, e) | ||||
|             modal.componentInstance.buttonsEnabled = true | ||||
|           }, | ||||
|         }) | ||||
|         this.clearSelection() | ||||
|       }) | ||||
|     } else { | ||||
|       this.tasksService.dismissTasks(tasks) | ||||
|       this.tasksService.dismissTasks(tasks).subscribe({ | ||||
|         error: (e) => | ||||
|           this.toastService.showError($localize`Error dismissing task`, e), | ||||
|       }) | ||||
|       this.clearSelection() | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -41,9 +41,3 @@ | ||||
|     min-width: 140px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .btn-group-xs { | ||||
|   > .btn { | ||||
|     border-radius: 0.15rem; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,10 @@ | ||||
| <div class="row pt-3 pb-3 pb-md-2 align-items-center"> | ||||
|   <div class="col-md text-truncate"> | ||||
|     <h3 class="text-truncate" style="line-height: 1.4"> | ||||
|     <h3 class="text-truncate d-flex align-items-center" style="line-height: 1.4"> | ||||
|       {{title}} | ||||
|       @if (id) { | ||||
|         <span class="badge bg-primary text-primary-text-contrast ms-2 small fs-normal">ID: {{id}}</span> | ||||
|       } | ||||
|       @if (subTitle) { | ||||
|         <span class="h6 mb-0 d-block d-md-inline fw-normal ms-md-3 text-truncate" style="line-height: 1.4">{{subTitle}}</span> | ||||
|       } | ||||
|   | ||||
| @@ -1,5 +1,10 @@ | ||||
| h3 { | ||||
|     min-height: calc(1.325rem + 0.9vw); | ||||
|  | ||||
|     .badge { | ||||
|         font-size: 0.65rem; | ||||
|         line-height: 1; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @media (min-width: 1200px) { | ||||
|   | ||||
| @@ -26,6 +26,9 @@ export class PageHeaderComponent { | ||||
|     return this._title | ||||
|   } | ||||
|  | ||||
|   @Input() | ||||
|   id: number | ||||
|  | ||||
|   @Input() | ||||
|   subTitle: string = '' | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| <pngx-page-header [(title)]="title"> | ||||
| <pngx-page-header [(title)]="title" [id]="documentId"> | ||||
|   @if (archiveContentRenderType === ContentRenderType.PDF && !useNativePdfViewer) { | ||||
|     @if (previewNumPages) { | ||||
|       <div class="input-group input-group-sm d-none d-md-flex"> | ||||
|   | ||||
| @@ -1212,7 +1212,7 @@ describe('DocumentDetailComponent', () => { | ||||
|   it('should support keyboard shortcuts', () => { | ||||
|     initNormally() | ||||
|  | ||||
|     jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     const hasNextSpy = jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     const nextSpy = jest.spyOn(component, 'nextDoc') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 'arrowright', ctrlKey: true }) | ||||
| @@ -1226,21 +1226,32 @@ describe('DocumentDetailComponent', () => { | ||||
|     ) | ||||
|     expect(prevSpy).toHaveBeenCalled() | ||||
|  | ||||
|     jest.spyOn(openDocumentsService, 'isDirty').mockReturnValue(true) | ||||
|     const isDirtySpy = jest | ||||
|       .spyOn(openDocumentsService, 'isDirty') | ||||
|       .mockReturnValue(true) | ||||
|     const saveSpy = jest.spyOn(component, 'save') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true }) | ||||
|     ) | ||||
|     expect(saveSpy).toHaveBeenCalled() | ||||
|  | ||||
|     jest.spyOn(openDocumentsService, 'isDirty').mockReturnValue(true) | ||||
|     jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     hasNextSpy.mockReturnValue(true) | ||||
|     const saveNextSpy = jest.spyOn(component, 'saveEditNext') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true, shiftKey: true }) | ||||
|     ) | ||||
|     expect(saveNextSpy).toHaveBeenCalled() | ||||
|  | ||||
|     saveSpy.mockClear() | ||||
|     saveNextSpy.mockClear() | ||||
|     isDirtySpy.mockReturnValue(true) | ||||
|     hasNextSpy.mockReturnValue(false) | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true, shiftKey: true }) | ||||
|     ) | ||||
|     expect(saveNextSpy).not.toHaveBeenCalled() | ||||
|     expect(saveSpy).toHaveBeenCalledWith(true) | ||||
|  | ||||
|     const closeSpy = jest.spyOn(component, 'close') | ||||
|     document.dispatchEvent(new KeyboardEvent('keydown', { key: 'escape' })) | ||||
|     expect(closeSpy).toHaveBeenCalled() | ||||
|   | ||||
| @@ -615,7 +615,10 @@ export class DocumentDetailComponent | ||||
|       }) | ||||
|       .pipe(takeUntil(this.unsubscribeNotifier)) | ||||
|       .subscribe(() => { | ||||
|         if (this.openDocumentService.isDirty(this.document)) this.saveEditNext() | ||||
|         if (this.openDocumentService.isDirty(this.document)) { | ||||
|           if (this.hasNext()) this.saveEditNext() | ||||
|           else this.save(true) | ||||
|         } | ||||
|       }) | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -51,7 +51,7 @@ describe('TasksService', () => { | ||||
|   }) | ||||
|  | ||||
|   it('calls acknowledge_tasks api endpoint on dismiss and reloads', () => { | ||||
|     tasksService.dismissTasks(new Set([1, 2, 3])) | ||||
|     tasksService.dismissTasks(new Set([1, 2, 3])).subscribe() | ||||
|     const req = httpTestingController.expectOne( | ||||
|       `${environment.apiBaseUrl}tasks/acknowledge/` | ||||
|     ) | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { HttpClient } from '@angular/common/http' | ||||
| import { Injectable, inject } from '@angular/core' | ||||
| import { Observable, Subject } from 'rxjs' | ||||
| import { first, takeUntil } from 'rxjs/operators' | ||||
| import { first, takeUntil, tap } from 'rxjs/operators' | ||||
| import { | ||||
|   PaperlessTask, | ||||
|   PaperlessTaskName, | ||||
| @@ -68,14 +68,17 @@ export class TasksService { | ||||
|   } | ||||
|  | ||||
|   public dismissTasks(task_ids: Set<number>) { | ||||
|     this.http | ||||
|     return this.http | ||||
|       .post(`${this.baseUrl}tasks/acknowledge/`, { | ||||
|         tasks: [...task_ids], | ||||
|       }) | ||||
|       .pipe(first()) | ||||
|       .subscribe((r) => { | ||||
|         this.reload() | ||||
|       }) | ||||
|       .pipe( | ||||
|         first(), | ||||
|         takeUntil(this.unsubscribeNotifer), | ||||
|         tap(() => { | ||||
|           this.reload() | ||||
|         }) | ||||
|       ) | ||||
|   } | ||||
|  | ||||
|   public cancelPending(): void { | ||||
|   | ||||
| @@ -161,3 +161,21 @@ class PaperlessNotePermissions(BasePermission): | ||||
|         perms = self.perms_map[request.method] | ||||
|  | ||||
|         return request.user.has_perms(perms) | ||||
|  | ||||
|  | ||||
| class AcknowledgeTasksPermissions(BasePermission): | ||||
|     """ | ||||
|     Permissions class that checks for model permissions for acknowledging tasks. | ||||
|     """ | ||||
|  | ||||
|     perms_map = { | ||||
|         "POST": ["documents.change_paperlesstask"], | ||||
|     } | ||||
|  | ||||
|     def has_permission(self, request, view): | ||||
|         if not request.user or not request.user.is_authenticated:  # pragma: no cover | ||||
|             return False | ||||
|  | ||||
|         perms = self.perms_map.get(request.method, []) | ||||
|  | ||||
|         return request.user.has_perms(perms) | ||||
|   | ||||
| @@ -76,7 +76,9 @@ def check_sanity(*, progress=False, scheduled=True) -> SanityCheckMessages: | ||||
|     messages = SanityCheckMessages() | ||||
|  | ||||
|     present_files = { | ||||
|         x.resolve() for x in Path(settings.MEDIA_ROOT).glob("**/*") if not x.is_dir() | ||||
|         x.resolve() | ||||
|         for x in Path(settings.MEDIA_ROOT).glob("**/*") | ||||
|         if not x.is_dir() and x.name not in settings.IGNORABLE_FILES | ||||
|     } | ||||
|  | ||||
|     lockfile = Path(settings.MEDIA_LOCK).resolve() | ||||
|   | ||||
| @@ -135,6 +135,44 @@ class TestTasks(DirectoriesMixin, APITestCase): | ||||
|         response = self.client.get(self.ENDPOINT + "?acknowledged=false") | ||||
|         self.assertEqual(len(response.data), 0) | ||||
|  | ||||
|     def test_acknowledge_tasks_requires_change_permission(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - A regular user initially without change permissions | ||||
|             - A regular user with change permissions | ||||
|         WHEN: | ||||
|             - API call is made to acknowledge tasks | ||||
|         THEN: | ||||
|             - The first user is forbidden from acknowledging tasks | ||||
|             - The second user is allowed to acknowledge tasks | ||||
|         """ | ||||
|         regular_user = User.objects.create_user(username="test") | ||||
|         self.client.force_authenticate(user=regular_user) | ||||
|  | ||||
|         task = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
|  | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT + "acknowledge/", | ||||
|             {"tasks": [task.id]}, | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_403_FORBIDDEN) | ||||
|  | ||||
|         regular_user2 = User.objects.create_user(username="test2") | ||||
|         regular_user2.user_permissions.add( | ||||
|             Permission.objects.get(codename="change_paperlesstask"), | ||||
|         ) | ||||
|         regular_user2.save() | ||||
|         self.client.force_authenticate(user=regular_user2) | ||||
|  | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT + "acknowledge/", | ||||
|             {"tasks": [task.id]}, | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|  | ||||
|     def test_tasks_owner_aware(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|   | ||||
| @@ -169,6 +169,13 @@ class TestSanityCheck(DirectoriesMixin, TestCase): | ||||
|         messages = check_sanity() | ||||
|         self.assertFalse(messages.has_warning) | ||||
|  | ||||
|     def test_ignore_ignorable_files(self): | ||||
|         self.make_test_data() | ||||
|         Path(self.dirs.media_dir, ".DS_Store").touch() | ||||
|         Path(self.dirs.media_dir, "desktop.ini").touch() | ||||
|         messages = check_sanity() | ||||
|         self.assertFalse(messages.has_warning) | ||||
|  | ||||
|     def test_archive_filename_no_checksum(self): | ||||
|         doc = self.make_test_data() | ||||
|         doc.archive_checksum = None | ||||
|   | ||||
| @@ -136,6 +136,7 @@ from documents.models import WorkflowAction | ||||
| from documents.models import WorkflowTrigger | ||||
| from documents.parsers import get_parser_class_for_mime_type | ||||
| from documents.parsers import parse_date_generator | ||||
| from documents.permissions import AcknowledgeTasksPermissions | ||||
| from documents.permissions import PaperlessAdminPermissions | ||||
| from documents.permissions import PaperlessNotePermissions | ||||
| from documents.permissions import PaperlessObjectPermissions | ||||
| @@ -2487,7 +2488,11 @@ class TasksViewSet(ReadOnlyModelViewSet): | ||||
|             queryset = PaperlessTask.objects.filter(task_id=task_id) | ||||
|         return queryset | ||||
|  | ||||
|     @action(methods=["post"], detail=False) | ||||
|     @action( | ||||
|         methods=["post"], | ||||
|         detail=False, | ||||
|         permission_classes=[IsAuthenticated, AcknowledgeTasksPermissions], | ||||
|     ) | ||||
|     def acknowledge(self, request): | ||||
|         serializer = AcknowledgeTasksViewSerializer(data=request.data) | ||||
|         serializer.is_valid(raise_exception=True) | ||||
|   | ||||
| @@ -1003,6 +1003,18 @@ THREADS_PER_WORKER = os.getenv( | ||||
| # Paperless Specific Settings                                                 # | ||||
| ############################################################################### | ||||
|  | ||||
| IGNORABLE_FILES: Final[list[str]] = [ | ||||
|     ".DS_Store", | ||||
|     ".DS_STORE", | ||||
|     "._*", | ||||
|     ".stfolder/*", | ||||
|     ".stversions/*", | ||||
|     ".localized/*", | ||||
|     "desktop.ini", | ||||
|     "@eaDir/*", | ||||
|     "Thumbs.db", | ||||
| ] | ||||
|  | ||||
| CONSUMER_POLLING = int(os.getenv("PAPERLESS_CONSUMER_POLLING", 0)) | ||||
|  | ||||
| CONSUMER_POLLING_DELAY = int(os.getenv("PAPERLESS_CONSUMER_POLLING_DELAY", 5)) | ||||
| @@ -1025,7 +1037,7 @@ CONSUMER_IGNORE_PATTERNS = list( | ||||
|     json.loads( | ||||
|         os.getenv( | ||||
|             "PAPERLESS_CONSUMER_IGNORE_PATTERNS", | ||||
|             '[".DS_Store", ".DS_STORE", "._*", ".stfolder/*", ".stversions/*", ".localized/*", "desktop.ini", "@eaDir/*", "Thumbs.db"]', | ||||
|             json.dumps(IGNORABLE_FILES), | ||||
|         ), | ||||
|     ), | ||||
| ) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user