[project] name = "paperless-ngx" version = "2.19.1" description = "A community-supported supercharged document management system: scan, index and archive all your physical documents" readme = "README.md" requires-python = ">=3.10" classifiers = [ "Programming Language :: Python :: 3 :: Only", "Programming Language :: Python :: 3.10", "Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.12", "Programming Language :: Python :: 3.13", "Programming Language :: Python :: 3.14", ] # TODO: Move certain things to groups and then utilize that further # This will allow testing to not install a webserver, mysql, etc dependencies = [ "babel>=2.17", "bleach~=6.2.0", "celery[redis]~=5.5.1", "channels~=4.2", "channels-redis~=4.2", "concurrent-log-handler~=0.9.25", "dateparser~=1.2", # WARNING: django does not use semver. # Only patch versions are guaranteed to not introduce breaking changes. "django~=5.2.5", "django-allauth[mfa,socialaccount]~=65.4.0", "django-auditlog~=3.2.1", "django-cachalot~=2.8.0", "django-celery-results~=2.6.0", "django-compression-middleware~=0.5.0", "django-cors-headers~=4.9.0", "django-extensions~=4.1", "django-filter~=25.1", "django-guardian~=3.2.0", "django-multiselectfield~=1.0.1", "django-soft-delete~=1.0.18", "django-treenode>=0.23.2", "djangorestframework~=3.16", "djangorestframework-guardian~=0.4.0", "drf-spectacular~=0.28", "drf-spectacular-sidecar~=2025.9.1", "drf-writable-nested~=0.7.1", "filelock~=3.20.0", "flower~=2.0.1", "gotenberg-client~=0.12.0", "httpx-oauth~=0.16", "imap-tools~=1.11.0", "inotifyrecursive~=0.3", "jinja2~=3.1.5", "langdetect~=1.0.9", "nltk~=3.9.1", "ocrmypdf~=16.11.0", "pathvalidate~=3.3.1", "pdf2image~=1.17.0", "python-dateutil~=2.9.0", "python-dotenv~=1.1.0", "python-gnupg~=0.5.4", "python-ipware~=3.0.0", "python-magic~=0.4.27", "pyzbar~=0.1.9", "rapidfuzz~=3.14.0", "redis[hiredis]~=5.2.1", "scikit-learn~=1.7.0", "setproctitle~=1.3.4", "tika-client~=0.10.0", "tqdm~=4.67.1", "watchdog~=6.0", "whitenoise~=6.9", "whoosh-reloaded>=2.7.5", "zxing-cpp~=2.3.0", ] optional-dependencies.mariadb = [ "mysqlclient~=2.2.7", ] optional-dependencies.postgres = [ "psycopg[c,pool]==3.2.9", # Direct dependency for proper resolution of the pre-built wheels "psycopg-c==3.2.9", "psycopg-pool==3.2.6", ] optional-dependencies.webserver = [ "granian[uvloop]~=2.5.1", ] [dependency-groups] dev = [ { "include-group" = "docs" }, { "include-group" = "testing" }, { "include-group" = "lint" }, ] docs = [ "mkdocs-glightbox~=0.5.1", "mkdocs-material~=9.6.4", ] testing = [ "daphne", "factory-boy~=3.3.1", "imagehash", "pytest~=8.4.1", "pytest-cov~=7.0.0", "pytest-django~=4.11.1", "pytest-env", "pytest-httpx", "pytest-mock", "pytest-rerunfailures", "pytest-sugar", "pytest-xdist", ] lint = [ "pre-commit~=4.3.0", "pre-commit-uv~=4.2.0", "ruff~=0.14.0", ] typing = [ "celery-types", "django-filter-stubs", "django-stubs[compatible-mypy]", "djangorestframework-stubs[compatible-mypy]", "lxml-stubs", "mypy", "types-bleach", "types-colorama", "types-dateparser", "types-markdown", "types-pygments", "types-python-dateutil", "types-pytz", "types-redis", "types-setuptools", "types-tqdm", ] [tool.uv] required-version = ">=0.5.14" package = false environments = [ "sys_platform == 'darwin'", "sys_platform == 'linux'", ] [tool.uv.sources] # Markers are chosen to select these almost exclusively when building the Docker image psycopg-c = [ { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.9/psycopg_c-3.2.9-cp312-cp312-linux_x86_64.whl", marker = "sys_platform == 'linux' and platform_machine == 'x86_64' and python_version == '3.12'" }, { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.9/psycopg_c-3.2.9-cp312-cp312-linux_aarch64.whl", marker = "sys_platform == 'linux' and platform_machine == 'aarch64' and python_version == '3.12'" }, ] zxing-cpp = [ { url = "https://github.com/paperless-ngx/builder/releases/download/zxing-2.3.0/zxing_cpp-2.3.0-cp312-cp312-linux_x86_64.whl", marker = "sys_platform == 'linux' and platform_machine == 'x86_64' and python_version == '3.12'" }, { url = "https://github.com/paperless-ngx/builder/releases/download/zxing-2.3.0/zxing_cpp-2.3.0-cp312-cp312-linux_aarch64.whl", marker = "sys_platform == 'linux' and platform_machine == 'aarch64' and python_version == '3.12'" }, ] [tool.ruff] target-version = "py310" line-length = 88 src = [ "src", ] respect-gitignore = true # https://docs.astral.sh/ruff/settings/ fix = true show-fixes = true output-format = "grouped" # https://docs.astral.sh/ruff/rules/ lint.extend-select = [ "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 "FBT", # https://docs.astral.sh/ruff/rules/#flake8-boolean-trap-fbt "FLY", # https://docs.astral.sh/ruff/rules/#flynt-fly "G201", # https://docs.astral.sh/ruff/rules/#flake8-logging-format-g "I", # https://docs.astral.sh/ruff/rules/#isort-i "ICN", # https://docs.astral.sh/ruff/rules/#flake8-import-conventions-icn "INP", # https://docs.astral.sh/ruff/rules/#flake8-no-pep420-inp "ISC", # https://docs.astral.sh/ruff/rules/#flake8-implicit-str-concat-isc "PIE", # https://docs.astral.sh/ruff/rules/#flake8-pie-pie "PLC", # https://docs.astral.sh/ruff/rules/#pylint-pl "PLE", # https://docs.astral.sh/ruff/rules/#pylint-pl "PTH", # https://docs.astral.sh/ruff/rules/#flake8-use-pathlib-pth "Q", # https://docs.astral.sh/ruff/rules/#flake8-quotes-q "RSE", # https://docs.astral.sh/ruff/rules/#flake8-raise-rse "RUF", # https://docs.astral.sh/ruff/rules/#ruff-specific-rules-ruf "SIM", # https://docs.astral.sh/ruff/rules/#flake8-simplify-sim "T20", # https://docs.astral.sh/ruff/rules/#flake8-print-t20 "TC", # https://docs.astral.sh/ruff/rules/#flake8-type-checking-tc "TID", # https://docs.astral.sh/ruff/rules/#flake8-tidy-imports-tid "UP", # https://docs.astral.sh/ruff/rules/#pyupgrade-up "W", # https://docs.astral.sh/ruff/rules/#pycodestyle-e-w ] lint.ignore = [ "DJ001", "PLC0415", "RUF012", "SIM105", ] # Migrations lint.per-file-ignores."*/migrations/*.py" = [ "E501", "SIM", "T201", ] # Testing lint.per-file-ignores."*/tests/*.py" = [ "E501", "SIM117", ] lint.per-file-ignores.".github/scripts/*.py" = [ "E501", "INP001", "SIM117", ] # Docker specific lint.per-file-ignores."docker/rootfs/usr/local/bin/wait-for-redis.py" = [ "INP001", "T201", ] lint.per-file-ignores."docker/wait-for-redis.py" = [ "INP001", "T201", ] lint.per-file-ignores."src/documents/models.py" = [ "SIM115", ] lint.per-file-ignores."src/paperless_tesseract/tests/test_parser.py" = [ "RUF001", ] lint.isort.force-single-line = true [tool.codespell] write-changes = true ignore-words-list = "criterias,afterall,valeu,ureue,equest,ure,assertIn,Oktober" skip = "src-ui/src/locale/*,src-ui/pnpm-lock.yaml,src-ui/e2e/*,src/paperless_mail/tests/samples/*,src/documents/tests/samples/*,*.po,*.json" [tool.pytest.ini_options] minversion = "8.0" pythonpath = [ "src", ] testpaths = [ "src/documents/tests/", "src/paperless/tests/", "src/paperless_mail/tests/", "src/paperless_tesseract/tests/", "src/paperless_tika/tests", "src/paperless_text/tests/", ] addopts = [ "--pythonwarnings=all", "--cov", "--cov-report=html", "--cov-report=xml", "--numprocesses=auto", "--maxprocesses=16", "--quiet", "--durations=50", "--junitxml=junit.xml", "-o junit_family=legacy", ] norecursedirs = [ "src/locale/", ".venv/", "src-ui/" ] DJANGO_SETTINGS_MODULE = "paperless.settings" [tool.pytest_env] PAPERLESS_DISABLE_DBHANDLER = "true" PAPERLESS_CACHE_BACKEND = "django.core.cache.backends.locmem.LocMemCache" [tool.coverage.run] source = [ "src/", ] omit = [ "*/tests/*", "manage.py", "paperless/wsgi.py", "paperless/auth.py", ] [tool.coverage.report] exclude_also = [ "if settings.AUDIT_LOG_ENABLED:", "if AUDIT_LOG_ENABLED:", "if TYPE_CHECKING:", ] [tool.mypy] mypy_path = "src" plugins = [ "mypy_django_plugin.main", "mypy_drf_plugin.main", ] check_untyped_defs = true disallow_any_generics = true disallow_incomplete_defs = true disallow_untyped_defs = true warn_redundant_casts = true warn_unused_ignores = true [tool.django-stubs] django_settings_module = "paperless.settings"