diff --git a/docs/api.md b/docs/api.md index ed00ab276..c2a83938d 100644 --- a/docs/api.md +++ b/docs/api.md @@ -376,6 +376,18 @@ The following methods are supported: - `"merge": true or false` (defaults to false) - The `merge` flag determines if the supplied permissions will overwrite all existing permissions (including removing them) or be merged with existing permissions. +- `merge` + - No additional `parameters` required. + - The ordering of the merged document is determined by the list of IDs. + - Optional `parameters`: + - `"metadata_document_id": DOC_ID` apply metadata (tags, correspondent, etc.) from this document to the merged document. +- `split` + - Requires `parameters`: + - `"pages": [..]` The list should be a list of pages and/or a ranges, separated by commas e.g. `"[1,2-3,4,5-7]"` + - The split operation only accepts a single document. +- `rotate` + - Requires `parameters`: + - `"degrees": DEGREES`. Must be an integer i.e. 90, 180, 270 ### Objects diff --git a/docs/usage.md b/docs/usage.md index 4db1c94de..e5ef3b2c0 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -456,6 +456,18 @@ Paperless-ngx added the ability to create shareable links to files in version 2. If your paperless-ngx instance is behind a reverse-proxy you may want to create an exception to bypass any authentication layers that are part of your setup in order to make links truly publicly-accessible. Of course, do so with caution. +## PDF Actions + +Paperless-ngx supports 3 basic editing operations for PDFs (these operations cannot be performed on non-PDF files): + +- Merging documents: available when selecting multiple documents for 'bulk editing' +- Rotating documents: available when selecting multiple documents for 'bulk editing' and from an individual document's details page. +- Splitting documents: available from an individual document's details page + +!!! important + + Note that rotation alters the Paperless-ngx original file. + ## Best practices {#basic-searching} Paperless offers a couple tools that help you organize your document diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index ae34385a1..95adeb28b 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -287,7 +287,7 @@ src/app/components/document-detail/document-detail.component.html - 81 + 89 @@ -447,7 +447,7 @@ src/app/components/document-detail/document-detail.component.html - 312 + 320 @@ -506,7 +506,7 @@ src/app/components/document-detail/document-detail.component.html - 304 + 312 src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -636,7 +636,7 @@ src/app/components/document-detail/document-detail.component.html - 321 + 329 src/app/components/document-list/document-list.component.html @@ -977,7 +977,7 @@ src/app/components/document-detail/document-detail.component.html - 280 + 288 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -1464,7 +1464,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 140 + 142 src/app/components/manage/custom-fields/custom-fields.component.html @@ -2032,15 +2032,15 @@ src/app/components/document-detail/document-detail.component.ts - 766 + 768 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 580 + 591 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 619 + 630 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2075,11 +2075,27 @@ src/app/components/document-detail/document-detail.component.ts - 768 + 770 + + + src/app/components/document-detail/document-detail.component.ts + 1052 + + + src/app/components/document-detail/document-detail.component.ts + 1090 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 621 + 632 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 665 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 684 src/app/components/manage/custom-fields/custom-fields.component.ts @@ -2522,19 +2538,19 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 356 + 367 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 396 + 407 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 434 + 445 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 472 + 483 @@ -2604,6 +2620,74 @@ 20 + + Documents: + + src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html + 9 + + + + Use metadata from: + + src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html + 22 + + + + Regenerate all metadata + + src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html + 24 + + + + Note that only PDFs will be included. + + src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html + 30 + + + + Note that only PDFs will be rotated. + + src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html + 35 + + + + Page + + src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html + 11 + + + src/app/components/document-detail/document-detail.component.html + 4 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 11 + + + + of + + src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html + 13 + + + src/app/components/document-detail/document-detail.component.html + 6,7 + + + + Add Split + + src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html + 28 + + Create New Field @@ -4709,7 +4793,7 @@ src/app/components/document-detail/document-detail.component.html - 94 + 102 src/app/components/document-list/document-list.component.html @@ -4732,7 +4816,7 @@ src/app/components/document-detail/document-detail.component.html - 98 + 106 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -4770,7 +4854,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 106 + 114 src/app/components/document-list/document-card-large/document-card-large.component.html @@ -4903,7 +4987,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 301 + 312 this string is used to separate processing, failed and added on the file upload widget @@ -4949,24 +5033,6 @@ 1 - - Page - - src/app/components/document-detail/document-detail.component.html - 4 - - - src/app/components/document-list/bulk-editor/bulk-editor.component.html - 11 - - - - of - - src/app/components/document-detail/document-detail.component.html - 6,7 - - - @@ -4996,7 +5062,7 @@ src/app/components/document-list/bulk-editor/bulk-editor.component.html - 91 + 92 @@ -5010,11 +5076,33 @@ 50 + + Split + + src/app/components/document-detail/document-detail.component.html + 55 + + + + Rotate + + src/app/components/document-detail/document-detail.component.html + 59 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 95 + + Close src/app/components/document-detail/document-detail.component.html - 75 + 83 + + + src/app/components/document-detail/document-detail.component.ts + 1108 src/app/guards/dirty-saved-view.guard.ts @@ -5025,35 +5113,35 @@ Previous src/app/components/document-detail/document-detail.component.html - 78 + 86 Details src/app/components/document-detail/document-detail.component.html - 91 + 99 Archive serial number src/app/components/document-detail/document-detail.component.html - 95 + 103 Date created src/app/components/document-detail/document-detail.component.html - 96 + 104 Document type src/app/components/document-detail/document-detail.component.html - 100 + 108 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -5076,7 +5164,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 102 + 110 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -5095,21 +5183,21 @@ Default src/app/components/document-detail/document-detail.component.html - 103 + 111 Content src/app/components/document-detail/document-detail.component.html - 187 + 195 Metadata src/app/components/document-detail/document-detail.component.html - 196 + 204 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -5120,190 +5208,190 @@ Date modified src/app/components/document-detail/document-detail.component.html - 203 + 211 Date added src/app/components/document-detail/document-detail.component.html - 207 + 215 Media filename src/app/components/document-detail/document-detail.component.html - 211 + 219 Original filename src/app/components/document-detail/document-detail.component.html - 215 + 223 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 219 + 227 Original file size src/app/components/document-detail/document-detail.component.html - 223 + 231 Original mime type src/app/components/document-detail/document-detail.component.html - 227 + 235 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 232 + 240 Archive file size src/app/components/document-detail/document-detail.component.html - 238 + 246 Original document metadata src/app/components/document-detail/document-detail.component.html - 247 + 255 Archived document metadata src/app/components/document-detail/document-detail.component.html - 250 + 258 Preview src/app/components/document-detail/document-detail.component.html - 257 + 265 Notes src/app/components/document-detail/document-detail.component.html - 269,272 + 277,280 Save & next src/app/components/document-detail/document-detail.component.html - 306 + 314 Save & close src/app/components/document-detail/document-detail.component.html - 309 + 317 Enter Password src/app/components/document-detail/document-detail.component.html - 360 + 368 An error occurred loading content: src/app/components/document-detail/document-detail.component.ts - 325,327 + 327,329 Document changes detected src/app/components/document-detail/document-detail.component.ts - 348 + 350 The version of this document in your browser session appears older than the existing version. src/app/components/document-detail/document-detail.component.ts - 349 + 351 Saving the document here may overwrite other changes that were made. To restore the existing version, discard your changes or close the document. src/app/components/document-detail/document-detail.component.ts - 350 + 352 Ok src/app/components/document-detail/document-detail.component.ts - 352 + 354 Error retrieving metadata src/app/components/document-detail/document-detail.component.ts - 492 + 494 Error retrieving suggestions. src/app/components/document-detail/document-detail.component.ts - 517 + 519 Document saved successfully. src/app/components/document-detail/document-detail.component.ts - 638 + 640 src/app/components/document-detail/document-detail.component.ts - 649 + 651 Error saving document src/app/components/document-detail/document-detail.component.ts - 653 + 655 src/app/components/document-detail/document-detail.component.ts - 694 + 696 Confirm delete src/app/components/document-detail/document-detail.component.ts - 721 + 723 src/app/components/manage/management-list/management-list.component.ts @@ -5318,67 +5406,138 @@ Do you really want to delete document ""? src/app/components/document-detail/document-detail.component.ts - 722 + 724 The files for this document will be deleted permanently. This operation cannot be undone. src/app/components/document-detail/document-detail.component.ts - 723 + 725 Delete document src/app/components/document-detail/document-detail.component.ts - 725 + 727 Error deleting document src/app/components/document-detail/document-detail.component.ts - 744 + 746 Redo OCR confirm src/app/components/document-detail/document-detail.component.ts - 764 + 766 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 617 + 628 This operation will permanently redo OCR for this document. src/app/components/document-detail/document-detail.component.ts - 765 + 767 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 - 776 + 778 Error executing operation src/app/components/document-detail/document-detail.component.ts - 787 + 789 Page Fit src/app/components/document-detail/document-detail.component.ts - 856 + 858 + + + + Split confirm + + src/app/components/document-detail/document-detail.component.ts + 1050 + + + + This operation will split the selected document(s) into new documents. + + src/app/components/document-detail/document-detail.component.ts + 1051 + + + + Split operation will begin in the background. + + src/app/components/document-detail/document-detail.component.ts + 1066 + + + + Error executing split operation + + src/app/components/document-detail/document-detail.component.ts + 1075 + + + + Rotate confirm + + src/app/components/document-detail/document-detail.component.ts + 1087 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 661 + + + + This operation will permanently rotate the current document. + + src/app/components/document-detail/document-detail.component.ts + 1088 + + + + This will alter the original copy. + + src/app/components/document-detail/document-detail.component.ts + 1089 + + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 663 + + + + Rotation will begin in the background. Close and re-open the document after the operation has completed to see the changes. + + src/app/components/document-detail/document-detail.component.ts + 1105 + + + + Error executing rotate operation + + src/app/components/document-detail/document-detail.component.ts + 1117 @@ -5439,57 +5598,64 @@ 65 + + Merge + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 98 + + Include: src/app/components/document-list/bulk-editor/bulk-editor.component.html - 112 + 120 - - Archived files + + Archived files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 116,118 + 124 - - Original files + + Original files src/app/components/document-list/bulk-editor/bulk-editor.component.html - 122,124 + 128 - - Use formatted filename + + Use formatted filename src/app/components/document-list/bulk-editor/bulk-editor.component.html - 129,131 + 133 Error executing bulk operation src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 218 + 229 "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 293 + 304 src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 299 + 310 "" and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 295 + 306 This is for messages like 'modify "tag1" and "tag2"' @@ -5497,7 +5663,7 @@ and "" src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 303,305 + 314,316 this is for messages like 'modify "tag1", "tag2" and "tag3"' @@ -5505,14 +5671,14 @@ Confirm tags assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 320 + 331 This operation will add the tag "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 326 + 337 @@ -5521,14 +5687,14 @@ )"/> to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 331,333 + 342,344 This operation will remove the tag "" from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 339 + 350 @@ -5537,7 +5703,7 @@ )"/> from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 344,346 + 355,357 @@ -5548,98 +5714,126 @@ )"/> on selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 348,352 + 359,363 Confirm correspondent assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 389 + 400 This operation will assign the correspondent "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 391 + 402 This operation will remove the correspondent from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 393 + 404 Confirm document type assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 427 + 438 This operation will assign the document type "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 429 + 440 This operation will remove the document type from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 431 + 442 Confirm storage path assignment src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 465 + 476 This operation will assign the storage path "" to selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 467 + 478 This operation will remove the storage path from selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 469 + 480 Delete confirm src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 578 + 589 This operation will permanently delete selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 579 + 590 Delete document(s) src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 582 + 593 This operation will permanently redo OCR for selected document(s). src/app/components/document-list/bulk-editor/bulk-editor.component.ts - 618 + 629 + + + + This operation will permanently rotate selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 662 + + + + Merge confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 682 + + + + This operation will merge selected documents into a new document. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 683 + + + + Merged document will be queued for consumption. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 696 diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 568b2bc0e..f990122dd 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -116,9 +116,13 @@ import { ConfirmButtonComponent } from './components/common/confirm-button/confi import { MonetaryComponent } from './components/common/input/monetary/monetary.component' import { SystemStatusDialogComponent } from './components/common/system-status-dialog/system-status-dialog.component' import { NgxFilesizeModule } from 'ngx-filesize' +import { RotateConfirmDialogComponent } from './components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' +import { MergeConfirmDialogComponent } from './components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component' +import { SplitConfirmDialogComponent } from './components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component' import { airplane, archive, + arrowClockwise, arrowCounterclockwise, arrowDown, arrowLeft, @@ -127,6 +131,7 @@ import { arrowRightShort, arrowUpRight, asterisk, + bodyText, boxArrowUp, boxArrowUpRight, boxes, @@ -174,6 +179,7 @@ import { hddStack, house, infoCircle, + journals, link, listTask, listUl, @@ -188,6 +194,7 @@ import { plus, plusCircle, questionCircle, + scissors, search, slashCircle, sliders2Vertical, @@ -209,6 +216,7 @@ import { const icons = { airplane, archive, + arrowClockwise, arrowCounterclockwise, arrowDown, arrowLeft, @@ -217,6 +225,7 @@ const icons = { arrowRightShort, arrowUpRight, asterisk, + bodyText, boxArrowUp, boxArrowUpRight, boxes, @@ -264,6 +273,7 @@ const icons = { hddStack, house, infoCircle, + journals, link, listTask, listUl, @@ -278,6 +288,7 @@ const icons = { plus, plusCircle, questionCircle, + scissors, search, slashCircle, sliders2Vertical, @@ -458,6 +469,9 @@ function initializeApp(settings: SettingsService) { ConfirmButtonComponent, MonetaryComponent, SystemStatusDialogComponent, + RotateConfirmDialogComponent, + MergeConfirmDialogComponent, + SplitConfirmDialogComponent, ], imports: [ BrowserModule, diff --git a/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html new file mode 100644 index 000000000..0da291c94 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.html @@ -0,0 +1,39 @@ + + + diff --git a/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.scss b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.scss new file mode 100644 index 000000000..c780e5a35 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.scss @@ -0,0 +1,3 @@ +.list-group-item { + cursor: move; +} diff --git a/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.spec.ts b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.spec.ts new file mode 100644 index 000000000..8b9bf3898 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.spec.ts @@ -0,0 +1,73 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { MergeConfirmDialogComponent } from './merge-confirm-dialog.component' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { of } from 'rxjs' +import { DocumentService } from 'src/app/services/rest/document.service' +import { FormsModule, ReactiveFormsModule } from '@angular/forms' + +describe('MergeConfirmDialogComponent', () => { + let component: MergeConfirmDialogComponent + let fixture: ComponentFixture + let documentService: DocumentService + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [MergeConfirmDialogComponent], + providers: [NgbActiveModal], + imports: [ + HttpClientTestingModule, + NgxBootstrapIconsModule.pick(allIcons), + ReactiveFormsModule, + FormsModule, + ], + }).compileComponents() + + fixture = TestBed.createComponent(MergeConfirmDialogComponent) + documentService = TestBed.inject(DocumentService) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should fetch documents on ngOnInit', () => { + const documents = [ + { id: 1, name: 'Document 1' }, + { id: 2, name: 'Document 2' }, + { id: 3, name: 'Document 3' }, + ] + jest.spyOn(documentService, 'getCachedMany').mockReturnValue(of(documents)) + + component.ngOnInit() + + expect(component.documents).toEqual(documents) + expect(documentService.getCachedMany).toHaveBeenCalledWith( + component.documentIDs + ) + }) + + it('should move documentIDs on drop', () => { + component.documentIDs = [1, 2, 3] + const event = { + previousIndex: 1, + currentIndex: 2, + } + + component.onDrop(event as any) + + expect(component.documentIDs).toEqual([1, 3, 2]) + }) + + it('should get document by ID', () => { + const documents = [ + { id: 1, name: 'Document 1' }, + { id: 2, name: 'Document 2' }, + { id: 3, name: 'Document 3' }, + ] + jest.spyOn(documentService, 'getCachedMany').mockReturnValue(of(documents)) + + component.ngOnInit() + + expect(component.getDocument(2)).toEqual({ id: 2, name: 'Document 2' }) + }) +}) diff --git a/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.ts b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.ts new file mode 100644 index 000000000..fd52459e0 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component.ts @@ -0,0 +1,51 @@ +import { Component, OnInit } from '@angular/core' +import { ConfirmDialogComponent } from '../confirm-dialog.component' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { DocumentService } from 'src/app/services/rest/document.service' +import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop' +import { Subject, takeUntil } from 'rxjs' +import { Document } from 'src/app/data/document' + +@Component({ + selector: 'pngx-merge-confirm-dialog', + templateUrl: './merge-confirm-dialog.component.html', + styleUrl: './merge-confirm-dialog.component.scss', +}) +export class MergeConfirmDialogComponent + extends ConfirmDialogComponent + implements OnInit +{ + public documentIDs: number[] = [] + private _documents: Document[] = [] + get documents(): Document[] { + return this._documents + } + + public metadataDocumentID: number = -1 + + private unsubscribeNotifier: Subject = new Subject() + + constructor( + activeModal: NgbActiveModal, + private documentService: DocumentService + ) { + super(activeModal) + } + + ngOnInit() { + this.documentService + .getCachedMany(this.documentIDs) + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe((documents) => { + this._documents = documents + }) + } + + onDrop(event: CdkDragDrop) { + moveItemInArray(this.documentIDs, event.previousIndex, event.currentIndex) + } + + getDocument(documentID: number): Document { + return this.documents.find((d) => d.id === documentID) + } +} diff --git a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html new file mode 100644 index 000000000..8a6eacef4 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.html @@ -0,0 +1,48 @@ + + + diff --git a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.scss b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.scss new file mode 100644 index 000000000..93e950ac1 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.scss @@ -0,0 +1,3 @@ +img { + transition: all 0.25s ease; +} diff --git a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.spec.ts b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.spec.ts new file mode 100644 index 000000000..d70e73747 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.spec.ts @@ -0,0 +1,60 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' +import { RotateConfirmDialogComponent } from './rotate-confirm-dialog.component' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' + +describe('RotateConfirmDialogComponent', () => { + let component: RotateConfirmDialogComponent + let fixture: ComponentFixture + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [RotateConfirmDialogComponent, SafeHtmlPipe], + providers: [NgbActiveModal, SafeHtmlPipe], + imports: [ + HttpClientTestingModule, + NgxBootstrapIconsModule.pick(allIcons), + ], + }).compileComponents() + + fixture = TestBed.createComponent(RotateConfirmDialogComponent) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should support rotating the image', () => { + component.documentID = 1 + fixture.detectChanges() + component.rotate() + fixture.detectChanges() + expect(component.degrees).toBe(90) + expect(fixture.nativeElement.querySelector('img').style.transform).toBe( + 'rotate(90deg)' + ) + component.rotate() + fixture.detectChanges() + expect(fixture.nativeElement.querySelector('img').style.transform).toBe( + 'rotate(180deg)' + ) + }) + + it('should normalize degrees', () => { + expect(component.degrees).toBe(0) + component.rotate() + expect(component.degrees).toBe(90) + component.rotate() + expect(component.degrees).toBe(180) + component.rotate() + expect(component.degrees).toBe(270) + component.rotate() + expect(component.degrees).toBe(0) + component.rotate() + expect(component.degrees).toBe(90) + component.rotate(false) + expect(component.degrees).toBe(0) + component.rotate(false) + expect(component.degrees).toBe(270) + }) +}) diff --git a/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.ts b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.ts new file mode 100644 index 000000000..7cef2b72e --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component.ts @@ -0,0 +1,34 @@ +import { Component } from '@angular/core' +import { ConfirmDialogComponent } from '../confirm-dialog.component' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { DocumentService } from 'src/app/services/rest/document.service' + +@Component({ + selector: 'pngx-rotate-confirm-dialog', + templateUrl: './rotate-confirm-dialog.component.html', + styleUrl: './rotate-confirm-dialog.component.scss', +}) +export class RotateConfirmDialogComponent extends ConfirmDialogComponent { + public documentID: number + public showPDFNote: boolean = true + + // animation is better if we dont normalize yet + public rotation: number = 0 + + public get degrees(): number { + let degrees = this.rotation % 360 + if (degrees < 0) degrees += 360 + return degrees + } + + constructor( + activeModal: NgbActiveModal, + public documentService: DocumentService + ) { + super(activeModal) + } + + rotate(clockwise: boolean = true) { + this.rotation += clockwise ? 90 : -90 + } +} diff --git a/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html new file mode 100644 index 000000000..36bd7796d --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.html @@ -0,0 +1,55 @@ + + + diff --git a/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.scss b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.scss new file mode 100644 index 000000000..c2fc99d55 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.scss @@ -0,0 +1,9 @@ +.pdf-viewer-container { + background-color: gray; + height: 300px; + + pngx-pdf-viewer { + width: 100%; + height: 100%; + } + } diff --git a/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.spec.ts b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.spec.ts new file mode 100644 index 000000000..b88835895 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.spec.ts @@ -0,0 +1,81 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' + +import { SplitConfirmDialogComponent } from './split-confirm-dialog.component' +import { HttpClientTestingModule } from '@angular/common/http/testing' +import { ReactiveFormsModule, FormsModule } from '@angular/forms' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { DocumentService } from 'src/app/services/rest/document.service' +import { PdfViewerComponent } from '../../pdf-viewer/pdf-viewer.component' + +describe('SplitConfirmDialogComponent', () => { + let component: SplitConfirmDialogComponent + let fixture: ComponentFixture + let documentService: DocumentService + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [SplitConfirmDialogComponent, PdfViewerComponent], + providers: [NgbActiveModal], + imports: [ + HttpClientTestingModule, + NgxBootstrapIconsModule.pick(allIcons), + ReactiveFormsModule, + FormsModule, + ], + }).compileComponents() + + fixture = TestBed.createComponent(SplitConfirmDialogComponent) + documentService = TestBed.inject(DocumentService) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should update pagesString when pages are added', () => { + component.totalPages = 5 + component.page = 2 + component.addSplit() + expect(component.pagesString).toEqual('1-2,3-5') + component.page = 4 + component.addSplit() + expect(component.pagesString).toEqual('1-2,3-4,5') + }) + + it('should update pagesString when pages are removed', () => { + component.totalPages = 5 + component.page = 2 + component.addSplit() + component.page = 4 + component.addSplit() + expect(component.pagesString).toEqual('1-2,3-4,5') + component.removeSplit(0) + expect(component.pagesString).toEqual('1-4,5') + }) + + it('should enable confirm button when pages are added', () => { + component.totalPages = 5 + component.page = 2 + component.addSplit() + expect(component.confirmButtonEnabled).toBeTruthy() + }) + + it('should disable confirm button when all pages are removed', () => { + component.totalPages = 5 + component.page = 2 + component.addSplit() + component.removeSplit(0) + expect(component.confirmButtonEnabled).toBeFalsy() + }) + + it('should not add split if page is the last page', () => { + component.totalPages = 5 + component.page = 5 + component.addSplit() + expect(component.pagesString).toEqual('1-5') + }) + + it('should update totalPages when pdf is loaded', () => { + component.pdfPreviewLoaded({ numPages: 5 } as any) + expect(component.totalPages).toEqual(5) + }) +}) diff --git a/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.ts b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.ts new file mode 100644 index 000000000..42b574b93 --- /dev/null +++ b/src-ui/src/app/components/common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component.ts @@ -0,0 +1,66 @@ +import { Component } from '@angular/core' +import { ConfirmDialogComponent } from '../confirm-dialog.component' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { DocumentService } from 'src/app/services/rest/document.service' +import { PDFDocumentProxy } from '../../pdf-viewer/typings' + +@Component({ + selector: 'pngx-split-confirm-dialog', + templateUrl: './split-confirm-dialog.component.html', + styleUrl: './split-confirm-dialog.component.scss', +}) +export class SplitConfirmDialogComponent extends ConfirmDialogComponent { + public get pagesString(): string { + let pagesStr = '' + + let lastPage = 1 + for (let i = 1; i <= this.totalPages; i++) { + if (this.pages.has(i) || i === this.totalPages) { + if (lastPage === i) { + pagesStr += `${i},` + lastPage = Math.min(i + 1, this.totalPages) + } else { + pagesStr += `${lastPage}-${i},` + lastPage = Math.min(i + 1, this.totalPages) + } + } + } + + return pagesStr.replace(/,$/, '') + } + + private pages: Set = new Set() + + public documentID: number + public page: number = 1 + public totalPages: number + + public get pdfSrc(): string { + return this.documentService.getPreviewUrl(this.documentID) + } + + constructor( + activeModal: NgbActiveModal, + private documentService: DocumentService + ) { + super(activeModal) + this.confirmButtonEnabled = this.pages.size > 0 + } + + pdfPreviewLoaded(pdf: PDFDocumentProxy) { + this.totalPages = pdf.numPages + } + + addSplit() { + if (this.page === this.totalPages) return + this.pages.add(this.page) + this.pages = new Set(Array.from(this.pages).sort()) + this.confirmButtonEnabled = this.pages.size > 0 + } + + removeSplit(i: number) { + let page = Array.from(this.pages)[Math.min(i, this.pages.size - 1)] + this.pages.delete(page) + this.confirmButtonEnabled = this.pages.size > 0 + } +} diff --git a/src-ui/src/app/components/document-detail/document-detail.component.html b/src-ui/src/app/components/document-detail/document-detail.component.html index 5b27a51ac..f3a616fef 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.html +++ b/src-ui/src/app/components/document-detail/document-detail.component.html @@ -44,11 +44,19 @@
+ + + +
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 e0da11a3e..b26ad9024 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 @@ -75,6 +75,8 @@ import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service import { PdfViewerComponent } from '../common/pdf-viewer/pdf-viewer.component' import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' import { environment } from 'src/environments/environment' +import { RotateConfirmDialogComponent } from '../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' +import { SplitConfirmDialogComponent } from '../common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component' const doc: Document = { id: 3, @@ -171,6 +173,8 @@ describe('DocumentDetailComponent', () => { ShareLinksDropdownComponent, CustomFieldsDropdownComponent, PdfViewerComponent, + SplitConfirmDialogComponent, + RotateConfirmDialogComponent, ], providers: [ DocumentTitlePipe, @@ -1070,6 +1074,58 @@ describe('DocumentDetailComponent', () => { ).not.toBeUndefined() }) + it('should support split', () => { + let modal: NgbModalRef + modalService.activeInstances.subscribe((m) => (modal = m[0])) + initNormally() + component.splitDocument() + expect(modal).not.toBeUndefined() + modal.componentInstance.documentID = doc.id + modal.componentInstance.totalPages = 5 + modal.componentInstance.page = 2 + modal.componentInstance.addSplit() + modal.componentInstance.confirm() + let req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + expect(req.request.body).toEqual({ + documents: [doc.id], + method: 'split', + parameters: { pages: '1-2,3-5' }, + }) + req.error(new ProgressEvent('failed')) + modal.componentInstance.confirm() + req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + req.flush(true) + }) + + it('should support rotate', () => { + let modal: NgbModalRef + modalService.activeInstances.subscribe((m) => (modal = m[0])) + initNormally() + component.rotateDocument() + expect(modal).not.toBeUndefined() + modal.componentInstance.documentID = doc.id + modal.componentInstance.rotate() + modal.componentInstance.confirm() + let req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + expect(req.request.body).toEqual({ + documents: [doc.id], + method: 'rotate', + parameters: { degrees: 90 }, + }) + req.error(new ProgressEvent('failed')) + modal.componentInstance.confirm() + req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + req.flush(true) + }) + function initNormally() { jest .spyOn(activatedRoute, 'paramMap', 'get') 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 4eae47615..5c5efdc9f 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 @@ -67,6 +67,8 @@ import { CustomField, CustomFieldDataType } from 'src/app/data/custom-field' import { CustomFieldInstance } from 'src/app/data/custom-field-instance' import { CustomFieldsService } from 'src/app/services/rest/custom-fields.service' import { PDFDocumentProxy } from '../common/pdf-viewer/typings' +import { SplitConfirmDialogComponent } from '../common/confirm-dialog/split-confirm-dialog/split-confirm-dialog.component' +import { RotateConfirmDialogComponent } from '../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' enum DocumentDetailNavIDs { Details = 1, @@ -1040,4 +1042,83 @@ export class DocumentDetailComponent this.updateFormForCustomFields(true) this.documentForm.updateValueAndValidity() } + + splitDocument() { + let modal = this.modalService.open(SplitConfirmDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.title = $localize`Split confirm` + modal.componentInstance.messageBold = $localize`This operation will split the selected document(s) into new documents.` + modal.componentInstance.btnCaption = $localize`Proceed` + modal.componentInstance.documentID = this.document.id + modal.componentInstance.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + modal.componentInstance.buttonsEnabled = false + this.documentsService + .bulkEdit([this.document.id], 'split', { + pages: modal.componentInstance.pagesString, + }) + .pipe(first(), takeUntil(this.unsubscribeNotifier)) + .subscribe({ + next: () => { + this.toastService.showInfo( + $localize`Split operation will begin in the background.` + ) + modal.close() + }, + error: (error) => { + if (modal) { + modal.componentInstance.buttonsEnabled = true + } + this.toastService.showError( + $localize`Error executing split operation`, + error + ) + }, + }) + }) + } + + rotateDocument() { + let modal = this.modalService.open(RotateConfirmDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.title = $localize`Rotate confirm` + modal.componentInstance.messageBold = $localize`This operation will permanently rotate the current document.` + modal.componentInstance.message = $localize`This will alter the original copy.` + modal.componentInstance.btnCaption = $localize`Proceed` + modal.componentInstance.documentID = this.document.id + modal.componentInstance.showPDFNote = false + modal.componentInstance.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + modal.componentInstance.buttonsEnabled = false + this.documentsService + .bulkEdit([this.document.id], 'rotate', { + degrees: modal.componentInstance.degrees, + }) + .pipe(first(), takeUntil(this.unsubscribeNotifier)) + .subscribe({ + next: () => { + this.toastService.show({ + content: $localize`Rotation will begin in the background. Close and re-open the document after the operation has completed to see the changes.`, + delay: 8000, + action: this.close.bind(this), + actionName: $localize`Close`, + }) + modal.close() + }, + error: (error) => { + if (modal) { + modal.componentInstance.buttonsEnabled = true + } + this.toastService.showError( + $localize`Error executing rotate operation`, + error + ) + }, + }) + }) + } } diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html index 686c07bb3..865502569 100644 --- a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -80,18 +80,26 @@ + -
- +
+ + + -
- -
+
diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts index 4da9f36df..e38138df1 100644 --- a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts @@ -52,6 +52,9 @@ import { StoragePath } from 'src/app/data/storage-path' import { CorrespondentEditDialogComponent } from '../../common/edit-dialog/correspondent-edit-dialog/correspondent-edit-dialog.component' import { DocumentTypeEditDialogComponent } from '../../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component' import { StoragePathEditDialogComponent } from '../../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' +import { IsNumberPipe } from 'src/app/pipes/is-number.pipe' +import { RotateConfirmDialogComponent } from '../../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' +import { MergeConfirmDialogComponent } from '../../common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component' const selectionData: SelectionData = { selected_tags: [ @@ -97,6 +100,9 @@ describe('BulkEditorComponent', () => { PermissionsGroupComponent, PermissionsUserComponent, SwitchComponent, + RotateConfirmDialogComponent, + IsNumberPipe, + MergeConfirmDialogComponent, ], providers: [ PermissionsService, @@ -818,6 +824,79 @@ describe('BulkEditorComponent', () => { ) // listAllFilteredIds }) + it('should support rotate', () => { + let modal: NgbModalRef + modalService.activeInstances.subscribe((m) => (modal = m[0])) + jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) + jest + .spyOn(documentListViewService, 'documents', 'get') + .mockReturnValue([{ id: 3 }, { id: 4 }]) + jest + .spyOn(documentListViewService, 'selected', 'get') + .mockReturnValue(new Set([3, 4])) + jest + .spyOn(permissionsService, 'currentUserHasObjectPermissions') + .mockReturnValue(true) + fixture.detectChanges() + component.rotateSelected() + expect(modal).not.toBeUndefined() + modal.componentInstance.rotate() + modal.componentInstance.confirm() + let req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + req.flush(true) + expect(req.request.body).toEqual({ + documents: [3, 4], + method: 'rotate', + parameters: { degrees: 90 }, + }) + httpTestingController.match( + `${environment.apiBaseUrl}documents/?page=1&page_size=50&ordering=-created&truncate_content=true` + ) // list reload + httpTestingController.match( + `${environment.apiBaseUrl}documents/?page=1&page_size=100000&fields=id` + ) // listAllFilteredIds + }) + + it('should support merge', () => { + let modal: NgbModalRef + modalService.activeInstances.subscribe((m) => (modal = m[0])) + jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) + jest + .spyOn(documentListViewService, 'documents', 'get') + .mockReturnValue([{ id: 3 }, { id: 4 }]) + jest + .spyOn(documentService, 'getCachedMany') + .mockReturnValue(of([{ id: 3 }, { id: 4 }])) + jest + .spyOn(documentListViewService, 'selected', 'get') + .mockReturnValue(new Set([3, 4])) + jest + .spyOn(permissionsService, 'currentUserHasObjectPermissions') + .mockReturnValue(true) + fixture.detectChanges() + component.mergeSelected() + expect(modal).not.toBeUndefined() + modal.componentInstance.metadataDocumentID = 3 + modal.componentInstance.confirm() + let req = httpTestingController.expectOne( + `${environment.apiBaseUrl}documents/bulk_edit/` + ) + req.flush(true) + expect(req.request.body).toEqual({ + documents: [3, 4], + method: 'merge', + parameters: { metadata_document_id: 3 }, + }) + httpTestingController.match( + `${environment.apiBaseUrl}documents/?page=1&page_size=50&ordering=-created&truncate_content=true` + ) // list reload + httpTestingController.match( + `${environment.apiBaseUrl}documents/?page=1&page_size=100000&fields=id` + ) // listAllFilteredIds + }) + it('should support bulk download with archive, originals or both and file formatting', () => { jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) jest diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts index 0bfb287cb..46a4980a6 100644 --- a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -6,7 +6,7 @@ import { TagService } from 'src/app/services/rest/tag.service' import { CorrespondentService } from 'src/app/services/rest/correspondent.service' import { DocumentTypeService } from 'src/app/services/rest/document-type.service' import { DocumentListViewService } from 'src/app/services/document-list-view.service' -import { NgbModal } from '@ng-bootstrap/ng-bootstrap' +import { NgbModal, NgbModalRef } from '@ng-bootstrap/ng-bootstrap' import { DocumentService, SelectionDataItem, @@ -39,6 +39,8 @@ import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component' import { TagEditDialogComponent } from '../../common/edit-dialog/tag-edit-dialog/tag-edit-dialog.component' import { DocumentTypeEditDialogComponent } from '../../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component' import { StoragePathEditDialogComponent } from '../../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' +import { RotateConfirmDialogComponent } from '../../common/confirm-dialog/rotate-confirm-dialog/rotate-confirm-dialog.component' +import { MergeConfirmDialogComponent } from '../../common/confirm-dialog/merge-confirm-dialog/merge-confirm-dialog.component' @Component({ selector: 'pngx-bulk-editor', @@ -192,12 +194,21 @@ export class BulkEditorComponent this.unsubscribeNotifier.complete() } - private executeBulkOperation(modal, method: string, args) { + private executeBulkOperation( + modal: NgbModalRef, + method: string, + args: any, + overrideDocumentIDs?: number[] + ) { if (modal) { modal.componentInstance.buttonsEnabled = false } this.documentService - .bulkEdit(Array.from(this.list.selected), method, args) + .bulkEdit( + overrideDocumentIDs ?? Array.from(this.list.selected), + method, + args + ) .pipe(first()) .subscribe({ next: () => { @@ -641,4 +652,49 @@ export class BulkEditorComponent } ) } + + rotateSelected() { + let modal = this.modalService.open(RotateConfirmDialogComponent, { + backdrop: 'static', + }) + const rotateDialog = modal.componentInstance as RotateConfirmDialogComponent + rotateDialog.title = $localize`Rotate confirm` + rotateDialog.messageBold = $localize`This operation will permanently rotate ${this.list.selected.size} selected document(s).` + rotateDialog.message = $localize`This will alter the original copy.` + rotateDialog.btnClass = 'btn-danger' + rotateDialog.btnCaption = $localize`Proceed` + rotateDialog.documentID = Array.from(this.list.selected)[0] + rotateDialog.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + rotateDialog.buttonsEnabled = false + this.executeBulkOperation(modal, 'rotate', { + degrees: rotateDialog.degrees, + }) + }) + } + + mergeSelected() { + let modal = this.modalService.open(MergeConfirmDialogComponent, { + backdrop: 'static', + }) + const mergeDialog = modal.componentInstance as MergeConfirmDialogComponent + mergeDialog.title = $localize`Merge confirm` + mergeDialog.messageBold = $localize`This operation will merge ${this.list.selected.size} selected documents into a new document.` + mergeDialog.btnCaption = $localize`Proceed` + mergeDialog.documentIDs = Array.from(this.list.selected) + mergeDialog.confirmClicked + .pipe(takeUntil(this.unsubscribeNotifier)) + .subscribe(() => { + const args = {} + if (mergeDialog.metadataDocumentID > -1) { + args['metadata_document_id'] = mergeDialog.metadataDocumentID + } + mergeDialog.buttonsEnabled = false + this.executeBulkOperation(modal, 'merge', args, mergeDialog.documentIDs) + this.toastService.showInfo( + $localize`Merged document will be queued for consumption.` + ) + }) + } } diff --git a/src/documents/bulk_edit.py b/src/documents/bulk_edit.py index ba001fd14..6b676733d 100644 --- a/src/documents/bulk_edit.py +++ b/src/documents/bulk_edit.py @@ -1,15 +1,27 @@ +import hashlib import itertools +import logging +import os +from typing import Optional +from celery import chord +from django.conf import settings from django.db.models import Q +from documents.data_models import ConsumableDocument +from documents.data_models import DocumentMetadataOverrides +from documents.data_models import DocumentSource from documents.models import Correspondent from documents.models import Document from documents.models import DocumentType from documents.models import StoragePath from documents.permissions import set_permissions_for_object from documents.tasks import bulk_update_documents +from documents.tasks import consume_file from documents.tasks import update_document_archive_file +logger = logging.getLogger("paperless.bulk_edit") + def set_correspondent(doc_ids, correspondent): if correspondent: @@ -146,3 +158,137 @@ def set_permissions(doc_ids, set_permissions, owner=None, merge=False): bulk_update_documents.delay(document_ids=affected_docs) return "OK" + + +def rotate(doc_ids: list[int], degrees: int): + logger.info( + f"Attempting to rotate {len(doc_ids)} documents by {degrees} degrees.", + ) + qs = Document.objects.filter(id__in=doc_ids) + affected_docs = [] + import pikepdf + + rotate_tasks = [] + for doc in qs: + if doc.mime_type != "application/pdf": + logger.warning( + f"Document {doc.id} is not a PDF, skipping rotation.", + ) + continue + try: + with pikepdf.open(doc.source_path, allow_overwriting_input=True) as pdf: + for page in pdf.pages: + page.rotate(degrees, relative=True) + pdf.save() + doc.checksum = hashlib.md5(doc.source_path.read_bytes()).hexdigest() + doc.save() + rotate_tasks.append( + update_document_archive_file.s( + document_id=doc.id, + ), + ) + logger.info( + f"Rotated document {doc.id} by {degrees} degrees", + ) + affected_docs.append(doc.id) + except Exception as e: + logger.exception(f"Error rotating document {doc.id}: {e}") + + if len(affected_docs) > 0: + bulk_update_task = bulk_update_documents.s(document_ids=affected_docs) + chord(header=rotate_tasks, body=bulk_update_task).delay() + + return "OK" + + +def merge(doc_ids: list[int], metadata_document_id: Optional[int] = None): + logger.info( + f"Attempting to merge {len(doc_ids)} documents into a single document.", + ) + qs = Document.objects.filter(id__in=doc_ids) + affected_docs = [] + import pikepdf + + merged_pdf = pikepdf.new() + version = merged_pdf.pdf_version + # use doc_ids to preserve order + for doc_id in doc_ids: + doc = qs.get(id=doc_id) + try: + with pikepdf.open(str(doc.source_path)) as pdf: + version = max(version, pdf.pdf_version) + merged_pdf.pages.extend(pdf.pages) + affected_docs.append(doc.id) + except Exception as e: + logger.exception( + f"Error merging document {doc.id}, it will not be included in the merge: {e}", + ) + if len(affected_docs) == 0: + logger.warning("No documents were merged") + return "OK" + + filepath = os.path.join( + settings.SCRATCH_DIR, + f"{'_'.join([str(doc_id) for doc_id in doc_ids])[:100]}_merged.pdf", + ) + merged_pdf.remove_unreferenced_resources() + merged_pdf.save(filepath, min_version=version) + merged_pdf.close() + + if metadata_document_id: + metadata_document = qs.get(id=metadata_document_id) + if metadata_document is not None: + overrides = DocumentMetadataOverrides.from_document(metadata_document) + overrides.title = metadata_document.title + " (merged)" + else: + overrides = DocumentMetadataOverrides() + + logger.info("Adding merged document to the task queue.") + consume_file.delay( + ConsumableDocument( + source=DocumentSource.ConsumeFolder, + original_file=filepath, + ), + overrides, + ) + + return "OK" + + +def split(doc_ids: list[int], pages: list[list[int]]): + logger.info( + f"Attempting to split document {doc_ids[0]} into {len(pages)} documents", + ) + doc = Document.objects.get(id=doc_ids[0]) + import pikepdf + + try: + with pikepdf.open(doc.source_path) as pdf: + for idx, split_doc in enumerate(pages): + dst = pikepdf.new() + for page in split_doc: + dst.pages.append(pdf.pages[page - 1]) + filepath = os.path.join( + settings.SCRATCH_DIR, + f"{doc.id}_{split_doc[0]}-{split_doc[-1]}.pdf", + ) + dst.remove_unreferenced_resources() + dst.save(filepath) + dst.close() + + overrides = DocumentMetadataOverrides().from_document(doc) + overrides.title = f"{doc.title} (split {idx + 1})" + logger.info( + f"Adding split document with pages {split_doc} to the task queue.", + ) + consume_file.delay( + ConsumableDocument( + source=DocumentSource.ConsumeFolder, + original_file=filepath, + ), + overrides, + ) + except Exception as e: + logger.exception(f"Error splitting document {doc.id}: {e}") + + return "OK" diff --git a/src/documents/caching.py b/src/documents/caching.py index a1f20d086..9abd3bc65 100644 --- a/src/documents/caching.py +++ b/src/documents/caching.py @@ -189,13 +189,21 @@ def refresh_metadata_cache( cache.touch(doc_key, timeout) -def clear_metadata_cache(document_id: int) -> None: - doc_key = get_metadata_cache_key(document_id) - cache.delete(doc_key) - - def get_thumbnail_modified_key(document_id: int) -> str: """ Builds the key to store a thumbnail's timestamp """ return f"doc_{document_id}_thumbnail_modified" + + +def clear_document_caches(document_id: int) -> None: + """ + Removes all cached items for the given document + """ + cache.delete_many( + [ + get_suggestion_cache_key(document_id), + get_metadata_cache_key(document_id), + get_thumbnail_modified_key(document_id), + ], + ) diff --git a/src/documents/data_models.py b/src/documents/data_models.py index 6bf3f4f96..4922b72dd 100644 --- a/src/documents/data_models.py +++ b/src/documents/data_models.py @@ -5,6 +5,8 @@ from pathlib import Path from typing import Optional import magic +from guardian.shortcuts import get_groups_with_perms +from guardian.shortcuts import get_users_with_perms @dataclasses.dataclass @@ -88,6 +90,44 @@ class DocumentMetadataOverrides: return self + @staticmethod + def from_document(doc) -> "DocumentMetadataOverrides": + """ + Fills in the overrides from a document object + """ + overrides = DocumentMetadataOverrides() + overrides.title = doc.title + overrides.correspondent_id = doc.correspondent.id if doc.correspondent else None + overrides.document_type_id = doc.document_type.id if doc.document_type else None + overrides.storage_path_id = doc.storage_path.id if doc.storage_path else None + overrides.owner_id = doc.owner.id if doc.owner else None + overrides.tag_ids = list(doc.tags.values_list("id", flat=True)) + + overrides.view_users = get_users_with_perms( + doc, + only_with_perms_in=["view_document"], + ).values_list("id", flat=True) + overrides.change_users = get_users_with_perms( + doc, + only_with_perms_in=["change_document"], + ).values_list("id", flat=True) + overrides.custom_field_ids = list( + doc.custom_fields.values_list("id", flat=True), + ) + + groups_with_perms = get_groups_with_perms( + doc, + attach_perms=True, + ) + overrides.view_groups = [ + group.id for group, perms in groups_with_perms if "view_document" in perms + ] + overrides.change_groups = [ + group.id for group, perms in groups_with_perms if "change_document" in perms + ] + + return overrides + class DocumentSource(IntEnum): """ diff --git a/src/documents/serialisers.py b/src/documents/serialisers.py index 1c2c6a095..bdac7660e 100644 --- a/src/documents/serialisers.py +++ b/src/documents/serialisers.py @@ -869,6 +869,9 @@ class BulkEditSerializer(DocumentListSerializer, SetPermissionsMixin): "delete", "redo_ocr", "set_permissions", + "rotate", + "merge", + "split", ], label="Method", write_only=True, @@ -906,6 +909,12 @@ class BulkEditSerializer(DocumentListSerializer, SetPermissionsMixin): return bulk_edit.redo_ocr elif method == "set_permissions": return bulk_edit.set_permissions + elif method == "rotate": + return bulk_edit.rotate + elif method == "merge": + return bulk_edit.merge + elif method == "split": + return bulk_edit.split else: raise serializers.ValidationError("Unsupported method.") @@ -984,6 +993,39 @@ class BulkEditSerializer(DocumentListSerializer, SetPermissionsMixin): if "merge" not in parameters: parameters["merge"] = False + def _validate_parameters_rotate(self, parameters): + try: + if ( + "degrees" not in parameters + or not float(parameters["degrees"]).is_integer() + ): + raise serializers.ValidationError("invalid rotation degrees") + except ValueError: + raise serializers.ValidationError("invalid rotation degrees") + + def _validate_parameters_split(self, parameters): + if "pages" not in parameters: + raise serializers.ValidationError("pages not specified") + try: + pages = [] + docs = parameters["pages"].split(",") + for doc in docs: + if "-" in doc: + pages.append( + [ + x + for x in range( + int(doc.split("-")[0]), + int(doc.split("-")[1]) + 1, + ) + ], + ) + else: + pages.append([int(doc)]) + parameters["pages"] = pages + except ValueError: + raise serializers.ValidationError("invalid pages specified") + def validate(self, attrs): method = attrs["method"] parameters = attrs["parameters"] @@ -1000,6 +1042,14 @@ class BulkEditSerializer(DocumentListSerializer, SetPermissionsMixin): self._validate_storage_path(parameters) elif method == bulk_edit.set_permissions: self._validate_parameters_set_permissions(parameters) + elif method == bulk_edit.rotate: + self._validate_parameters_rotate(parameters) + elif method == bulk_edit.split: + if len(attrs["documents"]) > 1: + raise serializers.ValidationError( + "Split method only supports one document", + ) + self._validate_parameters_split(parameters) return attrs diff --git a/src/documents/signals/handlers.py b/src/documents/signals/handlers.py index 85e8126c1..cdfedcb4c 100644 --- a/src/documents/signals/handlers.py +++ b/src/documents/signals/handlers.py @@ -23,7 +23,7 @@ from filelock import FileLock from guardian.shortcuts import remove_perm from documents import matching -from documents.caching import clear_metadata_cache +from documents.caching import clear_document_caches from documents.classifier import DocumentClassifier from documents.consumer import parse_doc_title_w_placeholders from documents.file_handling import create_source_path_directory @@ -439,7 +439,8 @@ def update_filename_and_move_files(sender, instance: Document, **kwargs): archive_filename=instance.archive_filename, modified=timezone.now(), ) - clear_metadata_cache(instance.pk) + # Clear any caching for this document. Slightly overkill, but not terrible + clear_document_caches(instance.pk) except (OSError, DatabaseError, CannotMoveFilesException) as e: logger.warning(f"Exception during file handling: {e}") diff --git a/src/documents/tasks.py b/src/documents/tasks.py index a83c2e6cd..0ab55ac45 100644 --- a/src/documents/tasks.py +++ b/src/documents/tasks.py @@ -18,6 +18,7 @@ from whoosh.writing import AsyncWriter from documents import index from documents import sanity_checker from documents.barcodes import BarcodePlugin +from documents.caching import clear_document_caches from documents.classifier import DocumentClassifier from documents.classifier import load_classifier from documents.consumer import Consumer @@ -213,6 +214,7 @@ def bulk_update_documents(document_ids): ix = index.open_index() for doc in documents: + clear_document_caches(doc.pk) document_updated.send( sender=None, document=doc, @@ -305,6 +307,8 @@ def update_document_archive_file(document_id): with index.open_index_writer() as writer: index.update_document(writer, document) + clear_document_caches(document.pk) + except Exception: logger.exception( f"Error while parsing document {document} (ID: {document_id})", diff --git a/src/documents/tests/test_api_bulk_edit.py b/src/documents/tests/test_api_bulk_edit.py index 10093eb44..d659c82e8 100644 --- a/src/documents/tests/test_api_bulk_edit.py +++ b/src/documents/tests/test_api_bulk_edit.py @@ -781,3 +781,153 @@ class TestBulkEditAPI(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, status.HTTP_200_OK) m.assert_called_once() + + @mock.patch("documents.serialisers.bulk_edit.rotate") + def test_rotate(self, m): + m.return_value = "OK" + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id, self.doc3.id], + "method": "rotate", + "parameters": {"degrees": 90}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_200_OK) + + m.assert_called_once() + args, kwargs = m.call_args + self.assertCountEqual(args[0], [self.doc2.id, self.doc3.id]) + self.assertEqual(kwargs["degrees"], 90) + + @mock.patch("documents.serialisers.bulk_edit.rotate") + def test_rotate_invalid_params(self, m): + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id, self.doc3.id], + "method": "rotate", + "parameters": {"degrees": "foo"}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id, self.doc3.id], + "method": "rotate", + "parameters": {"degrees": 90.5}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + + m.assert_not_called() + + @mock.patch("documents.serialisers.bulk_edit.merge") + def test_merge(self, m): + m.return_value = "OK" + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id, self.doc3.id], + "method": "merge", + "parameters": {"metadata_document_id": self.doc3.id}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_200_OK) + + m.assert_called_once() + args, kwargs = m.call_args + self.assertCountEqual(args[0], [self.doc2.id, self.doc3.id]) + self.assertEqual(kwargs["metadata_document_id"], self.doc3.id) + + @mock.patch("documents.serialisers.bulk_edit.split") + def test_split(self, m): + m.return_value = "OK" + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "split", + "parameters": {"pages": "1,2-4,5-6,7"}, + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_200_OK) + + m.assert_called_once() + args, kwargs = m.call_args + self.assertCountEqual(args[0], [self.doc2.id]) + self.assertEqual(kwargs["pages"], [[1], [2, 3, 4], [5, 6], [7]]) + + def test_split_invalid_params(self): + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "split", + "parameters": {}, # pages not specified + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"pages not specified", response.content) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [self.doc2.id], + "method": "split", + "parameters": {"pages": "1:7"}, # wrong format + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"invalid pages specified", response.content) + + response = self.client.post( + "/api/documents/bulk_edit/", + json.dumps( + { + "documents": [ + self.doc1.id, + self.doc2.id, + ], # only one document supported + "method": "split", + "parameters": {"pages": "1-2,3-7"}, # wrong format + }, + ), + content_type="application/json", + ) + + self.assertEqual(response.status_code, status.HTTP_400_BAD_REQUEST) + self.assertIn(b"Split method only supports one document", response.content) diff --git a/src/documents/tests/test_bulk_edit.py b/src/documents/tests/test_bulk_edit.py index f73835302..aca492649 100644 --- a/src/documents/tests/test_bulk_edit.py +++ b/src/documents/tests/test_bulk_edit.py @@ -1,3 +1,5 @@ +import shutil +from pathlib import Path from unittest import mock from django.contrib.auth.models import Group @@ -275,3 +277,262 @@ class TestBulkEdit(DirectoriesMixin, TestCase): self.doc1, ) self.assertEqual(groups_with_perms.count(), 2) + + +class TestPDFActions(DirectoriesMixin, TestCase): + def setUp(self): + super().setUp() + sample1 = self.dirs.scratch_dir / "sample.pdf" + shutil.copy( + Path(__file__).parent + / "samples" + / "documents" + / "originals" + / "0000001.pdf", + sample1, + ) + sample1_archive = self.dirs.archive_dir / "sample_archive.pdf" + shutil.copy( + Path(__file__).parent + / "samples" + / "documents" + / "originals" + / "0000001.pdf", + sample1_archive, + ) + sample2 = self.dirs.scratch_dir / "sample2.pdf" + shutil.copy( + Path(__file__).parent + / "samples" + / "documents" + / "originals" + / "0000002.pdf", + sample2, + ) + sample2_archive = self.dirs.archive_dir / "sample2_archive.pdf" + shutil.copy( + Path(__file__).parent + / "samples" + / "documents" + / "originals" + / "0000002.pdf", + sample2_archive, + ) + sample3 = self.dirs.scratch_dir / "sample3.pdf" + shutil.copy( + Path(__file__).parent + / "samples" + / "documents" + / "originals" + / "0000003.pdf", + sample3, + ) + self.doc1 = Document.objects.create( + checksum="A", + title="A", + filename=sample1, + mime_type="application/pdf", + ) + self.doc1.archive_filename = sample1_archive + self.doc1.save() + self.doc2 = Document.objects.create( + checksum="B", + title="B", + filename=sample2, + mime_type="application/pdf", + ) + self.doc2.archive_filename = sample2_archive + self.doc2.save() + self.doc3 = Document.objects.create( + checksum="C", + title="C", + filename=sample3, + mime_type="application/pdf", + ) + img_doc = self.dirs.scratch_dir / "sample_image.jpg" + shutil.copy( + Path(__file__).parent / "samples" / "simple.jpg", + img_doc, + ) + self.img_doc = Document.objects.create( + checksum="D", + title="D", + filename=img_doc, + mime_type="image/jpeg", + ) + + @mock.patch("documents.tasks.consume_file.delay") + def test_merge(self, mock_consume_file): + """ + GIVEN: + - Existing documents + WHEN: + - Merge action is called with 3 documents + THEN: + - Consume file should be called + """ + doc_ids = [self.doc1.id, self.doc2.id, self.doc3.id] + metadata_document_id = self.doc1.id + + result = bulk_edit.merge(doc_ids) + + expected_filename = ( + f"{'_'.join([str(doc_id) for doc_id in doc_ids])[:100]}_merged.pdf" + ) + + mock_consume_file.assert_called() + consume_file_args, _ = mock_consume_file.call_args + self.assertEqual( + Path(consume_file_args[0].original_file).name, + expected_filename, + ) + self.assertEqual(consume_file_args[1].title, None) + + # With metadata_document_id overrides + result = bulk_edit.merge(doc_ids, metadata_document_id=metadata_document_id) + consume_file_args, _ = mock_consume_file.call_args + self.assertEqual(consume_file_args[1].title, "A (merged)") + + self.assertEqual(result, "OK") + + @mock.patch("documents.tasks.consume_file.delay") + @mock.patch("pikepdf.open") + def test_merge_with_errors(self, mock_open_pdf, mock_consume_file): + """ + GIVEN: + - Existing documents + WHEN: + - Merge action is called with 2 documents + - Error occurs when opening both files + THEN: + - Consume file should not be called + """ + mock_open_pdf.side_effect = Exception("Error opening PDF") + doc_ids = [self.doc2.id, self.doc3.id] + + with self.assertLogs("paperless.bulk_edit", level="ERROR") as cm: + bulk_edit.merge(doc_ids) + error_str = cm.output[0] + expected_str = ( + "Error merging document 2, it will not be included in the merge" + ) + self.assertIn(expected_str, error_str) + + mock_consume_file.assert_not_called() + + @mock.patch("documents.tasks.consume_file.delay") + def test_split(self, mock_consume_file): + """ + GIVEN: + - Existing documents + WHEN: + - Split action is called with 1 document and 2 pages + THEN: + - Consume file should be called twice + """ + doc_ids = [self.doc2.id] + pages = [[1, 2], [3]] + result = bulk_edit.split(doc_ids, pages) + self.assertEqual(mock_consume_file.call_count, 2) + consume_file_args, _ = mock_consume_file.call_args + self.assertEqual(consume_file_args[1].title, "B (split 2)") + + self.assertEqual(result, "OK") + + @mock.patch("documents.tasks.consume_file.delay") + @mock.patch("pikepdf.Pdf.save") + def test_split_with_errors(self, mock_save_pdf, mock_consume_file): + """ + GIVEN: + - Existing documents + WHEN: + - Split action is called with 1 document and 2 page groups + - Error occurs when saving the files + THEN: + - Consume file should not be called + """ + mock_save_pdf.side_effect = Exception("Error saving PDF") + doc_ids = [self.doc2.id] + pages = [[1, 2], [3]] + + with self.assertLogs("paperless.bulk_edit", level="ERROR") as cm: + bulk_edit.split(doc_ids, pages) + error_str = cm.output[0] + expected_str = "Error splitting document 2" + self.assertIn(expected_str, error_str) + + mock_consume_file.assert_not_called() + + @mock.patch("documents.tasks.bulk_update_documents.s") + @mock.patch("documents.tasks.update_document_archive_file.s") + @mock.patch("celery.chord.delay") + def test_rotate(self, mock_chord, mock_update_document, mock_update_documents): + """ + GIVEN: + - Existing documents + WHEN: + - Rotate action is called with 2 documents + THEN: + - Rotate action should be called twice + """ + doc_ids = [self.doc1.id, self.doc2.id] + result = bulk_edit.rotate(doc_ids, 90) + self.assertEqual(mock_update_document.call_count, 2) + mock_update_documents.assert_called_once() + mock_chord.assert_called_once() + self.assertEqual(result, "OK") + + @mock.patch("documents.tasks.bulk_update_documents.s") + @mock.patch("documents.tasks.update_document_archive_file.s") + @mock.patch("pikepdf.Pdf.save") + def test_rotate_with_error( + self, + mock_pdf_save, + mock_update_archive_file, + mock_update_documents, + ): + """ + GIVEN: + - Existing documents + WHEN: + - Rotate action is called with 2 documents + - PikePDF raises an error + THEN: + - Rotate action should be called 0 times + """ + mock_pdf_save.side_effect = Exception("Error saving PDF") + doc_ids = [self.doc2.id, self.doc3.id] + + with self.assertLogs("paperless.bulk_edit", level="ERROR") as cm: + bulk_edit.rotate(doc_ids, 90) + error_str = cm.output[0] + expected_str = "Error rotating document" + self.assertIn(expected_str, error_str) + mock_update_archive_file.assert_not_called() + + @mock.patch("documents.tasks.bulk_update_documents.s") + @mock.patch("documents.tasks.update_document_archive_file.s") + @mock.patch("celery.chord.delay") + def test_rotate_non_pdf( + self, + mock_chord, + mock_update_document, + mock_update_documents, + ): + """ + GIVEN: + - Existing documents + WHEN: + - Rotate action is called with 2 documents, one of which is not a PDF + THEN: + - Rotate action should be performed 1 time, with the non-PDF document skipped + """ + with self.assertLogs("paperless.bulk_edit", level="INFO") as cm: + result = bulk_edit.rotate([self.doc2.id, self.img_doc.id], 90) + output_str = cm.output[1] + expected_str = "Document 4 is not a PDF, skipping rotation" + self.assertIn(expected_str, output_str) + self.assertEqual(mock_update_document.call_count, 1) + mock_update_documents.assert_called_once() + mock_chord.assert_called_once() + self.assertEqual(result, "OK") diff --git a/src/documents/views.py b/src/documents/views.py index 5fa0f7eb1..3e1996215 100644 --- a/src/documents/views.py +++ b/src/documents/views.py @@ -891,7 +891,8 @@ class BulkEditView(GenericAPIView, PassUserMixin): document_objs = Document.objects.filter(pk__in=documents) has_perms = ( all((doc.owner == user or doc.owner is None) for doc in document_objs) - if method == bulk_edit.set_permissions + if method + in [bulk_edit.set_permissions, bulk_edit.delete, bulk_edit.rotate] else all( has_perms_owner_aware(user, "change_document", doc) for doc in document_objs