diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf index 7d42085c3..ae9abe847 100644 --- a/src-ui/messages.xlf +++ b/src-ui/messages.xlf @@ -385,7 +385,7 @@ src/app/components/document-detail/document-detail.component.html - 100 + 117 @@ -534,7 +534,7 @@ src/app/components/document-detail/document-detail.component.html - 353 + 370 @@ -593,7 +593,7 @@ src/app/components/document-detail/document-detail.component.html - 346 + 363 src/app/components/document-list/bulk-editor/custom-fields-bulk-edit-dialog/custom-fields-bulk-edit-dialog.component.html @@ -739,7 +739,7 @@ src/app/components/document-detail/document-detail.component.html - 366 + 383 src/app/components/document-list/document-list.component.html @@ -1190,7 +1190,7 @@ src/app/components/document-detail/document-detail.component.html - 322 + 339 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -2077,8 +2077,8 @@ 19 - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 37 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 36 src/app/components/document-detail/document-detail.component.html @@ -3391,7 +3391,7 @@ src/app/components/document-detail/document-detail.component.html - 94 + 111 src/app/components/document-detail/document-detail.component.ts @@ -4288,7 +4288,7 @@ src/app/components/document-detail/document-detail.component.html - 288 + 305 @@ -4390,6 +4390,10 @@ src/app/components/common/profile-edit-dialog/profile-edit-dialog.component.html 10 + + src/app/components/document-detail/document-detail.component.html + 96 + First name @@ -5061,6 +5065,62 @@ 229 + + Email address(es) + + src/app/components/common/email-document-dialog/email-document-dialog.component.html + 7 + + + + Subject + + src/app/components/common/email-document-dialog/email-document-dialog.component.html + 11 + + + + Message + + src/app/components/common/email-document-dialog/email-document-dialog.component.html + 15 + + + + Use archive version + + src/app/components/common/email-document-dialog/email-document-dialog.component.html + 23 + + + + Send email + + src/app/components/common/email-document-dialog/email-document-dialog.component.html + 29 + + + + Email Document + + src/app/components/common/email-document-dialog/email-document-dialog.component.ts + 17 + + + + Email sent + + src/app/components/common/email-document-dialog/email-document-dialog.component.ts + 65 + + + + Error emailing document + + src/app/components/common/email-document-dialog/email-document-dialog.component.ts + 69 + + Include @@ -5082,8 +5142,8 @@ 58 - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 64 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 65 src/app/components/manage/management-list/management-list.component.html @@ -5543,8 +5603,8 @@ 155 - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 29 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 28 src/app/components/common/system-status-dialog/system-status-dialog.component.html @@ -5585,8 +5645,8 @@ 162 - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 40 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 39 @@ -5765,103 +5825,103 @@ 320 - - Share Links - - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 4 - - - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 32 - - No existing links - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 9,11 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 8,10 Share - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 33 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 32 Share archive version - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 47 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 48 Expires - src/app/components/common/share-links-dropdown/share-links-dropdown.component.html - 51 + src/app/components/common/share-links-dialog/share-links-dialog.component.html + 52 1 day - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 25 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 20 - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 111 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 104 7 days - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 26 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 21 30 days - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 27 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 22 Never - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 28 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 23 + + + + Share Links + + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 27 + + + src/app/components/document-detail/document-detail.component.html + 92 Error retrieving links - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 92 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 85 days - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 111 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 104 Error deleting link - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 140 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 133 Error creating link - src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts - 168 + src/app/components/common/share-links-dialog/share-links-dialog.component.ts + 161 @@ -6379,25 +6439,32 @@ 70 + + Send + + src/app/components/document-detail/document-detail.component.html + 88 + + Previous src/app/components/document-detail/document-detail.component.html - 97 + 114 Details src/app/components/document-detail/document-detail.component.html - 110 + 127 Title src/app/components/document-detail/document-detail.component.html - 113 + 130 src/app/components/document-list/document-list.component.html @@ -6420,21 +6487,21 @@ Archive serial number src/app/components/document-detail/document-detail.component.html - 114 + 131 Date created src/app/components/document-detail/document-detail.component.html - 115 + 132 Correspondent src/app/components/document-detail/document-detail.component.html - 117 + 134 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6461,7 +6528,7 @@ Document type src/app/components/document-detail/document-detail.component.html - 119 + 136 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6488,7 +6555,7 @@ Storage path src/app/components/document-detail/document-detail.component.html - 121 + 138 src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -6511,7 +6578,7 @@ Default src/app/components/document-detail/document-detail.component.html - 122 + 139 src/app/components/manage/saved-views/saved-views.component.html @@ -6522,14 +6589,14 @@ Content src/app/components/document-detail/document-detail.component.html - 218 + 235 Metadata src/app/components/document-detail/document-detail.component.html - 227 + 244 src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -6540,119 +6607,119 @@ Date modified src/app/components/document-detail/document-detail.component.html - 234 + 251 Date added src/app/components/document-detail/document-detail.component.html - 238 + 255 Media filename src/app/components/document-detail/document-detail.component.html - 242 + 259 Original filename src/app/components/document-detail/document-detail.component.html - 246 + 263 Original MD5 checksum src/app/components/document-detail/document-detail.component.html - 250 + 267 Original file size src/app/components/document-detail/document-detail.component.html - 254 + 271 Original mime type src/app/components/document-detail/document-detail.component.html - 258 + 275 Archive MD5 checksum src/app/components/document-detail/document-detail.component.html - 263 + 280 Archive file size src/app/components/document-detail/document-detail.component.html - 269 + 286 Original document metadata src/app/components/document-detail/document-detail.component.html - 278 + 295 Archived document metadata src/app/components/document-detail/document-detail.component.html - 281 + 298 Notes src/app/components/document-detail/document-detail.component.html - 300,303 + 317,320 History src/app/components/document-detail/document-detail.component.html - 311 + 328 Save & next src/app/components/document-detail/document-detail.component.html - 348 + 365 Save & close src/app/components/document-detail/document-detail.component.html - 351 + 368 Document loading... src/app/components/document-detail/document-detail.component.html - 361 + 378 Enter Password src/app/components/document-detail/document-detail.component.html - 415 + 432 @@ -6949,11 +7016,11 @@ An error occurred loading tiff: src/app/components/document-detail/document-detail.component.ts - 1461 + 1481 src/app/components/document-detail/document-detail.component.ts - 1465 + 1485 diff --git a/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.html b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.html new file mode 100644 index 000000000..56d404fd5 --- /dev/null +++ b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.html @@ -0,0 +1,32 @@ + + + diff --git a/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.scss b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.spec.ts b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.spec.ts new file mode 100644 index 000000000..7a3659205 --- /dev/null +++ b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.spec.ts @@ -0,0 +1,72 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing' + +import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' +import { provideHttpClientTesting } from '@angular/common/http/testing' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' +import { of, throwError } from 'rxjs' +import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' +import { PermissionsService } from 'src/app/services/permissions.service' +import { DocumentService } from 'src/app/services/rest/document.service' +import { ToastService } from 'src/app/services/toast.service' +import { EmailDocumentDialogComponent } from './email-document-dialog.component' + +describe('EmailDocumentDialogComponent', () => { + let component: EmailDocumentDialogComponent + let fixture: ComponentFixture + let documentService: DocumentService + let permissionsService: PermissionsService + let toastService: ToastService + + beforeEach(async () => { + await TestBed.configureTestingModule({ + imports: [ + EmailDocumentDialogComponent, + IfPermissionsDirective, + NgxBootstrapIconsModule.pick(allIcons), + ], + providers: [ + provideHttpClient(withInterceptorsFromDi()), + provideHttpClientTesting(), + NgbActiveModal, + ], + }).compileComponents() + + fixture = TestBed.createComponent(EmailDocumentDialogComponent) + documentService = TestBed.inject(DocumentService) + toastService = TestBed.inject(ToastService) + component = fixture.componentInstance + fixture.detectChanges() + }) + + it('should set hasArchiveVersion and useArchiveVersion', () => { + expect(component.hasArchiveVersion).toBeTruthy() + component.hasArchiveVersion = false + expect(component.hasArchiveVersion).toBeFalsy() + expect(component.useArchiveVersion).toBeFalsy() + }) + + it('should support sending document via email, showing error if needed', () => { + const toastErrorSpy = jest.spyOn(toastService, 'showError') + const toastSuccessSpy = jest.spyOn(toastService, 'showInfo') + component.emailAddress = 'hello@paperless-ngx.com' + component.emailSubject = 'Hello' + component.emailMessage = 'World' + jest + .spyOn(documentService, 'emailDocument') + .mockReturnValue(throwError(() => new Error('Unable to email document'))) + component.emailDocument() + expect(toastErrorSpy).toHaveBeenCalled() + + jest.spyOn(documentService, 'emailDocument').mockReturnValue(of(true)) + component.emailDocument() + expect(toastSuccessSpy).toHaveBeenCalled() + }) + + it('should close the dialog', () => { + const activeModal = TestBed.inject(NgbActiveModal) + const closeSpy = jest.spyOn(activeModal, 'close') + component.close() + expect(closeSpy).toHaveBeenCalled() + }) +}) diff --git a/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.ts b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.ts new file mode 100644 index 000000000..ab8b9768b --- /dev/null +++ b/src-ui/src/app/components/common/email-document-dialog/email-document-dialog.component.ts @@ -0,0 +1,77 @@ +import { Component, Input } from '@angular/core' +import { FormsModule } from '@angular/forms' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' +import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' +import { DocumentService } from 'src/app/services/rest/document.service' +import { ToastService } from 'src/app/services/toast.service' +import { LoadingComponentWithPermissions } from '../../loading-component/loading.component' + +@Component({ + selector: 'pngx-email-document-dialog', + templateUrl: './email-document-dialog.component.html', + styleUrl: './email-document-dialog.component.scss', + imports: [FormsModule, NgxBootstrapIconsModule], +}) +export class EmailDocumentDialogComponent extends LoadingComponentWithPermissions { + @Input() + title = $localize`Email Document` + + @Input() + documentId: number + + private _hasArchiveVersion: boolean = true + + @Input() + set hasArchiveVersion(value: boolean) { + this._hasArchiveVersion = value + this.useArchiveVersion = value + } + + get hasArchiveVersion(): boolean { + return this._hasArchiveVersion + } + + public useArchiveVersion: boolean = true + + public emailAddress: string = '' + public emailSubject: string = '' + public emailMessage: string = '' + + constructor( + private activeModal: NgbActiveModal, + private documentService: DocumentService, + private toastService: ToastService + ) { + super() + this.loading = false + } + + public emailDocument() { + this.loading = true + this.documentService + .emailDocument( + this.documentId, + this.emailAddress, + this.emailSubject, + this.emailMessage, + this.useArchiveVersion + ) + .subscribe({ + next: () => { + this.loading = false + this.emailAddress = '' + this.emailSubject = '' + this.emailMessage = '' + this.toastService.showInfo($localize`Email sent`) + }, + error: (e) => { + this.loading = false + this.toastService.showError($localize`Error emailing document`, e) + }, + }) + } + + public close() { + this.activeModal.close() + } +} diff --git a/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.html b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.html new file mode 100644 index 000000000..fe3f9b9c3 --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.html @@ -0,0 +1,68 @@ + + + diff --git a/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.scss b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.scss new file mode 100644 index 000000000..df5024ecd --- /dev/null +++ b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.scss @@ -0,0 +1,3 @@ +.copied-badge { + right: 15em; +} diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.spec.ts similarity index 92% rename from src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts rename to src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.spec.ts index b7b0305be..3f60b6733 100644 --- a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.spec.ts +++ b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.spec.ts @@ -11,17 +11,18 @@ import { tick, } from '@angular/core/testing' import { By } from '@angular/platform-browser' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' import { of, throwError } from 'rxjs' import { FileVersion, ShareLink } from 'src/app/data/share-link' import { ShareLinkService } from 'src/app/services/rest/share-link.service' import { ToastService } from 'src/app/services/toast.service' import { environment } from 'src/environments/environment' -import { ShareLinksDropdownComponent } from './share-links-dropdown.component' +import { ShareLinksDialogComponent } from './share-links-dialog.component' -describe('ShareLinksDropdownComponent', () => { - let component: ShareLinksDropdownComponent - let fixture: ComponentFixture +describe('ShareLinksDialogComponent', () => { + let component: ShareLinksDialogComponent + let fixture: ComponentFixture let shareLinkService: ShareLinkService let toastService: ToastService let httpController: HttpTestingController @@ -30,16 +31,17 @@ describe('ShareLinksDropdownComponent', () => { beforeEach(() => { TestBed.configureTestingModule({ imports: [ - ShareLinksDropdownComponent, + ShareLinksDialogComponent, NgxBootstrapIconsModule.pick(allIcons), ], providers: [ provideHttpClient(withInterceptorsFromDi()), provideHttpClientTesting(), + NgbActiveModal, ], }) - fixture = TestBed.createComponent(ShareLinksDropdownComponent) + fixture = TestBed.createComponent(ShareLinksDialogComponent) shareLinkService = TestBed.inject(ShareLinkService) toastService = TestBed.inject(ToastService) httpController = TestBed.inject(HttpTestingController) @@ -232,4 +234,11 @@ describe('ShareLinksDropdownComponent', () => { ] ).toBeTruthy() }) + + it('should support close', () => { + const activeModal = TestBed.inject(NgbActiveModal) + const closeSpy = jest.spyOn(activeModal, 'close') + component.close() + expect(closeSpy).toHaveBeenCalled() + }) }) diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.ts similarity index 90% rename from src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts rename to src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.ts index 5e65eed73..19123f73e 100644 --- a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.ts +++ b/src-ui/src/app/components/common/share-links-dialog/share-links-dialog.component.ts @@ -1,7 +1,7 @@ import { Clipboard } from '@angular/cdk/clipboard' import { Component, Input, OnInit } from '@angular/core' import { FormsModule, ReactiveFormsModule } from '@angular/forms' -import { NgbDropdownModule } from '@ng-bootstrap/ng-bootstrap' +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' import { first } from 'rxjs' import { FileVersion, ShareLink } from 'src/app/data/share-link' @@ -10,17 +10,12 @@ import { ToastService } from 'src/app/services/toast.service' import { environment } from 'src/environments/environment' @Component({ - selector: 'pngx-share-links-dropdown', - templateUrl: './share-links-dropdown.component.html', - styleUrls: ['./share-links-dropdown.component.scss'], - imports: [ - FormsModule, - ReactiveFormsModule, - NgbDropdownModule, - NgxBootstrapIconsModule, - ], + selector: 'pngx-share-links-dialog', + templateUrl: './share-links-dialog.component.html', + styleUrls: ['./share-links-dialog.component.scss'], + imports: [FormsModule, ReactiveFormsModule, NgxBootstrapIconsModule], }) -export class ShareLinksDropdownComponent implements OnInit { +export class ShareLinksDialogComponent implements OnInit { EXPIRATION_OPTIONS = [ { label: $localize`1 day`, value: 1 }, { label: $localize`7 days`, value: 7 }, @@ -41,9 +36,6 @@ export class ShareLinksDropdownComponent implements OnInit { } } - @Input() - disabled: boolean = false - private _hasArchiveVersion: boolean = true @Input() @@ -67,6 +59,7 @@ export class ShareLinksDropdownComponent implements OnInit { useArchiveVersion: boolean = true constructor( + private activeModal: NgbActiveModal, private shareLinkService: ShareLinkService, private toastService: ToastService, private clipboard: Clipboard @@ -169,4 +162,8 @@ export class ShareLinksDropdownComponent implements OnInit { }, }) } + + close() { + this.activeModal.close() + } } diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html deleted file mode 100644 index 08298abc7..000000000 --- a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.html +++ /dev/null @@ -1,70 +0,0 @@ -
- - -
diff --git a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss b/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss deleted file mode 100644 index 47e19d871..000000000 --- a/src-ui/src/app/components/common/share-links-dropdown/share-links-dropdown.component.scss +++ /dev/null @@ -1,14 +0,0 @@ -.share-links-dropdown { - min-width: 350px; - - // correct position on mobile - @media (max-width: 575.98px) { - &.show { - margin-left: -175px !important; - } - } -} - -.copied-badge { - right: 7.5em; -} 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 fc35bdb43..c99c35f01 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 @@ -81,7 +81,24 @@ (added)="addField($event)"> - + +
+ +
+ + @if (emailEnabled) { + + } +
+
+
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 349e213aa..b85a7eaf4 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 @@ -1330,4 +1330,18 @@ describe('DocumentDetailComponent', () => { expect(createSpy).toHaveBeenCalledWith('a') expect(urlRevokeSpy).toHaveBeenCalled() }) + + it('should get email enabled status from settings', () => { + jest.spyOn(settingsService, 'get').mockReturnValue(true) + expect(component.emailEnabled).toBeTruthy() + }) + + it('should support open share links and email modals', () => { + const modalSpy = jest.spyOn(modalService, 'open') + initNormally() + component.openShareLinks() + expect(modalSpy).toHaveBeenCalled() + component.openEmailDocument() + expect(modalSpy).toHaveBeenCalled() + }) }) 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 30e34d9cf..27a74cfcd 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 @@ -88,6 +88,7 @@ import { CorrespondentEditDialogComponent } from '../common/edit-dialog/correspo import { DocumentTypeEditDialogComponent } from '../common/edit-dialog/document-type-edit-dialog/document-type-edit-dialog.component' import { EditDialogMode } from '../common/edit-dialog/edit-dialog.component' import { StoragePathEditDialogComponent } from '../common/edit-dialog/storage-path-edit-dialog/storage-path-edit-dialog.component' +import { EmailDocumentDialogComponent } from '../common/email-document-dialog/email-document-dialog.component' import { CheckComponent } from '../common/input/check/check.component' import { DateComponent } from '../common/input/date/date.component' import { DocumentLinkComponent } from '../common/input/document-link/document-link.component' @@ -99,7 +100,7 @@ import { TagsComponent } from '../common/input/tags/tags.component' import { TextComponent } from '../common/input/text/text.component' import { UrlComponent } from '../common/input/url/url.component' import { PageHeaderComponent } from '../common/page-header/page-header.component' -import { ShareLinksDropdownComponent } from '../common/share-links-dropdown/share-links-dropdown.component' +import { ShareLinksDialogComponent } from '../common/share-links-dialog/share-links-dialog.component' import { DocumentHistoryComponent } from '../document-history/document-history.component' import { DocumentNotesComponent } from '../document-notes/document-notes.component' import { ComponentWithPermissions } from '../with-permissions/with-permissions.component' @@ -145,7 +146,6 @@ export enum ZoomSetting { CustomFieldsDropdownComponent, DocumentNotesComponent, DocumentHistoryComponent, - ShareLinksDropdownComponent, CheckComponent, DateComponent, DocumentLinkComponent, @@ -1426,6 +1426,26 @@ export class DocumentDetailComponent }) } + public openShareLinks() { + const modal = this.modalService.open(ShareLinksDialogComponent) + modal.componentInstance.documentId = this.document.id + modal.componentInstance.hasArchiveVersion = + !!this.document?.archived_file_name + } + + get emailEnabled(): boolean { + return this.settings.get(SETTINGS_KEYS.EMAIL_ENABLED) + } + + public openEmailDocument() { + const modal = this.modalService.open(EmailDocumentDialogComponent, { + backdrop: 'static', + }) + modal.componentInstance.documentId = this.document.id + modal.componentInstance.hasArchiveVersion = + !!this.document?.archived_file_name + } + private tryRenderTiff() { this.http.get(this.previewUrl, { responseType: 'arraybuffer' }).subscribe({ next: (res) => { diff --git a/src-ui/src/app/services/rest/document.service.spec.ts b/src-ui/src/app/services/rest/document.service.spec.ts index 4d7d7cef7..84f7f6f8a 100644 --- a/src-ui/src/app/services/rest/document.service.spec.ts +++ b/src-ui/src/app/services/rest/document.service.spec.ts @@ -355,6 +355,21 @@ it('should include custom fields in sort fields if user has permission', () => { ]) }) +it('should call appropriate api endpoint for email document', () => { + subscription = service + .emailDocument( + documents[0].id, + 'hello@paperless-ngx.com', + 'hello', + 'world', + true + ) + .subscribe() + httpTestingController.expectOne( + `${environment.apiBaseUrl}${endpoint}/${documents[0].id}/email/` + ) +}) + afterEach(() => { subscription?.unsubscribe() httpTestingController.verify() diff --git a/src-ui/src/app/services/rest/document.service.ts b/src-ui/src/app/services/rest/document.service.ts index bbb611adf..0c6c8cfa6 100644 --- a/src-ui/src/app/services/rest/document.service.ts +++ b/src-ui/src/app/services/rest/document.service.ts @@ -258,4 +258,19 @@ export class DocumentService extends AbstractPaperlessService { public get searchQuery(): string { return this._searchQuery } + + emailDocument( + documentId: number, + addresses: string, + subject: string, + message: string, + useArchiveVersion: boolean + ): Observable { + return this.http.post(this.getResourceUrl(documentId, 'email'), { + addresses: addresses, + subject: subject, + message: message, + use_archive_version: useArchiveVersion, + }) + } } diff --git a/src-ui/src/main.ts b/src-ui/src/main.ts index a9d446891..dd31a6b1e 100644 --- a/src-ui/src/main.ts +++ b/src-ui/src/main.ts @@ -112,6 +112,7 @@ import { questionCircle, scissors, search, + send, slashCircle, sliders2Vertical, sortAlphaDown, @@ -316,6 +317,7 @@ const icons = { questionCircle, scissors, search, + send, slashCircle, sliders2Vertical, sortAlphaDown, diff --git a/src/documents/tests/test_api_documents.py b/src/documents/tests/test_api_documents.py index 6247b0a6e..28261b392 100644 --- a/src/documents/tests/test_api_documents.py +++ b/src/documents/tests/test_api_documents.py @@ -15,6 +15,7 @@ from dateutil import parser from django.conf import settings from django.contrib.auth.models import Permission from django.contrib.auth.models import User +from django.core import mail from django.core.cache import cache from django.db import DataError from django.test import override_settings @@ -2651,6 +2652,153 @@ class TestDocumentApi(DirectoriesMixin, DocumentConsumeDelayMixin, APITestCase): self.assertEqual(resp.status_code, status.HTTP_200_OK) self.assertEqual(doc1.tags.count(), 2) + @override_settings( + EMAIL_ENABLED=True, + EMAIL_BACKEND="django.core.mail.backends.locmem.EmailBackend", + ) + def test_email_document(self): + """ + GIVEN: + - Existing document + WHEN: + - API request is made to email document action + THEN: + - Email is sent, with document (original or archive) attached + """ + doc = Document.objects.create( + title="test", + mime_type="application/pdf", + content="this is a document 1", + checksum="1", + filename="test.pdf", + archive_checksum="A", + archive_filename="archive.pdf", + ) + doc2 = Document.objects.create( + title="test2", + mime_type="application/pdf", + content="this is a document 2", + checksum="2", + filename="test2.pdf", + ) + + archive_file = Path(__file__).parent / "samples" / "simple.pdf" + source_file = Path(__file__).parent / "samples" / "simple.pdf" + + shutil.copy(archive_file, doc.archive_path) + shutil.copy(source_file, doc2.source_path) + + self.client.post( + f"/api/documents/{doc.pk}/email/", + { + "addresses": "hello@paperless-ngx.com", + "subject": "test", + "message": "hello", + }, + ) + + self.assertEqual(len(mail.outbox), 1) + self.assertEqual(mail.outbox[0].attachments[0][0], "archive.pdf") + + self.client.post( + f"/api/documents/{doc2.pk}/email/", + { + "addresses": "hello@paperless-ngx.com", + "subject": "test", + "message": "hello", + "use_archive_version": False, + }, + ) + + self.assertEqual(len(mail.outbox), 2) + self.assertEqual(mail.outbox[1].attachments[0][0], "test2.pdf") + + @mock.patch("django.core.mail.message.EmailMessage.send", side_effect=Exception) + def test_email_document_errors(self, mocked_send): + """ + GIVEN: + - Existing document + WHEN: + - API request is made to email document action with insufficient permissions + - API request is made to email document action with invalid document id + - API request is made to email document action with missing data + - API request is made to email document action with invalid email address + - API request is made to email document action and error occurs during email send + THEN: + - Error response is returned + """ + user1 = User.objects.create_user(username="test1") + user1.user_permissions.add(*Permission.objects.all()) + user1.save() + + doc = Document.objects.create( + title="test", + mime_type="application/pdf", + content="this is a document 1", + checksum="1", + filename="test.pdf", + archive_checksum="A", + archive_filename="archive.pdf", + ) + + doc2 = Document.objects.create( + title="test2", + mime_type="application/pdf", + content="this is a document 2", + checksum="2", + owner=self.user, + ) + + self.client.force_authenticate(user1) + + resp = self.client.post( + f"/api/documents/{doc2.pk}/email/", + { + "addresses": "hello@paperless-ngx.com", + "subject": "test", + "message": "hello", + }, + ) + self.assertEqual(resp.status_code, status.HTTP_403_FORBIDDEN) + + resp = self.client.post( + "/api/documents/999/email/", + { + "addresses": "hello@paperless-ngx.com", + "subject": "test", + "message": "hello", + }, + ) + self.assertEqual(resp.status_code, status.HTTP_404_NOT_FOUND) + + resp = self.client.post( + f"/api/documents/{doc.pk}/email/", + { + "addresses": "hello@paperless-ngx.com", + }, + ) + self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) + + resp = self.client.post( + f"/api/documents/{doc.pk}/email/", + { + "addresses": "hello@paperless-ngx.com,hello", + "subject": "test", + "message": "hello", + }, + ) + self.assertEqual(resp.status_code, status.HTTP_400_BAD_REQUEST) + + resp = self.client.post( + f"/api/documents/{doc.pk}/email/", + { + "addresses": "hello@paperless-ngx.com", + "subject": "test", + "message": "hello", + }, + ) + self.assertEqual(resp.status_code, status.HTTP_500_INTERNAL_SERVER_ERROR) + @mock.patch("django_softdelete.models.SoftDeleteModel.delete") def test_warn_on_delete_with_old_uuid_field(self, mocked_delete): """ diff --git a/src/documents/views.py b/src/documents/views.py index aceea6699..a4e35a2f4 100644 --- a/src/documents/views.py +++ b/src/documents/views.py @@ -37,6 +37,7 @@ from django.http import HttpResponse from django.http import HttpResponseBadRequest from django.http import HttpResponseForbidden from django.http import HttpResponseRedirect +from django.http import HttpResponseServerError from django.shortcuts import get_object_or_404 from django.utils import timezone from django.utils.decorators import method_decorator @@ -106,6 +107,7 @@ from documents.filters import ObjectOwnedPermissionsFilter from documents.filters import ShareLinkFilterSet from documents.filters import StoragePathFilterSet from documents.filters import TagFilterSet +from documents.mail import send_email from documents.matching import match_correspondents from documents.matching import match_document_types from documents.matching import match_storage_paths @@ -1023,6 +1025,57 @@ class DocumentViewSet( return Response(sorted(entries, key=lambda x: x["timestamp"], reverse=True)) + @action(methods=["post"], detail=True) + def email(self, request, pk=None): + try: + doc = Document.objects.select_related("owner").get(pk=pk) + if request.user is not None and not has_perms_owner_aware( + request.user, + "view_document", + doc, + ): + return HttpResponseForbidden("Insufficient permissions") + except Document.DoesNotExist: + raise Http404 + + try: + if ( + "addresses" not in request.data + or "subject" not in request.data + or "message" not in request.data + ): + return HttpResponseBadRequest("Missing required fields") + + use_archive_version = request.data.get("use_archive_version", True) + + addresses = request.data.get("addresses").split(",") + if not all( + re.match(r"[^@]+@[^@]+\.[^@]+", address.strip()) + for address in addresses + ): + return HttpResponseBadRequest("Invalid email address found") + + send_email( + subject=request.data.get("subject"), + body=request.data.get("message"), + to=addresses, + attachment=( + doc.archive_path + if use_archive_version and doc.has_archive_version + else doc.source_path + ), + attachment_mime_type=doc.mime_type, + ) + logger.debug( + f"Sent document {doc.id} via email to {addresses}", + ) + return Response({"message": "Email sent"}) + except Exception as e: + logger.warning(f"An error occurred emailing document: {e!s}") + return HttpResponseServerError( + "Error emailing document, check logs for more detail.", + ) + @extend_schema_view( list=extend_schema(