mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			8 Commits
		
	
	
		
			feature-re
			...
			feature-re
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ec12e71487 | ||
|   | 62b470f691 | ||
|   | a2e4977201 | ||
|   | 0fcd69b739 | ||
|   | af1c64e969 | ||
|   | 85c661dff2 | ||
|   | 3a7eee2c2e | ||
|   | bc4d3925cc | 
							
								
								
									
										18
									
								
								.codecov.yml
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								.codecov.yml
									
									
									
									
									
								
							| @@ -1,17 +1,19 @@ | |||||||
| codecov: | codecov: | ||||||
|   require_ci_to_pass: true |   require_ci_to_pass: true | ||||||
|   # https://docs.codecov.com/docs/components | # https://docs.codecov.com/docs/flags#recommended-automatic-flag-management | ||||||
| component_management: | # Require each flag to have 1 upload before notification | ||||||
|   individual_components: | flag_management: | ||||||
|     - component_id: backend |   individual_flags: | ||||||
|  |     - name: backend | ||||||
|       paths: |       paths: | ||||||
|         - src/** |         - src/ | ||||||
|     - component_id: frontend |     - name: frontend | ||||||
|       paths: |       paths: | ||||||
|         - src-ui/** |         - src-ui/ | ||||||
| # https://docs.codecov.com/docs/pull-request-comments | # https://docs.codecov.com/docs/pull-request-comments | ||||||
|  | # codecov will only comment if coverage changes | ||||||
| comment: | comment: | ||||||
|   layout: "header, diff, components, flags, files" |   require_changes: true | ||||||
|   # https://docs.codecov.com/docs/javascript-bundle-analysis |   # https://docs.codecov.com/docs/javascript-bundle-analysis | ||||||
|   require_bundle_changes: true |   require_bundle_changes: true | ||||||
|   bundle_change_threshold: "50Kb" |   bundle_change_threshold: "50Kb" | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.codespellrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.codespellrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | [codespell] | ||||||
|  | write-changes = True | ||||||
|  | ignore-words-list = criterias,afterall,valeu,ureue,equest,ure,assertIn | ||||||
| @@ -76,16 +76,18 @@ RUN set -eux \ | |||||||
|     && apt-get update \ |     && apt-get update \ | ||||||
|     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} |     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} | ||||||
|  |  | ||||||
| ARG PYTHON_PACKAGES="ca-certificates" | ARG PYTHON_PACKAGES="\ | ||||||
|  |   python3 \ | ||||||
|  |   python3-pip \ | ||||||
|  |   python3-wheel \ | ||||||
|  |   pipenv \ | ||||||
|  |   ca-certificates" | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   echo "Installing python packages" \ |   echo "Installing python packages" \ | ||||||
|     && apt-get update \ |     && apt-get update \ | ||||||
|     && apt-get install --yes --quiet ${PYTHON_PACKAGES} |     && apt-get install --yes --quiet ${PYTHON_PACKAGES} | ||||||
|  |  | ||||||
| COPY --from=ghcr.io/astral-sh/uv:0.7.8 /uv /bin/uv |  | ||||||
|  |  | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && echo "Installing pre-built updates" \ |   && echo "Installing pre-built updates" \ | ||||||
|     && echo "Installing qpdf ${QPDF_VERSION}" \ |     && echo "Installing qpdf ${QPDF_VERSION}" \ | ||||||
| @@ -121,14 +123,13 @@ RUN set -eux \ | |||||||
| WORKDIR /usr/src/paperless/src/docker/ | WORKDIR /usr/src/paperless/src/docker/ | ||||||
|  |  | ||||||
| COPY [ \ | COPY [ \ | ||||||
|   "docker/rootfs/etc/ImageMagick-6/paperless-policy.xml", \ |   "docker/imagemagick-policy.xml", \ | ||||||
|   "./" \ |   "./" \ | ||||||
| ] | ] | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && echo "Configuring ImageMagick" \ |   && echo "Configuring ImageMagick" \ | ||||||
|     && mv paperless-policy.xml /etc/ImageMagick-6/policy.xml |     && mv imagemagick-policy.xml /etc/ImageMagick-6/policy.xml | ||||||
|  |  | ||||||
|  |  | ||||||
| # Packages needed only for building a few quick Python | # Packages needed only for building a few quick Python | ||||||
| # dependencies | # dependencies | ||||||
| @@ -139,17 +140,18 @@ ARG BUILD_PACKAGES="\ | |||||||
|   libpq-dev \ |   libpq-dev \ | ||||||
|   # https://github.com/PyMySQL/mysqlclient#linux |   # https://github.com/PyMySQL/mysqlclient#linux | ||||||
|   default-libmysqlclient-dev \ |   default-libmysqlclient-dev \ | ||||||
|   pkg-config" |   pkg-config \ | ||||||
|  |   pre-commit" | ||||||
|  |  | ||||||
| # hadolint ignore=DL3042 | # hadolint ignore=DL3042 | ||||||
| RUN --mount=type=cache,target=/root/.cache/uv,id=pip-cache \ | RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \ | ||||||
|   set -eux \ |   set -eux \ | ||||||
|   && echo "Installing build system packages" \ |   && echo "Installing build system packages" \ | ||||||
|     && apt-get update \ |     && apt-get update \ | ||||||
|     && apt-get install --yes --quiet ${BUILD_PACKAGES} |     && apt-get install --yes --quiet ${BUILD_PACKAGES} | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && npm update -g pnpm |   && npm update npm -g | ||||||
|  |  | ||||||
| # add users, setup scripts | # add users, setup scripts | ||||||
| # Mount the compiled frontend to expected location | # Mount the compiled frontend to expected location | ||||||
| @@ -167,6 +169,9 @@ RUN set -eux \ | |||||||
|     && mkdir --parents --verbose /usr/src/paperless/paperless-ngx/.venv \ |     && mkdir --parents --verbose /usr/src/paperless/paperless-ngx/.venv \ | ||||||
|   && echo "Adjusting all permissions" \ |   && echo "Adjusting all permissions" \ | ||||||
|     && chown --from root:root --changes --recursive paperless:paperless /usr/src/paperless |     && chown --from root:root --changes --recursive paperless:paperless /usr/src/paperless | ||||||
|  | #  && echo "Collecting static files" \ | ||||||
|  | #    && gosu paperless python3 manage.py collectstatic --clear --no-input --link \ | ||||||
|  | #    && gosu paperless python3 manage.py compilemessages | ||||||
|  |  | ||||||
| VOLUME ["/usr/src/paperless/paperless-ngx/data", \ | VOLUME ["/usr/src/paperless/paperless-ngx/data", \ | ||||||
|         "/usr/src/paperless/paperless-ngx/media", \ |         "/usr/src/paperless/paperless-ngx/media", \ | ||||||
|   | |||||||
| @@ -1,8 +1,8 @@ | |||||||
| # Paperless-ngx Development Environment | # Paperless NGX Development Environment | ||||||
|  |  | ||||||
| ## Overview | ## Overview | ||||||
|  |  | ||||||
| Welcome to the Paperless-ngx development environment! This setup uses VSCode DevContainers to provide a consistent and seamless development experience. | Welcome to the Paperless NGX development environment! This setup uses VSCode DevContainers to provide a consistent and seamless development experience. | ||||||
|  |  | ||||||
| ### What are DevContainers? | ### What are DevContainers? | ||||||
|  |  | ||||||
| @@ -17,7 +17,7 @@ DevContainers are a feature in VSCode that allows you to develop within a Docker | |||||||
|  |  | ||||||
| ## DevContainer Setup | ## DevContainer Setup | ||||||
|  |  | ||||||
| The DevContainer configuration provides up all the necessary services for Paperless-ngx, including: | The DevContainer configuration provides up all the necessary services for Paperless NGX, including: | ||||||
|  |  | ||||||
| - Redis | - Redis | ||||||
| - Gotenberg | - Gotenberg | ||||||
| @@ -36,7 +36,7 @@ The setup includes debugging configurations (`launch.json`) and tasks (`tasks.js | |||||||
| - **Maintenance Tasks:** | - **Maintenance Tasks:** | ||||||
|   - Create superuser |   - Create superuser | ||||||
|   - Run migrations |   - Run migrations | ||||||
|   - Recreate virtual environment (`.venv` with `uv`) |   - Recreate virtual environment (`.venv` with pipenv) | ||||||
|   - Compile frontend assets |   - Compile frontend assets | ||||||
|  |  | ||||||
| ## Getting Started | ## Getting Started | ||||||
| @@ -47,25 +47,45 @@ To start the DevContainer: | |||||||
|  |  | ||||||
| 1. Open VSCode. | 1. Open VSCode. | ||||||
| 2. Open the project folder. | 2. Open the project folder. | ||||||
| 3. Open the command palette and choose `Dev Containers: Rebuild and Reopen in Container`. | 3. Open the command palette: | ||||||
|  |    - **Windows/Linux**: `Ctrl+Shift+P` | ||||||
|  |    - **Mac**: `Cmd+Shift+P` | ||||||
|  | 4. Type and select `Dev Containers: Rebuild and Reopen in Container`. | ||||||
|  |  | ||||||
| VSCode will build and start the DevContainer environment. | VSCode will build and start the DevContainer environment. | ||||||
|  |  | ||||||
| ### Step 2: Initial Setup | ### Step 2: Initial Setup | ||||||
|  |  | ||||||
| Once the DevContainer is up and running, run the `Project Setup: Run all Init Tasks` task to initialize the project. | Once the DevContainer is up and running, perform the following steps: | ||||||
|  |  | ||||||
| Alternatively, the Project Setup can be done with individual tasks: | 1. **Compile Frontend Assets**: | ||||||
|  |  | ||||||
| 1. **Compile Frontend Assets**: `Maintenance: Compile frontend for production`. |    - Open the command palette: | ||||||
| 2. **Run Database Migrations**: `Maintenance: manage.py migrate`. |      - **Windows/Linux**: `Ctrl+Shift+P` | ||||||
| 3. **Create Superuser**: `Maintenance: manage.py createsuperuser`. |      - **Mac**: `Cmd+Shift+P` | ||||||
|  |    - Select `Tasks: Run Task`. | ||||||
|  |    - Choose `Frontend Compile`. | ||||||
|  |  | ||||||
|  | 2. **Run Database Migrations**: | ||||||
|  |  | ||||||
|  |    - Open the command palette: | ||||||
|  |      - **Windows/Linux**: `Ctrl+Shift+P` | ||||||
|  |      - **Mac**: `Cmd+Shift+P` | ||||||
|  |    - Select `Tasks: Run Task`. | ||||||
|  |    - Choose `Migrate Database`. | ||||||
|  |  | ||||||
|  | 3. **Create Superuser**: | ||||||
|  |    - Open the command palette: | ||||||
|  |      - **Windows/Linux**: `Ctrl+Shift+P` | ||||||
|  |      - **Mac**: `Cmd+Shift+P` | ||||||
|  |    - Select `Tasks: Run Task`. | ||||||
|  |    - Choose `Create Superuser`. | ||||||
|  |  | ||||||
| ### Debugging and Running Services | ### Debugging and Running Services | ||||||
|  |  | ||||||
| You can start and debug backend services either as debugging sessions via `launch.json` or as tasks. | You can start and debug backend services either as debugging sessions via `launch.json` or as tasks. | ||||||
|  |  | ||||||
| #### Using `launch.json` | #### Using `launch.json`: | ||||||
|  |  | ||||||
| 1. Press `F5` or go to the **Run and Debug** view in VSCode. | 1. Press `F5` or go to the **Run and Debug** view in VSCode. | ||||||
| 2. Select the desired configuration: | 2. Select the desired configuration: | ||||||
| @@ -73,10 +93,13 @@ You can start and debug backend services either as debugging sessions via `launc | |||||||
|    - `Document Consumer` |    - `Document Consumer` | ||||||
|    - `Celery` |    - `Celery` | ||||||
|  |  | ||||||
| #### Using Tasks | #### Using Tasks: | ||||||
|  |  | ||||||
| 1. Open the command palette and select `Tasks: Run Task`. | 1. Open the command palette: | ||||||
| 2. Choose the desired task: |    - **Windows/Linux**: `Ctrl+Shift+P` | ||||||
|  |    - **Mac**: `Cmd+Shift+P` | ||||||
|  | 2. Select `Tasks: Run Task`. | ||||||
|  | 3. Choose the desired task: | ||||||
|    - `Runserver` |    - `Runserver` | ||||||
|    - `Document Consumer` |    - `Document Consumer` | ||||||
|    - `Celery` |    - `Celery` | ||||||
| @@ -85,7 +108,7 @@ You can start and debug backend services either as debugging sessions via `launc | |||||||
|  |  | ||||||
| Additional tasks are available for common maintenance operations: | Additional tasks are available for common maintenance operations: | ||||||
|  |  | ||||||
| - **Recreate .venv**: For setting up the virtual environment using `uv`. | - **Recreate .venv**: For setting up the virtual environment using pipenv. | ||||||
| - **Migrate Database**: To apply database migrations. | - **Migrate Database**: To apply database migrations. | ||||||
| - **Create Superuser**: To create an admin user for the application. | - **Create Superuser**: To create an admin user for the application. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,25 +3,13 @@ | |||||||
|     "dockerComposeFile": "docker-compose.devcontainer.sqlite-tika.yml", |     "dockerComposeFile": "docker-compose.devcontainer.sqlite-tika.yml", | ||||||
|     "service": "paperless-development", |     "service": "paperless-development", | ||||||
|     "workspaceFolder": "/usr/src/paperless/paperless-ngx", |     "workspaceFolder": "/usr/src/paperless/paperless-ngx", | ||||||
|     "postCreateCommand": "/bin/bash -c 'rm -rf .venv/.*  && uv sync --group dev && uv run pre-commit install'", |     "postCreateCommand": "/bin/bash -c pre-commit install && pipenv install --dev", | ||||||
|     "customizations": { |     "customizations": { | ||||||
|         "vscode": { |         "vscode": { | ||||||
|             "extensions": [ |             "extensions": [ | ||||||
|               "mhutchie.git-graph", |               "mhutchie.git-graph", | ||||||
|             "ms-python.python", |               "ms-python.python" | ||||||
|             "ms-vscode.js-debug-nightly", |             ] | ||||||
|             "eamodio.gitlens", |  | ||||||
|             "yzhang.markdown-all-in-one" |  | ||||||
|           ], |  | ||||||
|           "settings": { |  | ||||||
|             "python.defaultInterpreterPath": "/usr/src/paperless/paperless-ngx/.venv/bin/python", |  | ||||||
|             "python.pythonPath": "/usr/src/paperless/paperless-ngx/.venv/bin/python", |  | ||||||
|             "python.terminal.activateEnvInCurrentTerminal": true, |  | ||||||
|             "editor.formatOnPaste": false, |  | ||||||
|             "editor.formatOnSave": true, |  | ||||||
|             "editor.formatOnType": true, |  | ||||||
|             "files.trimTrailingWhitespace": true |  | ||||||
|           } |  | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
|     "remoteUser": "paperless" |     "remoteUser": "paperless" | ||||||
|   | |||||||
| @@ -20,12 +20,15 @@ | |||||||
| # | # | ||||||
| # This file is intended only to be used through VSCOde devcontainers. See README.md | # This file is intended only to be used through VSCOde devcontainers. See README.md | ||||||
| # in the folder .devcontainer. | # in the folder .devcontainer. | ||||||
|  |  | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:7 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - ./redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   # No ports need to be exposed; the VSCode DevContainer plugin manages them. |   # No ports need to be exposed; the VSCode DevContainer plugin manages them. | ||||||
|   paperless-development: |   paperless-development: | ||||||
|     image: paperless-ngx |     image: paperless-ngx | ||||||
| @@ -40,15 +43,14 @@ services: | |||||||
|     volumes: |     volumes: | ||||||
|       - ..:/usr/src/paperless/paperless-ngx:delegated |       - ..:/usr/src/paperless/paperless-ngx:delegated | ||||||
|       - ../.devcontainer/vscode:/usr/src/paperless/paperless-ngx/.vscode:delegated # VSCode config files |       - ../.devcontainer/vscode:/usr/src/paperless/paperless-ngx/.vscode:delegated # VSCode config files | ||||||
|       - virtualenv:/usr/src/paperless/paperless-ngx/.venv # Virtual environment persisted in volume |       - pipenv:/usr/src/paperless/paperless-ngx/.venv # Pipenv environment persisted in volume | ||||||
|       - /usr/src/paperless/paperless-ngx/src/documents/static/frontend # Static frontend files exist only in container |       - /usr/src/paperless/paperless-ngx/src/documents/static/frontend # Static frontend files exist only in container | ||||||
|       - /usr/src/paperless/paperless-ngx/src/.pytest_cache |       - /usr/src/paperless/paperless-ngx/src/.pytest_cache | ||||||
|       - /usr/src/paperless/paperless-ngx/.ruff_cache |       - /usr/src/paperless/paperless-ngx/.ruff_cache | ||||||
|       - /usr/src/paperless/paperless-ngx/htmlcov |       - /usr/src/paperless/paperless-ngx/htmlcov | ||||||
|       - /usr/src/paperless/paperless-ngx/.coverage |       - /usr/src/paperless/paperless-ngx/.coverage | ||||||
|       - ./data:/usr/src/paperless/paperless-ngx/data |       - data:/usr/src/paperless/paperless-ngx/data | ||||||
|       - ./media:/usr/src/paperless/paperless-ngx/media |       - media:/usr/src/paperless/paperless-ngx/media | ||||||
|       - ./consume:/usr/src/paperless/paperless-ngx/consume |  | ||||||
|     environment: |     environment: | ||||||
|       PAPERLESS_REDIS: redis://broker:6379 |       PAPERLESS_REDIS: redis://broker:6379 | ||||||
|       PAPERLESS_TIKA_ENABLED: 1 |       PAPERLESS_TIKA_ENABLED: 1 | ||||||
| @@ -56,22 +58,27 @@ services: | |||||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 |       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||||
|       PAPERLESS_STATICDIR: ./src/documents/static |       PAPERLESS_STATICDIR: ./src/documents/static | ||||||
|       PAPERLESS_DEBUG: true |       PAPERLESS_DEBUG: true | ||||||
|  |  | ||||||
|     # Overrides default command so things don't shut down after the process ends. |     # Overrides default command so things don't shut down after the process ends. | ||||||
|     command: /bin/sh -c "chown -R paperless:paperless /usr/src/paperless/paperless-ngx/src/documents/static/frontend && chown -R paperless:paperless /usr/src/paperless/paperless-ngx/.ruff_cache && while sleep 1000; do :; done" |     command: /bin/sh -c "chown -R paperless:paperless /usr/src/paperless/paperless-ngx/src/documents/static/frontend && chown -R paperless:paperless /usr/src/paperless/paperless-ngx/.ruff_cache && while sleep 1000; do :; done" | ||||||
|  |  | ||||||
|   gotenberg: |   gotenberg: | ||||||
|     image: docker.io/gotenberg/gotenberg:8.17 |     image: docker.io/gotenberg/gotenberg:7.10 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
|     # The Gotenberg Chromium route is used to convert .eml files. We do not |     # The Gotenberg Chromium route is used to convert .eml files. We do not | ||||||
|     # want to allow external content like tracking pixels or even JavaScript. |     # want to allow external content like tracking pixels or even JavaScript. | ||||||
|     command: |     command: | ||||||
|       - "gotenberg" |       - "gotenberg" | ||||||
|       - "--chromium-disable-javascript=true" |       - "--chromium-disable-javascript=true" | ||||||
|       - "--chromium-allow-list=file:///tmp/.*" |       - "--chromium-allow-list=file:///tmp/.*" | ||||||
|  |  | ||||||
|   tika: |   tika: | ||||||
|     image: docker.io/apache/tika:latest |     image: docker.io/apache/tika:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   redisdata: |   redisdata: | ||||||
|   virtualenv: |   pipenv: | ||||||
|   | |||||||
| @@ -2,57 +2,42 @@ | |||||||
|     "version": "0.2.0", |     "version": "0.2.0", | ||||||
|     "configurations": [ |     "configurations": [ | ||||||
|         { |         { | ||||||
|             "name": "Chrome: Debug Angular Frontend", |             "name": "manage.py runserver", | ||||||
| 						"description": "Debug the Angular Dev Frontend in Chrome", |  | ||||||
|             "type": "chrome", |  | ||||||
|             "request": "launch", |  | ||||||
|             "url": "http://localhost:4200", |  | ||||||
|             "webRoot": "${workspaceFolder}/src-ui", |  | ||||||
|             "preLaunchTask": "Start: Frontend Angular" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "name": "Debug: Backend Server (manage.py runserver)", |  | ||||||
| 						"description": "Debug the Django Backend Server", |  | ||||||
|             "type": "python", |             "type": "python", | ||||||
|             "request": "launch", |             "request": "launch", | ||||||
|             "program": "${workspaceFolder}/src/manage.py", |             "program": "${workspaceFolder}/src/manage.py", | ||||||
|             "args": [ |             "console": "integratedTerminal", | ||||||
|                 "runserver" |             "justMyCode": true, | ||||||
|             ], |             "args": ["runserver"], | ||||||
|             "django": true, |             "django": true | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "manage.py document_consumer", | ||||||
|  |             "type": "python", | ||||||
|  |             "request": "launch", | ||||||
|  |             "program": "${workspaceFolder}/src/manage.py", | ||||||
|  |             "console": "integratedTerminal", | ||||||
|  |             "justMyCode": true, | ||||||
|  |             "args": ["document_consumer"], | ||||||
|  |             "django": true | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "celery", | ||||||
|  |             "type": "python", | ||||||
|  |             "cwd": "${workspaceFolder}/src", | ||||||
|  |             "request": "launch", | ||||||
|  |             "module": "celery", | ||||||
|             "console": "integratedTerminal", |             "console": "integratedTerminal", | ||||||
|             "env": { |             "env": { | ||||||
|                 "PYTHONPATH": "${workspaceFolder}/src" |                 "PYTHONPATH": "${workspaceFolder}/src" | ||||||
|               }, |               }, | ||||||
|             "python": "${workspaceFolder}/.venv/bin/python" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "name": "Debug: Consumer Service (manage.py document_consumer)", |  | ||||||
| 						"description": "Debug the Consumer Service which processes files from a directory", |  | ||||||
|             "type": "python", |  | ||||||
|             "request": "launch", |  | ||||||
|             "program": "${workspaceFolder}/src/manage.py", |  | ||||||
|             "args": [ |             "args": [ | ||||||
|                 "document_consumer" |                 "-A", | ||||||
|             ], |                 "paperless", | ||||||
|             "django": true, |                 "worker", | ||||||
|             "console": "integratedTerminal", |                 "-l", | ||||||
|             "env": { |                 "DEBUG" | ||||||
|                 "PYTHONPATH": "${workspaceFolder}/src" |             ] | ||||||
|             }, |  | ||||||
|             "python": "${workspaceFolder}/.venv/bin/python" |  | ||||||
|         } |  | ||||||
|     ], |  | ||||||
|     "compounds": [ |  | ||||||
|         { |  | ||||||
|             "name": "Debug: FullStack", |  | ||||||
| 						"description": "Debug run the Angular dev frontend, Django backend, and consumer service", |  | ||||||
|             "configurations": [ |  | ||||||
|                 "Chrome: Debug Angular Frontend", |  | ||||||
|                 "Debug: Backend Server (manage.py runserver)", |  | ||||||
|                 "Debug: Consumer Service (manage.py document_consumer)" |  | ||||||
|             ], |  | ||||||
|             "preLaunchTask": "Start: Celery Worker" |  | ||||||
|         } |         } | ||||||
|     ] |     ] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| { | { | ||||||
|     "python.testing.pytestArgs": [], |     "python.testing.pytestArgs": [ | ||||||
|  |         "src" | ||||||
|  |     ], | ||||||
|     "python.testing.unittestEnabled": false, |     "python.testing.unittestEnabled": false, | ||||||
|     "python.testing.pytestEnabled": true, |     "python.testing.pytestEnabled": true, | ||||||
|     "files.watcherExclude": { |     "files.watcherExclude": { | ||||||
|         "**/.venv/**": true, |         "**/.venv/**": true, | ||||||
|         "**/pytest_cache/**": true |         "**/pytest_cache/**": true | ||||||
|     }, |     } | ||||||
|     "python.testing.cwd": "${workspaceFolder}/src" |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,66 +2,9 @@ | |||||||
| 	"version": "2.0.0", | 	"version": "2.0.0", | ||||||
| 	"tasks": [ | 	"tasks": [ | ||||||
| 	{ | 	{ | ||||||
| 			"label": "Start: Celery Worker", | 		"label": "manage.py document_consumer", | ||||||
| 			"description": "Start the Celery Worker which processes background and consume tasks", |  | ||||||
| 		"type": "shell", | 		"type": "shell", | ||||||
| 			"command": "uv run celery --app paperless worker -l DEBUG", | 		"command": "pipenv run python manage.py document_consumer", | ||||||
| 			"isBackground": true, |  | ||||||
| 			"options": { |  | ||||||
| 				"cwd": "${workspaceFolder}/src" |  | ||||||
| 			}, |  | ||||||
| 			"problemMatcher": [ |  | ||||||
| 				{ |  | ||||||
| 					"owner": "custom", |  | ||||||
| 					"pattern": [ |  | ||||||
| 						{ |  | ||||||
| 							"regexp": ".", |  | ||||||
| 							"file": 1, |  | ||||||
| 							"location": 2, |  | ||||||
| 							"message": 3 |  | ||||||
| 						} |  | ||||||
| 					], |  | ||||||
| 					"background": { |  | ||||||
| 						"activeOnStart": true, |  | ||||||
| 						"beginsPattern": "celery.*", |  | ||||||
| 						"endsPattern": "ready" |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			] |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"label": "Start: Frontend Angular", |  | ||||||
| 			"description": "Start the Frontend Angular Dev Server", |  | ||||||
| 			"type": "shell", |  | ||||||
| 			"command": "pnpm start", |  | ||||||
| 			"isBackground": true, |  | ||||||
| 			"options": { |  | ||||||
| 				"cwd": "${workspaceFolder}/src-ui" |  | ||||||
| 			}, |  | ||||||
| 			"problemMatcher": [ |  | ||||||
| 				{ |  | ||||||
| 					"owner": "custom", |  | ||||||
| 					"pattern": [ |  | ||||||
| 						{ |  | ||||||
| 							"regexp": ".", |  | ||||||
| 							"file": 1, |  | ||||||
| 							"location": 2, |  | ||||||
| 							"message": 3 |  | ||||||
| 						} |  | ||||||
| 					], |  | ||||||
| 					"background": { |  | ||||||
| 						"activeOnStart": true, |  | ||||||
| 						"beginsPattern": ".*", |  | ||||||
| 						"endsPattern": "Compiled successfully" |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 			] |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"label": "Start: Consumer Service (manage.py document_consumer)", |  | ||||||
| 			"description": "Start the Consumer Service which processes files from a directory", |  | ||||||
| 			"type": "shell", |  | ||||||
| 			"command": "uv run python manage.py document_consumer", |  | ||||||
| 		"group": "build", | 		"group": "build", | ||||||
| 		"presentation": { | 		"presentation": { | ||||||
| 			"echo": true, | 			"echo": true, | ||||||
| @@ -75,12 +18,12 @@ | |||||||
| 		"options": { | 		"options": { | ||||||
| 			"cwd": "${workspaceFolder}/src" | 			"cwd": "${workspaceFolder}/src" | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		}, | 		}, | ||||||
| 		{ | 		{ | ||||||
| 			"label": "Start: Backend Server (manage.py runserver)", | 			"label": "manage.py runserver", | ||||||
| 			"description": "Start the Backend Server which serves the Django API and the compiled Angular frontend", |  | ||||||
| 			"type": "shell", | 			"type": "shell", | ||||||
| 			"command": "uv run python manage.py runserver", | 			"command": "pipenv run python manage.py runserver", | ||||||
| 			"group": "build", | 			"group": "build", | ||||||
| 			"presentation": { | 			"presentation": { | ||||||
| 				"echo": true, | 				"echo": true, | ||||||
| @@ -94,12 +37,12 @@ | |||||||
| 			"options": { | 			"options": { | ||||||
| 				"cwd": "${workspaceFolder}/src" | 				"cwd": "${workspaceFolder}/src" | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			}, | 			}, | ||||||
| 	  { | 	  { | ||||||
| 		"label": "Maintenance: manage.py migrate", | 		"label": "Maintenance: manage.py migrate", | ||||||
| 			"description": "Apply database migrations", |  | ||||||
| 		"type": "shell", | 		"type": "shell", | ||||||
| 			"command": "uv run python manage.py migrate", | 		"command": "pipenv run python manage.py migrate", | ||||||
| 		"group": "none", | 		"group": "none", | ||||||
| 		"presentation": { | 		"presentation": { | ||||||
| 			"echo": true, | 			"echo": true, | ||||||
| @@ -114,30 +57,10 @@ | |||||||
| 			"cwd": "${workspaceFolder}/src" | 			"cwd": "${workspaceFolder}/src" | ||||||
| 		} | 		} | ||||||
| 	  }, | 	  }, | ||||||
| 		{ |  | ||||||
| 			"label": "Maintenance: Build Documentation", |  | ||||||
| 			"description": "Build the documentation with MkDocs", |  | ||||||
| 			"type": "shell", |  | ||||||
| 			"command": "uv run mkdocs build --config-file mkdocs.yml && uv run mkdocs serve", |  | ||||||
| 			"group": "none", |  | ||||||
| 			"presentation": { |  | ||||||
| 				"echo": true, |  | ||||||
| 				"reveal": "always", |  | ||||||
| 				"focus": true, |  | ||||||
| 				"panel": "shared", |  | ||||||
| 				"showReuseMessage": false, |  | ||||||
| 				"clear": true, |  | ||||||
| 				"revealProblems": "onProblem" |  | ||||||
| 			}, |  | ||||||
| 			"options": { |  | ||||||
| 				"cwd": "${workspaceFolder}" |  | ||||||
| 			} |  | ||||||
| 		}, |  | ||||||
| 	  { | 	  { | ||||||
| 		"label": "Maintenance: manage.py createsuperuser", | 		"label": "Maintenance: manage.py createsuperuser", | ||||||
| 			"description": "Create a superuser", |  | ||||||
| 		"type": "shell", | 		"type": "shell", | ||||||
| 			"command": "uv run python manage.py createsuperuser", | 		"command": "pipenv run python manage.py createsuperuser", | ||||||
| 		"group": "none", | 		"group": "none", | ||||||
| 		"presentation": { | 		"presentation": { | ||||||
| 			"echo": true, | 			"echo": true, | ||||||
| @@ -153,39 +76,9 @@ | |||||||
| 		} | 		} | ||||||
| 	  }, | 	  }, | ||||||
| 	  { | 	  { | ||||||
| 			"label": "Maintenance: recreate .venv", | 		"label": "compile frontend", | ||||||
| 			"description": "Recreate the python virtual environment and install python dependencies", |  | ||||||
| 		"type": "shell", | 		"type": "shell", | ||||||
| 			"command": "rm -rf .venv && uv venv && uv sync --dev", | 		"command": "npm ci && ./node_modules/.bin/ng build --configuration production", | ||||||
| 			"group": "none", |  | ||||||
| 			"presentation": { |  | ||||||
| 				"echo": true, |  | ||||||
| 				"reveal": "always", |  | ||||||
| 				"focus": true, |  | ||||||
| 				"panel": "shared", |  | ||||||
| 				"showReuseMessage": false, |  | ||||||
| 				"clear": true, |  | ||||||
| 				"revealProblems": "onProblem" |  | ||||||
| 			}, |  | ||||||
| 			"options": { |  | ||||||
| 				"cwd": "${workspaceFolder}" |  | ||||||
| 			} |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"label": "Maintenance: Install Frontend Dependencies", |  | ||||||
| 			"description": "Install frontend (pnpm) dependencies", |  | ||||||
| 			"type": "pnpm", |  | ||||||
| 			"script": "install", |  | ||||||
| 			"path": "src-ui", |  | ||||||
| 			"group": "clean", |  | ||||||
| 			"problemMatcher": [], |  | ||||||
| 			"detail": "install dependencies from package" |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"description": "Clean install frontend dependencies and build the frontend for production", |  | ||||||
| 			"label": "Maintenance: Compile frontend for production", |  | ||||||
| 			"type": "shell", |  | ||||||
| 			"command": "pnpm install && ./node_modules/.bin/ng build --configuration production", |  | ||||||
| 		"group": "none", | 		"group": "none", | ||||||
| 		"presentation": { | 		"presentation": { | ||||||
| 			"echo": true, | 			"echo": true, | ||||||
| @@ -201,23 +94,43 @@ | |||||||
| 		} | 		} | ||||||
| 	  }, | 	  }, | ||||||
| 	  { | 	  { | ||||||
| 			"label": "Project Setup: Run all Init Tasks", | 		"label": "Maintenance: recreate .venv", | ||||||
| 			"description": "Runs all init tasks to setup the project including migrate the database, create a superuser and compile the frontend for production", | 		"type": "shell", | ||||||
| 			"dependsOrder": "sequence", | 		"command": "rm -R -v .venv/* || pipenv install --dev", | ||||||
| 			"dependsOn": [ | 		"group": "none", | ||||||
| 				"Maintenance: manage.py migrate", | 		"presentation": { | ||||||
| 				"Maintenance: manage.py createsuperuser", | 			"echo": true, | ||||||
| 				"Maintenance: Compile frontend for production" | 			"reveal": "always", | ||||||
| 			] | 			"focus": true, | ||||||
|  | 			"panel": "shared", | ||||||
|  | 			"showReuseMessage": false, | ||||||
|  | 			"clear": true, | ||||||
|  | 			"revealProblems": "onProblem" | ||||||
|  | 		}, | ||||||
|  | 		"options": { | ||||||
|  | 			"cwd": "${workspaceFolder}" | ||||||
|  | 		} | ||||||
| 	  }, | 	  }, | ||||||
| 	  { | 	  { | ||||||
| 			"label": "Project Start: Run all Services", | 		"label": "Celery Worker", | ||||||
| 			"description": "Runs all services required to start the project including the Celery Worker, the Consumer Service and the Backend Server", | 		"type": "shell", | ||||||
| 			"dependsOn": [ | 		"command": "pipenv run celery --app paperless worker -l DEBUG", | ||||||
| 				"Start: Celery Worker", | 		"group": { | ||||||
| 				"Start: Consumer Service (manage.py document_consumer)", | 		  "kind": "build", | ||||||
| 				"Start: Backend Server (manage.py runserver)" | 		  "isDefault": true | ||||||
| 			] | 		}, | ||||||
|  | 		"presentation": { | ||||||
|  | 			"echo": true, | ||||||
|  | 			"reveal": "always", | ||||||
|  | 			"focus": true, | ||||||
|  | 			"panel": "shared", | ||||||
|  | 			"showReuseMessage": false, | ||||||
|  | 			"clear": true, | ||||||
|  | 			"revealProblems": "onProblem" | ||||||
|  | 		}, | ||||||
|  | 		"options": { | ||||||
|  | 			"cwd": "${workspaceFolder}/src" | ||||||
|  | 		} | ||||||
| 	  } | 	  } | ||||||
| 	] | 	] | ||||||
| } |   } | ||||||
|   | |||||||
| @@ -26,5 +26,3 @@ | |||||||
| ./dist | ./dist | ||||||
| ./scripts | ./scripts | ||||||
| ./resources | ./resources | ||||||
| # Other stuff |  | ||||||
| **/*.drawio.png |  | ||||||
|   | |||||||
| @@ -27,6 +27,9 @@ indent_style = space | |||||||
| [*.md] | [*.md] | ||||||
| indent_style = space | indent_style = space | ||||||
|  |  | ||||||
|  | [Pipfile.lock] | ||||||
|  | indent_style = space | ||||||
|  |  | ||||||
| # Tests don't get a line width restriction.  It's still a good idea to follow | # Tests don't get a line width restriction.  It's still a good idea to follow | ||||||
| # the 79 character rule, but in the interests of clarity, tests often need to | # the 79 character rule, but in the interests of clarity, tests often need to | ||||||
| # violate it. | # violate it. | ||||||
|   | |||||||
							
								
								
									
										55
									
								
								.github/DISCUSSION_TEMPLATE/support.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										55
									
								
								.github/DISCUSSION_TEMPLATE/support.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,55 +0,0 @@ | |||||||
