mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			18 Commits
		
	
	
		
			6aeb5a5503
			...
			bd73555ecc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd73555ecc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 613c922dd2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1659aa08e4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 68dfb4a930 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3c439b970f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 962f7994d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 93eea80f3e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5bc27eb4b2 | ||
|   | b19701cb96 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c552bc2d7 | ||
|   | 80fabb0b56 | ||
|   | af1c235af5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 92ee906701 | ||
|   | d6710de486 | ||
|   | f71b13b82a | ||
|   | 3df43d828a | ||
|   | 643e2b4a8e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6fa896df39 | 
							
								
								
									
										22
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -25,7 +25,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Check if workflow should run |       - name: Check if workflow should run | ||||||
|         id: check |         id: check | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           github-token: ${{ secrets.GITHUB_TOKEN }} |           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           script: | |           script: | | ||||||
| @@ -69,7 +69,7 @@ jobs: | |||||||
|       - name: Checkout repository |       - name: Checkout repository | ||||||
|         uses: actions/checkout@v5 |         uses: actions/checkout@v5 | ||||||
|       - name: Install python |       - name: Install python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Check files |       - name: Check files | ||||||
| @@ -84,7 +84,7 @@ jobs: | |||||||
|         uses: actions/checkout@v5 |         uses: actions/checkout@v5 | ||||||
|       - name: Set up Python |       - name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |       - name: Install uv | ||||||
| @@ -138,7 +138,7 @@ jobs: | |||||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml up --detach |           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml up --detach | ||||||
|       - name: Set up Python |       - name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|         with: |         with: | ||||||
|           python-version: "${{ matrix.python-version }}" |           python-version: "${{ matrix.python-version }}" | ||||||
|       - name: Install uv |       - name: Install uv | ||||||
| @@ -207,7 +207,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           version: 10 |           version: 10 | ||||||
|       - name: Use Node.js 20 |       - name: Use Node.js 20 | ||||||
|         uses: actions/setup-node@v4 |         uses: actions/setup-node@v5 | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'pnpm' | ||||||
| @@ -240,7 +240,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           version: 10 |           version: 10 | ||||||
|       - name: Use Node.js 20 |       - name: Use Node.js 20 | ||||||
|         uses: actions/setup-node@v4 |         uses: actions/setup-node@v5 | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'pnpm' | ||||||
| @@ -288,7 +288,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           version: 10 |           version: 10 | ||||||
|       - name: Use Node.js 20 |       - name: Use Node.js 20 | ||||||
|         uses: actions/setup-node@v4 |         uses: actions/setup-node@v5 | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'pnpm' | ||||||
| @@ -331,7 +331,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           version: 10 |           version: 10 | ||||||
|       - name: Use Node.js 20 |       - name: Use Node.js 20 | ||||||
|         uses: actions/setup-node@v4 |         uses: actions/setup-node@v5 | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'pnpm' | ||||||
| @@ -473,7 +473,7 @@ jobs: | |||||||
|         uses: actions/checkout@v5 |         uses: actions/checkout@v5 | ||||||
|       - name: Set up Python |       - name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |       - name: Install uv | ||||||
| @@ -621,7 +621,7 @@ jobs: | |||||||
|           ref: main |           ref: main | ||||||
|       - name: Set up Python |       - name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |       - name: Install uv | ||||||
| @@ -653,7 +653,7 @@ jobs: | |||||||
|           git commit -am "Changelog ${{ needs.publish-release.outputs.version }} - GHA" |           git commit -am "Changelog ${{ needs.publish-release.outputs.version }} - GHA" | ||||||
|           git push origin ${{ needs.publish-release.outputs.version }}-changelog |           git push origin ${{ needs.publish-release.outputs.version }}-changelog | ||||||
|       - name: Create Pull Request |       - name: Create Pull Request | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const { repo, owner } = context.repo; |             const { repo, owner } = context.repo; | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Label PR by file path or branch name |       - name: Label PR by file path or branch name | ||||||
|         # see .github/labeler.yml for the labeler config |         # see .github/labeler.yml for the labeler config | ||||||
|         uses: actions/labeler@v5 |         uses: actions/labeler@v6 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} |           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Label by size |       - name: Label by size | ||||||
| @@ -26,7 +26,7 @@ jobs: | |||||||
|           fail_if_xl: 'false' |           fail_if_xl: 'false' | ||||||
|           excluded_files: /\.lock$/ /\.txt$/ ^src-ui/pnpm-lock\.yaml$ ^src-ui/messages\.xlf$ ^src/locale/en_US/LC_MESSAGES/django\.po$ |           excluded_files: /\.lock$/ /\.txt$/ ^src-ui/pnpm-lock\.yaml$ ^src-ui/messages\.xlf$ ^src/locale/en_US/LC_MESSAGES/django\.po$ | ||||||
|       - name: Label by PR title |       - name: Label by PR title | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const pr = context.payload.pull_request; |             const pr = context.payload.pull_request; | ||||||
| @@ -52,7 +52,7 @@ jobs: | |||||||
|             } |             } | ||||||
|       - name: Label bot-generated PRs |       - name: Label bot-generated PRs | ||||||
|         if: ${{ contains(github.actor, 'dependabot') || contains(github.actor, 'crowdin-bot') }} |         if: ${{ contains(github.actor, 'dependabot') || contains(github.actor, 'crowdin-bot') }} | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const pr = context.payload.pull_request; |             const pr = context.payload.pull_request; | ||||||
| @@ -77,7 +77,7 @@ jobs: | |||||||
|             } |             } | ||||||
|       - name: Welcome comment |       - name: Welcome comment | ||||||
|         if: ${{ !contains(github.actor, 'bot') }} |         if: ${{ !contains(github.actor, 'bot') }} | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const pr = context.payload.pull_request; |             const pr = context.payload.pull_request; | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,7 @@ jobs: | |||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/stale@v9 |       - uses: actions/stale@v10 | ||||||
|         with: |         with: | ||||||
|           days-before-stale: 7 |           days-before-stale: 7 | ||||||
|           days-before-close: 14 |           days-before-close: 14 | ||||||
| @@ -57,7 +57,7 @@ jobs: | |||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v7 |       - uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
| @@ -114,7 +114,7 @@ jobs: | |||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v7 |       - uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
| @@ -206,7 +206,7 @@ jobs: | |||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-24.04 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v7 |       - uses: actions/github-script@v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,7 +17,7 @@ jobs: | |||||||
|           ref: ${{ github.head_ref }} |           ref: ${{ github.head_ref }} | ||||||
|       - name: Set up Python |       - name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v6 | ||||||
|       - name: Install system dependencies |       - name: Install system dependencies | ||||||
|         run: | |         run: | | ||||||
|           sudo apt-get update -qq |           sudo apt-get update -qq | ||||||
| @@ -38,7 +38,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           version: 10 |           version: 10 | ||||||
|       - name: Use Node.js 20 |       - name: Use Node.js 20 | ||||||
|         uses: actions/setup-node@v4 |         uses: actions/setup-node@v5 | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'pnpm' | ||||||
|   | |||||||
| @@ -49,7 +49,7 @@ repos: | |||||||
|           - 'prettier-plugin-organize-imports@4.1.0' |           - 'prettier-plugin-organize-imports@4.1.0' | ||||||
|   # Python hooks |   # Python hooks | ||||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit |   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||||
|     rev: v0.13.0 |     rev: v0.13.2 | ||||||
|     hooks: |     hooks: | ||||||
|       - id: ruff-check |       - id: ruff-check | ||||||
|       - id: ruff-format |       - id: ruff-format | ||||||
| @@ -59,7 +59,7 @@ repos: | |||||||
|       - id: pyproject-fmt |       - id: pyproject-fmt | ||||||
|   # Dockerfile hooks |   # Dockerfile hooks | ||||||
|   - repo: https://github.com/AleksaC/hadolint-py |   - repo: https://github.com/AleksaC/hadolint-py | ||||||
|     rev: v2.12.1b3 |     rev: v2.14.0 | ||||||
|     hooks: |     hooks: | ||||||
|       - id: hadolint |       - id: hadolint | ||||||
|   # Shell script hooks |   # Shell script hooks | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ services: | |||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:17 |     image: docker.io/library/postgres:18 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql/data |       - pgdata:/var/lib/postgresql/data | ||||||
|   | |||||||
| @@ -35,7 +35,7 @@ services: | |||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:17 |     image: docker.io/library/postgres:18 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql/data |       - pgdata:/var/lib/postgresql/data | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ services: | |||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:17 |     image: docker.io/library/postgres:18 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql/data |       - pgdata:/var/lib/postgresql/data | ||||||
|   | |||||||
| @@ -637,7 +637,7 @@ When you first delete a document it is moved to the 'trash' until either it is e | |||||||
| You can set how long documents remain in the trash before being automatically deleted with [`PAPERLESS_EMPTY_TRASH_DELAY`](configuration.md#PAPERLESS_EMPTY_TRASH_DELAY), which defaults | You can set how long documents remain in the trash before being automatically deleted with [`PAPERLESS_EMPTY_TRASH_DELAY`](configuration.md#PAPERLESS_EMPTY_TRASH_DELAY), which defaults | ||||||
| to 30 days. Until the file is actually deleted (e.g. the trash is emptied), all files and database content remains intact and can be restored at any point up until that time. | to 30 days. Until the file is actually deleted (e.g. the trash is emptied), all files and database content remains intact and can be restored at any point up until that time. | ||||||
|  |  | ||||||
| Additionally you may configure a directory where deleted files are moved to when they the trash is emptied with [`PAPERLESS_EMPTY_TRASH_DIR`](configuration.md#PAPERLESS_EMPTY_TRASH_DIR). | Additionally you may configure a directory where deleted files are moved to when the trash is emptied with [`PAPERLESS_EMPTY_TRASH_DIR`](configuration.md#PAPERLESS_EMPTY_TRASH_DIR). | ||||||
| Note that the empty trash directory only stores the original file, the archive file and all database information is permanently removed once a document is fully deleted. | Note that the empty trash directory only stores the original file, the archive file and all database information is permanently removed once a document is fully deleted. | ||||||
|  |  | ||||||
| ## Best practices {#basic-searching} | ## Best practices {#basic-searching} | ||||||
|   | |||||||
| @@ -54,7 +54,6 @@ dependencies = [ | |||||||
|   "ocrmypdf~=16.11.0", |   "ocrmypdf~=16.11.0", | ||||||
|   "pathvalidate~=3.3.1", |   "pathvalidate~=3.3.1", | ||||||
|   "pdf2image~=1.17.0", |   "pdf2image~=1.17.0", | ||||||
|   "psycopg-pool", |  | ||||||
|   "python-dateutil~=2.9.0", |   "python-dateutil~=2.9.0", | ||||||
|   "python-dotenv~=1.1.0", |   "python-dotenv~=1.1.0", | ||||||
|   "python-gnupg~=0.5.4", |   "python-gnupg~=0.5.4", | ||||||
|   | |||||||
| @@ -5,14 +5,14 @@ | |||||||
|       <trans-unit id="ngb.alert.close" datatype="html"> |       <trans-unit id="ngb.alert.close" datatype="html"> | ||||||
|         <source>Close</source> |         <source>Close</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/alert/alert.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/alert/alert.ts</context> | ||||||
|           <context context-type="linenumber">50</context> |           <context context-type="linenumber">50</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.carousel.slide-number" datatype="html"> |       <trans-unit id="ngb.carousel.slide-number" datatype="html"> | ||||||
|         <source> Slide <x id="INTERPOLATION" equiv-text="ueryList<NgbSli"/> of <x id="INTERPOLATION_1" equiv-text="EventSource = N"/> </source> |         <source> Slide <x id="INTERPOLATION" equiv-text="ueryList<NgbSli"/> of <x id="INTERPOLATION_1" equiv-text="EventSource = N"/> </source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/carousel/carousel.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/carousel/carousel.ts</context> | ||||||
|           <context context-type="linenumber">131,135</context> |           <context context-type="linenumber">131,135</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|         <note priority="1" from="description">Currently selected slide number read by screen reader</note> |         <note priority="1" from="description">Currently selected slide number read by screen reader</note> | ||||||
| @@ -20,212 +20,212 @@ | |||||||
|       <trans-unit id="ngb.carousel.previous" datatype="html"> |       <trans-unit id="ngb.carousel.previous" datatype="html"> | ||||||
|         <source>Previous</source> |         <source>Previous</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/carousel/carousel.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/carousel/carousel.ts</context> | ||||||
|           <context context-type="linenumber">157,159</context> |           <context context-type="linenumber">157,159</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.carousel.next" datatype="html"> |       <trans-unit id="ngb.carousel.next" datatype="html"> | ||||||
|         <source>Next</source> |         <source>Next</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/carousel/carousel.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/carousel/carousel.ts</context> | ||||||
|           <context context-type="linenumber">198</context> |           <context context-type="linenumber">198</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.datepicker.previous-month" datatype="html"> |       <trans-unit id="ngb.datepicker.previous-month" datatype="html"> | ||||||
|         <source>Previous month</source> |         <source>Previous month</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/datepicker/datepicker-navigation.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/datepicker/datepicker-navigation.ts</context> | ||||||
|           <context context-type="linenumber">83,85</context> |           <context context-type="linenumber">83,85</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/datepicker/datepicker-navigation.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/datepicker/datepicker-navigation.ts</context> | ||||||
|           <context context-type="linenumber">112</context> |           <context context-type="linenumber">112</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.datepicker.next-month" datatype="html"> |       <trans-unit id="ngb.datepicker.next-month" datatype="html"> | ||||||
|         <source>Next month</source> |         <source>Next month</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/datepicker/datepicker-navigation.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/datepicker/datepicker-navigation.ts</context> | ||||||
|           <context context-type="linenumber">112</context> |           <context context-type="linenumber">112</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/datepicker/datepicker-navigation.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/datepicker/datepicker-navigation.ts</context> | ||||||
|           <context context-type="linenumber">112</context> |           <context context-type="linenumber">112</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.HH" datatype="html"> |       <trans-unit id="ngb.timepicker.HH" datatype="html"> | ||||||
|         <source>HH</source> |         <source>HH</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.toast.close-aria" datatype="html"> |       <trans-unit id="ngb.toast.close-aria" datatype="html"> | ||||||
|         <source>Close</source> |         <source>Close</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.datepicker.select-month" datatype="html"> |       <trans-unit id="ngb.datepicker.select-month" datatype="html"> | ||||||
|         <source>Select month</source> |         <source>Select month</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.first" datatype="html"> |       <trans-unit id="ngb.pagination.first" datatype="html"> | ||||||
|         <source>««</source> |         <source>««</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.hours" datatype="html"> |       <trans-unit id="ngb.timepicker.hours" datatype="html"> | ||||||
|         <source>Hours</source> |         <source>Hours</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.previous" datatype="html"> |       <trans-unit id="ngb.pagination.previous" datatype="html"> | ||||||
|         <source>«</source> |         <source>«</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.MM" datatype="html"> |       <trans-unit id="ngb.timepicker.MM" datatype="html"> | ||||||
|         <source>MM</source> |         <source>MM</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.next" datatype="html"> |       <trans-unit id="ngb.pagination.next" datatype="html"> | ||||||
|         <source>»</source> |         <source>»</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.datepicker.select-year" datatype="html"> |       <trans-unit id="ngb.datepicker.select-year" datatype="html"> | ||||||
|         <source>Select year</source> |         <source>Select year</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.minutes" datatype="html"> |       <trans-unit id="ngb.timepicker.minutes" datatype="html"> | ||||||
|         <source>Minutes</source> |         <source>Minutes</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.last" datatype="html"> |       <trans-unit id="ngb.pagination.last" datatype="html"> | ||||||
|         <source>»»</source> |         <source>»»</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.first-aria" datatype="html"> |       <trans-unit id="ngb.pagination.first-aria" datatype="html"> | ||||||
|         <source>First</source> |         <source>First</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.increment-hours" datatype="html"> |       <trans-unit id="ngb.timepicker.increment-hours" datatype="html"> | ||||||
|         <source>Increment hours</source> |         <source>Increment hours</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.previous-aria" datatype="html"> |       <trans-unit id="ngb.pagination.previous-aria" datatype="html"> | ||||||
|         <source>Previous</source> |         <source>Previous</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.decrement-hours" datatype="html"> |       <trans-unit id="ngb.timepicker.decrement-hours" datatype="html"> | ||||||
|         <source>Decrement hours</source> |         <source>Decrement hours</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.next-aria" datatype="html"> |       <trans-unit id="ngb.pagination.next-aria" datatype="html"> | ||||||
|         <source>Next</source> |         <source>Next</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.increment-minutes" datatype="html"> |       <trans-unit id="ngb.timepicker.increment-minutes" datatype="html"> | ||||||
|         <source>Increment minutes</source> |         <source>Increment minutes</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.pagination.last-aria" datatype="html"> |       <trans-unit id="ngb.pagination.last-aria" datatype="html"> | ||||||
|         <source>Last</source> |         <source>Last</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.decrement-minutes" datatype="html"> |       <trans-unit id="ngb.timepicker.decrement-minutes" datatype="html"> | ||||||
|         <source>Decrement minutes</source> |         <source>Decrement minutes</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.SS" datatype="html"> |       <trans-unit id="ngb.timepicker.SS" datatype="html"> | ||||||
|         <source>SS</source> |         <source>SS</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.seconds" datatype="html"> |       <trans-unit id="ngb.timepicker.seconds" datatype="html"> | ||||||
|         <source>Seconds</source> |         <source>Seconds</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.increment-seconds" datatype="html"> |       <trans-unit id="ngb.timepicker.increment-seconds" datatype="html"> | ||||||
|         <source>Increment seconds</source> |         <source>Increment seconds</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.decrement-seconds" datatype="html"> |       <trans-unit id="ngb.timepicker.decrement-seconds" datatype="html"> | ||||||
|         <source>Decrement seconds</source> |         <source>Decrement seconds</source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|       <trans-unit id="ngb.timepicker.PM" datatype="html"> |       <trans-unit id="ngb.timepicker.PM" datatype="html"> | ||||||
|         <source><x id="INTERPOLATION"/></source> |         <source><x id="INTERPOLATION"/></source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/ngb-config.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/ngb-config.ts</context> | ||||||
|           <context context-type="linenumber">13</context> |           <context context-type="linenumber">13</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
| @@ -233,7 +233,7 @@ | |||||||
|         <source><x id="INTERPOLATION" equiv-text="barConfig); |         <source><x id="INTERPOLATION" equiv-text="barConfig); | ||||||
| 	pu"/></source> | 	pu"/></source> | ||||||
|         <context-group purpose="location"> |         <context-group purpose="location"> | ||||||
|           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.2.4_@angular+core@20.2.4_@angular+_db9461b4835bfc9061e01150e14e6256/node_modules/src/progressbar/progressbar.ts</context> |           <context context-type="sourcefile">node_modules/.pnpm/@ng-bootstrap+ng-bootstrap@19.0.1_@angular+common@20.3.2_@angular+core@20.3.2_@angular+_4a8591e6ee586bf00b666f6438778cc7/node_modules/src/progressbar/progressbar.ts</context> | ||||||
|           <context context-type="linenumber">41,42</context> |           <context context-type="linenumber">41,42</context> | ||||||
|         </context-group> |         </context-group> | ||||||
|       </trans-unit> |       </trans-unit> | ||||||
|   | |||||||
| @@ -11,17 +11,17 @@ | |||||||
|   }, |   }, | ||||||
|   "private": true, |   "private": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@angular/cdk": "^20.2.2", |     "@angular/cdk": "^20.2.6", | ||||||
|     "@angular/common": "~20.2.4", |     "@angular/common": "~20.3.2", | ||||||
|     "@angular/compiler": "~20.2.4", |     "@angular/compiler": "~20.3.2", | ||||||
|     "@angular/core": "~20.2.4", |     "@angular/core": "~20.3.2", | ||||||
|     "@angular/forms": "~20.2.4", |     "@angular/forms": "~20.3.2", | ||||||
|     "@angular/localize": "~20.2.4", |     "@angular/localize": "~20.3.2", | ||||||
|     "@angular/platform-browser": "~20.2.4", |     "@angular/platform-browser": "~20.3.2", | ||||||
|     "@angular/platform-browser-dynamic": "~20.2.4", |     "@angular/platform-browser-dynamic": "~20.3.2", | ||||||
|     "@angular/router": "~20.2.4", |     "@angular/router": "~20.3.2", | ||||||
|     "@ng-bootstrap/ng-bootstrap": "^19.0.1", |     "@ng-bootstrap/ng-bootstrap": "^19.0.1", | ||||||
|     "@ng-select/ng-select": "^20.1.3", |     "@ng-select/ng-select": "^20.2.2", | ||||||
|     "@ngneat/dirty-check-forms": "^3.0.3", |     "@ngneat/dirty-check-forms": "^3.0.3", | ||||||
|     "@popperjs/core": "^2.11.8", |     "@popperjs/core": "^2.11.8", | ||||||
|     "bootstrap": "^5.3.8", |     "bootstrap": "^5.3.8", | ||||||
| @@ -29,47 +29,48 @@ | |||||||
|     "mime-names": "^1.0.0", |     "mime-names": "^1.0.0", | ||||||
|     "ng2-pdf-viewer": "^10.4.0", |     "ng2-pdf-viewer": "^10.4.0", | ||||||
|     "ngx-bootstrap-icons": "^1.9.3", |     "ngx-bootstrap-icons": "^1.9.3", | ||||||
|     "ngx-color": "^10.0.0", |     "ngx-color": "^10.1.0", | ||||||
|     "ngx-cookie-service": "^20.1.0", |     "ngx-cookie-service": "^20.1.0", | ||||||
|     "ngx-device-detector": "^10.1.0", |     "ngx-device-detector": "^10.1.0", | ||||||
|     "ngx-ui-tour-ng-bootstrap": "^17.0.1", |     "ngx-ui-tour-ng-bootstrap": "^17.0.1", | ||||||
|     "rxjs": "^7.8.2", |     "rxjs": "^7.8.2", | ||||||
|     "tslib": "^2.8.1", |     "tslib": "^2.8.1", | ||||||
|     "utif": "^3.1.0", |     "utif": "^3.1.0", | ||||||
|     "uuid": "^11.1.0", |     "uuid": "^13.0.0", | ||||||
|     "zone.js": "^0.15.1" |     "zone.js": "^0.15.1" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@angular-builders/custom-webpack": "^20.0.0", |     "@angular-builders/custom-webpack": "^20.0.0", | ||||||
|     "@angular-builders/jest": "^20.0.0", |     "@angular-builders/jest": "^20.0.0", | ||||||
|     "@angular-devkit/core": "^20.2.2", |     "@angular-devkit/core": "^20.3.3", | ||||||
|     "@angular-devkit/schematics": "^20.2.2", |     "@angular-devkit/schematics": "^20.3.3", | ||||||
|     "@angular-eslint/builder": "20.2.0", |     "@angular-eslint/builder": "20.3.0", | ||||||
|     "@angular-eslint/eslint-plugin": "20.2.0", |     "@angular-eslint/eslint-plugin": "20.3.0", | ||||||
|     "@angular-eslint/eslint-plugin-template": "20.2.0", |     "@angular-eslint/eslint-plugin-template": "20.3.0", | ||||||
|     "@angular-eslint/schematics": "20.2.0", |     "@angular-eslint/schematics": "20.3.0", | ||||||
|     "@angular-eslint/template-parser": "20.2.0", |     "@angular-eslint/template-parser": "20.3.0", | ||||||
|     "@angular/build": "^20.2.2", |     "@angular/build": "^20.3.3", | ||||||
|     "@angular/cli": "~20.2.2", |     "@angular/cli": "~20.3.3", | ||||||
|     "@angular/compiler-cli": "~20.2.4", |     "@angular/compiler-cli": "~20.3.2", | ||||||
|     "@codecov/webpack-plugin": "^1.9.1", |     "@codecov/webpack-plugin": "^1.9.1", | ||||||
|     "@playwright/test": "^1.55.0", |     "@playwright/test": "^1.55.1", | ||||||
|     "@types/jest": "^30.0.0", |     "@types/jest": "^30.0.0", | ||||||
|     "@types/node": "^24.3.0", |     "@types/node": "^24.6.1", | ||||||
|     "@typescript-eslint/eslint-plugin": "^8.41.0", |     "@typescript-eslint/eslint-plugin": "^8.45.0", | ||||||
|     "@typescript-eslint/parser": "^8.41.0", |     "@typescript-eslint/parser": "^8.45.0", | ||||||
|     "@typescript-eslint/utils": "^8.41.0", |     "@typescript-eslint/utils": "^8.45.0", | ||||||
|     "eslint": "^9.34.0", |     "eslint": "^9.36.0", | ||||||
|     "jest": "30.1.3", |     "jest": "30.2.0", | ||||||
|     "jest-environment-jsdom": "^30.1.2", |     "jest-environment-jsdom": "^30.2.0", | ||||||
|     "jest-junit": "^16.0.0", |     "jest-junit": "^16.0.0", | ||||||
|     "jest-preset-angular": "^15.0.0", |     "jest-preset-angular": "^15.0.2", | ||||||
|     "jest-websocket-mock": "^2.5.0", |     "jest-websocket-mock": "^2.5.0", | ||||||
|     "prettier-plugin-organize-imports": "^4.2.0", |     "prettier-plugin-organize-imports": "^4.3.0", | ||||||
|     "ts-node": "~10.9.1", |     "ts-node": "~10.9.1", | ||||||
|     "typescript": "^5.8.3", |     "typescript": "^5.8.3", | ||||||
|     "webpack": "^5.101.3" |     "webpack": "^5.102.0" | ||||||
|   }, |   }, | ||||||
|  |   "packageManager": "pnpm@10.17.1", | ||||||
|   "pnpm": { |   "pnpm": { | ||||||
|     "onlyBuiltDependencies": [ |     "onlyBuiltDependencies": [ | ||||||
|       "@parcel/watcher", |       "@parcel/watcher", | ||||||
|   | |||||||
							
								
								
									
										3498
									
								
								src-ui/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										3498
									
								
								src-ui/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -145,4 +145,14 @@ HTMLCanvasElement.prototype.getContext = < | |||||||
|   typeof HTMLCanvasElement.prototype.getContext |   typeof HTMLCanvasElement.prototype.getContext | ||||||
| >jest.fn() | >jest.fn() | ||||||
|  |  | ||||||
|  | jest.mock('uuid', () => ({ | ||||||
|  |   v4: jest.fn(() => | ||||||
|  |     'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (char: string) => { | ||||||
|  |       const random = Math.floor(Math.random() * 16) | ||||||
|  |       const value = char === 'x' ? random : (random & 0x3) | 0x8 | ||||||
|  |       return value.toString(16) | ||||||
|  |     }) | ||||||
|  |   ), | ||||||
|  | })) | ||||||
|  |  | ||||||
| jest.mock('pdfjs-dist') | jest.mock('pdfjs-dist') | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import re | |||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from decimal import Decimal | from decimal import Decimal | ||||||
| from typing import TYPE_CHECKING | from typing import TYPE_CHECKING | ||||||
|  | from typing import Literal | ||||||
|  |  | ||||||
| import magic | import magic | ||||||
| from celery import states | from celery import states | ||||||
| @@ -252,6 +253,35 @@ class OwnedObjectSerializer( | |||||||
|             except KeyError: |             except KeyError: | ||||||
|                 pass |                 pass | ||||||
|  |  | ||||||
|  |     def _get_perms(self, obj, codename: str, target: Literal["users", "groups"]): | ||||||
|  |         """ | ||||||
|  |         Get the given permissions from context or from django-guardian. | ||||||
|  |  | ||||||
|  |         :param codename: The permission codename, e.g. 'view' or 'change' | ||||||
|  |         :param target: 'users' or 'groups' | ||||||
|  |         """ | ||||||
|  |         key = f"{target}_{codename}_perms" | ||||||
|  |         cached = self.context.get(key, {}).get(obj.pk) | ||||||
|  |         if cached is not None: | ||||||
|  |             return list(cached) | ||||||
|  |  | ||||||
|  |         # Permission not found in the context, get it from guardian | ||||||
|  |         if target == "users": | ||||||
|  |             return list( | ||||||
|  |                 get_users_with_perms( | ||||||
|  |                     obj, | ||||||
|  |                     only_with_perms_in=[f"{codename}_{obj.__class__.__name__.lower()}"], | ||||||
|  |                     with_group_users=False, | ||||||
|  |                 ).values_list("id", flat=True), | ||||||
|  |             ) | ||||||
|  |         else:  # groups | ||||||
|  |             return list( | ||||||
|  |                 get_groups_with_only_permission( | ||||||
|  |                     obj, | ||||||
|  |                     codename=f"{codename}_{obj.__class__.__name__.lower()}", | ||||||
|  |                 ).values_list("id", flat=True), | ||||||
|  |             ) | ||||||
|  |  | ||||||
|     @extend_schema_field( |     @extend_schema_field( | ||||||
|         field={ |         field={ | ||||||
|             "type": "object", |             "type": "object", | ||||||
| @@ -286,31 +316,14 @@ class OwnedObjectSerializer( | |||||||
|         }, |         }, | ||||||
|     ) |     ) | ||||||
|     def get_permissions(self, obj) -> dict: |     def get_permissions(self, obj) -> dict: | ||||||
|         view_codename = f"view_{obj.__class__.__name__.lower()}" |  | ||||||
|         change_codename = f"change_{obj.__class__.__name__.lower()}" |  | ||||||
|  |  | ||||||
|         return { |         return { | ||||||
|             "view": { |             "view": { | ||||||
|                 "users": get_users_with_perms( |                 "users": self._get_perms(obj, "view", "users"), | ||||||
|                     obj, |                 "groups": self._get_perms(obj, "view", "groups"), | ||||||
|                     only_with_perms_in=[view_codename], |  | ||||||
|                     with_group_users=False, |  | ||||||
|                 ).values_list("id", flat=True), |  | ||||||
|                 "groups": get_groups_with_only_permission( |  | ||||||
|                     obj, |  | ||||||
|                     codename=view_codename, |  | ||||||
|                 ).values_list("id", flat=True), |  | ||||||
|             }, |             }, | ||||||
|             "change": { |             "change": { | ||||||
|                 "users": get_users_with_perms( |                 "users": self._get_perms(obj, "change", "users"), | ||||||
|                     obj, |                 "groups": self._get_perms(obj, "change", "groups"), | ||||||
|                     only_with_perms_in=[change_codename], |  | ||||||
|                     with_group_users=False, |  | ||||||
|                 ).values_list("id", flat=True), |  | ||||||
|                 "groups": get_groups_with_only_permission( |  | ||||||
|                     obj, |  | ||||||
|                     codename=change_codename, |  | ||||||
|                 ).values_list("id", flat=True), |  | ||||||
|             }, |             }, | ||||||
|         } |         } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,17 +1,23 @@ | |||||||
|  | import json | ||||||
| import tempfile | import tempfile | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.contrib.auth.models import Group | ||||||
| from django.contrib.auth.models import Permission | from django.contrib.auth.models import Permission | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|  | from django.db import connection | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| from django.test import override_settings | from django.test import override_settings | ||||||
|  | from django.test.utils import CaptureQueriesContext | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
|  | from guardian.shortcuts import assign_perm | ||||||
| from rest_framework import status | from rest_framework import status | ||||||
|  |  | ||||||
| from documents.models import Document | from documents.models import Document | ||||||
| from documents.models import ShareLink | from documents.models import ShareLink | ||||||
|  | from documents.models import Tag | ||||||
| from documents.tests.utils import DirectoriesMixin | from documents.tests.utils import DirectoriesMixin | ||||||
| from paperless.models import ApplicationConfiguration | from paperless.models import ApplicationConfiguration | ||||||
|  |  | ||||||
| @@ -154,3 +160,113 @@ class TestViews(DirectoriesMixin, TestCase): | |||||||
|         response.render() |         response.render() | ||||||
|         self.assertEqual(response.request["PATH_INFO"], "/accounts/login/") |         self.assertEqual(response.request["PATH_INFO"], "/accounts/login/") | ||||||
|         self.assertContains(response, b"Share link has expired") |         self.assertContains(response, b"Share link has expired") | ||||||
|  |  | ||||||
|  |     def test_list_with_full_permissions(self): | ||||||
|  |         """ | ||||||
|  |         GIVEN: | ||||||
|  |             - Tags with different permissions | ||||||
|  |         WHEN: | ||||||
|  |             - Request to get tag list with full permissions is made | ||||||
|  |         THEN: | ||||||
|  |             - Tag list is returned with the right permission information | ||||||
|  |         """ | ||||||
|  |         user2 = User.objects.create(username="user2") | ||||||
|  |         user3 = User.objects.create(username="user3") | ||||||
|  |         group1 = Group.objects.create(name="group1") | ||||||
|  |         group2 = Group.objects.create(name="group2") | ||||||
|  |         group3 = Group.objects.create(name="group3") | ||||||
|  |         t1 = Tag.objects.create(name="invoice", pk=1) | ||||||
|  |         assign_perm("view_tag", self.user, t1) | ||||||
|  |         assign_perm("view_tag", user2, t1) | ||||||
|  |         assign_perm("view_tag", user3, t1) | ||||||
|  |         assign_perm("view_tag", group1, t1) | ||||||
|  |         assign_perm("view_tag", group2, t1) | ||||||
|  |         assign_perm("view_tag", group3, t1) | ||||||
|  |         assign_perm("change_tag", self.user, t1) | ||||||
|  |         assign_perm("change_tag", user2, t1) | ||||||
|  |         assign_perm("change_tag", group1, t1) | ||||||
|  |         assign_perm("change_tag", group2, t1) | ||||||
|  |  | ||||||
|  |         Tag.objects.create(name="bank statement", pk=2) | ||||||
|  |         d1 = Document.objects.create( | ||||||
|  |             title="Invoice 1", | ||||||
|  |             content="This is the invoice of a very expensive item", | ||||||
|  |             checksum="A", | ||||||
|  |         ) | ||||||
|  |         d1.tags.add(t1) | ||||||
|  |         d2 = Document.objects.create( | ||||||
|  |             title="Invoice 2", | ||||||
|  |             content="Internet invoice, I should pay it to continue contributing", | ||||||
|  |             checksum="B", | ||||||
|  |         ) | ||||||
|  |         d2.tags.add(t1) | ||||||
|  |  | ||||||
|  |         view_permissions = Permission.objects.filter( | ||||||
|  |             codename__contains="view_tag", | ||||||
|  |         ) | ||||||
|  |         self.user.user_permissions.add(*view_permissions) | ||||||
|  |         self.user.save() | ||||||
|  |  | ||||||
|  |         self.client.force_login(self.user) | ||||||
|  |         response = self.client.get("/api/tags/?page=1&full_perms=true") | ||||||
|  |         results = json.loads(response.content)["results"] | ||||||
|  |         for tag in results: | ||||||
|  |             if tag["name"] == "invoice": | ||||||
|  |                 assert tag["permissions"] == { | ||||||
|  |                     "view": { | ||||||
|  |                         "users": [self.user.pk, user2.pk, user3.pk], | ||||||
|  |                         "groups": [group1.pk, group2.pk, group3.pk], | ||||||
|  |                     }, | ||||||
|  |                     "change": { | ||||||
|  |                         "users": [self.user.pk, user2.pk], | ||||||
|  |                         "groups": [group1.pk, group2.pk], | ||||||
|  |                     }, | ||||||
|  |                 } | ||||||
|  |             elif tag["name"] == "bank statement": | ||||||
|  |                 assert tag["permissions"] == { | ||||||
|  |                     "view": {"users": [], "groups": []}, | ||||||
|  |                     "change": {"users": [], "groups": []}, | ||||||
|  |                 } | ||||||
|  |             else: | ||||||
|  |                 assert False, f"Unexpected tag found: {tag['name']}" | ||||||
|  |  | ||||||
|  |     def test_list_no_n_plus_1_queries(self): | ||||||
|  |         """ | ||||||
|  |         GIVEN: | ||||||
|  |             - Tags with different permissions | ||||||
|  |         WHEN: | ||||||
|  |             - Request to get tag list with full permissions is made | ||||||
|  |         THEN: | ||||||
|  |             - Permissions are not queried in database tag by tag, | ||||||
|  |              i.e. there are no N+1 queries | ||||||
|  |         """ | ||||||
|  |         view_permissions = Permission.objects.filter( | ||||||
|  |             codename__contains="view_tag", | ||||||
|  |         ) | ||||||
|  |         self.user.user_permissions.add(*view_permissions) | ||||||
|  |         self.user.save() | ||||||
|  |         self.client.force_login(self.user) | ||||||
|  |  | ||||||
|  |         # Start by a small list, and count the number of SQL queries | ||||||
|  |         for i in range(2): | ||||||
|  |             Tag.objects.create(name=f"tag_{i}") | ||||||
|  |  | ||||||
|  |         with CaptureQueriesContext(connection) as ctx_small: | ||||||
|  |             response_small = self.client.get("/api/tags/?full_perms=true") | ||||||
|  |             assert response_small.status_code == 200 | ||||||
|  |         num_queries_small = len(ctx_small.captured_queries) | ||||||
|  |  | ||||||
|  |         # Complete the list, and count the number of SQL queries again | ||||||
|  |         for i in range(2, 50): | ||||||
|  |             Tag.objects.create(name=f"tag_{i}") | ||||||
|  |  | ||||||
|  |         with CaptureQueriesContext(connection) as ctx_large: | ||||||
|  |             response_large = self.client.get("/api/tags/?full_perms=true") | ||||||
|  |             assert response_large.status_code == 200 | ||||||
|  |         num_queries_large = len(ctx_large.captured_queries) | ||||||
|  |  | ||||||
|  |         # A few additional queries are allowed, but not a linear explosion | ||||||
|  |         assert num_queries_large <= num_queries_small + 5, ( | ||||||
|  |             f"Possible N+1 queries detected: {num_queries_small} queries for 2 tags, " | ||||||
|  |             f"but {num_queries_large} queries for 50 tags" | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -5,9 +5,11 @@ import platform | |||||||
| import re | import re | ||||||
| import tempfile | import tempfile | ||||||
| import zipfile | import zipfile | ||||||
|  | from collections import defaultdict | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from time import mktime | from time import mktime | ||||||
|  | from typing import Literal | ||||||
| from unicodedata import normalize | from unicodedata import normalize | ||||||
| from urllib.parse import quote | from urllib.parse import quote | ||||||
| from urllib.parse import urlparse | from urllib.parse import urlparse | ||||||
| @@ -19,6 +21,7 @@ from celery import states | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.models import Group | from django.contrib.auth.models import Group | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|  | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.db import connections | from django.db import connections | ||||||
| from django.db.migrations.loader import MigrationLoader | from django.db.migrations.loader import MigrationLoader | ||||||
| from django.db.migrations.recorder import MigrationRecorder | from django.db.migrations.recorder import MigrationRecorder | ||||||
| @@ -56,6 +59,8 @@ from drf_spectacular.utils import OpenApiParameter | |||||||
| from drf_spectacular.utils import extend_schema | from drf_spectacular.utils import extend_schema | ||||||
| from drf_spectacular.utils import extend_schema_view | from drf_spectacular.utils import extend_schema_view | ||||||
| from drf_spectacular.utils import inline_serializer | from drf_spectacular.utils import inline_serializer | ||||||
|  | from guardian.utils import get_group_obj_perms_model | ||||||
|  | from guardian.utils import get_user_obj_perms_model | ||||||
| from langdetect import detect | from langdetect import detect | ||||||
| from packaging import version as packaging_version | from packaging import version as packaging_version | ||||||
| from redis import Redis | from redis import Redis | ||||||
| @@ -254,7 +259,104 @@ class PassUserMixin(GenericAPIView): | |||||||
|         return super().get_serializer(*args, **kwargs) |         return super().get_serializer(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class PermissionsAwareDocumentCountMixin(PassUserMixin): | class BulkPermissionMixin: | ||||||
|  |     """ | ||||||
|  |     Prefetch Django-Guardian permissions for a list before serialization, to avoid N+1 queries. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def _get_object_perms( | ||||||
|  |         self, | ||||||
|  |         objects: list, | ||||||
|  |         perm_codenames: list[str], | ||||||
|  |         actor: Literal["users", "groups"], | ||||||
|  |     ) -> dict[int, dict[str, list[int]]]: | ||||||
|  |         """ | ||||||
|  |         Collect object-level permissions for either users or groups. | ||||||
|  |         """ | ||||||
|  |         model = self.queryset.model | ||||||
|  |         obj_perm_model = ( | ||||||
|  |             get_user_obj_perms_model(model) | ||||||
|  |             if actor == "users" | ||||||
|  |             else get_group_obj_perms_model(model) | ||||||
|  |         ) | ||||||
|  |         id_field = "user_id" if actor == "users" else "group_id" | ||||||
|  |         ctype = ContentType.objects.get_for_model(model) | ||||||
|  |         object_pks = [obj.pk for obj in objects] | ||||||
|  |  | ||||||
|  |         perms_qs = obj_perm_model.objects.filter( | ||||||
|  |             content_type=ctype, | ||||||
|  |             object_pk__in=object_pks, | ||||||
|  |             permission__codename__in=perm_codenames, | ||||||
|  |         ).values_list("object_pk", id_field, "permission__codename") | ||||||
|  |  | ||||||
|  |         perms: dict[int, dict[str, list[int]]] = defaultdict(lambda: defaultdict(list)) | ||||||
|  |         for object_pk, actor_id, codename in perms_qs: | ||||||
|  |             perms[int(object_pk)][codename].append(actor_id) | ||||||
|  |  | ||||||
|  |         # Ensure that all objects have all codenames, even if empty | ||||||
|  |         for pk in object_pks: | ||||||
|  |             for codename in perm_codenames: | ||||||
|  |                 perms[pk][codename] | ||||||
|  |  | ||||||
|  |         return perms | ||||||
|  |  | ||||||
|  |     def get_serializer_context(self): | ||||||
|  |         """ | ||||||
|  |         Get all permissions of the current list of objects at once and pass them to the serializer. | ||||||
|  |         This avoid fetching permissions object by object in database. | ||||||
|  |         """ | ||||||
|  |         context = super().get_serializer_context() | ||||||
|  |         try: | ||||||
|  |             full_perms = get_boolean( | ||||||
|  |                 str(self.request.query_params.get("full_perms", "false")), | ||||||
|  |             ) | ||||||
|  |         except ValueError: | ||||||
|  |             full_perms = False | ||||||
|  |  | ||||||
|  |         if not full_perms: | ||||||
|  |             return context | ||||||
|  |  | ||||||
|  |         # Check which objects are being paginated | ||||||
|  |         page = getattr(self, "paginator", None) | ||||||
|  |         if page and hasattr(page, "page"): | ||||||
|  |             queryset = page.page.object_list | ||||||
|  |         elif hasattr(self, "page"): | ||||||
|  |             queryset = self.page | ||||||
|  |         else: | ||||||
|  |             queryset = self.filter_queryset(self.get_queryset()) | ||||||
|  |  | ||||||
|  |         model_name = self.queryset.model.__name__.lower() | ||||||
|  |         permission_name_view = f"view_{model_name}" | ||||||
|  |         permission_name_change = f"change_{model_name}" | ||||||
|  |  | ||||||
|  |         user_perms = self._get_object_perms( | ||||||
|  |             objects=queryset, | ||||||
|  |             perm_codenames=[permission_name_view, permission_name_change], | ||||||
|  |             actor="users", | ||||||
|  |         ) | ||||||
|  |         group_perms = self._get_object_perms( | ||||||
|  |             objects=queryset, | ||||||
|  |             perm_codenames=[permission_name_view, permission_name_change], | ||||||
|  |             actor="groups", | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         context["users_view_perms"] = { | ||||||
|  |             pk: user_perms[pk][permission_name_view] for pk in user_perms | ||||||
|  |         } | ||||||
|  |         context["users_change_perms"] = { | ||||||
|  |             pk: user_perms[pk][permission_name_change] for pk in user_perms | ||||||
|  |         } | ||||||
|  |         context["groups_view_perms"] = { | ||||||
|  |             pk: group_perms[pk][permission_name_view] for pk in group_perms | ||||||
|  |         } | ||||||
|  |         context["groups_change_perms"] = { | ||||||
|  |             pk: group_perms[pk][permission_name_change] for pk in group_perms | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return context | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PermissionsAwareDocumentCountMixin(BulkPermissionMixin, PassUserMixin): | ||||||
|     """ |     """ | ||||||
|     Mixin to add document count to queryset, permissions-aware if needed |     Mixin to add document count to queryset, permissions-aware if needed | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ msgid "" | |||||||
| msgstr "" | msgstr "" | ||||||
| "Project-Id-Version: paperless-ngx\n" | "Project-Id-Version: paperless-ngx\n" | ||||||
| "Report-Msgid-Bugs-To: \n" | "Report-Msgid-Bugs-To: \n" | ||||||
| "POT-Creation-Date: 2025-09-22 18:20+0000\n" | "POT-Creation-Date: 2025-09-30 16:50+0000\n" | ||||||
| "PO-Revision-Date: 2022-02-17 04:17\n" | "PO-Revision-Date: 2022-02-17 04:17\n" | ||||||
| "Last-Translator: \n" | "Last-Translator: \n" | ||||||
| "Language-Team: English\n" | "Language-Team: English\n" | ||||||
| @@ -1191,44 +1191,44 @@ msgstr "" | |||||||
| msgid "workflow runs" | msgid "workflow runs" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:140 | #: documents/serialisers.py:141 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "Invalid regular expression: %(error)s" | msgid "Invalid regular expression: %(error)s" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:594 | #: documents/serialisers.py:607 | ||||||
| msgid "Invalid color." | msgid "Invalid color." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:623 | #: documents/serialisers.py:636 | ||||||
| msgid "Invalid parent tag." | msgid "Invalid parent tag." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1780 | #: documents/serialisers.py:1793 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "File type %(type)s not supported" | msgid "File type %(type)s not supported" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1824 | #: documents/serialisers.py:1837 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "Custom field id must be an integer: %(id)s" | msgid "Custom field id must be an integer: %(id)s" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1831 | #: documents/serialisers.py:1844 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "Custom field with id %(id)s does not exist" | msgid "Custom field with id %(id)s does not exist" | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1848 documents/serialisers.py:1858 | #: documents/serialisers.py:1861 documents/serialisers.py:1871 | ||||||
| msgid "" | msgid "" | ||||||
| "Custom fields must be a list of integers or an object mapping ids to values." | "Custom fields must be a list of integers or an object mapping ids to values." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1853 | #: documents/serialisers.py:1866 | ||||||
| msgid "Some custom fields don't exist or were specified twice." | msgid "Some custom fields don't exist or were specified twice." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
| #: documents/serialisers.py:1923 | #: documents/serialisers.py:1936 | ||||||
| msgid "Invalid variable detected." | msgid "Invalid variable detected." | ||||||
| msgstr "" | msgstr "" | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user