diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 6614a9971..b99f699de 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -30,7 +30,7 @@ jobs: github.repository name: Linting Checks - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 steps: - name: Checkout repository @@ -46,7 +46,7 @@ jobs: documentation: name: "Build & Deploy Documentation" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - pre-commit steps: @@ -95,7 +95,7 @@ jobs: tests-backend: name: "Backend Tests (Python ${{ matrix.python-version }})" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - pre-commit strategy: @@ -170,7 +170,7 @@ jobs: install-frontend-depedendencies: name: "Install Frontend Dependencies" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - pre-commit steps: @@ -201,7 +201,7 @@ jobs: tests-frontend: name: "Frontend Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - install-frontend-depedendencies strategy: @@ -261,7 +261,7 @@ jobs: tests-coverage-upload: name: "Upload to Codecov" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - tests-backend - tests-frontend @@ -333,7 +333,7 @@ jobs: build-docker-image: name: Build Docker image for ${{ github.ref_name }} - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v')) concurrency: group: ${{ github.workflow }}-build-docker-image-${{ github.ref_name }} @@ -461,7 +461,7 @@ jobs: needs: - build-docker-image - documentation - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 steps: - name: Checkout @@ -569,7 +569,7 @@ jobs: publish-release: name: "Publish Release" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 outputs: prerelease: ${{ steps.get_version.outputs.prerelease }} changelog: ${{ steps.create-release.outputs.body }} @@ -619,7 +619,7 @@ jobs: append-changelog: name: "Append Changelog" - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - publish-release if: needs.publish-release.outputs.prerelease == 'false' diff --git a/.github/workflows/cleanup-tags.yml b/.github/workflows/cleanup-tags.yml index 4eba9127a..5d2c32984 100644 --- a/.github/workflows/cleanup-tags.yml +++ b/.github/workflows/cleanup-tags.yml @@ -21,7 +21,7 @@ jobs: cleanup-images: name: Cleanup Image Tags for ${{ matrix.primary-name }} if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 strategy: fail-fast: false matrix: @@ -33,7 +33,7 @@ jobs: - name: Clean temporary images if: "${{ env.TOKEN != '' }}" - uses: stumpylog/image-cleaner-action/ephemeral@v0.8.0 + uses: stumpylog/image-cleaner-action/ephemeral@v0.9.0 with: token: "${{ env.TOKEN }}" owner: "${{ github.repository_owner }}" @@ -47,7 +47,7 @@ jobs: cleanup-untagged-images: name: Cleanup Untagged Images Tags for ${{ matrix.primary-name }} if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 needs: - cleanup-images strategy: @@ -61,7 +61,7 @@ jobs: - name: Clean untagged images if: "${{ env.TOKEN != '' }}" - uses: stumpylog/image-cleaner-action/untagged@v0.8.0 + uses: stumpylog/image-cleaner-action/untagged@v0.9.0 with: token: "${{ env.TOKEN }}" owner: "${{ github.repository_owner }}" diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml index fc521843f..cdc0e7d6d 100644 --- a/.github/workflows/codeql-analysis.yml +++ b/.github/workflows/codeql-analysis.yml @@ -23,7 +23,7 @@ on: jobs: analyze: name: Analyze - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 permissions: actions: read contents: read diff --git a/.github/workflows/crowdin.yml b/.github/workflows/crowdin.yml index d5d358d9a..308b646d7 100644 --- a/.github/workflows/crowdin.yml +++ b/.github/workflows/crowdin.yml @@ -16,7 +16,7 @@ jobs: synchronize-with-crowdin: name: Crowdin Sync if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - name: Checkout diff --git a/.github/workflows/project-actions.yml b/.github/workflows/project-actions.yml index bd788c5b3..aa6ad2d38 100644 --- a/.github/workflows/project-actions.yml +++ b/.github/workflows/project-actions.yml @@ -15,7 +15,7 @@ permissions: jobs: pr_opened_or_reopened: name: pr_opened_or_reopened - runs-on: ubuntu-22.04 + runs-on: ubuntu-24.04 permissions: # write permission is required for autolabeler pull-requests: write diff --git a/.github/workflows/repo-maintenance.yml b/.github/workflows/repo-maintenance.yml index 13458713b..930b2a7b5 100644 --- a/.github/workflows/repo-maintenance.yml +++ b/.github/workflows/repo-maintenance.yml @@ -17,7 +17,7 @@ jobs: stale: name: 'Stale' if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: actions/stale@v9 with: @@ -33,7 +33,7 @@ jobs: lock-threads: name: 'Lock Old Threads' if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: dessant/lock-threads@v5 with: @@ -59,7 +59,7 @@ jobs: close-answered-discussions: name: 'Close Answered Discussions' if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: actions/github-script@v7 with: @@ -116,7 +116,7 @@ jobs: close-outdated-discussions: name: 'Close Outdated Discussions' if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: actions/github-script@v7 with: @@ -208,7 +208,7 @@ jobs: close-unsupported-feature-requests: name: 'Close Unsupported Feature Requests' if: github.repository_owner == 'paperless-ngx' - runs-on: ubuntu-latest + runs-on: ubuntu-24.04 steps: - uses: actions/github-script@v7 with: diff --git a/src-ui/src/app/components/admin/tasks/tasks.component.html b/src-ui/src/app/components/admin/tasks/tasks.component.html index 4178bb2c8..3d40c7897 100644 --- a/src-ui/src/app/components/admin/tasks/tasks.component.html +++ b/src-ui/src/app/components/admin/tasks/tasks.component.html @@ -43,7 +43,7 @@ - @for (task of tasks | slice: (page-1) * pageSize : page * pageSize; track task) { + @for (task of tasks | slice: (page-1) * pageSize : page * pageSize; track task.id) {
diff --git a/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss b/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss index c44e29f43..14439b8fb 100644 --- a/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss +++ b/src-ui/src/app/components/common/preview-popup/preview-popup.component.scss @@ -19,6 +19,7 @@ object:not(.pdf) { mix-blend-mode: difference; + background: white !important; &.p-2 { padding: 0 !important; } 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 6cd26b368..4fa535ea3 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 @@ -710,7 +710,10 @@ export class DocumentDetailComponent next: (docValues) => { // in case data changed while saving eg removing inbox_tags this.documentForm.patchValue(docValues) - this.store.next(this.documentForm.value) + const newValues = Object.assign({}, this.documentForm.value) + newValues.tags = [...docValues.tags] + newValues.custom_fields = [...docValues.custom_fields] + this.store.next(newValues) this.openDocumentService.setDirty(this.document, false) this.openDocumentService.save() this.toastService.showInfo($localize`Document saved successfully.`) diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html index 4885212ad..2ac5cb43f 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html @@ -1,6 +1,6 @@
-
+
diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index 895e7085b..eafb3be90 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -501,7 +501,7 @@ ul.pagination { border-color:var(--bs-primary); .document-card-check { - display: block; + display: block !important; } .doc-img-container { diff --git a/src-ui/src/theme.scss b/src-ui/src/theme.scss index 877bf6507..228d78f04 100644 --- a/src-ui/src/theme.scss +++ b/src-ui/src/theme.scss @@ -204,7 +204,7 @@ $form-check-radio-checked-bg-image-dark: url("data:image/svg+xml, 0 + and Document.deleted_objects.filter( + archive_serial_number=attrs["archive_serial_number"], + ).exists() + ): + raise serializers.ValidationError( + { + "archive_serial_number": [ + "Document with this Archive Serial Number already exists in the trash.", + ], + }, + ) + return super().validate(attrs) + def update(self, instance: Document, validated_data): if "created_date" in validated_data and "created" not in validated_data: new_datetime = datetime.datetime.combine( diff --git a/src/documents/tests/test_api_documents.py b/src/documents/tests/test_api_documents.py index 2e2b02f0d..08d86d24e 100644 --- a/src/documents/tests/test_api_documents.py +++ b/src/documents/tests/test_api_documents.py @@ -2540,6 +2540,50 @@ class TestDocumentApi(DirectoriesMixin, DocumentConsumeDelayMixin, APITestCase): self.assertEqual(resp.status_code, status.HTTP_200_OK) self.assertEqual(resp.content, b"1") + def test_asn_not_unique_with_trashed_doc(self): + """ + GIVEN: + - Existing document with ASN that is trashed + WHEN: + - API request to update document with same ASN + THEN: + - Explicit error is returned + """ + user1 = User.objects.create_superuser(username="test1") + + self.client.force_authenticate(user1) + + doc1 = Document.objects.create( + title="test", + mime_type="application/pdf", + content="this is a document 1", + checksum="1", + archive_serial_number=1, + ) + doc1.delete() + + doc2 = Document.objects.create( + title="test2", + mime_type="application/pdf", + content="this is a document 2", + checksum="2", + ) + result = self.client.patch( + f"/api/documents/{doc2.pk}/", + { + "archive_serial_number": 1, + }, + ) + self.assertEqual(result.status_code, status.HTTP_400_BAD_REQUEST) + self.assertEqual( + result.json(), + { + "archive_serial_number": [ + "Document with this Archive Serial Number already exists in the trash.", + ], + }, + ) + def test_remove_inbox_tags(self): """ GIVEN: diff --git a/src/paperless_tesseract/parsers.py b/src/paperless_tesseract/parsers.py index 6b9ec3d93..95c1dbfcc 100644 --- a/src/paperless_tesseract/parsers.py +++ b/src/paperless_tesseract/parsers.py @@ -43,10 +43,15 @@ class RasterisedDocumentParser(DocumentParser): def get_page_count(self, document_path, mime_type): page_count = None if mime_type == "application/pdf": - import pikepdf + try: + import pikepdf - with pikepdf.Pdf.open(document_path) as pdf: - page_count = len(pdf.pages) + with pikepdf.Pdf.open(document_path) as pdf: + page_count = len(pdf.pages) + except Exception as e: + self.log.warning( + f"Unable to determine PDF page count {document_path}: {e}", + ) return page_count def extract_metadata(self, document_path, mime_type): diff --git a/src/paperless_tesseract/tests/test_parser.py b/src/paperless_tesseract/tests/test_parser.py index 45a5939ab..f7490fbbf 100644 --- a/src/paperless_tesseract/tests/test_parser.py +++ b/src/paperless_tesseract/tests/test_parser.py @@ -81,6 +81,24 @@ class TestParser(DirectoriesMixin, FileSystemAssertsMixin, TestCase): ) self.assertEqual(page_count, 6) + def test_get_page_count_password_protected(self): + """ + GIVEN: + - Password protected PDF file + WHEN: + - The number of pages is requested + THEN: + - The method returns None + """ + parser = RasterisedDocumentParser(uuid.uuid4()) + with self.assertLogs("paperless.parsing.tesseract", level="WARNING") as cm: + page_count = parser.get_page_count( + os.path.join(self.SAMPLE_FILES, "password-protected.pdf"), + "application/pdf", + ) + self.assertEqual(page_count, None) + self.assertIn("Unable to determine PDF page count", cm.output[0]) + def test_thumbnail(self): parser = RasterisedDocumentParser(uuid.uuid4()) thumb = parser.get_thumbnail(