| title: "[Support] " |  | ||||||
| body: |  | ||||||
|   - type: textarea |  | ||||||
|     id: description |  | ||||||
|     attributes: |  | ||||||
|       label: What's your question or issue? |  | ||||||
|       description: Provide a clear and concise description of what you're trying to do, and what's going wrong. |  | ||||||
|       placeholder: | |  | ||||||
|         I'm trying to... |  | ||||||
|  |  | ||||||
|         [Include screenshots if helpful] |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: textarea |  | ||||||
|     id: steps |  | ||||||
|     attributes: |  | ||||||
|       label: What have you tried? |  | ||||||
|       description: Describe any steps you've already taken to troubleshoot or solve the issue. |  | ||||||
|       placeholder: | |  | ||||||
|         - I checked the logs and saw... |  | ||||||
|         - I followed the install guide and tried... |  | ||||||
|   - type: input |  | ||||||
|     id: version |  | ||||||
|     attributes: |  | ||||||
|       label: Paperless-ngx version |  | ||||||
|       placeholder: e.g. 1.14.0 |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|   - type: input |  | ||||||
|     id: host-os |  | ||||||
|     attributes: |  | ||||||
|       label: Host OS |  | ||||||
|       description: Include architecture if relevant. |  | ||||||
|       placeholder: e.g. Ubuntu 22.04 / Raspberry Pi arm64 |  | ||||||
|   - type: dropdown |  | ||||||
|     id: install-method |  | ||||||
|     attributes: |  | ||||||
|       label: Installation method |  | ||||||
|       options: |  | ||||||
|         - Docker - official image |  | ||||||
|         - Docker - linuxserver.io image |  | ||||||
|         - Bare metal |  | ||||||
|         - Other (please describe above) |  | ||||||
|   - type: textarea |  | ||||||
|     id: system-status |  | ||||||
|     attributes: |  | ||||||
|       label: System status |  | ||||||
|       description: If available, copy & paste the system status output from Settings > System Status > Copy |  | ||||||
|       render: json |  | ||||||
|   - type: textarea |  | ||||||
|     id: logs |  | ||||||
|     attributes: |  | ||||||
|       label: Relevant logs or output |  | ||||||
|       description: If you have logs, errors that might help, paste it here. |  | ||||||
|       render: bash |  | ||||||
							
								
								
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | |||||||
| github: [shamoon, stumpylog] |  | ||||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -98,7 +98,7 @@ body: | |||||||
|       label: Browser |       label: Browser | ||||||
|       description: Which browser you are using, if relevant. |       description: Which browser you are using, if relevant. | ||||||
|       placeholder: e.g. Chrome, Safari |       placeholder: e.g. Chrome, Safari | ||||||
|   - type: textarea |   - type: input | ||||||
|     id: config-changes |     id: config-changes | ||||||
|     attributes: |     attributes: | ||||||
|       label: Configuration changes |       label: Configuration changes | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -2,10 +2,10 @@ blank_issues_enabled: false | |||||||
| contact_links: | contact_links: | ||||||
|   - name: 🤔 Questions and Help |   - name: 🤔 Questions and Help | ||||||
|     url: https://github.com/paperless-ngx/paperless-ngx/discussions |     url: https://github.com/paperless-ngx/paperless-ngx/discussions | ||||||
|     about: General questions or support for using Paperless-ngx. |     about: This issue tracker is not for support questions. Please refer to our Discussions. | ||||||
|   - name: 💬 Chat |   - name: 💬 Chat | ||||||
|     url: https://matrix.to/#/#paperlessngx:matrix.org |     url: https://matrix.to/#/#paperlessngx:matrix.org | ||||||
|     about: Want to discuss Paperless-ngx with others? Check out our chat. |     about: Want to discuss Paperless-ngx with others? Check out our chat. | ||||||
|   - name: 🚀 Feature Request |   - name: 🚀 Feature Request | ||||||
|     url: https://github.com/paperless-ngx/paperless-ngx/discussions/new?category=feature-requests |     url: https://github.com/paperless-ngx/paperless-ngx/discussions/new?category=feature-requests | ||||||
|     about: Remember to search for existing feature requests and "up-vote" those that you like. |     about: Remember to search for existing feature requests and "up-vote" any you like | ||||||
|   | |||||||
							
								
								
									
										77
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										77
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,13 +1,12 @@ | |||||||
| # Please see the documentation for all configuration options: | # https://docs.github.com/en/code-security/supply-chain-security/keeping-your-dependencies-updated-automatically/configuration-options-for-dependency-updates#package-ecosystem | ||||||
| # https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates |  | ||||||
| version: 2 | version: 2 | ||||||
| # Required for uv support for now |  | ||||||
| enable-beta-ecosystems: true |  | ||||||
| updates: | updates: | ||||||
|   # Enable version updates for pnpm |  | ||||||
|  |   # Enable version updates for npm | ||||||
|   - package-ecosystem: "npm" |   - package-ecosystem: "npm" | ||||||
|     target-branch: "dev" |     target-branch: "dev" | ||||||
|     # Look for `pnpm-lock.yaml` file in the `/src-ui` directory |     # Look for `package.json` and `lock` files in the `/src-ui` directory | ||||||
|     directory: "/src-ui" |     directory: "/src-ui" | ||||||
|     open-pull-requests-limit: 10 |     open-pull-requests-limit: 10 | ||||||
|     schedule: |     schedule: | ||||||
| @@ -15,6 +14,9 @@ updates: | |||||||
|     labels: |     labels: | ||||||
|       - "frontend" |       - "frontend" | ||||||
|       - "dependencies" |       - "dependencies" | ||||||
|  |     # Add reviewers | ||||||
|  |     reviewers: | ||||||
|  |       - "paperless-ngx/frontend" | ||||||
|     groups: |     groups: | ||||||
|       frontend-angular-dependencies: |       frontend-angular-dependencies: | ||||||
|         patterns: |         patterns: | ||||||
| @@ -30,9 +32,11 @@ updates: | |||||||
|         patterns: |         patterns: | ||||||
|           - "@typescript-eslint*" |           - "@typescript-eslint*" | ||||||
|           - "eslint" |           - "eslint" | ||||||
|  |  | ||||||
|   # Enable version updates for Python |   # Enable version updates for Python | ||||||
|   - package-ecosystem: "uv" |   - package-ecosystem: "pip" | ||||||
|     target-branch: "dev" |     target-branch: "dev" | ||||||
|  |     # Look for a `Pipfile` in the `root` directory | ||||||
|     directory: "/" |     directory: "/" | ||||||
|     # Check for updates once a week |     # Check for updates once a week | ||||||
|     schedule: |     schedule: | ||||||
| @@ -40,17 +44,20 @@ updates: | |||||||
|     labels: |     labels: | ||||||
|       - "backend" |       - "backend" | ||||||
|       - "dependencies" |       - "dependencies" | ||||||
|  |     # Add reviewers | ||||||
|  |     reviewers: | ||||||
|  |       - "paperless-ngx/backend" | ||||||
|  |     ignore: | ||||||
|  |       - dependency-name: "uvicorn" | ||||||
|     groups: |     groups: | ||||||
|       development: |       development: | ||||||
|         patterns: |         patterns: | ||||||
|           - "*pytest*" |           - "*pytest*" | ||||||
|           - "ruff" |           - "ruff" | ||||||
|           - "mkdocs-material" |           - "mkdocs-material" | ||||||
|           - "pre-commit*" |  | ||||||
|       django: |       django: | ||||||
|         patterns: |         patterns: | ||||||
|           - "*django*" |           - "*django*" | ||||||
|           - "drf-*" |  | ||||||
|       major-versions: |       major-versions: | ||||||
|         update-types: |         update-types: | ||||||
|           - "major" |           - "major" | ||||||
| @@ -58,13 +65,7 @@ updates: | |||||||
|         update-types: |         update-types: | ||||||
|           - "minor" |           - "minor" | ||||||
|           - "patch" |           - "patch" | ||||||
|         exclude-patterns: |  | ||||||
|           - "*django*" |  | ||||||
|           - "drf-*" |  | ||||||
|       pre-built: |  | ||||||
|         patterns: |  | ||||||
|           - psycopg* |  | ||||||
|           - zxing-cpp |  | ||||||
|   # Enable updates for GitHub Actions |   # Enable updates for GitHub Actions | ||||||
|   - package-ecosystem: "github-actions" |   - package-ecosystem: "github-actions" | ||||||
|     target-branch: "dev" |     target-branch: "dev" | ||||||
| @@ -75,50 +76,12 @@ updates: | |||||||
|     labels: |     labels: | ||||||
|       - "ci-cd" |       - "ci-cd" | ||||||
|       - "dependencies" |       - "dependencies" | ||||||
|  |     # Add reviewers | ||||||
|  |     reviewers: | ||||||
|  |       - "paperless-ngx/ci-cd" | ||||||
|     groups: |     groups: | ||||||
|       actions: |       actions: | ||||||
|         update-types: |         update-types: | ||||||
|           - "major" |           - "major" | ||||||
|           - "minor" |           - "minor" | ||||||
|           - "patch" |           - "patch" | ||||||
|   # Update Dockerfile in root directory |  | ||||||
|   - package-ecosystem: "docker" |  | ||||||
|     directories: |  | ||||||
|       - "/" |  | ||||||
|       - "/.devcontainer/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
|     open-pull-requests-limit: 5 |  | ||||||
|     labels: |  | ||||||
|       - "dependencies" |  | ||||||
|     commit-message: |  | ||||||
|       prefix: "docker" |  | ||||||
|       include: "scope" |  | ||||||
|   # Update Docker Compose files in docker/compose directory |  | ||||||
|   - package-ecosystem: "docker-compose" |  | ||||||
|     directory: "/docker/compose/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
|     open-pull-requests-limit: 5 |  | ||||||
|     labels: |  | ||||||
|       - "dependencies" |  | ||||||
|     commit-message: |  | ||||||
|       prefix: "docker-compose" |  | ||||||
|       include: "scope" |  | ||||||
|     groups: |  | ||||||
|       # Individual groups for each image |  | ||||||
|       gotenberg: |  | ||||||
|         patterns: |  | ||||||
|           - "docker.io/gotenberg/gotenberg*" |  | ||||||
|       tika: |  | ||||||
|         patterns: |  | ||||||
|           - "docker.io/apache/tika*" |  | ||||||
|       redis: |  | ||||||
|         patterns: |  | ||||||
|           - "docker.io/library/redis*" |  | ||||||
|       mariadb: |  | ||||||
|         patterns: |  | ||||||
|           - "docker.io/library/mariadb*" |  | ||||||
|       postgres: |  | ||||||
|         patterns: |  | ||||||
|           - "docker.io/library/postgres*" |  | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,26 +0,0 @@ | |||||||
| backend: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - 'src/**' |  | ||||||
|           - 'pyproject.toml' |  | ||||||
|           - 'uv.lock' |  | ||||||
|           - 'requirements.txt' |  | ||||||
| frontend: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - 'src-ui/**' |  | ||||||
| documentation: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - 'docs/**' |  | ||||||
| ci-cd: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - '.github/**' |  | ||||||
| # pr types |  | ||||||
| bug: |  | ||||||
|   - head-branch: |  | ||||||
|       - ['^fix'] |  | ||||||
| enhancement: |  | ||||||
|   - head-branch: |  | ||||||
|       - ['^feature'] |  | ||||||
							
								
								
									
										14
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,15 @@ | |||||||
|  | autolabeler: | ||||||
|  |   - label: "bug" | ||||||
|  |     branch: | ||||||
|  |       - '/^fix/' | ||||||
|  |     title: | ||||||
|  |       - "/^fix/i" | ||||||
|  |       - "/^Bugfix/i" | ||||||
|  |   - label: "enhancement" | ||||||
|  |     branch: | ||||||
|  |       - '/^feature/' | ||||||
|  |     title: | ||||||
|  |       - "/^feature/i" | ||||||
| categories: | categories: | ||||||
|   - title: 'Breaking Changes' |   - title: 'Breaking Changes' | ||||||
|     labels: |     labels: | ||||||
| @@ -5,7 +17,7 @@ categories: | |||||||
|   - title: 'Notable Changes' |   - title: 'Notable Changes' | ||||||
|     labels: |     labels: | ||||||
|       - 'notable' |       - 'notable' | ||||||
|   - title: 'Features / Enhancements' |   - title: 'Features' | ||||||
|     labels: |     labels: | ||||||
|       - 'enhancement' |       - 'enhancement' | ||||||
|   - title: 'Bug Fixes' |   - title: 'Bug Fixes' | ||||||
|   | |||||||
							
								
								
									
										651
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										651
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| name: ci | name: ci | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     tags: |     tags: | ||||||
| @@ -11,118 +12,90 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|     branches-ignore: |     branches-ignore: | ||||||
|       - 'translations**' |       - 'translations**' | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DEFAULT_UV_VERSION: "0.9.x" |   # This is the version of pipenv all the steps will use | ||||||
|  |   # If changing this, change Dockerfile | ||||||
|  |   DEFAULT_PIP_ENV_VERSION: "2024.0.3" | ||||||
|   # This is the default version of Python to use in most steps which aren't specific |   # This is the default version of Python to use in most steps which aren't specific | ||||||
|   DEFAULT_PYTHON_VERSION: "3.11" |   DEFAULT_PYTHON_VERSION: "3.11" | ||||||
|   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@v8 |  | ||||||
|         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: | ||||||
|     needs: |     # We want to run on external PRs, but not on our own internal PRs as they'll be run | ||||||
|       - detect-duplicate |     # by the push to the branch. Without this if check, checks are duplicated since | ||||||
|     if: needs.detect-duplicate.outputs.should_run == 'true' |     # internal PRs match both the push and pull_request events. | ||||||
|  |     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-22.04 | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout repository | ||||||
|       - name: Install python |         uses: actions/checkout@v4 | ||||||
|         uses: actions/setup-python@v6 |       - | ||||||
|  |         name: Install python | ||||||
|  |         uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Check files |       - | ||||||
|  |         name: Check files | ||||||
|         uses: pre-commit/action@v3.0.1 |         uses: pre-commit/action@v3.0.1 | ||||||
|  |  | ||||||
|   documentation: |   documentation: | ||||||
|     name: "Build & Deploy Documentation" |     name: "Build & Deploy Documentation" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - pre-commit |       - pre-commit | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout | ||||||
|       - name: Set up Python |         uses: actions/checkout@v4 | ||||||
|  |       - | ||||||
|  |         name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v6 |         uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |           cache: "pipenv" | ||||||
|         uses: astral-sh/setup-uv@v6 |           cache-dependency-path: 'Pipfile.lock' | ||||||
|         with: |       - | ||||||
|           version: ${{ env.DEFAULT_UV_VERSION }} |         name: Install pipenv | ||||||
|           enable-cache: true |  | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |  | ||||||
|       - name: Install Python dependencies |  | ||||||
|         run: | |         run: | | ||||||
|           uv sync --python ${{ steps.setup-python.outputs.python-version }} --dev --frozen |           pip install --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} | ||||||
|       - name: Make documentation |       - | ||||||
|  |         name: Install dependencies | ||||||
|         run: | |         run: | | ||||||
|           uv run \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} sync --dev | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |       - | ||||||
|             --dev \ |         name: List installed Python dependencies | ||||||
|             --frozen \ |         run: | | ||||||
|             mkdocs build --config-file ./mkdocs.yml |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run pip list | ||||||
|       - name: Deploy documentation |       - | ||||||
|  |         name: Make documentation | ||||||
|  |         run: | | ||||||
|  |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run mkdocs build --config-file ./mkdocs.yml | ||||||
|  |       - | ||||||
|  |         name: Deploy documentation | ||||||
|         if: github.event_name == 'push' && github.ref == 'refs/heads/main' |         if: github.event_name == 'push' && github.ref == 'refs/heads/main' | ||||||
|         run: | |         run: | | ||||||
|           echo "docs.paperless-ngx.com" > "${{ github.workspace }}/docs/CNAME" |           echo "docs.paperless-ngx.com" > "${{ github.workspace }}/docs/CNAME" | ||||||
|           git config --global user.name "${{ github.actor }}" |           git config --global user.name "${{ github.actor }}" | ||||||
|           git config --global user.email "${{ github.actor }}@users.noreply.github.com" |           git config --global user.email "${{ github.actor }}@users.noreply.github.com" | ||||||
|           uv run \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run mkdocs gh-deploy --force --no-history | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |       - | ||||||
|             --dev \ |         name: Upload artifact | ||||||
|             --frozen \ |  | ||||||
|             mkdocs gh-deploy --force --no-history |  | ||||||
|       - name: Upload artifact |  | ||||||
|         uses: actions/upload-artifact@v4 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: documentation |           name: documentation | ||||||
|           path: site/ |           path: site/ | ||||||
|           retention-days: 7 |           retention-days: 7 | ||||||
|  |  | ||||||
|   tests-backend: |   tests-backend: | ||||||
|     name: "Backend Tests (Python ${{ matrix.python-version }})" |     name: "Backend Tests (Python ${{ matrix.python-version }})" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - pre-commit |       - pre-commit | ||||||
|     strategy: |     strategy: | ||||||
| @@ -130,103 +103,107 @@ jobs: | |||||||
|         python-version: ['3.10', '3.11', '3.12'] |         python-version: ['3.10', '3.11', '3.12'] | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout | ||||||
|       - name: Start containers |         uses: actions/checkout@v4 | ||||||
|  |       - | ||||||
|  |         name: Start containers | ||||||
|         run: | |         run: | | ||||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml pull --quiet |           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml pull --quiet | ||||||
|           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@v6 |         uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: "${{ matrix.python-version }}" |           python-version: "${{ matrix.python-version }}" | ||||||
|       - name: Install uv |           cache: "pipenv" | ||||||
|         uses: astral-sh/setup-uv@v6 |           cache-dependency-path: 'Pipfile.lock' | ||||||
|         with: |       - | ||||||
|           version: ${{ env.DEFAULT_UV_VERSION }} |         name: Install pipenv | ||||||
|           enable-cache: true |         run: | | ||||||
|           python-version: ${{ steps.setup-python.outputs.python-version }} |           pip install --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} | ||||||
|       - name: Install system dependencies |       - | ||||||
|  |         name: Install system dependencies | ||||||
|         run: | |         run: | | ||||||
|           sudo apt-get update -qq |           sudo apt-get update -qq | ||||||
|           sudo apt-get install -qq --no-install-recommends unpaper tesseract-ocr imagemagick ghostscript libzbar0 poppler-utils |           sudo apt-get install -qq --no-install-recommends unpaper tesseract-ocr imagemagick ghostscript libzbar0 poppler-utils | ||||||
|       - name: Configure ImageMagick |       - | ||||||
|  |         name: Configure ImageMagick | ||||||
|         run: | |         run: | | ||||||
|           sudo cp docker/rootfs/etc/ImageMagick-6/paperless-policy.xml /etc/ImageMagick-6/policy.xml |           sudo cp docker/imagemagick-policy.xml /etc/ImageMagick-6/policy.xml | ||||||
|       - name: Install Python dependencies |       - | ||||||
|  |         name: Install Python dependencies | ||||||
|         run: | |         run: | | ||||||
|           uv sync \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run python --version | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} sync --dev | ||||||
|             --group testing \ |       - | ||||||
|             --frozen |         name: List installed Python dependencies | ||||||
|       - name: List installed Python dependencies |  | ||||||
|         run: | |         run: | | ||||||
|           uv pip list |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run pip list | ||||||
|       - name: Install or update NLTK dependencies |       - | ||||||
|         run: uv run python -m nltk.downloader punkt punkt_tab snowball_data stopwords -d ${{ env.NLTK_DATA }} |         name: Tests | ||||||
|       - name: Tests |  | ||||||
|         env: |         env: | ||||||
|           NLTK_DATA: ${{ env.NLTK_DATA }} |  | ||||||
|           PAPERLESS_CI_TEST: 1 |           PAPERLESS_CI_TEST: 1 | ||||||
|           # Enable paperless_mail testing against real server |           # Enable paperless_mail testing against real server | ||||||
|           PAPERLESS_MAIL_TEST_HOST: ${{ secrets.TEST_MAIL_HOST }} |           PAPERLESS_MAIL_TEST_HOST: ${{ secrets.TEST_MAIL_HOST }} | ||||||
|           PAPERLESS_MAIL_TEST_USER: ${{ secrets.TEST_MAIL_USER }} |           PAPERLESS_MAIL_TEST_USER: ${{ secrets.TEST_MAIL_USER }} | ||||||
|           PAPERLESS_MAIL_TEST_PASSWD: ${{ secrets.TEST_MAIL_PASSWD }} |           PAPERLESS_MAIL_TEST_PASSWD: ${{ secrets.TEST_MAIL_PASSWD }} | ||||||
|         run: | |         run: | | ||||||
|           uv run \ |           cd src/ | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run pytest -ra | ||||||
|             --dev \ |       - | ||||||
|             --frozen \ |         name: Upload coverage | ||||||
|             pytest |         if: ${{ matrix.python-version == env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Upload backend test results to Codecov |         uses: actions/upload-artifact@v4 | ||||||
|         if: always() |  | ||||||
|         uses: codecov/test-results-action@v1 |  | ||||||
|         with: |         with: | ||||||
|           flags: backend-python-${{ matrix.python-version }} |           name: backend-coverage-report | ||||||
|           files: junit.xml |           path: src/coverage.xml | ||||||
|       - name: Upload backend coverage to Codecov |           retention-days: 7 | ||||||
|         uses: codecov/codecov-action@v5 |           if-no-files-found: warn | ||||||
|         with: |       - | ||||||
|           flags: backend-python-${{ matrix.python-version }} |         name: Stop containers | ||||||
|           files: coverage.xml |  | ||||||
|       - name: Stop containers |  | ||||||
|         if: always() |         if: always() | ||||||
|         run: | |         run: | | ||||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml logs |           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml logs | ||||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml down |           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml down | ||||||
|   install-frontend-dependencies: |  | ||||||
|  |   install-frontend-depedendencies: | ||||||
|     name: "Install Frontend Dependencies" |     name: "Install Frontend Dependencies" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - pre-commit |       - pre-commit | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v5 |       - uses: actions/checkout@v4 | ||||||
|       - name: Install pnpm |       - | ||||||
|         uses: pnpm/action-setup@v4 |         name: Use Node.js 20 | ||||||
|         with: |         uses: actions/setup-node@v4 | ||||||
|           version: 10 |  | ||||||
|       - name: Use Node.js 20 |  | ||||||
|         uses: actions/setup-node@v5 |  | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'npm' | ||||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' |           cache-dependency-path: 'src-ui/package-lock.json' | ||||||
|       - name: Cache frontend dependencies |       - name: Cache frontend dependencies | ||||||
|         id: cache-frontend-deps |         id: cache-frontend-deps | ||||||
|         uses: actions/cache@v4 |         uses: actions/cache@v4 | ||||||
|         with: |         with: | ||||||
|           path: | |           path: | | ||||||
|             ~/.pnpm-store |             ~/.npm | ||||||
|             ~/.cache |             ~/.cache | ||||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} |           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||||
|       - name: Install dependencies |       - | ||||||
|         run: cd src-ui && pnpm install |         name: Install dependencies | ||||||
|  |         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||||
|  |         run: cd src-ui && npm ci | ||||||
|  |       - | ||||||
|  |         name: Install Playwright | ||||||
|  |         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||||
|  |         run: cd src-ui && npx playwright install --with-deps | ||||||
|  |  | ||||||
|   tests-frontend: |   tests-frontend: | ||||||
|     name: "Frontend Unit Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" |     name: "Frontend Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - install-frontend-dependencies |       - install-frontend-depedendencies | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
| @@ -234,135 +211,139 @@ jobs: | |||||||
|         shard-index: [1, 2, 3, 4] |         shard-index: [1, 2, 3, 4] | ||||||
|         shard-count: [4] |         shard-count: [4] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v5 |       - uses: actions/checkout@v4 | ||||||
|       - name: Install pnpm |       - | ||||||
|         uses: pnpm/action-setup@v4 |         name: Use Node.js 20 | ||||||
|         with: |         uses: actions/setup-node@v4 | ||||||
|           version: 10 |  | ||||||
|       - name: Use Node.js 20 |  | ||||||
|         uses: actions/setup-node@v5 |  | ||||||
|         with: |         with: | ||||||
|           node-version: 20.x |           node-version: 20.x | ||||||
|           cache: 'pnpm' |           cache: 'npm' | ||||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' |           cache-dependency-path: 'src-ui/package-lock.json' | ||||||
|       - name: Cache frontend dependencies |       - name: Cache frontend dependencies | ||||||
|         id: cache-frontend-deps |         id: cache-frontend-deps | ||||||
|         uses: actions/cache@v4 |         uses: actions/cache@v4 | ||||||
|         with: |         with: | ||||||
|           path: | |           path: | | ||||||
|             ~/.pnpm-store |             ~/.npm | ||||||
|             ~/.cache |  | ||||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} |  | ||||||
|       - name: Re-link Angular cli |  | ||||||
|         run: cd src-ui && pnpm link @angular/cli |  | ||||||
|       - name: Linting checks |  | ||||||
|         run: cd src-ui && pnpm run lint |  | ||||||
|       - name: Run Jest unit tests |  | ||||||
|         run: cd src-ui && pnpm run test --max-workers=2 --shard=${{ matrix.shard-index }}/${{ matrix.shard-count }} |  | ||||||
|       - name: Upload frontend test results to Codecov |  | ||||||
|         uses: codecov/test-results-action@v1 |  | ||||||
|         if: always() |  | ||||||
|         with: |  | ||||||
|           flags: frontend-node-${{ matrix.node-version }} |  | ||||||
|           directory: src-ui/ |  | ||||||
|       - name: Upload frontend coverage to Codecov |  | ||||||
|         uses: codecov/codecov-action@v5 |  | ||||||
|         with: |  | ||||||
|           flags: frontend-node-${{ matrix.node-version }} |  | ||||||
|           directory: src-ui/coverage/ |  | ||||||
|   tests-frontend-e2e: |  | ||||||
|     name: "Frontend E2E Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" |  | ||||||
|     runs-on: ubuntu-24.04 |  | ||||||
|     needs: |  | ||||||
|       - install-frontend-dependencies |  | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         node-version: [20.x] |  | ||||||
|         shard-index: [1, 2] |  | ||||||
|         shard-count: [2] |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v5 |  | ||||||
|       - name: Install pnpm |  | ||||||
|         uses: pnpm/action-setup@v4 |  | ||||||
|         with: |  | ||||||
|           version: 10 |  | ||||||
|       - name: Use Node.js 20 |  | ||||||
|         uses: actions/setup-node@v5 |  | ||||||
|         with: |  | ||||||
|           node-version: 20.x |  | ||||||
|           cache: 'pnpm' |  | ||||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' |  | ||||||
|       - name: Cache frontend dependencies |  | ||||||
|         id: cache-frontend-deps |  | ||||||
|         uses: actions/cache@v4 |  | ||||||
|         with: |  | ||||||
|           path: | |  | ||||||
|             ~/.pnpm-store |  | ||||||
|             ~/.cache |  | ||||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} |  | ||||||
|       - name: Re-link Angular cli |  | ||||||
|         run: cd src-ui && pnpm link @angular/cli |  | ||||||
|       - name: Cache Playwright browsers |  | ||||||
|         uses: actions/cache@v4 |  | ||||||
|         with: |  | ||||||
|           path: ~/.cache/ms-playwright |  | ||||||
|           key: ${{ runner.os }}-playwright-${{ hashFiles('src-ui/pnpm-lock.yaml') }} |  | ||||||
|           restore-keys: | |  | ||||||
|             ${{ runner.os }}-playwright- |  | ||||||
|       - name: Install Playwright system dependencies |  | ||||||
|         run: npx playwright install-deps |  | ||||||
|       - name: Install dependencies |  | ||||||
|         run: cd src-ui && pnpm install --no-frozen-lockfile |  | ||||||
|       - name: Install Playwright |  | ||||||
|         run: cd src-ui && pnpm exec playwright install |  | ||||||
|       - name: Run Playwright e2e tests |  | ||||||
|         run: cd src-ui && pnpm exec playwright test --shard ${{ matrix.shard-index }}/${{ matrix.shard-count }} |  | ||||||
|   frontend-bundle-analysis: |  | ||||||
|     name: "Frontend Bundle Analysis" |  | ||||||
|     runs-on: ubuntu-24.04 |  | ||||||
|     needs: |  | ||||||
|       - tests-frontend |  | ||||||
|       - tests-frontend-e2e |  | ||||||
|     steps: |  | ||||||
|       - uses: actions/checkout@v5 |  | ||||||
|       - name: Install pnpm |  | ||||||
|         uses: pnpm/action-setup@v4 |  | ||||||
|         with: |  | ||||||
|           version: 10 |  | ||||||
|       - name: Use Node.js 20 |  | ||||||
|         uses: actions/setup-node@v5 |  | ||||||
|         with: |  | ||||||
|           node-version: 20.x |  | ||||||
|           cache: 'pnpm' |  | ||||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' |  | ||||||
|       - name: Cache frontend dependencies |  | ||||||
|         id: cache-frontend-deps |  | ||||||
|         uses: actions/cache@v4 |  | ||||||
|         with: |  | ||||||
|           path: | |  | ||||||
|             ~/.pnpm-store |  | ||||||
|             ~/.cache |             ~/.cache | ||||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} |           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||||
|       - name: Re-link Angular cli |       - name: Re-link Angular cli | ||||||
|         run: cd src-ui && pnpm link @angular/cli |         run: cd src-ui && npm link @angular/cli | ||||||
|       - name: Build frontend and upload analysis |       - | ||||||
|  |         name: Linting checks | ||||||
|  |         run: cd src-ui && npm run lint | ||||||
|  |       - | ||||||
|  |         name: Run Jest unit tests | ||||||
|  |         run: cd src-ui && npm run test -- --max-workers=2 --shard=${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||||
|  |       - | ||||||
|  |         name: Upload Jest coverage | ||||||
|  |         if: always() | ||||||
|  |         uses: actions/upload-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: jest-coverage-report-${{ matrix.shard-index }} | ||||||
|  |           path: | | ||||||
|  |             src-ui/coverage/coverage-final.json | ||||||
|  |             src-ui/coverage/lcov.info | ||||||
|  |             src-ui/coverage/clover.xml | ||||||
|  |           retention-days: 7 | ||||||
|  |           if-no-files-found: warn | ||||||
|  |       - | ||||||
|  |         name: Run Playwright e2e tests | ||||||
|  |         run: cd src-ui && npx playwright test --shard ${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||||
|  |       - | ||||||
|  |         name: Upload Playwright test results | ||||||
|  |         if: always() | ||||||
|  |         uses: actions/upload-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: playwright-report-${{ matrix.shard-index }} | ||||||
|  |           path: src-ui/playwright-report | ||||||
|  |           retention-days: 7 | ||||||
|  |  | ||||||
|  |   tests-coverage-upload: | ||||||
|  |     name: "Upload to Codecov" | ||||||
|  |     runs-on: ubuntu-22.04 | ||||||
|  |     needs: | ||||||
|  |       - tests-backend | ||||||
|  |       - tests-frontend | ||||||
|  |     steps: | ||||||
|  |       - | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|  |       - | ||||||
|  |         name: Download frontend jest coverage | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           path: src-ui/coverage/ | ||||||
|  |           pattern: jest-coverage-report-* | ||||||
|  |       - | ||||||
|  |         name: Download frontend playwright coverage | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           path: src-ui/coverage/ | ||||||
|  |           pattern: playwright-report-* | ||||||
|  |           merge-multiple: true | ||||||
|  |       - | ||||||
|  |         name: Upload frontend coverage to Codecov | ||||||
|  |         uses: codecov/codecov-action@v4 | ||||||
|  |         with: | ||||||
|  |           # not required for public repos, but intermittently fails otherwise | ||||||
|  |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|  |           flags: frontend | ||||||
|  |           directory: src-ui/coverage/ | ||||||
|  |           # dont include backend coverage files here | ||||||
|  |           files: '!coverage.xml' | ||||||
|  |       - | ||||||
|  |         name: Download backend coverage | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: backend-coverage-report | ||||||
|  |           path: src/ | ||||||
|  |       - | ||||||
|  |         name: Upload coverage to Codecov | ||||||
|  |         uses: codecov/codecov-action@v4 | ||||||
|  |         with: | ||||||
|  |           # not required for public repos, but intermittently fails otherwise | ||||||
|  |           token: ${{ secrets.CODECOV_TOKEN }} | ||||||
|  |           # future expansion | ||||||
|  |           flags: backend | ||||||
|  |           directory: src/ | ||||||
|  |       - | ||||||
|  |         name: Use Node.js 20 | ||||||
|  |         uses: actions/setup-node@v4 | ||||||
|  |         with: | ||||||
|  |           node-version: 20.x | ||||||
|  |           cache: 'npm' | ||||||
|  |           cache-dependency-path: 'src-ui/package-lock.json' | ||||||
|  |       - | ||||||
|  |         name: Cache frontend dependencies | ||||||
|  |         id: cache-frontend-deps | ||||||
|  |         uses: actions/cache@v4 | ||||||
|  |         with: | ||||||
|  |           path: | | ||||||
|  |             ~/.npm | ||||||
|  |             ~/.cache | ||||||
|  |           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||||
|  |       - | ||||||
|  |         name: Re-link Angular cli | ||||||
|  |         run: cd src-ui && npm link @angular/cli | ||||||
|  |       - | ||||||
|  |         name: Build frontend and upload analysis | ||||||
|         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 && ng build --configuration=production | ||||||
|  |  | ||||||
|   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-22.04 | ||||||
|     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v') || startsWith(github.ref, 'refs/heads/l10n_')) |     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v')) | ||||||
|     concurrency: |     concurrency: | ||||||
|       group: ${{ github.workflow }}-build-docker-image-${{ github.ref_name }} |       group: ${{ github.workflow }}-build-docker-image-${{ github.ref_name }} | ||||||
|       cancel-in-progress: true |       cancel-in-progress: true | ||||||
|     needs: |     needs: | ||||||
|       - tests-backend |       - tests-backend | ||||||
|       - tests-frontend |       - tests-frontend | ||||||
|       - tests-frontend-e2e |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Check pushing to Docker Hub |       - | ||||||
|  |         name: Check pushing to Docker Hub | ||||||
|         id: push-other-places |         id: push-other-places | ||||||
|         # Only push to Dockerhub from the main repo AND the ref is either: |         # Only push to Dockerhub from the main repo AND the ref is either: | ||||||
|         #  main |         #  main | ||||||
| @@ -378,13 +359,15 @@ jobs: | |||||||
|             echo "Not pushing to DockerHub" |             echo "Not pushing to DockerHub" | ||||||
|             echo "enable=false" >> $GITHUB_OUTPUT |             echo "enable=false" >> $GITHUB_OUTPUT | ||||||
|           fi |           fi | ||||||
|       - name: Set ghcr repository name |       - | ||||||
|  |         name: Set ghcr repository name | ||||||
|         id: set-ghcr-repository |         id: set-ghcr-repository | ||||||
|         run: | |         run: | | ||||||
|           ghcr_name=$(echo "${{ github.repository }}" | awk '{ print tolower($0) }') |           ghcr_name=$(echo "${{ github.repository }}" | awk '{ print tolower($0) }') | ||||||
|           echo "Name is ${ghcr_name}" |           echo "Name is ${ghcr_name}" | ||||||
|           echo "ghcr-repository=${ghcr_name}" >> $GITHUB_OUTPUT |           echo "ghcr-repository=${ghcr_name}" >> $GITHUB_OUTPUT | ||||||
|       - name: Gather Docker metadata |       - | ||||||
|  |         name: Gather Docker metadata | ||||||
|         id: docker-meta |         id: docker-meta | ||||||
|         uses: docker/metadata-action@v5 |         uses: docker/metadata-action@v5 | ||||||
|         with: |         with: | ||||||
| @@ -399,39 +382,46 @@ jobs: | |||||||
|             # For a tag x.y.z or vX.Y.Z, output an x.y.z and x.y image tag |             # For a tag x.y.z or vX.Y.Z, output an x.y.z and x.y image tag | ||||||
|             type=semver,pattern={{version}} |             type=semver,pattern={{version}} | ||||||
|             type=semver,pattern={{major}}.{{minor}} |             type=semver,pattern={{major}}.{{minor}} | ||||||
|       - name: Checkout |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|       # If https://github.com/docker/buildx/issues/1044 is resolved, |       # If https://github.com/docker/buildx/issues/1044 is resolved, | ||||||
|       # the append input with a native arm64 arch could be used to |       # the append input with a native arm64 arch could be used to | ||||||
|       # significantly speed up building |       # significantly speed up building | ||||||
|       - name: Set up Docker Buildx |       - | ||||||
|  |         name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v3 | ||||||
|       - name: Set up QEMU |       - | ||||||
|  |         name: Set up QEMU | ||||||
|         uses: docker/setup-qemu-action@v3 |         uses: docker/setup-qemu-action@v3 | ||||||
|         with: |         with: | ||||||
|           platforms: arm64 |           platforms: arm64 | ||||||
|       - name: Login to GitHub Container Registry |       - | ||||||
|  |         name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v3 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.actor }} |           username: ${{ github.actor }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Login to Docker Hub |       - | ||||||
|  |         name: Login to Docker Hub | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v3 | ||||||
|         # Don't attempt to login if not pushing to Docker Hub |         # Don't attempt to login is not pushing to Docker Hub | ||||||
|         if: steps.push-other-places.outputs.enable == 'true' |         if: steps.push-other-places.outputs.enable == 'true' | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} |           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} |           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|       - name: Login to Quay.io |       - | ||||||
|  |         name: Login to Quay.io | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v3 | ||||||
|         # Don't attempt to login if not pushing to Quay.io |         # Don't attempt to login is not pushing to Quay.io | ||||||
|         if: steps.push-other-places.outputs.enable == 'true' |         if: steps.push-other-places.outputs.enable == 'true' | ||||||
|         with: |         with: | ||||||
|           registry: quay.io |           registry: quay.io | ||||||
|           username: ${{ secrets.QUAY_USERNAME }} |           username: ${{ secrets.QUAY_USERNAME }} | ||||||
|           password: ${{ secrets.QUAY_ROBOT_TOKEN }} |           password: ${{ secrets.QUAY_ROBOT_TOKEN }} | ||||||
|       - name: Build and push |       - | ||||||
|  |         name: Build and push | ||||||
|         uses: docker/build-push-action@v6 |         uses: docker/build-push-action@v6 | ||||||
|         with: |         with: | ||||||
|           context: . |           context: . | ||||||
| @@ -449,72 +439,82 @@ jobs: | |||||||
|             type=registry,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:dev |             type=registry,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:dev | ||||||
|           cache-to: | |           cache-to: | | ||||||
|             type=registry,mode=max,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:${{ github.ref_name }} |             type=registry,mode=max,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:${{ github.ref_name }} | ||||||
|       - name: Inspect image |       - | ||||||
|  |         name: Inspect image | ||||||
|         run: | |         run: | | ||||||
|           docker buildx imagetools inspect ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} |           docker buildx imagetools inspect ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} | ||||||
|       - name: Export frontend artifact from docker |       - | ||||||
|  |         name: Export frontend artifact from docker | ||||||
|         run: | |         run: | | ||||||
|           docker create --name frontend-extract ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} |           docker create --name frontend-extract ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} | ||||||
|           docker cp frontend-extract:/usr/src/paperless/src/documents/static/frontend src/documents/static/frontend/ |           docker cp frontend-extract:/usr/src/paperless/src/documents/static/frontend src/documents/static/frontend/ | ||||||
|       - name: Upload frontend artifact |       - | ||||||
|  |         name: Upload frontend artifact | ||||||
|         uses: actions/upload-artifact@v4 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: frontend-compiled |           name: frontend-compiled | ||||||
|           path: src/documents/static/frontend/ |           path: src/documents/static/frontend/ | ||||||
|           retention-days: 7 |           retention-days: 7 | ||||||
|  |  | ||||||
|   build-release: |   build-release: | ||||||
|     name: "Build Release" |     name: "Build Release" | ||||||
|     needs: |     needs: | ||||||
|       - build-docker-image |       - build-docker-image | ||||||
|       - documentation |       - documentation | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout | ||||||
|       - name: Set up Python |         uses: actions/checkout@v4 | ||||||
|  |       - | ||||||
|  |         name: Set up Python | ||||||
|         id: setup-python |         id: setup-python | ||||||
|         uses: actions/setup-python@v6 |         uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |           cache: "pipenv" | ||||||
|         uses: astral-sh/setup-uv@v6 |           cache-dependency-path: 'Pipfile.lock' | ||||||
|         with: |       - | ||||||
|           version: ${{ env.DEFAULT_UV_VERSION }} |         name: Install pipenv + tools | ||||||
|           enable-cache: true |  | ||||||
|           python-version: ${{ steps.setup-python.outputs.python-version }} |  | ||||||
|       - name: Install Python dependencies |  | ||||||
|         run: | |         run: | | ||||||
|           uv sync --python ${{ steps.setup-python.outputs.python-version }} --dev --frozen |           pip install --upgrade --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} setuptools wheel | ||||||
|       - name: Install system dependencies |       - | ||||||
|  |         name: Install Python dependencies | ||||||
|  |         run: | | ||||||
|  |           pipenv --python ${{ steps.setup-python.outputs.python-version }} sync --dev | ||||||
|  |       - | ||||||
|  |         name: Install system dependencies | ||||||
|         run: | |         run: | | ||||||
|           sudo apt-get update -qq |           sudo apt-get update -qq | ||||||
|           sudo apt-get install -qq --no-install-recommends gettext liblept5 |           sudo apt-get install -qq --no-install-recommends gettext liblept5 | ||||||
|       - name: Download frontend artifact |       - | ||||||
|         uses: actions/download-artifact@v5 |         name: Download frontend artifact | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: frontend-compiled |           name: frontend-compiled | ||||||
|           path: src/documents/static/frontend/ |           path: src/documents/static/frontend/ | ||||||
|       - name: Download documentation artifact |       - | ||||||
|         uses: actions/download-artifact@v5 |         name: Download documentation artifact | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: documentation |           name: documentation | ||||||
|           path: docs/_build/html/ |           path: docs/_build/html/ | ||||||
|       - name: Generate requirements file |       - | ||||||
|  |         name: Generate requirements file | ||||||
|         run: | |         run: | | ||||||
|           uv export --quiet --no-dev --all-extras --format requirements-txt --output-file requirements.txt |           pipenv --python ${{ steps.setup-python.outputs.python-version }} requirements > requirements.txt | ||||||
|       - name: Compile messages |       - | ||||||
|  |         name: Compile messages | ||||||
|         run: | |         run: | | ||||||
|           cd src/ |           cd src/ | ||||||
|           uv run \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run python3 manage.py compilemessages | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |       - | ||||||
|             manage.py compilemessages |         name: Collect static files | ||||||
|       - name: Collect static files |  | ||||||
|         run: | |         run: | | ||||||
|           cd src/ |           cd src/ | ||||||
|           uv run \ |           pipenv --python ${{ steps.setup-python.outputs.python-version }} run python3 manage.py collectstatic --no-input | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |       - | ||||||
|             manage.py collectstatic --no-input |         name: Move files | ||||||
|       - name: Move files |  | ||||||
|         run: | |         run: | | ||||||
|           echo "Making dist folders" |           echo "Making dist folders" | ||||||
|           for directory in dist \ |           for directory in dist \ | ||||||
| @@ -528,12 +528,13 @@ jobs: | |||||||
|           for file_name in .dockerignore \ |           for file_name in .dockerignore \ | ||||||
|                           .env \ |                           .env \ | ||||||
|                           Dockerfile \ |                           Dockerfile \ | ||||||
|                           pyproject.toml \ |                           Pipfile \ | ||||||
|                           uv.lock \ |                           Pipfile.lock \ | ||||||
|                           requirements.txt \ |                           requirements.txt \ | ||||||
|                           LICENSE \ |                           LICENSE \ | ||||||
|                           README.md \ |                           README.md \ | ||||||
|                           paperless.conf.example |                           paperless.conf.example \ | ||||||
|  |                           gunicorn.conf.py | ||||||
|           do |           do | ||||||
|             cp --verbose ${file_name} dist/paperless-ngx/ |             cp --verbose ${file_name} dist/paperless-ngx/ | ||||||
|           done |           done | ||||||
| @@ -551,21 +552,24 @@ jobs: | |||||||
|           cp --recursive docs/_build/html/ dist/paperless-ngx/docs |           cp --recursive docs/_build/html/ dist/paperless-ngx/docs | ||||||
|  |  | ||||||
|           mv --verbose static dist/paperless-ngx |           mv --verbose static dist/paperless-ngx | ||||||
|       - name: Make release package |       - | ||||||
|  |         name: Make release package | ||||||
|         run: | |         run: | | ||||||
|           echo "Creating release archive" |           echo "Creating release archive" | ||||||
|           cd dist |           cd dist | ||||||
|           sudo chown -R 1000:1000 paperless-ngx/ |           sudo chown -R 1000:1000 paperless-ngx/ | ||||||
|           tar -cJf paperless-ngx.tar.xz paperless-ngx/ |           tar -cJf paperless-ngx.tar.xz paperless-ngx/ | ||||||
|       - name: Upload release artifact |       - | ||||||
|  |         name: Upload release artifact | ||||||
|         uses: actions/upload-artifact@v4 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: release |           name: release | ||||||
|           path: dist/paperless-ngx.tar.xz |           path: dist/paperless-ngx.tar.xz | ||||||
|           retention-days: 7 |           retention-days: 7 | ||||||
|  |  | ||||||
|   publish-release: |   publish-release: | ||||||
|     name: "Publish Release" |     name: "Publish Release" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     outputs: |     outputs: | ||||||
|       prerelease: ${{ steps.get_version.outputs.prerelease }} |       prerelease: ${{ steps.get_version.outputs.prerelease }} | ||||||
|       changelog: ${{ steps.create-release.outputs.body }} |       changelog: ${{ steps.create-release.outputs.body }} | ||||||
| @@ -574,12 +578,14 @@ jobs: | |||||||
|       - build-release |       - build-release | ||||||
|     if: github.ref_type == 'tag' && (startsWith(github.ref_name, 'v') || contains(github.ref_name, '-beta.rc')) |     if: github.ref_type == 'tag' && (startsWith(github.ref_name, 'v') || contains(github.ref_name, '-beta.rc')) | ||||||
|     steps: |     steps: | ||||||
|       - name: Download release artifact |       - | ||||||
|         uses: actions/download-artifact@v5 |         name: Download release artifact | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: release |           name: release | ||||||
|           path: ./ |           path: ./ | ||||||
|       - name: Get version |       - | ||||||
|  |         name: Get version | ||||||
|         id: get_version |         id: get_version | ||||||
|         run: | |         run: | | ||||||
|           echo "version=${{ github.ref_name }}" >> $GITHUB_OUTPUT |           echo "version=${{ github.ref_name }}" >> $GITHUB_OUTPUT | ||||||
| @@ -588,7 +594,8 @@ jobs: | |||||||
|           else |           else | ||||||
|             echo "prerelease=false" >> $GITHUB_OUTPUT |             echo "prerelease=false" >> $GITHUB_OUTPUT | ||||||
|           fi |           fi | ||||||
|       - name: Create Release and Changelog |       - | ||||||
|  |         name: Create Release and Changelog | ||||||
|         id: create-release |         id: create-release | ||||||
|         uses: release-drafter/release-drafter@v6 |         uses: release-drafter/release-drafter@v6 | ||||||
|         with: |         with: | ||||||
| @@ -599,7 +606,8 @@ jobs: | |||||||
|           publish: true # ensures release is not marked as draft |           publish: true # ensures release is not marked as draft | ||||||
|         env: |         env: | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Upload release archive |       - | ||||||
|  |         name: Upload release archive | ||||||
|         id: upload-release-asset |         id: upload-release-asset | ||||||
|         uses: shogo82148/actions-upload-release-asset@v1 |         uses: shogo82148/actions-upload-release-asset@v1 | ||||||
|         with: |         with: | ||||||
| @@ -608,29 +616,32 @@ jobs: | |||||||
|           asset_path: ./paperless-ngx.tar.xz |           asset_path: ./paperless-ngx.tar.xz | ||||||
|           asset_name: paperless-ngx-${{ steps.get_version.outputs.version }}.tar.xz |           asset_name: paperless-ngx-${{ steps.get_version.outputs.version }}.tar.xz | ||||||
|           asset_content_type: application/x-xz |           asset_content_type: application/x-xz | ||||||
|  |  | ||||||
|   append-changelog: |   append-changelog: | ||||||
|     name: "Append Changelog" |     name: "Append Changelog" | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - publish-release |       - publish-release | ||||||
|     if: needs.publish-release.outputs.prerelease == 'false' |     if: needs.publish-release.outputs.prerelease == 'false' | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - | ||||||
|         uses: actions/checkout@v5 |         name: Checkout | ||||||
|  |         uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           ref: main |           ref: main | ||||||
|       - name: Set up Python |       - | ||||||
|         id: setup-python |         name: Set up Python | ||||||
|         uses: actions/setup-python@v6 |         uses: actions/setup-python@v5 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||||
|       - name: Install uv |           cache: "pipenv" | ||||||
|         uses: astral-sh/setup-uv@v6 |           cache-dependency-path: 'Pipfile.lock' | ||||||
|         with: |       - | ||||||
|           version: ${{ env.DEFAULT_UV_VERSION }} |         name: Install pipenv + tools | ||||||
|           enable-cache: true |         run: | | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} |           pip install --upgrade --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} setuptools wheel | ||||||
|       - name: Append Changelog to docs |       - | ||||||
|  |         name: Append Changelog to docs | ||||||
|         id: append-Changelog |         id: append-Changelog | ||||||
|         working-directory: docs |         working-directory: docs | ||||||
|         run: | |         run: | | ||||||
| @@ -644,16 +655,14 @@ jobs: | |||||||
|           CURRENT_CHANGELOG=`tail --lines +2 changelog.md` |           CURRENT_CHANGELOG=`tail --lines +2 changelog.md` | ||||||
|           echo -e "$CURRENT_CHANGELOG" >> changelog-new.md |           echo -e "$CURRENT_CHANGELOG" >> changelog-new.md | ||||||
|           mv changelog-new.md changelog.md |           mv changelog-new.md changelog.md | ||||||
|           uv run \ |           pipenv run pre-commit run --files changelog.md || true | ||||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ |  | ||||||
|             --dev \ |  | ||||||
|             pre-commit run --files changelog.md || true |  | ||||||
|           git config --global user.name "github-actions" |           git config --global user.name "github-actions" | ||||||
|           git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com" |           git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com" | ||||||
|           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 |       - | ||||||
|         uses: actions/github-script@v8 |         name: Create Pull Request | ||||||
|  |         uses: actions/github-script@v7 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const { repo, owner } = context.repo; |             const { repo, owner } = context.repo; | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,19 +4,24 @@ | |||||||
| # Requires a PAT with the correct scope set in the secrets. | # Requires a PAT with the correct scope set in the secrets. | ||||||
| # | # | ||||||
| # This workflow will not trigger runs on forked repos. | # This workflow will not trigger runs on forked repos. | ||||||
|  |  | ||||||
| name: Cleanup Image Tags | name: Cleanup Image Tags | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   workflow_dispatch: |   delete: | ||||||
|   schedule: |   push: | ||||||
|     - cron: '0 0 * * 0' |     paths: | ||||||
|  |       - ".github/workflows/cleanup-tags.yml" | ||||||
|  |  | ||||||
| concurrency: | concurrency: | ||||||
|   group: registry-tags-cleanup |   group: registry-tags-cleanup | ||||||
|   cancel-in-progress: false |   cancel-in-progress: false | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   cleanup-images: |   cleanup-images: | ||||||
|     name: Cleanup Image Tags for ${{ matrix.primary-name }} |     name: Cleanup Image Tags for ${{ matrix.primary-name }} | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
| @@ -25,9 +30,10 @@ jobs: | |||||||
|       # Requires a personal access token with the OAuth scope delete:packages |       # Requires a personal access token with the OAuth scope delete:packages | ||||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} |       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Clean temporary images |       - | ||||||
|  |         name: Clean temporary images | ||||||
|         if: "${{ env.TOKEN != '' }}" |         if: "${{ env.TOKEN != '' }}" | ||||||
|         uses: stumpylog/image-cleaner-action/ephemeral@v0.11.0 |         uses: stumpylog/image-cleaner-action/ephemeral@v0.8.0 | ||||||
|         with: |         with: | ||||||
|           token: "${{ env.TOKEN }}" |           token: "${{ env.TOKEN }}" | ||||||
|           owner: "${{ github.repository_owner }}" |           owner: "${{ github.repository_owner }}" | ||||||
| @@ -37,10 +43,11 @@ jobs: | |||||||
|           repo_name: "paperless-ngx" |           repo_name: "paperless-ngx" | ||||||
|           match_regex: "(feature|fix)" |           match_regex: "(feature|fix)" | ||||||
|           do_delete: "true" |           do_delete: "true" | ||||||
|  |  | ||||||
|   cleanup-untagged-images: |   cleanup-untagged-images: | ||||||
|     name: Cleanup Untagged Images Tags for ${{ matrix.primary-name }} |     name: Cleanup Untagged Images Tags for ${{ matrix.primary-name }} | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     needs: |     needs: | ||||||
|       - cleanup-images |       - cleanup-images | ||||||
|     strategy: |     strategy: | ||||||
| @@ -51,9 +58,10 @@ jobs: | |||||||
|       # Requires a personal access token with the OAuth scope delete:packages |       # Requires a personal access token with the OAuth scope delete:packages | ||||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} |       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Clean untagged images |       - | ||||||
|  |         name: Clean untagged images | ||||||
|         if: "${{ env.TOKEN != '' }}" |         if: "${{ env.TOKEN != '' }}" | ||||||
|         uses: stumpylog/image-cleaner-action/untagged@v0.11.0 |         uses: stumpylog/image-cleaner-action/untagged@v0.8.0 | ||||||
|         with: |         with: | ||||||
|           token: "${{ env.TOKEN }}" |           token: "${{ env.TOKEN }}" | ||||||
|           owner: "${{ github.repository_owner }}" |           owner: "${{ github.repository_owner }}" | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,31 +10,36 @@ | |||||||
| # supported CodeQL languages. | # supported CodeQL languages. | ||||||
| # | # | ||||||
| name: "CodeQL" | name: "CodeQL" | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: [main, dev] |     branches: [ main, dev ] | ||||||
|   pull_request: |   pull_request: | ||||||
|     # The branches below must be a subset of the branches above |     # The branches below must be a subset of the branches above | ||||||
|     branches: [dev] |     branches: [ dev ] | ||||||
|   schedule: |   schedule: | ||||||
|     - cron: '28 13 * * 5' |     - cron: '28 13 * * 5' | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   analyze: |   analyze: | ||||||
|     name: Analyze |     name: Analyze | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     permissions: |     permissions: | ||||||
|       actions: read |       actions: read | ||||||
|       contents: read |       contents: read | ||||||
|       security-events: write |       security-events: write | ||||||
|  |  | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         language: ['javascript', 'python'] |         language: [ 'javascript', 'python' ] | ||||||
|         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] |         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] | ||||||
|         # Learn more about CodeQL language support at https://git.io/codeql-language-support |         # Learn more about CodeQL language support at https://git.io/codeql-language-support | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout repository |     - name: Checkout repository | ||||||
|         uses: actions/checkout@v5 |       uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|     # Initializes the CodeQL tools for scanning. |     # Initializes the CodeQL tools for scanning. | ||||||
|     - name: Initialize CodeQL |     - name: Initialize CodeQL | ||||||
|       uses: github/codeql-action/init@v3 |       uses: github/codeql-action/init@v3 | ||||||
| @@ -44,5 +49,6 @@ jobs: | |||||||
|         # By default, queries listed here will override any specified in a config file. |         # By default, queries listed here will override any specified in a config file. | ||||||
|         # Prefix the list here with "+" to use these queries and those in the config file. |         # Prefix the list here with "+" to use these queries and those in the config file. | ||||||
|         # queries: ./path/to/local/query, your-org/your-repo/queries@main |         # queries: ./path/to/local/query, your-org/your-repo/queries@main | ||||||
|  |  | ||||||
|     - name: Perform CodeQL Analysis |     - name: Perform CodeQL Analysis | ||||||
|       uses: github/codeql-action/analyze@v3 |       uses: github/codeql-action/analyze@v3 | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,21 +1,26 @@ | |||||||
| name: Crowdin Action | name: Crowdin Action | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|   schedule: |   schedule: | ||||||
|     - cron: '2 */12 * * *' |     - cron: '2 */12 * * *' | ||||||
|   push: |   push: | ||||||
|     paths: ['src/locale/**', 'src-ui/messages.xlf', 'src-ui/src/locale/**'] |     paths: [ | ||||||
|     branches: [dev] |       'src/locale/**', | ||||||
|  |       'src-ui/messages.xlf', | ||||||
|  |       'src-ui/src/locale/**' | ||||||
|  |     ] | ||||||
|  |     branches: [ dev ] | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   synchronize-with-crowdin: |   synchronize-with-crowdin: | ||||||
|     name: Crowdin Sync |     name: Crowdin Sync | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout |     - name: Checkout | ||||||
|         uses: actions/checkout@v5 |       uses: actions/checkout@v4 | ||||||
|         with: |  | ||||||
|           token: ${{ secrets.PNGX_BOT_PAT }} |  | ||||||
|     - name: crowdin action |     - name: crowdin action | ||||||
|       uses: crowdin/github-action@v2 |       uses: crowdin/github-action@v2 | ||||||
|       with: |       with: | ||||||
|   | |||||||
							
								
								
									
										112
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										112
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,112 +0,0 @@ | |||||||
| name: PR Bot |  | ||||||
| on: |  | ||||||
|   pull_request_target: |  | ||||||
|     types: [opened] |  | ||||||
| permissions: |  | ||||||
|   contents: read |  | ||||||
|   pull-requests: write |  | ||||||
| jobs: |  | ||||||
|   pr-bot: |  | ||||||
|     name: Automated PR Bot |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Label PR by file path or branch name |  | ||||||
|         # see .github/labeler.yml for the labeler config |  | ||||||
|         uses: actions/labeler@v6 |  | ||||||
|         with: |  | ||||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|       - name: Label by size |  | ||||||
|         uses: Gascon1/pr-size-labeler@v1.3.0 |  | ||||||
|         with: |  | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |  | ||||||
|           xs_label: 'small-change' |  | ||||||
|           xs_diff: '9' |  | ||||||
|           s_label: 'non-trivial' |  | ||||||
|           s_diff: '99999' |  | ||||||
|           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$ |  | ||||||
|       - name: Label by PR title |  | ||||||
|         uses: actions/github-script@v8 |  | ||||||
|         with: |  | ||||||
|           script: | |  | ||||||
|             const pr = context.payload.pull_request; |  | ||||||
|             const title = pr.title.toLowerCase(); |  | ||||||
|             const labels = []; |  | ||||||
|  |  | ||||||
|             if (/^(fix|bugfix)/i.test(title)) { |  | ||||||
|               labels.push('bug'); |  | ||||||
|             } else if (/^feature/i.test(title)) { |  | ||||||
|               labels.push('enhancement'); |  | ||||||
|             } else if (!/^(dependabot)/i.test(title) && !/^(chore)/i.test(title)) { |  | ||||||
|               labels.push('enhancement'); // Default fallback |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (labels.length) { |  | ||||||
|               await github.rest.issues.addLabels({ |  | ||||||
|                 owner: context.repo.owner, |  | ||||||
|                 repo: context.repo.repo, |  | ||||||
|                 issue_number: pr.number, |  | ||||||
|                 labels, |  | ||||||
|               }); |  | ||||||
|               core.info(`Added labels based on title: ${labels.join(', ')}`); |  | ||||||
|             } |  | ||||||
|       - name: Label bot-generated PRs |  | ||||||
|         if: ${{ contains(github.actor, 'dependabot') || contains(github.actor, 'crowdin-bot') }} |  | ||||||
|         uses: actions/github-script@v8 |  | ||||||
|         with: |  | ||||||
|           script: | |  | ||||||
|             const pr = context.payload.pull_request; |  | ||||||
|             const user = pr.user.login.toLowerCase(); |  | ||||||
|             const labels = []; |  | ||||||
|  |  | ||||||
|             if (user.includes('dependabot')) { |  | ||||||
|               labels.push('dependencies'); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (user.includes('crowdin-bot')) { |  | ||||||
|               labels.push('translation', 'skip-changelog'); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (labels.length) { |  | ||||||
|               await github.rest.issues.addLabels({ |  | ||||||
|                 owner: context.repo.owner, |  | ||||||
|                 repo: context.repo.repo, |  | ||||||
|                 issue_number: pr.number, |  | ||||||
|                 labels, |  | ||||||
|               }); |  | ||||||
|             } |  | ||||||
|       - name: Welcome comment |  | ||||||
|         if: ${{ !contains(github.actor, 'bot') }} |  | ||||||
|         uses: actions/github-script@v8 |  | ||||||
|         with: |  | ||||||
|           script: | |  | ||||||
|             const pr = context.payload.pull_request; |  | ||||||
|             const user = pr.user.login; |  | ||||||
|  |  | ||||||
|             const { data: members } = await github.rest.orgs.listMembers({ |  | ||||||
|               org: 'paperless-ngx', |  | ||||||
|             }); |  | ||||||
|  |  | ||||||
|             const memberLogins = members.map(m => m.login.toLowerCase()); |  | ||||||
|             if (memberLogins.includes(user.toLowerCase())) { |  | ||||||
|               core.info('Skipping comment: user is org member'); |  | ||||||
|               return; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             const body = |  | ||||||
|                 "Hello @" + user + ",\n\n" + |  | ||||||
|                 "Thank you very much for submitting this PR to us!\n\n" + |  | ||||||
|                 "This is what will happen next:\n\n" + |  | ||||||
|                 "1. CI tests will run against your PR to ensure quality and consistency.\n" + |  | ||||||
|                 "2. Next, human contributors from paperless-ngx review your changes.\n" + |  | ||||||
|                 "3. Please address any issues that come up during the review as soon as you are able to.\n" + |  | ||||||
|                 "4. If accepted, your pull request will be merged into the `dev` branch and changes there will be tested further.\n" + |  | ||||||
|                 "5. Eventually, changes from you and other contributors will be merged into `main` and a new release will be made.\n\n" + |  | ||||||
|                 "You'll be hearing from us soon, and thank you again for contributing to our project."; |  | ||||||
|  |  | ||||||
|             await github.rest.issues.createComment({ |  | ||||||
|               issue_number: pr.number, |  | ||||||
|               owner: context.repo.owner, |  | ||||||
|               repo: context.repo.repo, |  | ||||||
|               body, |  | ||||||
|             }); |  | ||||||
							
								
								
									
										5
									
								
								.github/workflows/project-actions.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/project-actions.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| name: Project Automations | name: Project Automations | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   pull_request_target: #_target allows access to secrets |   pull_request_target: #_target allows access to secrets | ||||||
|     types: |     types: | ||||||
| @@ -7,12 +8,14 @@ on: | |||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - dev |       - dev | ||||||
|  |  | ||||||
| permissions: | permissions: | ||||||
|   contents: read |   contents: read | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   pr_opened_or_reopened: |   pr_opened_or_reopened: | ||||||
|     name: pr_opened_or_reopened |     name: pr_opened_or_reopened | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-22.04 | ||||||
|     permissions: |     permissions: | ||||||
|       # write permission is required for autolabeler |       # write permission is required for autolabeler | ||||||
|       pull-requests: write |       pull-requests: write | ||||||
|   | |||||||
							
								
								
									
										57
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										57
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,41 +1,39 @@ | |||||||
| name: 'Repository Maintenance' | name: 'Repository Maintenance' | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   schedule: |   schedule: | ||||||
|     - cron: '0 3 * * *' |     - cron: '0 3 * * *' | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| permissions: | permissions: | ||||||
|   issues: write |   issues: write | ||||||
|   pull-requests: write |   pull-requests: write | ||||||
|   discussions: write |   discussions: write | ||||||
|  |  | ||||||
| concurrency: | concurrency: | ||||||
|   group: lock |   group: lock | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   stale: |   stale: | ||||||
|     name: 'Stale' |     name: 'Stale' | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/stale@v10 |       - uses: actions/stale@v9 | ||||||
|         with: |         with: | ||||||
|           days-before-stale: 7 |           days-before-stale: 7 | ||||||
|           days-before-close: 14 |           days-before-close: 14 | ||||||
|           any-of-issue-labels: 'cant-reproduce,not a bug' |           any-of-labels: 'stale,cant-reproduce,not a bug' | ||||||
|           stale-issue-label: stale |           stale-issue-label: stale | ||||||
|           stale-issue-message: > |  | ||||||
|             This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. |  | ||||||
|  |  | ||||||
|           days-before-pr-stale: 14 |  | ||||||
|           days-before-pr-close: 7 |  | ||||||
|           stale-pr-message: "" |  | ||||||
|           stale-pr-label: stale |           stale-pr-label: stale | ||||||
|           exempt-pr-labels: 'notable' |           stale-issue-message: > | ||||||
|           close-pr-message: > |             This issue has been automatically marked as stale because it has not had | ||||||
|             This pull request has been automatically closed because it has not had recent activity. Thank you for your contributions. Please open a new pull request or discussion if you would like to continue working on this change. |             recent activity. It will be closed if no further activity occurs. Thank you | ||||||
|  |             for your contributions. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||||
|   lock-threads: |   lock-threads: | ||||||
|     name: 'Lock Old Threads' |     name: 'Lock Old Threads' | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: dessant/lock-threads@v5 |       - uses: dessant/lock-threads@v5 | ||||||
|         with: |         with: | ||||||
| @@ -44,20 +42,26 @@ jobs: | |||||||
|           discussion-inactive-days: '30' |           discussion-inactive-days: '30' | ||||||
|           log-output: true |           log-output: true | ||||||
|           issue-comment: > |           issue-comment: > | ||||||
|             This issue has been automatically locked since there has not been any recent activity after it was closed. Please open a new discussion or issue for related concerns. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. |             This issue has been automatically locked since there | ||||||
|  |             has not been any recent activity after it was closed. | ||||||
|  |             Please open a new discussion or issue for related concerns. | ||||||
|  |             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||||
|           pr-comment: > |           pr-comment: > | ||||||
|             This pull request has been automatically locked since there has not been any recent activity after it was closed. Please open a new discussion or issue for related concerns. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. |             This pull request has been automatically locked since there | ||||||
|  |             has not been any recent activity after it was closed. | ||||||
|  |             Please open a new discussion or issue for related concerns. | ||||||
|  |             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||||
|           discussion-comment: > |           discussion-comment: > | ||||||
|             This discussion has been automatically locked since there has not been any recent activity after it was closed. Please open a new discussion for related concerns. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. |             This discussion has been automatically locked since there | ||||||
|  |             has not been any recent activity after it was closed. | ||||||
|  |             Please open a new discussion for related concerns. | ||||||
|  |             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||||
|   close-answered-discussions: |   close-answered-discussions: | ||||||
|     name: 'Close Answered Discussions' |     name: 'Close Answered Discussions' | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v8 |       - uses: actions/github-script@v7 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
| @@ -112,9 +116,9 @@ jobs: | |||||||
|   close-outdated-discussions: |   close-outdated-discussions: | ||||||
|     name: 'Close Outdated Discussions' |     name: 'Close Outdated Discussions' | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v8 |       - uses: actions/github-script@v7 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
| @@ -204,9 +208,9 @@ jobs: | |||||||
|   close-unsupported-feature-requests: |   close-unsupported-feature-requests: | ||||||
|     name: 'Close Unsupported Feature Requests' |     name: 'Close Unsupported Feature Requests' | ||||||
|     if: github.repository_owner == 'paperless-ngx' |     if: github.repository_owner == 'paperless-ngx' | ||||||
|     runs-on: ubuntu-24.04 |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/github-script@v8 |       - uses: actions/github-script@v7 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             function sleep(ms) { |             function sleep(ms) { | ||||||
| @@ -241,7 +245,6 @@ jobs: | |||||||
|                 ) { |                 ) { | ||||||
|                   nodes { |                   nodes { | ||||||
|                     id, |                     id, | ||||||
|                     createdAt, |  | ||||||
|                     number, |                     number, | ||||||
|                     updatedAt, |                     updatedAt, | ||||||
|                     upvoteCount, |                     upvoteCount, | ||||||
|   | |||||||
							
								
								
									
										69
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										69
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,69 +0,0 @@ | |||||||
| name: Generate Translation Strings |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - dev |  | ||||||
| jobs: |  | ||||||
|   generate-translate-strings: |  | ||||||
|     name: Generate Translation Strings |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     permissions: |  | ||||||
|       contents: write |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout code |  | ||||||
|         uses: actions/checkout@v5 |  | ||||||
|         with: |  | ||||||
|           token: ${{ secrets.PNGX_BOT_PAT }} |  | ||||||
|           ref: ${{ github.head_ref }} |  | ||||||
|       - name: Set up Python |  | ||||||
|         id: setup-python |  | ||||||
|         uses: actions/setup-python@v6 |  | ||||||
|       - name: Install system dependencies |  | ||||||
|         run: | |  | ||||||
|           sudo apt-get update -qq |  | ||||||
|           sudo apt-get install -qq --no-install-recommends gettext |  | ||||||
|       - name: Install uv |  | ||||||
|         uses: astral-sh/setup-uv@v6 |  | ||||||
|         with: |  | ||||||
|           enable-cache: true |  | ||||||
|       - name: Install backend python dependencies |  | ||||||
|         run: | |  | ||||||
|           uv sync \ |  | ||||||
|             --group dev \ |  | ||||||
|             --frozen |  | ||||||
|       - name: Generate backend translation strings |  | ||||||
|         run: cd src/ && uv run manage.py makemessages -l en_US -i "samples*" |  | ||||||
|       - name: Install pnpm |  | ||||||
|         uses: pnpm/action-setup@v4 |  | ||||||
|         with: |  | ||||||
|           version: 10 |  | ||||||
|       - name: Use Node.js 20 |  | ||||||
|         uses: actions/setup-node@v5 |  | ||||||
|         with: |  | ||||||
|           node-version: 20.x |  | ||||||
|           cache: 'pnpm' |  | ||||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' |  | ||||||
|       - name: Cache frontend dependencies |  | ||||||
|         id: cache-frontend-deps |  | ||||||
|         uses: actions/cache@v4 |  | ||||||
|         with: |  | ||||||
|           path: | |  | ||||||
|             ~/.pnpm-store |  | ||||||
|             ~/.cache |  | ||||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} |  | ||||||
|       - name: Install frontend dependencies |  | ||||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' |  | ||||||
|         run: cd src-ui && pnpm install |  | ||||||
|       - name: Re-link Angular cli |  | ||||||
|         run: cd src-ui && pnpm link @angular/cli |  | ||||||
|       - name: Generate frontend translation strings |  | ||||||
|         run: | |  | ||||||
|           cd src-ui |  | ||||||
|           pnpm run ng extract-i18n |  | ||||||
|       - name: Commit changes |  | ||||||
|         uses: stefanzweifel/git-auto-commit-action@v6 |  | ||||||
|         with: |  | ||||||
|           file_pattern: 'src-ui/messages.xlf src/locale/en_US/LC_MESSAGES/django.po' |  | ||||||
|           commit_message: "Auto translate strings" |  | ||||||
|           commit_user_name: "GitHub Actions" |  | ||||||
|           commit_author: "GitHub Actions <41898282+github-actions[bot]@users.noreply.github.com>" |  | ||||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -44,7 +44,6 @@ nosetests.xml | |||||||
| coverage.xml | coverage.xml | ||||||
| *,cover | *,cover | ||||||
| .pytest_cache | .pytest_cache | ||||||
| junit.xml |  | ||||||
|  |  | ||||||
| # Translations | # Translations | ||||||
| *.mo | *.mo | ||||||
| @@ -101,12 +100,3 @@ scripts/nuke | |||||||
|  |  | ||||||
| # celery schedule file | # celery schedule file | ||||||
| celerybeat-schedule* | celerybeat-schedule* | ||||||
|  |  | ||||||
| # ignore .devcontainer sub folders |  | ||||||
| /.devcontainer/consume/ |  | ||||||
| /.devcontainer/data/ |  | ||||||
| /.devcontainer/media/ |  | ||||||
| /.devcontainer/redisdata/ |  | ||||||
|  |  | ||||||
| # ignore pnpm package store folder created when setting up the devcontainer |  | ||||||
| .pnpm-store/ |  | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| # This file configures pre-commit hooks. | # This file configures pre-commit hooks. | ||||||
| # See https://pre-commit.com/ for general information | # See https://pre-commit.com/ for general information | ||||||
| # See https://pre-commit.com/hooks.html for a listing of possible hooks | # See https://pre-commit.com/hooks.html for a listing of possible hooks | ||||||
|  |  | ||||||
| repos: | repos: | ||||||
|   # General hooks |   # General hooks | ||||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks |   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||||
|     rev: v6.0.0 |     rev: v4.6.0 | ||||||
|     hooks: |     hooks: | ||||||
|       - id: check-docstring-first |       - id: check-docstring-first | ||||||
|       - id: check-json |       - id: check-json | ||||||
| @@ -18,7 +19,7 @@ repos: | |||||||
|         exclude_types: |         exclude_types: | ||||||
|           - svg |           - svg | ||||||
|           - pofile |           - pofile | ||||||
|         exclude: "(^LICENSE$|^src/documents/static/bootstrap.min.css$)" |         exclude: "(^LICENSE$)" | ||||||
|       - id: mixed-line-ending |       - id: mixed-line-ending | ||||||
|         args: |         args: | ||||||
|           - "--fix=lf" |           - "--fix=lf" | ||||||
| @@ -28,38 +29,32 @@ repos: | |||||||
|       - id: check-case-conflict |       - id: check-case-conflict | ||||||
|       - id: detect-private-key |       - id: detect-private-key | ||||||
|   - repo: https://github.com/codespell-project/codespell |   - repo: https://github.com/codespell-project/codespell | ||||||
|     rev: v2.4.1 |     rev: v2.3.0 | ||||||
|     hooks: |     hooks: | ||||||
|       - id: codespell |       - id: codespell | ||||||
|         additional_dependencies: [tomli] |         exclude: "(^src-ui/src/locale/)|(^src-ui/e2e/)|(^src/paperless_mail/tests/samples/)" | ||||||
|         exclude_types: |         exclude_types: | ||||||
|           - pofile |           - pofile | ||||||
|           - json |           - json | ||||||
|   # See https://github.com/prettier/prettier/issues/15742 for the fork reason |   # See https://github.com/prettier/prettier/issues/15742 for the fork reason | ||||||
|   - repo: https://github.com/rbubley/mirrors-prettier |   - repo: https://github.com/rbubley/mirrors-prettier | ||||||
|     rev: 'v3.6.2' |     rev: 'v3.3.3' | ||||||
|     hooks: |     hooks: | ||||||
|       - id: prettier |       - id: prettier | ||||||
|         types_or: |         types_or: | ||||||
|           - javascript |           - javascript | ||||||
|           - ts |           - ts | ||||||
|           - markdown |           - markdown | ||||||
|         additional_dependencies: |         exclude: "(^Pipfile\\.lock$)" | ||||||
|           - prettier@3.3.3 |  | ||||||
|           - '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.14.0 |     rev: 'v0.6.8' | ||||||
|     hooks: |     hooks: | ||||||
|       - id: ruff-check |       - id: ruff | ||||||
|       - id: ruff-format |       - id: ruff-format | ||||||
|   - repo: https://github.com/tox-dev/pyproject-fmt |  | ||||||
|     rev: "v2.11.0" |  | ||||||
|     hooks: |  | ||||||
|       - id: pyproject-fmt |  | ||||||
|   # Dockerfile hooks |   # Dockerfile hooks | ||||||
|   - repo: https://github.com/AleksaC/hadolint-py |   - repo: https://github.com/AleksaC/hadolint-py | ||||||
|     rev: v2.14.0 |     rev: v2.12.0.3 | ||||||
|     hooks: |     hooks: | ||||||
|       - id: hadolint |       - id: hadolint | ||||||
|   # Shell script hooks |   # Shell script hooks | ||||||
| @@ -72,13 +67,6 @@ repos: | |||||||
|         args: |         args: | ||||||
|           - "--tab" |           - "--tab" | ||||||
|   - repo: https://github.com/shellcheck-py/shellcheck-py |   - repo: https://github.com/shellcheck-py/shellcheck-py | ||||||
|     rev: "v0.11.0.1" |     rev: "v0.10.0.1" | ||||||
|     hooks: |     hooks: | ||||||
|       - id: shellcheck |       - id: shellcheck | ||||||
|   - repo: https://github.com/google/yamlfmt |  | ||||||
|     rev: v0.18.0 |  | ||||||
|     hooks: |  | ||||||
|       - id: yamlfmt |  | ||||||
|         exclude: "^src-ui/pnpm-lock.yaml" |  | ||||||
|         types: |  | ||||||
|           - yaml |  | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.prettierrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.prettierrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | { | ||||||
|  |     # https://prettier.io/docs/en/options.html#semicolons | ||||||
|  |     "semi": false, | ||||||
|  |     # https://prettier.io/docs/en/options.html#quotes | ||||||
|  |     "singleQuote": true, | ||||||
|  |     # https://prettier.io/docs/en/options.html#trailing-commas | ||||||
|  |     "trailingComma": "es5", | ||||||
|  |     "overrides": [ | ||||||
|  |         { | ||||||
|  |             "files": ["docs/*.md"], | ||||||
|  |             "options": { | ||||||
|  |                 "tabWidth": 4, | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     ] | ||||||
|  | } | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| const config = { |  | ||||||
| 	// https://prettier.io/docs/en/options.html#semicolons |  | ||||||
| 	semi: false, |  | ||||||
| 	// https://prettier.io/docs/en/options.html#quotes |  | ||||||
| 	singleQuote: true, |  | ||||||
| 	// https://prettier.io/docs/en/options.html#trailing-commas |  | ||||||
| 	trailingComma: 'es5', |  | ||||||
| 	overrides: [ |  | ||||||
| 		{ |  | ||||||
| 			files: ['docs/*.md'], |  | ||||||
| 			options: { |  | ||||||
| 				tabWidth: 4, |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 	], |  | ||||||
| 	plugins: [require('prettier-plugin-organize-imports')], |  | ||||||
| } |  | ||||||
|  |  | ||||||
| module.exports = config |  | ||||||
							
								
								
									
										1
									
								
								.python-version
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.python-version
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | 3.10.15 | ||||||
							
								
								
									
										47
									
								
								.ruff.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								.ruff.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | |||||||
|  | fix = true | ||||||
|  | line-length = 88 | ||||||
|  | respect-gitignore = true | ||||||
|  | src = ["src"] | ||||||
|  | target-version = "py310" | ||||||
|  | output-format = "grouped" | ||||||
|  | show-fixes = true | ||||||
|  |  | ||||||
|  | # https://docs.astral.sh/ruff/settings/ | ||||||
|  | # https://docs.astral.sh/ruff/rules/ | ||||||
|  | [lint] | ||||||
|  | extend-select = [ | ||||||
|  |   "W",     # https://docs.astral.sh/ruff/rules/#pycodestyle-e-w | ||||||
|  |   "I",     # https://docs.astral.sh/ruff/rules/#isort-i | ||||||
|  |   "UP",    # https://docs.astral.sh/ruff/rules/#pyupgrade-up | ||||||
|  |   "COM",   # https://docs.astral.sh/ruff/rules/#flake8-commas-com | ||||||
|  |   "DJ",    # https://docs.astral.sh/ruff/rules/#flake8-django-dj | ||||||
|  |   "EXE",   # https://docs.astral.sh/ruff/rules/#flake8-executable-exe | ||||||
|  |   "ISC",   # https://docs.astral.sh/ruff/rules/#flake8-implicit-str-concat-isc | ||||||
|  |   "ICN",   # https://docs.astral.sh/ruff/rules/#flake8-import-conventions-icn | ||||||
|  |   "G201",  # https://docs.astral.sh/ruff/rules/#flake8-logging-format-g | ||||||
|  |   "INP",   # https://docs.astral.sh/ruff/rules/#flake8-no-pep420-inp | ||||||
|  |   "PIE",   # https://docs.astral.sh/ruff/rules/#flake8-pie-pie | ||||||
|  |   "Q",     # https://docs.astral.sh/ruff/rules/#flake8-quotes-q | ||||||
|  |   "RSE",   # https://docs.astral.sh/ruff/rules/#flake8-raise-rse | ||||||
|  |   "T20",   # https://docs.astral.sh/ruff/rules/#flake8-print-t20 | ||||||
|  |   "SIM",   # https://docs.astral.sh/ruff/rules/#flake8-simplify-sim | ||||||
|  |   "TID",   # https://docs.astral.sh/ruff/rules/#flake8-tidy-imports-tid | ||||||
|  |   "TCH",   # https://docs.astral.sh/ruff/rules/#flake8-type-checking-tch | ||||||
|  |   "PLC",   # https://docs.astral.sh/ruff/rules/#pylint-pl | ||||||
|  |   "PLE",   # https://docs.astral.sh/ruff/rules/#pylint-pl | ||||||
|  |   "RUF",   # https://docs.astral.sh/ruff/rules/#ruff-specific-rules-ruf | ||||||
|  |   "FLY",   # https://docs.astral.sh/ruff/rules/#flynt-fly | ||||||
|  | ] | ||||||
|  | # TODO PTH https://docs.astral.sh/ruff/rules/#flake8-use-pathlib-pth | ||||||
|  | ignore = ["DJ001", "SIM105", "RUF012"] | ||||||
|  |  | ||||||
|  | [lint.per-file-ignores] | ||||||
|  | ".github/scripts/*.py" = ["E501", "INP001", "SIM117"] | ||||||
|  | "docker/wait-for-redis.py" = ["INP001", "T201"] | ||||||
|  | "*/tests/*.py" = ["E501", "SIM117"] | ||||||
|  | "*/migrations/*.py" = ["E501", "SIM", "T201"] | ||||||
|  | "src/paperless_tesseract/tests/test_parser.py" = ["RUF001"] | ||||||
|  | "src/documents/models.py" = ["SIM115"] | ||||||
|  |  | ||||||
|  | [lint.isort] | ||||||
|  | force-single-line = true | ||||||
| @@ -5,6 +5,5 @@ | |||||||
| /src-ui/ @paperless-ngx/frontend | /src-ui/ @paperless-ngx/frontend | ||||||
|  |  | ||||||
| /src/ @paperless-ngx/backend | /src/ @paperless-ngx/backend | ||||||
| pyproject.toml @paperless-ngx/backend | Pipfile* @paperless-ngx/backend | ||||||
| uv.lock @paperless-ngx/backend |  | ||||||
| *.py @paperless-ngx/backend | *.py @paperless-ngx/backend | ||||||
|   | |||||||
| @@ -2,11 +2,9 @@ | |||||||
|  |  | ||||||
| If you feel like contributing to the project, please do! Bug fixes and improvements are always welcome. | If you feel like contributing to the project, please do! Bug fixes and improvements are always welcome. | ||||||
|  |  | ||||||
| ⚠️ Please note: Pull requests that implement a new feature or enhancement _should almost always target an existing feature request_ with evidence of community interest and discussion. This is in order to balance the work of implementing and maintaining new features / enhancements. Pull requests that are opened without meeting this requirement may not be merged. |  | ||||||
|  |  | ||||||
| If you want to implement something big: | If you want to implement something big: | ||||||
|  |  | ||||||
| - As above, please start with a discussion! Maybe something similar is already in development and we can make it happen together. | - Please start a discussion about that in the issues! Maybe something similar is already in development and we can make it happen together. | ||||||
| - When making additions to the project, consider if the majority of users will benefit from your change. If not, you're probably better of forking the project. | - When making additions to the project, consider if the majority of users will benefit from your change. If not, you're probably better of forking the project. | ||||||
| - Also consider if your change will get in the way of other users. A good change is a change that enhances the experience of some users who want that change and does not affect users who do not care about the change. | - Also consider if your change will get in the way of other users. A good change is a change that enhances the experience of some users who want that change and does not affect users who do not care about the change. | ||||||
| - Please see the [paperless-ngx merge process](#merging-prs) below. | - Please see the [paperless-ngx merge process](#merging-prs) below. | ||||||
| @@ -39,8 +37,6 @@ Before you can run `pytest`, ensure to [properly set up your local environment]( | |||||||
|  |  | ||||||
| Once you have submitted a **P**ull **R**equest it will be reviewed, approved, and merged by one or more community members of any team. Automated code tests and formatting checks must be passed. | Once you have submitted a **P**ull **R**equest it will be reviewed, approved, and merged by one or more community members of any team. Automated code tests and formatting checks must be passed. | ||||||
|  |  | ||||||
| Important: Pull requests that implement a new feature or enhancement _should almost always target an existing feature request_ with evidence of community interest and discussion. This is in order to balance the work of implementing and maintaining new features / enhancements. Instead of opening a PR which does not meet this requirement, please open a feature request instead, to gather feedback from both users and the project maintainers. |  | ||||||
|  |  | ||||||
| ## Non-Trivial Requests | ## Non-Trivial Requests | ||||||
|  |  | ||||||
| PRs deemed `non-trivial` will go through a stricter review process before being merged into `dev`. This is to ensure code quality and complete functionality (free of side effects). | PRs deemed `non-trivial` will go through a stricter review process before being merged into `dev`. This is to ensure code quality and complete functionality (free of side effects). | ||||||
| @@ -85,7 +81,7 @@ Some notes about translation: | |||||||
|  |  | ||||||
| If a language has already been added, and you would like to contribute new translations or change existing translations, please read the "Translation" section in the README.md file for further details on that. | If a language has already been added, and you would like to contribute new translations or change existing translations, please read the "Translation" section in the README.md file for further details on that. | ||||||
|  |  | ||||||
| If you would like the project to be translated to another language, first head over to https://crowdin.com/project/paperless-ngx to check if that language has already been enabled for translation. | If you would like the project to be translated to another language, first head over to https://crwd.in/paperless-ngx to check if that language has already been enabled for translation. | ||||||
| If not, please request the language to be added by creating an issue on GitHub. The issue should contain: | If not, please request the language to be added by creating an issue on GitHub. The issue should contain: | ||||||
|  |  | ||||||
| - English name of the language (the localized name can be added on Crowdin). | - English name of the language (the localized name can be added on Crowdin). | ||||||
| @@ -113,12 +109,28 @@ Paperless-ngx is a community project. We do our best to delegate permission and | |||||||
|  |  | ||||||
| ## Structure | ## Structure | ||||||
|  |  | ||||||
| There are currently 2 members in paperless-ngx with complete administrative privileges to the repo: | As of writing, there are 21 members in paperless-ngx. 4 of these people have complete administrative privileges to the repo: | ||||||
|  |  | ||||||
| - [@shamoon](https://github.com/shamoon) | - [@shamoon](https://github.com/shamoon) | ||||||
| - [@stumpylog](https://github.com/stumpylog) | - [@bauerj](https://github.com/bauerj) | ||||||
|  | - [@qcasey](https://github.com/qcasey) | ||||||
|  | - [@FrankStrieter](https://github.com/FrankStrieter) | ||||||
|  |  | ||||||
| There are other members who occasionally contribute but we are actively seeking more dedicated maintainers of the project. Please reach out if you are interested. | There are 5 teams collaborating on specific tasks within paperless-ngx: | ||||||
|  |  | ||||||
|  | - @paperless-ngx/backend (Python / django) | ||||||
|  | - @paperless-ngx/frontend (JavaScript / Typescript) | ||||||
|  | - @paperless-ngx/ci-cd (GitHub Actions / Deployment) | ||||||
|  | - @paperless-ngx/issues (Issue triage) | ||||||
|  | - @paperless-ngx/test (General testing for larger PRs) | ||||||
|  |  | ||||||
|  | ## Permissions | ||||||
|  |  | ||||||
|  | All team members are notified when mentioned or assigned to a relevant issue or pull request. Additionally, each team has slightly different access to paperless-ngx: | ||||||
|  |  | ||||||
|  | - The **test** team has no special permissions. | ||||||
|  | - The **issues** team has `triage` access. This means they can organize issues and pull requests. | ||||||
|  | - The **backend**, **frontend**, and **ci-cd** teams have `write` access. This means they can approve PRs and push code, containers, releases, and more. | ||||||
|  |  | ||||||
| ## Joining | ## Joining | ||||||
|  |  | ||||||
| @@ -129,13 +141,13 @@ The admins occasionally invite contributors directly if we believe having them o | |||||||
| # Automatic Repository Maintenance | # Automatic Repository Maintenance | ||||||
|  |  | ||||||
| The Paperless-ngx team appreciates all effort and interest from the community in filing bug reports, creating feature requests, sharing ideas and helping other | The Paperless-ngx team appreciates all effort and interest from the community in filing bug reports, creating feature requests, sharing ideas and helping other | ||||||
| community members. That said, in an effort to keep the repository organized and manageable the project uses automatic handling of certain areas: | community members. That said, in an effort to keep the repository organized and managebale the project uses automatic handling of certain areas: | ||||||
|  |  | ||||||
| - Issues that cannot be reproduced will be marked 'stale' after 7 days of inactivity and closed after 14 further days of inactivity. | - Issues that cannot be reproduced will be marked 'stale' after 7 days of inactivity and closed after 14 further days of inactivity. | ||||||
| - Issues, pull requests and discussions that are closed will be locked after 30 days of inactivity. | - Issues, pull requests and discussions that are closed will be locked after 30 days of inactivity. | ||||||
| - Discussions with a marked answer will be automatically closed. | - Discussions with a marked answer will be automatically closed. | ||||||
| - Discussions in the 'General' or 'Support' categories will be closed after 180 days of inactivity. | - Discussions in the 'General' or 'Support' categories will be closed after 180 days of inactivity. | ||||||
| - Feature requests that do not meet the following thresholds will be closed: 180 days of inactivity with less than 80 "up-votes", < 5 "up-votes" after 180 days, < 20 "up-votes" after 1 year or < 40 "up-votes" at 2 years. | - Feature requests that do not meet the following thresholds will be closed: 180 days of inactivity, < 5 "up-votes" after 180 days, < 20 "up-votes" after 1 year or < 80 "up-votes" at 2 years. | ||||||
|  |  | ||||||
| In all cases, threads can be re-opened by project maintainers and, of course, users can always create a new discussion for related concerns. | In all cases, threads can be re-opened by project maintainers and, of course, users can always create a new discussion for related concerns. | ||||||
| Finally, remember that all information remains searchable and 'closed' feature requests can still serve as inspiration for new features. | Finally, remember that all information remains searchable and 'closed' feature requests can still serve as inspiration for new features. | ||||||
|   | |||||||
							
								
								
									
										187
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -4,90 +4,50 @@ | |||||||
| # Stage: compile-frontend | # Stage: compile-frontend | ||||||
| # Purpose: Compiles the frontend | # Purpose: Compiles the frontend | ||||||
| # Notes: | # Notes: | ||||||
| #  - Does PNPM stuff with Typescript and such | #  - Does NPM stuff with Typescript and such | ||||||
| FROM --platform=$BUILDPLATFORM docker.io/node:20-bookworm-slim AS compile-frontend | FROM --platform=$BUILDPLATFORM docker.io/node:20-bookworm-slim AS compile-frontend | ||||||
|  |  | ||||||
| COPY ./src-ui /src/src-ui | COPY ./src-ui /src/src-ui | ||||||
|  |  | ||||||
| WORKDIR /src/src-ui | WORKDIR /src/src-ui | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && npm update -g pnpm \ |   && npm update npm -g \ | ||||||
|   && npm install -g corepack@latest \ |   && npm ci | ||||||
|   && corepack enable \ |  | ||||||
|   && pnpm install |  | ||||||
|  |  | ||||||
| ARG PNGX_TAG_VERSION= | ARG PNGX_TAG_VERSION= | ||||||
| # Add the tag to the environment file if its a tagged dev build | # Add the tag to the environment file if its a tagged dev build | ||||||
| RUN set -eux && \ | RUN set -eux && \ | ||||||
| case "${PNGX_TAG_VERSION}" in \ | case "${PNGX_TAG_VERSION}" in \ | ||||||
|   dev|beta|fix*|feature*) \ |   dev|beta|fix*|feature*) \ | ||||||
|     sed -i -E "s/tag: '([a-z\.]+)'/tag: '${PNGX_TAG_VERSION}'/g" /src/src-ui/src/environments/environment.prod.ts \ |     sed -i -E "s/version: '([0-9\.]+)'/version: '\1 #${PNGX_TAG_VERSION}'/g" /src/src-ui/src/environments/environment.prod.ts \ | ||||||
|     ;; \ |     ;; \ | ||||||
| esac | esac | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && ./node_modules/.bin/ng build --configuration production |   && ./node_modules/.bin/ng build --configuration production | ||||||
|  |  | ||||||
| # Stage: s6-overlay-base | # Stage: pipenv-base | ||||||
| # Purpose: Installs s6-overlay and rootfs | # Purpose: Generates a requirements.txt file for building | ||||||
| # Comments: | # Comments: | ||||||
| #  - Don't leave anything extra in here either | #  - pipenv dependencies are not left in the final image | ||||||
| FROM ghcr.io/astral-sh/uv:0.9.4-python3.12-bookworm-slim AS s6-overlay-base | #  - pipenv can't touch the final image somehow | ||||||
|  | FROM --platform=$BUILDPLATFORM docker.io/python:3.12-alpine AS pipenv-base | ||||||
|  |  | ||||||
| WORKDIR /usr/src/s6 | WORKDIR /usr/src/pipenv | ||||||
|  |  | ||||||
| # https://github.com/just-containers/s6-overlay#customizing-s6-overlay-behaviour | COPY Pipfile* ./ | ||||||
| ENV \ |  | ||||||
|     S6_BEHAVIOUR_IF_STAGE2_FAILS=2 \ |  | ||||||
|     S6_CMD_WAIT_FOR_SERVICES_MAXTIME=0 \ |  | ||||||
|     S6_VERBOSITY=1 \ |  | ||||||
|     PATH=/command:$PATH |  | ||||||
|  |  | ||||||
| # Buildx provided, must be defined to use though |  | ||||||
| ARG TARGETARCH |  | ||||||
| ARG TARGETVARIANT |  | ||||||
| # Lock this version |  | ||||||
| ARG S6_OVERLAY_VERSION=3.2.1.0 |  | ||||||
|  |  | ||||||
| ARG S6_BUILD_TIME_PKGS="curl \ |  | ||||||
|                         xz-utils" |  | ||||||
|  |  | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|     && echo "Installing build time packages" \ |   && echo "Installing pipenv" \ | ||||||
|       && apt-get update \ |     && python3 -m pip install --no-cache-dir --upgrade pipenv==2024.0.3 \ | ||||||
|       && apt-get install --yes --quiet --no-install-recommends ${S6_BUILD_TIME_PKGS} \ |   && echo "Generating requirement.txt" \ | ||||||
|     && echo "Determining arch" \ |     && pipenv requirements > requirements.txt | ||||||
|       && S6_ARCH="" \ |  | ||||||
|       && if [ "${TARGETARCH}${TARGETVARIANT}" = "amd64" ]; then S6_ARCH="x86_64"; \ |  | ||||||
|       elif [ "${TARGETARCH}${TARGETVARIANT}" = "arm64" ]; then S6_ARCH="aarch64"; fi\ |  | ||||||
|       && if [ -z "${S6_ARCH}" ]; then { echo "Error: Not able to determine arch"; exit 1; }; fi \ |  | ||||||
|     && echo "Installing s6-overlay for ${S6_ARCH}" \ |  | ||||||
|       && curl --fail --silent --no-progress-meter --show-error --location --remote-name-all --parallel --parallel-max 4 \ |  | ||||||
|         "https://github.com/just-containers/s6-overlay/releases/download/v${S6_OVERLAY_VERSION}/s6-overlay-noarch.tar.xz" \ |  | ||||||
|         "https://github.com/just-containers/s6-overlay/releases/download/v${S6_OVERLAY_VERSION}/s6-overlay-noarch.tar.xz.sha256" \ |  | ||||||
|         "https://github.com/just-containers/s6-overlay/releases/download/v${S6_OVERLAY_VERSION}/s6-overlay-${S6_ARCH}.tar.xz" \ |  | ||||||
|         "https://github.com/just-containers/s6-overlay/releases/download/v${S6_OVERLAY_VERSION}/s6-overlay-${S6_ARCH}.tar.xz.sha256" \ |  | ||||||
|       && echo "Validating s6-archive checksums" \ |  | ||||||
|         && sha256sum --check ./*.sha256 \ |  | ||||||
|       && echo "Unpacking archives" \ |  | ||||||
|         && tar --directory / -Jxpf s6-overlay-noarch.tar.xz \ |  | ||||||
|         && tar --directory / -Jxpf s6-overlay-${S6_ARCH}.tar.xz \ |  | ||||||
|       && echo "Removing downloaded archives" \ |  | ||||||
|         && rm ./*.tar.xz \ |  | ||||||
|         && rm ./*.sha256 \ |  | ||||||
|     && echo "Cleaning up image" \ |  | ||||||
|       && apt-get --yes purge ${S6_BUILD_TIME_PKGS} \ |  | ||||||
|       && apt-get --yes autoremove --purge \ |  | ||||||
|       && rm -rf /var/lib/apt/lists/* |  | ||||||
|  |  | ||||||
| # Copy our service defs and filesystem |  | ||||||
| COPY ./docker/rootfs / |  | ||||||
|  |  | ||||||
| # Stage: main-app | # Stage: main-app | ||||||
| # Purpose: The final image | # Purpose: The final image | ||||||
| # Comments: | # Comments: | ||||||
| #  - Don't leave anything extra in here | #  - Don't leave anything extra in here | ||||||
| FROM s6-overlay-base AS main-app | FROM docker.io/python:3.12-slim-bookworm AS main-app | ||||||
|  |  | ||||||
| LABEL org.opencontainers.image.authors="paperless-ngx team <hello@paperless-ngx.com>" | LABEL org.opencontainers.image.authors="paperless-ngx team <hello@paperless-ngx.com>" | ||||||
| LABEL org.opencontainers.image.documentation="https://docs.paperless-ngx.com/" | LABEL org.opencontainers.image.documentation="https://docs.paperless-ngx.com/" | ||||||
| @@ -101,19 +61,16 @@ ARG DEBIAN_FRONTEND=noninteractive | |||||||
| ARG TARGETARCH | ARG TARGETARCH | ||||||
|  |  | ||||||
| # Can be workflow provided, defaults set for manual building | # Can be workflow provided, defaults set for manual building | ||||||
| ARG JBIG2ENC_VERSION=0.30 | ARG JBIG2ENC_VERSION=0.29 | ||||||
| ARG QPDF_VERSION=11.9.0 | ARG QPDF_VERSION=11.9.0 | ||||||
| ARG GS_VERSION=10.03.1 | ARG GS_VERSION=10.03.1 | ||||||
|  |  | ||||||
| # Set Python environment variables | # Set Python environment variables | ||||||
| ENV PYTHONDONTWRITEBYTECODE=1 \ | ENV PYTHONDONTWRITEBYTECODE=1 \ | ||||||
|     PYTHONUNBUFFERED=1 \ |     PYTHONUNBUFFERED=1 \ | ||||||
|     # Ignore warning from Whitenoise about async iterators |     # Ignore warning from Whitenoise | ||||||
|     PYTHONWARNINGS="ignore:::django.http.response:517" \ |     PYTHONWARNINGS="ignore:::django.http.response:517" \ | ||||||
|     PNGX_CONTAINERIZED=1 \ |     PNGX_CONTAINERIZED=1 | ||||||
|     # https://docs.astral.sh/uv/reference/settings/#link-mode |  | ||||||
|     UV_LINK_MODE=copy \ |  | ||||||
|     UV_CACHE_DIR=/cache/uv/ |  | ||||||
|  |  | ||||||
| # | # | ||||||
| # Begin installation and configuration | # Begin installation and configuration | ||||||
| @@ -170,51 +127,118 @@ RUN set -eux \ | |||||||
|     && apt-get update \ |     && apt-get update \ | ||||||
|     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} \ |     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} \ | ||||||
|     && echo "Installing pre-built updates" \ |     && echo "Installing pre-built updates" \ | ||||||
|       && curl --fail --silent --no-progress-meter --show-error --location --remote-name-all --parallel --parallel-max 4 \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ |  | ||||||
|         https://github.com/paperless-ngx/builder/releases/download/jbig2enc-${JBIG2ENC_VERSION}/jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ |  | ||||||
|       && echo "Installing qpdf ${QPDF_VERSION}" \ |       && echo "Installing qpdf ${QPDF_VERSION}" \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|         && dpkg --install ./libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ |         && dpkg --install ./libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|         && dpkg --install ./qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ |         && dpkg --install ./qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|       && echo "Installing Ghostscript ${GS_VERSION}" \ |       && echo "Installing Ghostscript ${GS_VERSION}" \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ | ||||||
|         && dpkg --install ./libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ |         && dpkg --install ./libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ | ||||||
|         && dpkg --install ./libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ |         && dpkg --install ./libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|         && dpkg --install ./ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ |         && dpkg --install ./ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||||
|       && echo "Installing jbig2enc" \ |       && echo "Installing jbig2enc" \ | ||||||
|  |         && curl --fail --silent --show-error --location \ | ||||||
|  |           --output jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|  |           https://github.com/paperless-ngx/builder/releases/download/jbig2enc-${JBIG2ENC_VERSION}/jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|         && dpkg --install ./jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ |         && dpkg --install ./jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||||
|       && echo "Configuring imagemagick" \ |  | ||||||
|         && cp /etc/ImageMagick-6/paperless-policy.xml /etc/ImageMagick-6/policy.xml \ |  | ||||||
|       && echo "Cleaning up image layer" \ |       && echo "Cleaning up image layer" \ | ||||||
|         && rm --force --verbose *.deb \ |         && rm --force --verbose *.deb \ | ||||||
|     && rm --recursive --force --verbose /var/lib/apt/lists/* |     && rm --recursive --force --verbose /var/lib/apt/lists/* \ | ||||||
|  |   && echo "Installing supervisor" \ | ||||||
|  |     && python3 -m pip install --default-timeout=1000 --upgrade --no-cache-dir supervisor==4.2.5 | ||||||
|  |  | ||||||
|  | # Copy gunicorn config | ||||||
|  | # Changes very infrequently | ||||||
|  | WORKDIR /usr/src/paperless/ | ||||||
|  |  | ||||||
|  | COPY gunicorn.conf.py . | ||||||
|  |  | ||||||
|  | # setup docker-specific things | ||||||
|  | # These change sometimes, but rarely | ||||||
|  | WORKDIR /usr/src/paperless/src/docker/ | ||||||
|  |  | ||||||
|  | COPY [ \ | ||||||
|  |   "docker/imagemagick-policy.xml", \ | ||||||
|  |   "docker/supervisord.conf", \ | ||||||
|  |   "docker/docker-entrypoint.sh", \ | ||||||
|  |   "docker/docker-prepare.sh", \ | ||||||
|  |   "docker/paperless_cmd.sh", \ | ||||||
|  |   "docker/wait-for-redis.py", \ | ||||||
|  |   "docker/env-from-file.sh", \ | ||||||
|  |   "docker/management_script.sh", \ | ||||||
|  |   "docker/flower-conditional.sh", \ | ||||||
|  |   "docker/install_management_commands.sh", \ | ||||||
|  |   "/usr/src/paperless/src/docker/" \ | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | RUN set -eux \ | ||||||
|  |   && echo "Configuring ImageMagick" \ | ||||||
|  |     && mv imagemagick-policy.xml /etc/ImageMagick-6/policy.xml \ | ||||||
|  |   && echo "Configuring supervisord" \ | ||||||
|  |     && mkdir /var/log/supervisord /var/run/supervisord \ | ||||||
|  |     && mv supervisord.conf /etc/supervisord.conf \ | ||||||
|  |   && echo "Setting up Docker scripts" \ | ||||||
|  |     && mv docker-entrypoint.sh /sbin/docker-entrypoint.sh \ | ||||||
|  |     && chmod 755 /sbin/docker-entrypoint.sh \ | ||||||
|  |     && mv docker-prepare.sh /sbin/docker-prepare.sh \ | ||||||
|  |     && chmod 755 /sbin/docker-prepare.sh \ | ||||||
|  |     && mv wait-for-redis.py /sbin/wait-for-redis.py \ | ||||||
|  |     && chmod 755 /sbin/wait-for-redis.py \ | ||||||
|  |     && mv env-from-file.sh /sbin/env-from-file.sh \ | ||||||
|  |     && chmod 755 /sbin/env-from-file.sh \ | ||||||
|  |     && mv paperless_cmd.sh /usr/local/bin/paperless_cmd.sh \ | ||||||
|  |     && chmod 755 /usr/local/bin/paperless_cmd.sh \ | ||||||
|  |     && mv flower-conditional.sh /usr/local/bin/flower-conditional.sh \ | ||||||
|  |     && chmod 755 /usr/local/bin/flower-conditional.sh \ | ||||||
|  |   && echo "Installing management commands" \ | ||||||
|  |     && chmod +x install_management_commands.sh \ | ||||||
|  |     && ./install_management_commands.sh | ||||||
|  |  | ||||||
| WORKDIR /usr/src/paperless/src/ | WORKDIR /usr/src/paperless/src/ | ||||||
|  |  | ||||||
| # Python dependencies | # Python dependencies | ||||||
| # Change pretty frequently | # Change pretty frequently | ||||||
| COPY --chown=1000:1000 ["pyproject.toml", "uv.lock", "/usr/src/paperless/src/"] | COPY --from=pipenv-base /usr/src/pipenv/requirements.txt ./ | ||||||
|  |  | ||||||
| # Packages needed only for building a few quick Python | # Packages needed only for building a few quick Python | ||||||
| # dependencies | # dependencies | ||||||
| ARG BUILD_PACKAGES="\ | ARG BUILD_PACKAGES="\ | ||||||
|   build-essential \ |   build-essential \ | ||||||
|  |   git \ | ||||||
|  |   # https://www.psycopg.org/docs/install.html#prerequisites | ||||||
|  |   libpq-dev \ | ||||||
|   # https://github.com/PyMySQL/mysqlclient#linux |   # https://github.com/PyMySQL/mysqlclient#linux | ||||||
|   default-libmysqlclient-dev \ |   default-libmysqlclient-dev \ | ||||||
|   pkg-config" |   pkg-config" | ||||||
|  |  | ||||||
| # hadolint ignore=DL3042 | # hadolint ignore=DL3042 | ||||||
| RUN --mount=type=cache,target=${UV_CACHE_DIR},id=python-cache \ | RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \ | ||||||
|   set -eux \ |   set -eux \ | ||||||
|   && echo "Installing build system packages" \ |   && echo "Installing build system packages" \ | ||||||
|     && apt-get update \ |     && apt-get update \ | ||||||
|     && apt-get install --yes --quiet --no-install-recommends ${BUILD_PACKAGES} \ |     && apt-get install --yes --quiet --no-install-recommends ${BUILD_PACKAGES} \ | ||||||
|  |     && python3 -m pip install --no-cache-dir --upgrade wheel \ | ||||||
|   && echo "Installing Python requirements" \ |   && echo "Installing Python requirements" \ | ||||||
|     && uv export --quiet --no-dev --all-extras --format requirements-txt --output-file requirements.txt \ |     && curl --fail --silent --show-error --location \ | ||||||
|     && uv pip install --system --no-python-downloads --python-preference system --requirements requirements.txt \ |     --output psycopg_c-3.2.2-cp312-cp312-linux_x86_64.whl \ | ||||||
|  |     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.2/psycopg_c-3.2.2-cp312-cp312-linux_x86_64.whl \ | ||||||
|  |     && curl --fail --silent --show-error --location \ | ||||||
|  |     --output psycopg_c-3.2.2-cp312-cp312-linux_aarch64.whl  \ | ||||||
|  |     https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.2/psycopg_c-3.2.2-cp312-cp312-linux_aarch64.whl \ | ||||||
|  |     && python3 -m pip install --default-timeout=1000 --find-links . --requirement requirements.txt \ | ||||||
|   && echo "Installing NLTK data" \ |   && echo "Installing NLTK data" \ | ||||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \ |     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \ | ||||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" stopwords \ |     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" stopwords \ | ||||||
| @@ -239,7 +263,6 @@ COPY --from=compile-frontend --chown=1000:1000 /src/src/documents/static/fronten | |||||||
| # add users, setup scripts | # add users, setup scripts | ||||||
| # Mount the compiled frontend to expected location | # Mount the compiled frontend to expected location | ||||||
| RUN set -eux \ | RUN set -eux \ | ||||||
|   && sed -i '1s|^#!/usr/bin/env python3|#!/command/with-contenv python3|' manage.py \ |  | ||||||
|   && echo "Setting up user/group" \ |   && echo "Setting up user/group" \ | ||||||
|     && addgroup --gid 1000 paperless \ |     && addgroup --gid 1000 paperless \ | ||||||
|     && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ |     && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ | ||||||
| @@ -253,16 +276,18 @@ RUN set -eux \ | |||||||
|   && echo "Adjusting all permissions" \ |   && echo "Adjusting all permissions" \ | ||||||
|     && chown --from root:root --changes --recursive paperless:paperless /usr/src/paperless \ |     && chown --from root:root --changes --recursive paperless:paperless /usr/src/paperless \ | ||||||
|   && echo "Collecting static files" \ |   && echo "Collecting static files" \ | ||||||
|     && s6-setuidgid paperless python3 manage.py collectstatic --clear --no-input --link \ |     && gosu paperless python3 manage.py collectstatic --clear --no-input --link \ | ||||||
|     && s6-setuidgid paperless python3 manage.py compilemessages |     && gosu paperless python3 manage.py compilemessages | ||||||
|  |  | ||||||
| VOLUME ["/usr/src/paperless/data", \ | VOLUME ["/usr/src/paperless/data", \ | ||||||
|         "/usr/src/paperless/media", \ |         "/usr/src/paperless/media", \ | ||||||
|         "/usr/src/paperless/consume", \ |         "/usr/src/paperless/consume", \ | ||||||
|         "/usr/src/paperless/export"] |         "/usr/src/paperless/export"] | ||||||
|  |  | ||||||
| ENTRYPOINT ["/init"] | ENTRYPOINT ["/sbin/docker-entrypoint.sh"] | ||||||
|  |  | ||||||
| EXPOSE 8000 | EXPOSE 8000 | ||||||
|  |  | ||||||
| HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "-L", "--max-time", "2", "http://localhost:8000" ] | CMD ["/usr/local/bin/paperless_cmd.sh"] | ||||||
|  |  | ||||||
|  | HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000" ] | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								Pipfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | [[source]] | ||||||
|  | url = "https://pypi.python.org/simple" | ||||||
|  | verify_ssl = true | ||||||
|  | name = "pypi" | ||||||
|  |  | ||||||
|  | [packages] | ||||||
|  | dateparser = "~=1.2" | ||||||
|  | # WARNING: django does not use semver. | ||||||
|  | #          Only patch versions are guaranteed to not introduce breaking changes. | ||||||
|  | django = "~=5.1.1" | ||||||
|  | django-allauth = {extras = ["socialaccount"], version = "*"} | ||||||
|  | django-auditlog = "*" | ||||||
|  | django-celery-results = "*" | ||||||
|  | django-compression-middleware = "*" | ||||||
|  | django-cors-headers = "*" | ||||||
|  | django-extensions = "*" | ||||||
|  | django-filter = "~=24.3" | ||||||
|  | django-guardian = "*" | ||||||
|  | django-multiselectfield = "*" | ||||||
|  | django-soft-delete = "*" | ||||||
|  | djangorestframework = "==3.15.2" | ||||||
|  | djangorestframework-guardian = "*" | ||||||
|  | drf-writable-nested = "*" | ||||||
|  | bleach = "*" | ||||||
|  | celery = {extras = ["redis"], version = "*"} | ||||||
|  | channels = "~=4.1" | ||||||
|  | channels-redis = "*" | ||||||
|  | concurrent-log-handler = "*" | ||||||
|  | filelock = "*" | ||||||
|  | flower = "*" | ||||||
|  | gotenberg-client = "*" | ||||||
|  | gunicorn = "*" | ||||||
|  | httpx-oauth = "*" | ||||||
|  | imap-tools = "*" | ||||||
|  | inotifyrecursive = "~=0.3" | ||||||
|  | jinja2 = "~=3.1" | ||||||
|  | langdetect = "*" | ||||||
|  | mysqlclient = "*" | ||||||
|  | nltk = "*" | ||||||
|  | ocrmypdf = "~=16.5" | ||||||
|  | pathvalidate = "*" | ||||||
|  | pdf2image = "*" | ||||||
|  | psycopg = {version = "*", extras = ["c"]} | ||||||
|  | python-dateutil = "*" | ||||||
|  | python-dotenv = "*" | ||||||
|  | python-gnupg = "*" | ||||||
|  | python-ipware = "*" | ||||||
|  | python-magic = "*" | ||||||
|  | pyzbar = "*" | ||||||
|  | rapidfuzz = "*" | ||||||
|  | redis = {extras = ["hiredis"], version = "*"} | ||||||
|  | scikit-learn = "~=1.5" | ||||||
|  | setproctitle = "*" | ||||||
|  | tika-client = "*" | ||||||
|  | tqdm = "*" | ||||||
|  | # See https://github.com/paperless-ngx/paperless-ngx/issues/5494 | ||||||
|  | uvicorn = {extras = ["standard"], version = "==0.25.0"} | ||||||
|  | watchdog = "~=4.0" | ||||||
|  | whitenoise = "~=6.8" | ||||||
|  | whoosh = "~=2.7" | ||||||
|  | zxing-cpp = {version = "*", platform_machine = "== 'x86_64'"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | [dev-packages] | ||||||
|  | # Linting | ||||||
|  | pre-commit = "*" | ||||||
|  | ruff = "*" | ||||||
|  | factory-boy = "*" | ||||||
|  | # Testing | ||||||
|  | pytest = "*" | ||||||
|  | pytest-cov = "*" | ||||||
|  | pytest-django = "*" | ||||||
|  | pytest-httpx = "*" | ||||||
|  | pytest-env = "*" | ||||||
|  | pytest-sugar = "*" | ||||||
|  | pytest-xdist = "*" | ||||||
|  | pytest-mock = "*" | ||||||
|  | pytest-rerunfailures = "*" | ||||||
|  | imagehash = "*" | ||||||
|  | daphne = "*" | ||||||
|  | # Documentation | ||||||
|  | mkdocs-material = "*" | ||||||
|  | mkdocs-glightbox = "*" | ||||||
|  |  | ||||||
|  | [typing-dev] | ||||||
|  | mypy = "*" | ||||||
|  | types-Pillow = "*" | ||||||
|  | django-filter-stubs = "*" | ||||||
|  | types-python-dateutil = "*" | ||||||
|  | djangorestframework-stubs = {extras= ["compatible-mypy"], version="*"} | ||||||
|  | celery-types = "*" | ||||||
|  | django-stubs = {extras= ["compatible-mypy"], version="*"} | ||||||
|  | types-dateparser = "*" | ||||||
|  | types-bleach = "*" | ||||||
|  | types-redis = "*" | ||||||
|  | types-tqdm = "*" | ||||||
|  | types-Markdown = "*" | ||||||
|  | types-Pygments = "*" | ||||||
|  | types-colorama = "*" | ||||||
|  | types-setuptools = "*" | ||||||
							
								
								
									
										4489
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										4489
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -55,7 +55,7 @@ A full list of [features](https://docs.paperless-ngx.com/#features) and [screens | |||||||
|  |  | ||||||
| # Getting started | # Getting started | ||||||
|  |  | ||||||
| The easiest way to deploy paperless is `docker compose`. The files in the [`/docker/compose` directory](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) are configured to pull the image from the GitHub container registry. | The easiest way to deploy paperless is `docker compose`. The files in the [`/docker/compose` directory](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) are configured to pull the image from GitHub Packages. | ||||||
|  |  | ||||||
| If you'd like to jump right in, you can configure a `docker compose` environment with our install script: | If you'd like to jump right in, you can configure a `docker compose` environment with our install script: | ||||||
|  |  | ||||||
| @@ -83,7 +83,7 @@ People interested in continuing the work on paperless-ngx are encouraged to reac | |||||||
|  |  | ||||||
| ## Translation | ## Translation | ||||||
|  |  | ||||||
| Paperless-ngx is available in many languages that are coordinated on Crowdin. If you want to help out by translating paperless-ngx into your language, please head over to https://crowdin.com/project/paperless-ngx, and thank you! More details can be found in [CONTRIBUTING.md](https://github.com/paperless-ngx/paperless-ngx/blob/main/CONTRIBUTING.md#translating-paperless-ngx). | Paperless-ngx is available in many languages that are coordinated on Crowdin. If you want to help out by translating paperless-ngx into your language, please head over to https://crwd.in/paperless-ngx, and thank you! More details can be found in [CONTRIBUTING.md](https://github.com/paperless-ngx/paperless-ngx/blob/main/CONTRIBUTING.md#translating-paperless-ngx). | ||||||
|  |  | ||||||
| ## Feature Requests | ## Feature Requests | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| # Docker Compose file for running paperless testing with actual Gotenberg | # Docker Compose file for running paperless testing with actual gotenberg | ||||||
| # and Tika containers for a more end to end test of the Tika related functionality | # and Tika containers for a more end to end test of the Tika related functionality | ||||||
| # Can be used locally or by the CI to start the necessary containers with the | # Can be used locally or by the CI to start the necessary containers with the | ||||||
| # correct networking for the tests | # correct networking for the tests | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   gotenberg: |   gotenberg: | ||||||
|     image: docker.io/gotenberg/gotenberg:8.24 |     image: docker.io/gotenberg/gotenberg:8.7 | ||||||
|     hostname: gotenberg |     hostname: gotenberg | ||||||
|     container_name: gotenberg |     container_name: gotenberg | ||||||
|     network_mode: host |     network_mode: host | ||||||
|   | |||||||
| @@ -1,17 +1,26 @@ | |||||||
| ############################################################################### |  | ||||||
| # Paperless-ngx settings                                                      # |  | ||||||
| ############################################################################### |  | ||||||
|  |  | ||||||
| # See http://docs.paperless-ngx.com/configuration/ for all available options. |  | ||||||
|  |  | ||||||
| # The UID and GID of the user used to run paperless in the container. Set this | # The UID and GID of the user used to run paperless in the container. Set this | ||||||
| # to your UID and GID on the host so that you have write access to the | # to your UID and GID on the host so that you have write access to the | ||||||
| # consumption directory. | # consumption directory. | ||||||
| #USERMAP_UID=1000 | #USERMAP_UID=1000 | ||||||
| #USERMAP_GID=1000 | #USERMAP_GID=1000 | ||||||
|  |  | ||||||
| # See the documentation linked above for all options. A few commonly adjusted settings | # Additional languages to install for text recognition, separated by a | ||||||
| # are provided below. | # whitespace. Note that this is | ||||||
|  | # different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines the | ||||||
|  | # language used for OCR. | ||||||
|  | # The container installs English, German, Italian, Spanish and French by | ||||||
|  | # default. | ||||||
|  | # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster | ||||||
|  | # for available languages. | ||||||
|  | #PAPERLESS_OCR_LANGUAGES=tur ces | ||||||
|  |  | ||||||
|  | ############################################################################### | ||||||
|  | # Paperless-specific settings                                                 # | ||||||
|  | ############################################################################### | ||||||
|  |  | ||||||
|  | # All settings defined in the paperless.conf.example can be used here. The | ||||||
|  | # Docker setup does not use the configuration file. | ||||||
|  | # A few commonly adjusted settings are provided below. | ||||||
|  |  | ||||||
| # This is required if you will be exposing Paperless-ngx on a public domain | # This is required if you will be exposing Paperless-ngx on a public domain | ||||||
| # (if doing so please consider security measures such as reverse proxy) | # (if doing so please consider security measures such as reverse proxy) | ||||||
| @@ -21,17 +30,13 @@ | |||||||
| # be a very long sequence of random characters. You don't need to remember it. | # be a very long sequence of random characters. You don't need to remember it. | ||||||
| #PAPERLESS_SECRET_KEY=change-me | #PAPERLESS_SECRET_KEY=change-me | ||||||
|  |  | ||||||
| # Use this variable to set a timezone for the Paperless Docker containers. Defaults to UTC. | # Use this variable to set a timezone for the Paperless Docker containers. If not specified, defaults to UTC. | ||||||
| #PAPERLESS_TIME_ZONE=America/Los_Angeles | #PAPERLESS_TIME_ZONE=America/Los_Angeles | ||||||
|  |  | ||||||
| # The default language to use for OCR. Set this to the language most of your | # The default language to use for OCR. Set this to the language most of your | ||||||
| # documents are written in. | # documents are written in. | ||||||
| #PAPERLESS_OCR_LANGUAGE=eng | #PAPERLESS_OCR_LANGUAGE=eng | ||||||
|  |  | ||||||
| # Additional languages to install for text recognition, separated by a whitespace. | # Set if accessing paperless via a domain subpath e.g. https://domain.com/PATHPREFIX and using a reverse-proxy like traefik or nginx | ||||||
| # Note that this is different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines | #PAPERLESS_FORCE_SCRIPT_NAME=/PATHPREFIX | ||||||
| # the language used for OCR. | #PAPERLESS_STATIC_URL=/PATHPREFIX/static/ # trailing slash required | ||||||
| # The container installs English, German, Italian, Spanish and French by default. |  | ||||||
| # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names |  | ||||||
| # for available languages. |  | ||||||
| #PAPERLESS_OCR_LANGUAGES=tur ces |  | ||||||
|   | |||||||
| @@ -16,26 +16,29 @@ | |||||||
| # - Instead of SQLite (default), MariaDB is used as the database server. | # - Instead of SQLite (default), MariaDB is used as the database server. | ||||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||||
| #   is configured to use these services. These provide support for consuming | #   is configured to use these services. These provide support for consuming | ||||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||||
| #   parts). | #   parts. | ||||||
| # | # | ||||||
| # To install and update paperless with this file, do the following: | # To install and update paperless with this file, do the following: | ||||||
| # | # | ||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/mariadb:12 |     image: docker.io/library/mariadb:11 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - dbdata:/var/lib/mysql |       - dbdata:/var/lib/mysql | ||||||
| @@ -45,6 +48,7 @@ services: | |||||||
|       MARIADB_USER: paperless |       MARIADB_USER: paperless | ||||||
|       MARIADB_PASSWORD: paperless |       MARIADB_PASSWORD: paperless | ||||||
|       MARIADB_ROOT_PASSWORD: paperless |       MARIADB_ROOT_PASSWORD: paperless | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -71,8 +75,9 @@ services: | |||||||
|       PAPERLESS_TIKA_ENABLED: 1 |       PAPERLESS_TIKA_ENABLED: 1 | ||||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 |       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 |       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||||
|  |  | ||||||
|   gotenberg: |   gotenberg: | ||||||
|     image: docker.io/gotenberg/gotenberg:8.24 |     image: docker.io/gotenberg/gotenberg:8.7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     # The gotenberg chromium route is used to convert .eml files. We do not |     # The gotenberg chromium route is used to convert .eml files. We do not | ||||||
|     # want to allow external content like tracking pixels or even javascript. |     # want to allow external content like tracking pixels or even javascript. | ||||||
| @@ -80,9 +85,11 @@ services: | |||||||
|       - "gotenberg" |       - "gotenberg" | ||||||
|       - "--chromium-disable-javascript=true" |       - "--chromium-disable-javascript=true" | ||||||
|       - "--chromium-allow-list=file:///tmp/.*" |       - "--chromium-allow-list=file:///tmp/.*" | ||||||
|  |  | ||||||
|   tika: |   tika: | ||||||
|     image: docker.io/apache/tika:latest |     image: docker.io/apache/tika:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -20,18 +20,21 @@ | |||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/mariadb:12 |     image: docker.io/library/mariadb:11 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - dbdata:/var/lib/mysql |       - dbdata:/var/lib/mysql | ||||||
| @@ -41,6 +44,7 @@ services: | |||||||
|       MARIADB_USER: paperless |       MARIADB_USER: paperless | ||||||
|       MARIADB_PASSWORD: paperless |       MARIADB_PASSWORD: paperless | ||||||
|       MARIADB_ROOT_PASSWORD: paperless |       MARIADB_ROOT_PASSWORD: paperless | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -62,6 +66,7 @@ services: | |||||||
|       PAPERLESS_DBUSER: paperless # only needed if non-default username |       PAPERLESS_DBUSER: paperless # only needed if non-default username | ||||||
|       PAPERLESS_DBPASS: paperless # only needed if non-default password |       PAPERLESS_DBPASS: paperless # only needed if non-default password | ||||||
|       PAPERLESS_DBPORT: 3306 |       PAPERLESS_DBPORT: 3306 | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -19,27 +19,32 @@ | |||||||
| # | # | ||||||
| # - Open portainer Stacks list and click 'Add stack' | # - Open portainer Stacks list and click 'Add stack' | ||||||
| # - Paste the contents of this file and assign a name, e.g. 'paperless' | # - Paste the contents of this file and assign a name, e.g. 'paperless' | ||||||
| # - Upload 'docker-compose.env' by clicking on 'Load variables from .env file' |  | ||||||
| # - Modify the environment variables as needed |  | ||||||
| # - Click 'Deploy the stack' and wait for it to be deployed | # - Click 'Deploy the stack' and wait for it to be deployed | ||||||
|  | # - Open the list of containers, select paperless_webserver_1 | ||||||
|  | # - Click 'Console' and then 'Connect' to open the command line inside the container | ||||||
|  | # - Run 'python3 manage.py createsuperuser' to create a user | ||||||
|  | # - Exit the console | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:18 |     image: docker.io/library/postgres:16 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql |       - pgdata:/var/lib/postgresql/data | ||||||
|     environment: |     environment: | ||||||
|       POSTGRES_DB: paperless |       POSTGRES_DB: paperless | ||||||
|       POSTGRES_USER: paperless |       POSTGRES_USER: paperless | ||||||
|       POSTGRES_PASSWORD: paperless |       POSTGRES_PASSWORD: paperless | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -56,8 +61,29 @@ services: | |||||||
|     environment: |     environment: | ||||||
|       PAPERLESS_REDIS: redis://broker:6379 |       PAPERLESS_REDIS: redis://broker:6379 | ||||||
|       PAPERLESS_DBHOST: db |       PAPERLESS_DBHOST: db | ||||||
|     env_file: | # The UID and GID of the user used to run paperless in the container. Set this | ||||||
|       - stack.env | # to your UID and GID on the host so that you have write access to the | ||||||
|  | # consumption directory. | ||||||
|  |       USERMAP_UID: 1000 | ||||||
|  |       USERMAP_GID: 100 | ||||||
|  | # Additional languages to install for text recognition, separated by a | ||||||
|  | # whitespace. Note that this is | ||||||
|  | # different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines the | ||||||
|  | # language used for OCR. | ||||||
|  | # The container installs English, German, Italian, Spanish and French by | ||||||
|  | # default. | ||||||
|  | # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster | ||||||
|  | # for available languages. | ||||||
|  |       #PAPERLESS_OCR_LANGUAGES: tur ces | ||||||
|  | # Adjust this key if you plan to make paperless available publicly. It should | ||||||
|  | # be a very long sequence of random characters. You don't need to remember it. | ||||||
|  |       #PAPERLESS_SECRET_KEY: change-me | ||||||
|  | # Use this variable to set a timezone for the Paperless Docker containers. If not specified, defaults to UTC. | ||||||
|  |       #PAPERLESS_TIME_ZONE: America/Los_Angeles | ||||||
|  | # The default language to use for OCR. Set this to the language most of your | ||||||
|  | # documents are written in. | ||||||
|  |       #PAPERLESS_OCR_LANGUAGE: eng | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -16,33 +16,37 @@ | |||||||
| # - Instead of SQLite (default), PostgreSQL is used as the database server. | # - Instead of SQLite (default), PostgreSQL is used as the database server. | ||||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||||
| #   is configured to use these services. These provide support for consuming | #   is configured to use these services. These provide support for consuming | ||||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||||
| #   parts). | #   parts. | ||||||
| # | # | ||||||
| # To install and update paperless with this file, do the following: | # To install and update paperless with this file, do the following: | ||||||
| # | # | ||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:18 |     image: docker.io/library/postgres:16 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql |       - pgdata:/var/lib/postgresql/data | ||||||
|     environment: |     environment: | ||||||
|       POSTGRES_DB: paperless |       POSTGRES_DB: paperless | ||||||
|       POSTGRES_USER: paperless |       POSTGRES_USER: paperless | ||||||
|       POSTGRES_PASSWORD: paperless |       POSTGRES_PASSWORD: paperless | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -65,18 +69,22 @@ services: | |||||||
|       PAPERLESS_TIKA_ENABLED: 1 |       PAPERLESS_TIKA_ENABLED: 1 | ||||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 |       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 |       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||||
|  |  | ||||||
|   gotenberg: |   gotenberg: | ||||||
|     image: docker.io/gotenberg/gotenberg:8.24 |     image: docker.io/gotenberg/gotenberg:8.7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
|     # The gotenberg chromium route is used to convert .eml files. We do not |     # The gotenberg chromium route is used to convert .eml files. We do not | ||||||
|     # want to allow external content like tracking pixels or even javascript. |     # want to allow external content like tracking pixels or even javascript. | ||||||
|     command: |     command: | ||||||
|       - "gotenberg" |       - "gotenberg" | ||||||
|       - "--chromium-disable-javascript=true" |       - "--chromium-disable-javascript=true" | ||||||
|       - "--chromium-allow-list=file:///tmp/.*" |       - "--chromium-allow-list=file:///tmp/.*" | ||||||
|  |  | ||||||
|   tika: |   tika: | ||||||
|     image: docker.io/apache/tika:latest |     image: docker.io/apache/tika:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -20,25 +20,29 @@ | |||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   db: |   db: | ||||||
|     image: docker.io/library/postgres:18 |     image: docker.io/library/postgres:16 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - pgdata:/var/lib/postgresql |       - pgdata:/var/lib/postgresql/data | ||||||
|     environment: |     environment: | ||||||
|       POSTGRES_DB: paperless |       POSTGRES_DB: paperless | ||||||
|       POSTGRES_USER: paperless |       POSTGRES_USER: paperless | ||||||
|       POSTGRES_PASSWORD: paperless |       POSTGRES_PASSWORD: paperless | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -56,6 +60,7 @@ services: | |||||||
|     environment: |     environment: | ||||||
|       PAPERLESS_REDIS: redis://broker:6379 |       PAPERLESS_REDIS: redis://broker:6379 | ||||||
|       PAPERLESS_DBHOST: db |       PAPERLESS_DBHOST: db | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -16,24 +16,27 @@ | |||||||
| # | # | ||||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||||
| #   is configured to use these services. These provide support for consuming | #   is configured to use these services. These provide support for consuming | ||||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||||
| #   parts). | #   parts. | ||||||
| # | # | ||||||
| # To install and update paperless with this file, do the following: | # To install and update paperless with this file, do the following: | ||||||
| # | # | ||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -54,18 +57,22 @@ services: | |||||||
|       PAPERLESS_TIKA_ENABLED: 1 |       PAPERLESS_TIKA_ENABLED: 1 | ||||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 |       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 |       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||||
|  |  | ||||||
|   gotenberg: |   gotenberg: | ||||||
|     image: docker.io/gotenberg/gotenberg:8.24 |     image: docker.io/gotenberg/gotenberg:8.7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
|     # The gotenberg chromium route is used to convert .eml files. We do not |     # The gotenberg chromium route is used to convert .eml files. We do not | ||||||
|     # want to allow external content like tracking pixels or even javascript. |     # want to allow external content like tracking pixels or even javascript. | ||||||
|     command: |     command: | ||||||
|       - "gotenberg" |       - "gotenberg" | ||||||
|       - "--chromium-disable-javascript=true" |       - "--chromium-disable-javascript=true" | ||||||
|       - "--chromium-allow-list=file:///tmp/.*" |       - "--chromium-allow-list=file:///tmp/.*" | ||||||
|  |  | ||||||
|   tika: |   tika: | ||||||
|     image: docker.io/apache/tika:latest |     image: docker.io/apache/tika:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
| @@ -17,16 +17,19 @@ | |||||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||||
| #   and '.env' into a folder. | #   and '.env' into a folder. | ||||||
| # - Run 'docker compose pull'. | # - Run 'docker compose pull'. | ||||||
|  | # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||||
| # - Run 'docker compose up -d'. | # - Run 'docker compose up -d'. | ||||||
| # | # | ||||||
| # For more extensive installation and update instructions, refer to the | # For more extensive installation and update instructions, refer to the | ||||||
| # documentation. | # documentation. | ||||||
|  |  | ||||||
| services: | services: | ||||||
|   broker: |   broker: | ||||||
|     image: docker.io/library/redis:8 |     image: docker.io/library/redis:7 | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
|     volumes: |     volumes: | ||||||
|       - redisdata:/data |       - redisdata:/data | ||||||
|  |  | ||||||
|   webserver: |   webserver: | ||||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest |     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||||
|     restart: unless-stopped |     restart: unless-stopped | ||||||
| @@ -42,6 +45,7 @@ services: | |||||||
|     env_file: docker-compose.env |     env_file: docker-compose.env | ||||||
|     environment: |     environment: | ||||||
|       PAPERLESS_REDIS: redis://broker:6379 |       PAPERLESS_REDIS: redis://broker:6379 | ||||||
|  |  | ||||||
| volumes: | volumes: | ||||||
|   data: |   data: | ||||||
|   media: |   media: | ||||||
|   | |||||||
							
								
								
									
										179
									
								
								docker/docker-entrypoint.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										179
									
								
								docker/docker-entrypoint.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,179 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  | # Source: https://github.com/sameersbn/docker-gitlab/ | ||||||
|  | map_uidgid() { | ||||||
|  | 	local -r usermap_original_uid=$(id -u paperless) | ||||||
|  | 	local -r usermap_original_gid=$(id -g paperless) | ||||||
|  | 	local -r usermap_new_uid=${USERMAP_UID:-$usermap_original_uid} | ||||||
|  | 	local -r usermap_new_gid=${USERMAP_GID:-${usermap_original_gid:-$usermap_new_uid}} | ||||||
|  | 	if [[ ${usermap_new_uid} != "${usermap_original_uid}" || ${usermap_new_gid} != "${usermap_original_gid}" ]]; then | ||||||
|  | 		echo "Mapping UID and GID for paperless:paperless to $usermap_new_uid:$usermap_new_gid" | ||||||
|  | 		usermod --non-unique --uid "${usermap_new_uid}" paperless | ||||||
|  | 		groupmod --non-unique --gid "${usermap_new_gid}" paperless | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | map_folders() { | ||||||
|  | 	# Export these so they can be used in docker-prepare.sh | ||||||
|  | 	export DATA_DIR="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" | ||||||
|  | 	export MEDIA_ROOT_DIR="${PAPERLESS_MEDIA_ROOT:-/usr/src/paperless/media}" | ||||||
|  | 	export CONSUME_DIR="${PAPERLESS_CONSUMPTION_DIR:-/usr/src/paperless/consume}" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | custom_container_init() { | ||||||
|  | 	# Mostly borrowed from the LinuxServer.io base image | ||||||
|  | 	# https://github.com/linuxserver/docker-baseimage-ubuntu/tree/bionic/root/etc/cont-init.d | ||||||
|  | 	local -r custom_script_dir="/custom-cont-init.d" | ||||||
|  | 	# Tamper checking. | ||||||
|  | 	# Don't run files which are owned by anyone except root | ||||||
|  | 	# Don't run files which are writeable by others | ||||||
|  | 	if [ -d "${custom_script_dir}" ]; then | ||||||
|  | 		if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 ! -user root)" ]; then | ||||||
|  | 			echo "**** Potential tampering with custom scripts detected ****" | ||||||
|  | 			echo "**** The folder '${custom_script_dir}' must be owned by root ****" | ||||||
|  | 			return 0 | ||||||
|  | 		fi | ||||||
|  | 		if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 -perm -o+w)" ]; then | ||||||
|  | 			echo "**** The folder '${custom_script_dir}' or some of contents have write permissions for others, which is a security risk. ****" | ||||||
|  | 			echo "**** Please review the permissions and their contents to make sure they are owned by root, and can only be modified by root. ****" | ||||||
|  | 			return 0 | ||||||
|  | 		fi | ||||||
|  |  | ||||||
|  | 		# Make sure custom init directory has files in it | ||||||
|  | 		if [ -n "$(/bin/ls --almost-all "${custom_script_dir}" 2>/dev/null)" ]; then | ||||||
|  | 			echo "[custom-init] files found in ${custom_script_dir} executing" | ||||||
|  | 			# Loop over files in the directory | ||||||
|  | 			for SCRIPT in "${custom_script_dir}"/*; do | ||||||
|  | 				NAME="$(basename "${SCRIPT}")" | ||||||
|  | 				if [ -f "${SCRIPT}" ]; then | ||||||
|  | 					echo "[custom-init] ${NAME}: executing..." | ||||||
|  | 					/bin/bash "${SCRIPT}" | ||||||
|  | 					echo "[custom-init] ${NAME}: exited $?" | ||||||
|  | 				elif [ ! -f "${SCRIPT}" ]; then | ||||||
|  | 					echo "[custom-init] ${NAME}: is not a file" | ||||||
|  | 				fi | ||||||
|  | 			done | ||||||
|  | 		else | ||||||
|  | 			echo "[custom-init] no custom files found exiting..." | ||||||
|  | 		fi | ||||||
|  |  | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | initialize() { | ||||||
|  |  | ||||||
|  | 	# Setup environment from secrets before anything else | ||||||
|  | 	# Check for a version of this var with _FILE appended | ||||||
|  | 	# and convert the contents to the env var value | ||||||
|  | 	# Source it so export is persistent | ||||||
|  | 	# shellcheck disable=SC1091 | ||||||
|  | 	source /sbin/env-from-file.sh | ||||||
|  |  | ||||||
|  | 	# Change the user and group IDs if needed | ||||||
|  | 	map_uidgid | ||||||
|  |  | ||||||
|  | 	# Check for overrides of certain folders | ||||||
|  | 	map_folders | ||||||
|  |  | ||||||
|  | 	local -r export_dir="/usr/src/paperless/export" | ||||||
|  |  | ||||||
|  | 	for dir in \ | ||||||
|  | 		"${export_dir}" \ | ||||||
|  | 		"${DATA_DIR}" "${DATA_DIR}/index" \ | ||||||
|  | 		"${MEDIA_ROOT_DIR}" "${MEDIA_ROOT_DIR}/documents" "${MEDIA_ROOT_DIR}/documents/originals" "${MEDIA_ROOT_DIR}/documents/thumbnails" \ | ||||||
|  | 		"${CONSUME_DIR}"; do | ||||||
|  | 		if [[ ! -d "${dir}" ]]; then | ||||||
|  | 			echo "Creating directory ${dir}" | ||||||
|  | 			mkdir --parents --verbose "${dir}" | ||||||
|  | 		fi | ||||||
|  | 	done | ||||||
|  |  | ||||||
|  | 	local -r tmp_dir="${PAPERLESS_SCRATCH_DIR:=/tmp/paperless}" | ||||||
|  | 	echo "Creating directory scratch directory ${tmp_dir}" | ||||||
|  | 	mkdir --parents --verbose "${tmp_dir}" | ||||||
|  |  | ||||||
|  | 	set +e | ||||||
|  | 	echo "Adjusting permissions of paperless files. This may take a while." | ||||||
|  | 	chown -R paperless:paperless "${tmp_dir}" | ||||||
|  | 	for dir in \ | ||||||
|  | 		"${export_dir}" \ | ||||||
|  | 		"${DATA_DIR}" \ | ||||||
|  | 		"${MEDIA_ROOT_DIR}" \ | ||||||
|  | 		"${CONSUME_DIR}"; do | ||||||
|  | 		find "${dir}" -not \( -user paperless -and -group paperless \) -exec chown --changes paperless:paperless {} + | ||||||
|  | 	done | ||||||
|  | 	set -e | ||||||
|  |  | ||||||
|  | 	"${gosu_cmd[@]}" /sbin/docker-prepare.sh | ||||||
|  |  | ||||||
|  | 	# Leave this last thing | ||||||
|  | 	custom_container_init | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | install_languages() { | ||||||
|  | 	echo "Installing languages..." | ||||||
|  |  | ||||||
|  | 	read -ra langs <<<"$1" | ||||||
|  |  | ||||||
|  | 	# Check that it is not empty | ||||||
|  | 	if [ ${#langs[@]} -eq 0 ]; then | ||||||
|  | 		return | ||||||
|  | 	fi | ||||||
|  |  | ||||||
|  | 	# Build list of packages to install | ||||||
|  | 	to_install=() | ||||||
|  | 	for lang in "${langs[@]}"; do | ||||||
|  | 		pkg="tesseract-ocr-$lang" | ||||||
|  |  | ||||||
|  | 		if dpkg --status "$pkg" &>/dev/null; then | ||||||
|  | 			echo "Package $pkg already installed!" | ||||||
|  | 			continue | ||||||
|  | 		else | ||||||
|  | 			to_install+=("$pkg") | ||||||
|  | 		fi | ||||||
|  | 	done | ||||||
|  |  | ||||||
|  | 	# Use apt only when we install packages | ||||||
|  | 	if [ ${#to_install[@]} -gt 0 ]; then | ||||||
|  | 		apt-get update | ||||||
|  |  | ||||||
|  | 		for pkg in "${to_install[@]}"; do | ||||||
|  |  | ||||||
|  | 			if ! apt-cache show "$pkg" &>/dev/null; then | ||||||
|  | 				echo "Skipped $pkg: Package not found! :(" | ||||||
|  | 				continue | ||||||
|  | 			fi | ||||||
|  |  | ||||||
|  | 			echo "Installing package $pkg..." | ||||||
|  | 			if ! apt-get --assume-yes install "$pkg" &>/dev/null; then | ||||||
|  | 				echo "Could not install $pkg" | ||||||
|  | 				exit 1 | ||||||
|  | 			fi | ||||||
|  | 		done | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | echo "Paperless-ngx docker container starting..." | ||||||
|  |  | ||||||
|  | gosu_cmd=(gosu paperless) | ||||||
|  | if [ "$(id --user)" == "$(id --user paperless)" ]; then | ||||||
|  | 	gosu_cmd=() | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | # Install additional languages if specified | ||||||
|  | if [[ -n "$PAPERLESS_OCR_LANGUAGES" ]]; then | ||||||
|  | 	install_languages "$PAPERLESS_OCR_LANGUAGES" | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | initialize | ||||||
|  |  | ||||||
|  | if [[ "$1" != "/"* ]]; then | ||||||
|  | 	echo Executing management command "$@" | ||||||
|  | 	exec "${gosu_cmd[@]}" python3 manage.py "$@" | ||||||
|  | else | ||||||
|  | 	echo Executing "$@" | ||||||
|  | 	exec "$@" | ||||||
|  | fi | ||||||
							
								
								
									
										120
									
								
								docker/docker-prepare.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										120
									
								
								docker/docker-prepare.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,120 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | set -e | ||||||
|  |  | ||||||
|  | wait_for_postgres() { | ||||||
|  | 	local attempt_num=1 | ||||||
|  | 	local -r max_attempts=5 | ||||||
|  |  | ||||||
|  | 	echo "Waiting for PostgreSQL to start..." | ||||||
|  |  | ||||||
|  | 	local -r host="${PAPERLESS_DBHOST:-localhost}" | ||||||
|  | 	local -r port="${PAPERLESS_DBPORT:-5432}" | ||||||
|  |  | ||||||
|  | 	# Disable warning, host and port can't have spaces | ||||||
|  | 	# shellcheck disable=SC2086 | ||||||
|  | 	while [ ! "$(pg_isready --host ${host} --port ${port})" ]; do | ||||||
|  |  | ||||||
|  | 		if [ $attempt_num -eq $max_attempts ]; then | ||||||
|  | 			echo "Unable to connect to database." | ||||||
|  | 			exit 1 | ||||||
|  | 		else | ||||||
|  | 			echo "Attempt $attempt_num failed! Trying again in 5 seconds..." | ||||||
|  | 		fi | ||||||
|  |  | ||||||
|  | 		attempt_num=$(("$attempt_num" + 1)) | ||||||
|  | 		sleep 5 | ||||||
|  | 	done | ||||||
|  | 	echo "Connected to PostgreSQL" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | wait_for_mariadb() { | ||||||
|  | 	echo "Waiting for MariaDB to start..." | ||||||
|  |  | ||||||
|  | 	local -r host="${PAPERLESS_DBHOST:=localhost}" | ||||||
|  | 	local -r port="${PAPERLESS_DBPORT:=3306}" | ||||||
|  |  | ||||||
|  | 	local attempt_num=1 | ||||||
|  | 	local -r max_attempts=5 | ||||||
|  |  | ||||||
|  | 	# Disable warning, host and port can't have spaces | ||||||
|  | 	# shellcheck disable=SC2086 | ||||||
|  | 	while ! true > /dev/tcp/$host/$port; do | ||||||
|  |  | ||||||
|  | 		if [ $attempt_num -eq $max_attempts ]; then | ||||||
|  | 			echo "Unable to connect to database." | ||||||
|  | 			exit 1 | ||||||
|  | 		else | ||||||
|  | 			echo "Attempt $attempt_num failed! Trying again in 5 seconds..." | ||||||
|  |  | ||||||
|  | 		fi | ||||||
|  |  | ||||||
|  | 		attempt_num=$(("$attempt_num" + 1)) | ||||||
|  | 		sleep 5 | ||||||
|  | 	done | ||||||
|  | 	echo "Connected to MariaDB" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | wait_for_redis() { | ||||||
|  | 	# We use a Python script to send the Redis ping | ||||||
|  | 	# instead of installing redis-tools just for 1 thing | ||||||
|  | 	if ! python3 /sbin/wait-for-redis.py; then | ||||||
|  | 		exit 1 | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | migrations() { | ||||||
|  | 	( | ||||||
|  | 		# flock is in place to prevent multiple containers from doing migrations | ||||||
|  | 		# simultaneously. This also ensures that the db is ready when the command | ||||||
|  | 		# of the current container starts. | ||||||
|  | 		flock 200 | ||||||
|  | 		echo "Apply database migrations..." | ||||||
|  | 		python3 manage.py migrate --skip-checks --no-input | ||||||
|  | 	) 200>"${DATA_DIR}/migration_lock" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | django_checks() { | ||||||
|  | 	# Explicitly run the Django system checks | ||||||
|  | 	echo "Running Django checks" | ||||||
|  | 	python3 manage.py check | ||||||
|  | } | ||||||
|  |  | ||||||
|  | search_index() { | ||||||
|  |  | ||||||
|  | 	local -r index_version=9 | ||||||
|  | 	local -r index_version_file=${DATA_DIR}/.index_version | ||||||
|  |  | ||||||
|  | 	if [[ (! -f "${index_version_file}") || $(<"${index_version_file}") != "$index_version" ]]; then | ||||||
|  | 		echo "Search index out of date. Updating..." | ||||||
|  | 		python3 manage.py document_index reindex --no-progress-bar | ||||||
|  | 		echo ${index_version} | tee "${index_version_file}" >/dev/null | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | superuser() { | ||||||
|  | 	if [[ -n "${PAPERLESS_ADMIN_USER}" ]]; then | ||||||
|  | 		python3 manage.py manage_superuser | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | do_work() { | ||||||
|  | 	if [[ "${PAPERLESS_DBENGINE}" == "mariadb" ]]; then | ||||||
|  | 		wait_for_mariadb | ||||||
|  | 	elif [[ -n "${PAPERLESS_DBHOST}" ]]; then | ||||||
|  | 		wait_for_postgres | ||||||
|  | 	fi | ||||||
|  |  | ||||||
|  | 	wait_for_redis | ||||||
|  |  | ||||||
|  | 	migrations | ||||||
|  |  | ||||||
|  | 	django_checks | ||||||
|  |  | ||||||
|  | 	search_index | ||||||
|  |  | ||||||
|  | 	superuser | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | do_work | ||||||
							
								
								
									
										42
									
								
								docker/env-from-file.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								docker/env-from-file.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | # Scans the environment variables for those with the suffix _FILE | ||||||
|  | # When located, checks the file exists, and exports the contents | ||||||
|  | # of the file as the same name, minus the suffix | ||||||
|  | # This allows the use of Docker secrets or mounted files | ||||||
|  | # to fill in any of the settings configurable via environment | ||||||
|  | # variables | ||||||
|  |  | ||||||
|  | set -eu | ||||||
|  |  | ||||||
|  | for line in $(printenv) | ||||||
|  | do | ||||||
|  | 	# Extract the name of the environment variable | ||||||
|  | 	env_name=${line%%=*} | ||||||
|  | 	# Check if it starts with "PAPERLESS_" and ends in "_FILE" | ||||||
|  | 	if [[ ${env_name} == PAPERLESS_*_FILE ]]; then | ||||||
|  | 		# This should have been named different.. | ||||||
|  | 		if [[ ${env_name} == "PAPERLESS_OCR_SKIP_ARCHIVE_FILE" || ${env_name} == "PAPERLESS_MODEL_FILE" ]]; then | ||||||
|  | 			continue | ||||||
|  | 		fi | ||||||
|  | 		# Extract the value of the environment | ||||||
|  | 		env_value=${line#*=} | ||||||
|  |  | ||||||
|  | 		# Check the file exists | ||||||
|  | 		if [[ -f ${env_value} ]]; then | ||||||
|  |  | ||||||
|  | 			# Trim off the _FILE suffix | ||||||
|  | 			non_file_env_name=${env_name%"_FILE"} | ||||||
|  | 			echo "Setting ${non_file_env_name} from file" | ||||||
|  |  | ||||||
|  | 			# Reads the value from th file | ||||||
|  | 			val="$(< "${!env_name}")" | ||||||
|  |  | ||||||
|  | 			# Sets the normal name to the read file contents | ||||||
|  | 			export "${non_file_env_name}"="${val}" | ||||||
|  |  | ||||||
|  | 		else | ||||||
|  | 			echo "File ${env_value} referenced by ${env_name} doesn't exist" | ||||||
|  | 		fi | ||||||
|  | 	fi | ||||||
|  | done | ||||||
							
								
								
									
										12
									
								
								docker/flower-conditional.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								docker/flower-conditional.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | echo "Checking if we should start flower..." | ||||||
|  |  | ||||||
|  | if [[ -n  "${PAPERLESS_ENABLE_FLOWER}" ]]; then | ||||||
|  | 	# Small delay to allow celery to be up first | ||||||
|  | 	echo "Starting flower in 5s" | ||||||
|  | 	sleep 5 | ||||||
|  | 	celery --app paperless flower --conf=/usr/src/paperless/src/paperless/flowerconfig.py | ||||||
|  | else | ||||||
|  | 	echo "Not starting flower" | ||||||
|  | fi | ||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 30 KiB | 
| @@ -1,7 +1,5 @@ | |||||||
| #!/usr/bin/env bash | #!/usr/bin/env bash | ||||||
|  |  | ||||||
| # Run this script to generate the management commands again (for example if a new command is create or the template is updated) |  | ||||||
|  |  | ||||||
| set -eu | set -eu | ||||||
|  |  | ||||||
| for command in decrypt_documents \ | for command in decrypt_documents \ | ||||||
| @@ -17,11 +15,9 @@ for command in decrypt_documents \ | |||||||
| 	document_sanity_checker \ | 	document_sanity_checker \ | ||||||
| 	document_fuzzy_match \ | 	document_fuzzy_match \ | ||||||
| 	manage_superuser \ | 	manage_superuser \ | ||||||
| 	convert_mariadb_uuid \ | 	convert_mariadb_uuid; | ||||||
| 	prune_audit_logs \ |  | ||||||
| 	createsuperuser; |  | ||||||
| do | do | ||||||
| 	echo "installing $command..." | 	echo "installing $command..." | ||||||
| 	sed "s/management_command/$command/g" management_script.sh >"$PWD/rootfs/usr/local/bin/$command" | 	sed "s/management_command/$command/g" management_script.sh > /usr/local/bin/$command | ||||||
| 	chmod u=rwx,g=rwx,o=rx "$PWD/rootfs/usr/local/bin/$command" | 	chmod +x /usr/local/bin/$command | ||||||
| done | done | ||||||
|   | |||||||
| @@ -1,13 +1,17 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash | #!/usr/bin/env bash | ||||||
| # shellcheck shell=bash |  | ||||||
|  |  | ||||||
| set -e | set -e | ||||||
|  |  | ||||||
| cd "${PAPERLESS_SRC_DIR}" | cd /usr/src/paperless/src/ | ||||||
|  | # This ensures environment is setup | ||||||
|  | # shellcheck disable=SC1091 | ||||||
|  | source /sbin/env-from-file.sh | ||||||
|  |  | ||||||
| if [[ $(id -u) == 0 ]]; then | if [[ $(id -u) == 0 ]] ; | ||||||
| 	s6-setuidgid paperless python3 manage.py management_command "$@" | then | ||||||
| elif [[ $(id -un) == "paperless" ]]; then | 	gosu paperless python3 manage.py management_command "$@" | ||||||
|  | elif [[ $(id -un) == "paperless" ]] ; | ||||||
|  | then | ||||||
| 	python3 manage.py management_command "$@" | 	python3 manage.py management_command "$@" | ||||||
| else | else | ||||||
| 	echo "Unknown user." | 	echo "Unknown user." | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								docker/paperless_cmd.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										16
									
								
								docker/paperless_cmd.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  |  | ||||||
|  | SUPERVISORD_WORKING_DIR="${PAPERLESS_SUPERVISORD_WORKING_DIR:-$PWD}" | ||||||
|  | rootless_args=() | ||||||
|  | if [ "$(id -u)" == "$(id -u paperless)" ]; then | ||||||
|  | 	rootless_args=( | ||||||
|  | 		--user | ||||||
|  | 		paperless | ||||||
|  | 		--logfile | ||||||
|  | 		"${SUPERVISORD_WORKING_DIR}/supervisord.log" | ||||||
|  | 		--pidfile | ||||||
|  | 		"${SUPERVISORD_WORKING_DIR}/supervisord.pid" | ||||||
|  | 	) | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | exec /usr/local/bin/supervisord -c /etc/supervisord.conf "${rootless_args[@]}" | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
| declare -r log_prefix="[init-complete]" |  | ||||||
| declare -r end_time=$(date +%s) |  | ||||||
| declare -r start_time=${PAPERLESS_START_TIME_S} |  | ||||||
|  |  | ||||||
| echo "${log_prefix} paperless-ngx docker container init completed in $(($end_time-$start_time)) seconds" |  | ||||||
| echo "${log_prefix} Starting services" |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-complete/run |  | ||||||
| @@ -1,44 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
|  |  | ||||||
| declare -r log_prefix="[custom-init]" |  | ||||||
|  |  | ||||||
| # Mostly borrowed from the LinuxServer.io base image |  | ||||||
| # https://github.com/linuxserver/docker-baseimage-ubuntu/tree/bionic/root/etc/cont-init.d |  | ||||||
| declare -r custom_script_dir="/custom-cont-init.d" |  | ||||||
|  |  | ||||||
| # Tamper checking. |  | ||||||
| # Don't run files which are owned by anyone except root |  | ||||||
| # Don't run files which are writeable by others |  | ||||||
| if [ -d "${custom_script_dir}" ]; then |  | ||||||
| 	if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 ! -user root)" ]; then |  | ||||||
| 		echo "${log_prefix} **** Potential tampering with custom scripts detected ****" |  | ||||||
| 		echo "${log_prefix} **** The folder '${custom_script_dir}' must be owned by root ****" |  | ||||||
| 		exit 0 |  | ||||||
| 	fi |  | ||||||
| 	if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 -perm -o+w)" ]; then |  | ||||||
| 		echo "${log_prefix} **** The folder '${custom_script_dir}' or some of contents have write permissions for others, which is a security risk. ****" |  | ||||||
| 		echo "${log_prefix} **** Please review the permissions and their contents to make sure they are owned by root, and can only be modified by root. ****" |  | ||||||
| 		exit 0 |  | ||||||
| 	fi |  | ||||||
|  |  | ||||||
| 	# Make sure custom init directory has files in it |  | ||||||
| 	if [ -n "$(/bin/ls --almost-all "${custom_script_dir}" 2>/dev/null)" ]; then |  | ||||||
| 		echo "${log_prefix} files found in ${custom_script_dir} executing" |  | ||||||
| 		# Loop over files in the directory |  | ||||||
| 		for SCRIPT in "${custom_script_dir}"/*; do |  | ||||||
| 			NAME="$(basename "${SCRIPT}")" |  | ||||||
| 			if [ -f "${SCRIPT}" ]; then |  | ||||||
| 				echo "${log_prefix} ${NAME}: executing..." |  | ||||||
| 				/command/with-contenv /bin/bash "${SCRIPT}" |  | ||||||
| 				echo "${log_prefix} ${NAME}: exited $?" |  | ||||||
| 			elif [ ! -f "${SCRIPT}" ]; then |  | ||||||
| 				echo "${log_prefix} ${NAME}: is not a file" |  | ||||||
| 			fi |  | ||||||
| 		done |  | ||||||
| 	else |  | ||||||
| 		echo "${log_prefix} no custom files found exiting..." |  | ||||||
| 	fi |  | ||||||
| else |  | ||||||
| 	echo "${log_prefix} ${custom_script_dir} doesn't exist, nothing to do" |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-custom-init/run |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
|  |  | ||||||
| declare -r log_prefix="[env-init]" |  | ||||||
|  |  | ||||||
| echo "${log_prefix} Checking for environment from files" |  | ||||||
|  |  | ||||||
| if find /run/s6/container_environment/*"_FILE" -maxdepth 1 > /dev/null 2>&1; then |  | ||||||
| 	for FILENAME in /run/s6/container_environment/*; do |  | ||||||
| 		if [[ "${FILENAME##*/}" == PAPERLESS_*_FILE ]]; then |  | ||||||
| 			# This should have been named different.. |  | ||||||
| 			if [[ "${FILENAME##*/}" == "PAPERLESS_OCR_SKIP_ARCHIVE_FILE" || "${FILENAME##*/}" == "PAPERLESS_MODEL_FILE" ]]; then |  | ||||||
| 				continue |  | ||||||
| 			fi |  | ||||||
| 			SECRETFILE=$(cat "${FILENAME}") |  | ||||||
| 			# Check the file exists |  | ||||||
| 			if [[ -f ${SECRETFILE} ]]; then |  | ||||||
| 				# Trim off trailing _FILE |  | ||||||
| 				FILESTRIP=${FILENAME//_FILE/} |  | ||||||
| 				if [[ $(tail -n1 "${SECRETFILE}" | wc -l) != 0 ]]; then |  | ||||||
| 					echo "${log_prefix} Your secret: ${FILENAME##*/} contains a trailing newline and may not work as expected" |  | ||||||
| 				fi |  | ||||||
| 				# Set environment variable |  | ||||||
| 				cat "${SECRETFILE}" > "${FILESTRIP}" |  | ||||||
| 				echo "${log_prefix} ${FILESTRIP##*/} set from ${FILENAME##*/}" |  | ||||||
| 			else |  | ||||||
| 				echo "${log_prefix} cannot find secret in ${FILENAME##*/}" |  | ||||||
| 			fi |  | ||||||
| 		fi |  | ||||||
| 	done |  | ||||||
| else |  | ||||||
| 		echo "${log_prefix} No *_FILE environment found" |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-env-file/run |  | ||||||
| @@ -1,65 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
|  |  | ||||||
| declare -r log_prefix="[init-folders]" |  | ||||||
|  |  | ||||||
| declare -r export_dir="/usr/src/paperless/export" |  | ||||||
| declare -r data_dir="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" |  | ||||||
| declare -r media_root_dir="${PAPERLESS_MEDIA_ROOT:-/usr/src/paperless/media}" |  | ||||||
| declare -r consume_dir="${PAPERLESS_CONSUMPTION_DIR:-/usr/src/paperless/consume}" |  | ||||||
| declare -r tmp_dir="${PAPERLESS_SCRATCH_DIR:=/tmp/paperless}" |  | ||||||
|  |  | ||||||
| declare -r main_dirs=( |  | ||||||
| 	"${export_dir}" |  | ||||||
| 	"${data_dir}" |  | ||||||
| 	"${media_root_dir}" |  | ||||||
| 	"${consume_dir}" |  | ||||||
| 	"${tmp_dir}" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| declare -r extra_dirs=( |  | ||||||
| 	"${main_dirs[@]}" |  | ||||||
| 	"${data_dir}/index" |  | ||||||
| 	"${media_root_dir}/documents" |  | ||||||
| 	"${media_root_dir}/documents/originals" |  | ||||||
| 	"${media_root_dir}/documents/thumbnails" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| if [[ -n "${USER_IS_NON_ROOT}" ]]; then |  | ||||||
| 	# Non-root mode: Create directories as current user, warn about permission issues |  | ||||||
| 	echo "${log_prefix} Running in non-root mode, checking directories" |  | ||||||
| 	current_uid=$(id --user) |  | ||||||
| 	current_gid=$(id --group) |  | ||||||
|  |  | ||||||
| 	for dir in "${extra_dirs[@]}"; do |  | ||||||
| 		if [[ ! -d "${dir}" ]]; then |  | ||||||
| 			mkdir --parents --verbose "${dir}" || echo "${log_prefix} WARNING: Could not create ${dir} - permission denied" |  | ||||||
| 		fi |  | ||||||
| 		# Check permissions on existing directories too |  | ||||||
| 		if [[ -d "${dir}" && ! -w "${dir}" ]]; then |  | ||||||
| 			echo "${log_prefix} WARNING: No write permission to ${dir}" |  | ||||||
| 		fi |  | ||||||
| 	done |  | ||||||
|  |  | ||||||
| 	# Warn about ownership issues |  | ||||||
| 	for dir in "${main_dirs[@]}"; do |  | ||||||
| 		if [[ -d "${dir}" ]]; then |  | ||||||
| 			find "${dir}" -not \( -user ${current_uid} -and -group ${current_gid} \) -exec echo "${log_prefix} WARNING: Permission issue on {}: not owned by current user (${current_uid}:${current_gid})" \; 2>/dev/null || echo "${log_prefix} WARNING: Cannot check permissions on ${dir}" |  | ||||||
| 		fi |  | ||||||
| 	done |  | ||||||
| else |  | ||||||
| 	# Root mode: Create and fix permissions as needed |  | ||||||
| 	echo "${log_prefix} Running with root privileges, adjusting directories and permissions" |  | ||||||
|  |  | ||||||
| 	# First create directories |  | ||||||
| 	for dir in "${extra_dirs[@]}"; do |  | ||||||
| 		if [[ ! -d "${dir}" ]]; then |  | ||||||
| 			mkdir --parents --verbose "${dir}" |  | ||||||
| 		fi |  | ||||||
| 	done |  | ||||||
|  |  | ||||||
| 	# Then fix permissions on all directories |  | ||||||
| 	for dir in "${main_dirs[@]}"; do |  | ||||||
| 		find "${dir}" -not \( -user paperless -and -group paperless \) -exec chown --changes paperless:paperless {} + |  | ||||||
| 	done |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-folders/run |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
| declare -r log_prefix="[init-migrations]" |  | ||||||
|  |  | ||||||
| declare -r data_dir="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" |  | ||||||
|  |  | ||||||
| echo "${log_prefix} Apply database migrations..." |  | ||||||
|  |  | ||||||
| cd "${PAPERLESS_SRC_DIR}" |  | ||||||
|  |  | ||||||
| # The whole migrate, with flock, needs to run as the right user |  | ||||||
| if [[ -n "${USER_IS_NON_ROOT}" ]]; then |  | ||||||
| 	exec s6-setlock -n "${data_dir}/migration_lock" python3 manage.py migrate --skip-checks --no-input |  | ||||||
| else |  | ||||||
| 	exec s6-setuidgid paperless \ |  | ||||||
| 		s6-setlock -n "${data_dir}/migration_lock" \ |  | ||||||
| 		python3 manage.py migrate --skip-checks --no-input |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-migrations/run |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
| declare -r log_prefix="[init-user]" |  | ||||||
|  |  | ||||||
| declare -r usermap_original_uid=$(id -u paperless) |  | ||||||
| declare -r usermap_original_gid=$(id -g paperless) |  | ||||||
| declare -r usermap_new_uid=${USERMAP_UID:-$usermap_original_uid} |  | ||||||
| declare -r usermap_new_gid=${USERMAP_GID:-${usermap_original_gid:-$usermap_new_uid}} |  | ||||||
|  |  | ||||||
| if [[ ${usermap_new_uid} != "${usermap_original_uid}" ]]; then |  | ||||||
| 	echo "${log_prefix} Mapping UID for paperless to $usermap_new_uid" |  | ||||||
| 	usermod --non-unique --uid "${usermap_new_uid}" paperless |  | ||||||
| else |  | ||||||
| 	echo "${log_prefix} No UID changes for paperless" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if [[ ${usermap_new_gid} != "${usermap_original_gid}" ]]; then |  | ||||||
| 	echo "${log_prefix} Mapping GID for paperless to $usermap_new_gid" |  | ||||||
| 	groupmod --non-unique --gid "${usermap_new_gid}" paperless |  | ||||||
| else |  | ||||||
| 	echo "${log_prefix} No GID changes for paperless" |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-modify-user/run |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| #!/command/with-contenv /usr/bin/bash |  | ||||||
| # shellcheck shell=bash |  | ||||||
|  |  | ||||||
| declare -r log_prefix="[init-index]" |  | ||||||
|  |  | ||||||
| declare -r index_version=9 |  | ||||||
| declare -r data_dir="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" |  | ||||||
| declare -r index_version_file="${data_dir}/.index_version" |  | ||||||
|  |  | ||||||
| update_index () { |  | ||||||
| 	echo "${log_prefix} Search index out of date. Updating..." |  | ||||||
| 	cd "${PAPERLESS_SRC_DIR}" |  | ||||||
| 	if [[ -n "${USER_IS_NON_ROOT}" ]]; then |  | ||||||
| 		python3 manage.py document_index reindex --no-progress-bar |  | ||||||
| 		echo ${index_version} | tee "${index_version_file}" > /dev/null |  | ||||||
| 	else |  | ||||||
| 		s6-setuidgid paperless python3 manage.py document_index reindex --no-progress-bar |  | ||||||
| 		echo ${index_version} | s6-setuidgid paperless tee "${index_version_file}" > /dev/null |  | ||||||
| 	fi |  | ||||||
| } |  | ||||||
|  |  | ||||||
| if [[ (! -f "${index_version_file}") ]]; then |  | ||||||
| 	echo "${log_prefix} No index version file found" |  | ||||||
| 	update_index |  | ||||||
| elif [[ $(<"${index_version_file}") != "$index_version" ]]; then |  | ||||||
| 	echo "${log_prefix} index version updated" |  | ||||||
| 	update_index |  | ||||||
| fi |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| oneshot |  | ||||||
| @@ -1 +0,0 @@ | |||||||
| /etc/s6-overlay/s6-rc.d/init-search-index/run |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user