mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			6 Commits
		
	
	
		
			fb9200a344
			...
			0b7b6cb607
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0b7b6cb607 | ||
|   | 764ad059d1 | ||
|   | 5e47069934 | ||
|   | 4ff09c4cf4 | ||
|   | 53b393dab5 | ||
|   | 6119c215e7 | 
							
								
								
									
										140
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										140
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,11 +17,52 @@ env: | |||||||
|   DEFAULT_PYTHON_VERSION: "3.11" |   DEFAULT_PYTHON_VERSION: "3.11" | ||||||
|   NLTK_DATA: "/usr/share/nltk_data" |   NLTK_DATA: "/usr/share/nltk_data" | ||||||
| jobs: | jobs: | ||||||
|  |   detect-duplicate: | ||||||
|  |     name: Detect Duplicate Run | ||||||
|  |     runs-on: ubuntu-24.04 | ||||||
|  |     outputs: | ||||||
|  |       should_run: ${{ steps.check.outputs.should_run }} | ||||||
|  |     steps: | ||||||
|  |       - name: Check if workflow should run | ||||||
|  |         id: check | ||||||
|  |         uses: actions/github-script@v7 | ||||||
|  |         with: | ||||||
|  |           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           script: | | ||||||
|  |             if (context.eventName !== 'push') { | ||||||
|  |               core.info('Not a push event; running workflow.'); | ||||||
|  |               core.setOutput('should_run', 'true'); | ||||||
|  |               return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             const ref = context.ref || ''; | ||||||
|  |             if (!ref.startsWith('refs/heads/')) { | ||||||
|  |               core.info('Push is not to a branch; running workflow.'); | ||||||
|  |               core.setOutput('should_run', 'true'); | ||||||
|  |               return; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             const branch = ref.substring('refs/heads/'.length); | ||||||
|  |             const { owner, repo } = context.repo; | ||||||
|  |             const prs = await github.paginate(github.rest.pulls.list, { | ||||||
|  |               owner, | ||||||
|  |               repo, | ||||||
|  |               state: 'open', | ||||||
|  |               head: `${owner}:${branch}`, | ||||||
|  |               per_page: 100, | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             if (prs.length === 0) { | ||||||
|  |               core.info(`No open PR found for ${branch}; running workflow.`); | ||||||
|  |               core.setOutput('should_run', 'true'); | ||||||
|  |             } else { | ||||||
|  |               core.info(`Found ${prs.length} open PR(s) for ${branch}; skipping duplicate push run.`); | ||||||
|  |               core.setOutput('should_run', 'false'); | ||||||
|  |             } | ||||||
|   pre-commit: |   pre-commit: | ||||||
|     # We want to run on external PRs, but not on our own internal PRs as they'll be run |     needs: | ||||||
|     # by the push to the branch. Without this if check, checks are duplicated since |       - detect-duplicate | ||||||
|     # internal PRs match both the push and pull_request events. |     if: needs.detect-duplicate.outputs.should_run == 'true' | ||||||
|     if: github.event_name == 'push' || github.event.pull_request.head.repo.full_name != github.repository |  | ||||||
|     name: Linting Checks |     name: Linting Checks | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|     steps: |     steps: | ||||||
| @@ -151,18 +192,6 @@ jobs: | |||||||
|           token: ${{ secrets.CODECOV_TOKEN }} |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|           flags: backend-python-${{ matrix.python-version }} |           flags: backend-python-${{ matrix.python-version }} | ||||||
|           files: coverage.xml |           files: coverage.xml | ||||||
|       - name: Upload coverage artifacts |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         if: always() |  | ||||||
|         with: |  | ||||||
|           name: backend-coverage-${{ matrix.python-version }} |  | ||||||
|           path: | |  | ||||||
|             .coverage |  | ||||||
|             coverage.xml |  | ||||||
|             junit.xml |  | ||||||
|           retention-days: 1 |  | ||||||
|           include-hidden-files: true |  | ||||||
|           if-no-files-found: error |  | ||||||
|       - name: Stop containers |       - name: Stop containers | ||||||
|         if: always() |         if: always() | ||||||
|         run: | |         run: | | ||||||
| @@ -245,17 +274,6 @@ jobs: | |||||||
|           token: ${{ secrets.CODECOV_TOKEN }} |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|           flags: frontend-node-${{ matrix.node-version }} |           flags: frontend-node-${{ matrix.node-version }} | ||||||
|           directory: src-ui/coverage/ |           directory: src-ui/coverage/ | ||||||
|       - name: Upload coverage artifacts |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         if: always() |  | ||||||
|         with: |  | ||||||
|           name: frontend-coverage-${{ matrix.shard-index }} |  | ||||||
|           path: | |  | ||||||
|             src-ui/coverage/lcov.info |  | ||||||
|             src-ui/coverage/coverage-final.json |  | ||||||
|             src-ui/junit.xml |  | ||||||
|           retention-days: 1 |  | ||||||
|           if-no-files-found: error |  | ||||||
|   tests-frontend-e2e: |   tests-frontend-e2e: | ||||||
|     name: "Frontend E2E Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" |     name: "Frontend E2E Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
| @@ -336,74 +354,6 @@ jobs: | |||||||
|         env: |         env: | ||||||
|           CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} |           CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} | ||||||
|         run: cd src-ui && pnpm run build --configuration=production |         run: cd src-ui && pnpm run build --configuration=production | ||||||
|   sonarqube-analysis: |  | ||||||
|     name: "SonarQube Analysis" |  | ||||||
|     runs-on: ubuntu-24.04 |  | ||||||
|     needs: |  | ||||||
|       - tests-backend |  | ||||||
|       - tests-frontend |  | ||||||
|     if: github.repository_owner == 'paperless-ngx' |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout |  | ||||||
|         uses: actions/checkout@v5 |  | ||||||
|         with: |  | ||||||
|           fetch-depth: 0 |  | ||||||
|       - name: Download all backend coverage |  | ||||||
|         uses: actions/download-artifact@v5.0.0 |  | ||||||
|         with: |  | ||||||
|           pattern: backend-coverage-* |  | ||||||
|           path: ./coverage/ |  | ||||||
|       - name: Download all frontend coverage |  | ||||||
|         uses: actions/download-artifact@v5.0.0 |  | ||||||
|         with: |  | ||||||
|           pattern: frontend-coverage-* |  | ||||||
|           path: ./coverage/ |  | ||||||
|       - name: Set up Python |  | ||||||
|         uses: actions/setup-python@v5 |  | ||||||
|         with: |  | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |  | ||||||
|       - name: Install coverage tools |  | ||||||
|         run: | |  | ||||||
|           pip install coverage |  | ||||||
|           npm install -g nyc |  | ||||||
|       # Merge backend coverage from all Python versions |  | ||||||
|       - name: Merge backend coverage |  | ||||||
|         run: | |  | ||||||
|           coverage combine coverage/backend-coverage-*/.coverage |  | ||||||
|           coverage xml -o merged-backend-coverage.xml |  | ||||||
|       # Merge frontend coverage from all shards |  | ||||||
|       - name: Merge frontend coverage |  | ||||||
|         run: | |  | ||||||
|           # Find all coverage-final.json files from the shards, exit with error if none found |  | ||||||
|           shopt -s nullglob |  | ||||||
|           files=(coverage/frontend-coverage-*/coverage/coverage-final.json) |  | ||||||
|           if [ ${#files[@]} -eq 0 ]; then |  | ||||||
|             echo "No frontend coverage JSON found under coverage/" >&2 |  | ||||||
|             exit 1 |  | ||||||
|           fi |  | ||||||
|           # Create .nyc_output directory and copy each shard's coverage JSON into it with a unique name |  | ||||||
|           mkdir -p .nyc_output |  | ||||||
|           for coverage_json in "${files[@]}"; do |  | ||||||
|             shard=$(basename "$(dirname "$(dirname "$coverage_json")")") |  | ||||||
|             cp "$coverage_json" ".nyc_output/${shard}.json" |  | ||||||
|           done |  | ||||||
|           npx nyc merge .nyc_output .nyc_output/out.json |  | ||||||
|           npx nyc report --reporter=lcovonly --report-dir coverage |  | ||||||
|       - name: Upload coverage artifacts |  | ||||||
|         uses: actions/upload-artifact@v4.6.2 |  | ||||||
|         with: |  | ||||||
|           name: merged-coverage |  | ||||||
|           path: | |  | ||||||
|             merged-backend-coverage.xml |  | ||||||
|             .nyc_output/* |  | ||||||
|             coverage/lcov.info |  | ||||||
|           retention-days: 7 |  | ||||||
|           if-no-files-found: error |  | ||||||
|           include-hidden-files: true |  | ||||||
|       - name: SonarQube Analysis |  | ||||||
|         uses: SonarSource/sonarqube-scan-action@v5 |  | ||||||
|         env: |  | ||||||
|           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} |  | ||||||
|   build-docker-image: |   build-docker-image: | ||||||
|     name: Build Docker image for ${{ github.ref_name }} |     name: Build Docker image for ${{ github.ref_name }} | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|   | |||||||
| @@ -255,7 +255,6 @@ PAPERLESS_DISABLE_DBHANDLER = "true" | |||||||
| PAPERLESS_CACHE_BACKEND = "django.core.cache.backends.locmem.LocMemCache" | PAPERLESS_CACHE_BACKEND = "django.core.cache.backends.locmem.LocMemCache" | ||||||
|  |  | ||||||
| [tool.coverage.run] | [tool.coverage.run] | ||||||
| relative_files = true |  | ||||||
| source = [ | source = [ | ||||||
|   "src/", |   "src/", | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| sonar.projectKey=paperless-ngx_paperless-ngx |  | ||||||
| sonar.organization=paperless-ngx |  | ||||||
| sonar.projectName=Paperless-ngx |  | ||||||
| sonar.projectVersion=1.0 |  | ||||||
|  |  | ||||||
| # Source and test directories |  | ||||||
| sonar.sources=src/,src-ui/ |  | ||||||
| sonar.test.inclusions=**/test_*.py,**/tests.py,**/*.spec.ts,**/*.test.ts |  | ||||||
|  |  | ||||||
| # Language specific settings |  | ||||||
| sonar.python.version=3.10,3.11,3.12,3.13 |  | ||||||
|  |  | ||||||
| # Coverage reports |  | ||||||
| sonar.python.coverage.reportPaths=merged-backend-coverage.xml |  | ||||||
| sonar.javascript.lcov.reportPaths=coverage/lcov.info |  | ||||||
|  |  | ||||||
| # Test execution reports |  | ||||||
| sonar.junit.reportPaths=**/junit.xml,**/test-results.xml |  | ||||||
|  |  | ||||||
| # Encoding |  | ||||||
| sonar.sourceEncoding=UTF-8 |  | ||||||
|  |  | ||||||
| # Exclusions |  | ||||||
| sonar.exclusions=**/migrations/**,**/node_modules/**,**/static/**,**/venv/**,**/.venv/**,**/dist/** |  | ||||||
| @@ -177,10 +177,16 @@ export class CustomFieldEditDialogComponent | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   public removeSelectOption(index: number) { |   public removeSelectOption(index: number) { | ||||||
|     this.selectOptions.removeAt(index) |     const globalIndex = | ||||||
|     this._allSelectOptions.splice( |       index + (this.selectOptionsPage - 1) * SELECT_OPTION_PAGE_SIZE | ||||||
|       index + (this.selectOptionsPage - 1) * SELECT_OPTION_PAGE_SIZE, |     this._allSelectOptions.splice(globalIndex, 1) | ||||||
|       1 |  | ||||||
|  |     const totalPages = Math.max( | ||||||
|  |       1, | ||||||
|  |       Math.ceil(this._allSelectOptions.length / SELECT_OPTION_PAGE_SIZE) | ||||||
|     ) |     ) | ||||||
|  |     const targetPage = Math.min(this.selectOptionsPage, totalPages) | ||||||
|  |  | ||||||
|  |     this.selectOptionsPage = targetPage | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -164,6 +164,9 @@ class BarcodePlugin(ConsumeTaskPlugin): | |||||||
|                         mailrule_id=self.input_doc.mailrule_id, |                         mailrule_id=self.input_doc.mailrule_id, | ||||||
|                         # Can't use same folder or the consume might grab it again |                         # Can't use same folder or the consume might grab it again | ||||||
|                         original_file=(tmp_dir / new_document.name).resolve(), |                         original_file=(tmp_dir / new_document.name).resolve(), | ||||||
|  |                         # Adding optional original_path for later uses in | ||||||
|  |                         # workflow matching | ||||||
|  |                         original_path=self.input_doc.original_file, | ||||||
|                     ), |                     ), | ||||||
|                     # All the same metadata |                     # All the same metadata | ||||||
|                     self.metadata, |                     self.metadata, | ||||||
|   | |||||||
| @@ -156,6 +156,7 @@ class ConsumableDocument: | |||||||
|  |  | ||||||
|     source: DocumentSource |     source: DocumentSource | ||||||
|     original_file: Path |     original_file: Path | ||||||
|  |     original_path: Path | None = None | ||||||
|     mailrule_id: int | None = None |     mailrule_id: int | None = None | ||||||
|     mime_type: str = dataclasses.field(init=False, default=None) |     mime_type: str = dataclasses.field(init=False, default=None) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -92,6 +92,9 @@ class Command(MultiProcessMixin, ProgressBarMixin, BaseCommand): | |||||||
|                 # doc to doc is obviously not useful |                 # doc to doc is obviously not useful | ||||||
|                 if first_doc.pk == second_doc.pk: |                 if first_doc.pk == second_doc.pk: | ||||||
|                     continue |                     continue | ||||||
|  |                 # Skip empty documents (e.g. password-protected) | ||||||
|  |                 if first_doc.content.strip() == "" or second_doc.content.strip() == "": | ||||||
|  |                     continue | ||||||
|                 # Skip matching which have already been matched together |                 # Skip matching which have already been matched together | ||||||
|                 # doc 1 to doc 2 is the same as doc 2 to doc 1 |                 # doc 1 to doc 2 is the same as doc 2 to doc 1 | ||||||
|                 doc_1_to_doc_2 = (first_doc.pk, second_doc.pk) |                 doc_1_to_doc_2 = (first_doc.pk, second_doc.pk) | ||||||
|   | |||||||
| @@ -314,11 +314,19 @@ def consumable_document_matches_workflow( | |||||||
|         trigger_matched = False |         trigger_matched = False | ||||||
|  |  | ||||||
|     # Document path vs trigger path |     # Document path vs trigger path | ||||||
|  |  | ||||||
|  |     # Use the original_path if set, else us the original_file | ||||||
|  |     match_against = ( | ||||||
|  |         document.original_path | ||||||
|  |         if document.original_path is not None | ||||||
|  |         else document.original_file | ||||||
|  |     ) | ||||||
|  |  | ||||||
|     if ( |     if ( | ||||||
|         trigger.filter_path is not None |         trigger.filter_path is not None | ||||||
|         and len(trigger.filter_path) > 0 |         and len(trigger.filter_path) > 0 | ||||||
|         and not fnmatch( |         and not fnmatch( | ||||||
|             document.original_file, |             match_against, | ||||||
|             trigger.filter_path, |             trigger.filter_path, | ||||||
|         ) |         ) | ||||||
|     ): |     ): | ||||||
|   | |||||||
| @@ -614,14 +614,16 @@ class TestBarcodeNewConsume( | |||||||
|             self.assertIsNotFile(temp_copy) |             self.assertIsNotFile(temp_copy) | ||||||
|  |  | ||||||
|             # Check the split files exist |             # Check the split files exist | ||||||
|  |             # Check the original_path is set | ||||||
|             # Check the source is unchanged |             # Check the source is unchanged | ||||||
|             # Check the overrides are unchanged |             # Check the overrides are unchanged | ||||||
|             for ( |             for ( | ||||||
|                 new_input_doc, |                 new_input_doc, | ||||||
|                 new_doc_overrides, |                 new_doc_overrides, | ||||||
|             ) in self.get_all_consume_delay_call_args(): |             ) in self.get_all_consume_delay_call_args(): | ||||||
|                 self.assertEqual(new_input_doc.source, DocumentSource.ConsumeFolder) |  | ||||||
|                 self.assertIsFile(new_input_doc.original_file) |                 self.assertIsFile(new_input_doc.original_file) | ||||||
|  |                 self.assertEqual(new_input_doc.original_path, temp_copy) | ||||||
|  |                 self.assertEqual(new_input_doc.source, DocumentSource.ConsumeFolder) | ||||||
|                 self.assertEqual(overrides, new_doc_overrides) |                 self.assertEqual(overrides, new_doc_overrides) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -206,3 +206,29 @@ class TestFuzzyMatchCommand(TestCase): | |||||||
|         self.assertEqual(Document.objects.count(), 2) |         self.assertEqual(Document.objects.count(), 2) | ||||||
|         self.assertIsNotNone(Document.objects.get(pk=1)) |         self.assertIsNotNone(Document.objects.get(pk=1)) | ||||||
|         self.assertIsNotNone(Document.objects.get(pk=2)) |         self.assertIsNotNone(Document.objects.get(pk=2)) | ||||||
|  |  | ||||||
|  |     def test_empty_content(self): | ||||||
|  |         """ | ||||||
|  |         GIVEN: | ||||||
|  |             - 2 documents exist, content is empty (pw-protected) | ||||||
|  |         WHEN: | ||||||
|  |             - Command is called | ||||||
|  |         THEN: | ||||||
|  |             - No matches are found | ||||||
|  |         """ | ||||||
|  |         Document.objects.create( | ||||||
|  |             checksum="BEEFCAFE", | ||||||
|  |             title="A", | ||||||
|  |             content="", | ||||||
|  |             mime_type="application/pdf", | ||||||
|  |             filename="test.pdf", | ||||||
|  |         ) | ||||||
|  |         Document.objects.create( | ||||||
|  |             checksum="DEADBEAF", | ||||||
|  |             title="A", | ||||||
|  |             content="", | ||||||
|  |             mime_type="application/pdf", | ||||||
|  |             filename="other_test.pdf", | ||||||
|  |         ) | ||||||
|  |         stdout, _ = self.call_command() | ||||||
|  |         self.assertIn("No matches found", stdout) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user