mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			41 Commits
		
	
	
		
			feature-tr
			...
			1de7c52478
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1de7c52478 | ||
|   | 9aaaa6f069 | ||
|   | c3a20b7797 | ||
|   | 476556379b | ||
|   | e5cafff043 | ||
|   | 8e0d574e99 | ||
|   | 8a5820328e | ||
|   | 809d62a2f4 | ||
|   | 0d87f94b9b | ||
|   | 315b90f8e5 | ||
|   | 47b2d2964b | ||
|   | e05639ae4e | ||
|   | f400a8cb2f | ||
|   | 26abcf5612 | ||
|   | afde52430d | ||
|   | 716f2da652 | ||
|   | c54073b7c2 | ||
|   | 247e6f39dc | ||
|   | 1e6dfc4481 | ||
|   | 7cc0750066 | ||
|   | bd6585d3b4 | ||
|   | 717e828a1d | ||
|   | 07381d48e6 | ||
|   | dd0ffaf312 | ||
|   | 264504affc | ||
|   | 4feedf2add | ||
|   | 2f76cf9831 | ||
|   | 1002d37f6b | ||
|   | d260a94740 | ||
|   | 88c69b83ea | ||
|   | 2557ee2014 | ||
|   | 3c75deed80 | ||
|   | d05343c927 | ||
|   | e7972b7eaf | ||
|   | 75a091cc0d | ||
|   | dca74803fd | ||
|   | 3cf3d868d0 | ||
|   | bf4fc6604a | ||
|   | e8c1eb86fa | ||
|   | c3dad3cf69 | ||
|   | 811bd66088 | 
							
								
								
									
										7
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,9 +6,10 @@ | ||||
| # This workflow will not trigger runs on forked repos. | ||||
| name: Cleanup Image Tags | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: '0 0 * * 0' | ||||
|   delete: | ||||
|   push: | ||||
|     paths: | ||||
|       - ".github/workflows/cleanup-tags.yml" | ||||
| concurrency: | ||||
|   group: registry-tags-cleanup | ||||
|   cancel-in-progress: false | ||||
|   | ||||
							
								
								
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -5,7 +5,7 @@ | ||||
| # Purpose: Compiles the frontend | ||||
| # Notes: | ||||
| #  - Does PNPM stuff with Typescript and such | ||||
| FROM --platform=$BUILDPLATFORM docker.io/node:20-trixie-slim AS compile-frontend | ||||
| FROM --platform=$BUILDPLATFORM docker.io/node:20-bookworm-slim AS compile-frontend | ||||
|  | ||||
| COPY ./src-ui /src/src-ui | ||||
|  | ||||
| @@ -170,8 +170,20 @@ RUN set -eux \ | ||||
|     && apt-get update \ | ||||
|     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} \ | ||||
|     && echo "Installing pre-built updates" \ | ||||
|       && curl --fail --silent --no-progress-meter --show-error --location --remote-name-all \ | ||||
|         https://github.com/paperless-ngx/builder/releases/download/jbig2enc-v${JBIG2ENC_VERSION}/jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||
|       && 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}" \ | ||||
|         && dpkg --install ./libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|       && echo "Installing Ghostscript ${GS_VERSION}" \ | ||||
|         && dpkg --install ./libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ | ||||
|         && dpkg --install ./libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||
|       && echo "Installing jbig2enc" \ | ||||
|         && dpkg --install ./jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||
|       && echo "Configuring imagemagick" \ | ||||
|   | ||||
							
								
								
									
										319
									
								
								dev.txt
									
									
									
									
									
								
							
							
						
						
									
										319
									
								
								dev.txt
									
									
									
									
									
								
							| @@ -1,319 +0,0 @@ | ||||
| adduser 3.134 | ||||
| apt 2.6.1 | ||||
| base-files 12.4+deb12u11 | ||||
| base-passwd 3.6.1 | ||||
| bash 5.2.15-2+b8 | ||||
| bsdutils 1:2.38.1-5+deb12u3 | ||||
| ca-certificates 20230311+deb12u1 | ||||
| coreutils 9.1-1 | ||||
| curl 7.88.1-10+deb12u12 | ||||
| dash 0.5.12-2 | ||||
| debconf 1.5.82 | ||||
| debian-archive-keyring 2023.3+deb12u2 | ||||
| debianutils 5.7-0.5~deb12u1 | ||||
| diffutils 1:3.8-4 | ||||
| dirmngr 2.2.40-1.1 | ||||
| dpkg 1.21.22 | ||||
| e2fsprogs 1.47.0-2 | ||||
| file 1:5.44-3 | ||||
| findutils 4.9.0-4 | ||||
| fontconfig 2.14.1-4 | ||||
| fontconfig-config 2.14.1-4 | ||||
| fonts-liberation 1:1.07.4-11 | ||||
| fonts-urw-base35 20200910-7 | ||||
| gcc-12-base 12.2.0-14+deb12u1 | ||||
| gettext 0.21-12 | ||||
| gettext-base 0.21-12 | ||||
| ghostscript 10.03.1~dfsg-1 | ||||
| gnupg 2.2.40-1.1 | ||||
| gnupg-l10n 2.2.40-1.1 | ||||
| gnupg-utils 2.2.40-1.1 | ||||
| gosu 1.14-1+b10 | ||||
| gpg 2.2.40-1.1 | ||||
| gpg-agent 2.2.40-1.1 | ||||
| gpg-wks-client 2.2.40-1.1 | ||||
| gpg-wks-server 2.2.40-1.1 | ||||
| gpgconf 2.2.40-1.1 | ||||
| gpgsm 2.2.40-1.1 | ||||
| gpgv 2.2.40-1.1 | ||||
| grep 3.8-5 | ||||
| gzip 1.12-1 | ||||
| hicolor-icon-theme 0.17-2 | ||||
| hostname 3.23+nmu1 | ||||
| icc-profiles-free 2.0.1+dfsg-1.1 | ||||
| imagemagick 8:6.9.11.60+dfsg-1.6+deb12u3 | ||||
| imagemagick-6-common 8:6.9.11.60+dfsg-1.6+deb12u3 | ||||
| imagemagick-6.q16 8:6.9.11.60+dfsg-1.6+deb12u3 | ||||
| init-system-helpers 1.65.2 | ||||
| jbig2dec 0.19-3 | ||||
| jbig2enc 0.30-1 | ||||
| libacl1 2.3.1-3 | ||||
| libaom3 3.6.0-1+deb12u1 | ||||
| libapt-pkg6.0 2.6.1 | ||||
| libarchive13 3.6.2-1+deb12u2 | ||||
| libassuan0 2.5.5-5 | ||||
| libattr1 1:2.5.1-4 | ||||
| libaudit-common 1:3.0.9-1 | ||||
| libaudit1 1:3.0.9-1 | ||||
| libavahi-client3 0.8-10+deb12u1 | ||||
| libavahi-common-data 0.8-10+deb12u1 | ||||
| libavahi-common3 0.8-10+deb12u1 | ||||
| libavcodec59 7:5.1.6-0+deb12u1 | ||||
| libavformat59 7:5.1.6-0+deb12u1 | ||||
| libavutil57 7:5.1.6-0+deb12u1 | ||||
| libblkid1 2.38.1-5+deb12u3 | ||||
| libbluray2 1:1.3.4-1 | ||||
| libbrotli1 1.0.9-2+b6 | ||||
| libbsd0 0.11.7-2 | ||||
| libbz2-1.0 1.0.8-5+b1 | ||||
| libc-bin 2.36-9+deb12u10 | ||||
| libc6 2.36-9+deb12u10 | ||||
| libcairo-gobject2 1.16.0-7 | ||||
| libcairo2 1.16.0-7 | ||||
| libcap-ng0 0.8.3-1+b3 | ||||
| libcap2 1:2.66-4+deb12u1 | ||||
| libchromaprint1 1.5.1-2+b1 | ||||
| libcjson1 1.7.15-1+deb12u2 | ||||
| libcodec2-1.0 1.0.5-1 | ||||
| libcom-err2 1.47.0-2 | ||||
| libconfig-inifiles-perl 3.000003-2 | ||||
| libcrypt1 1:4.4.33-2 | ||||
| libcups2 2.4.2-3+deb12u8 | ||||
| libcurl4 7.88.1-10+deb12u12 | ||||
| libdatrie1 0.2.13-2+b1 | ||||
| libdav1d6 1.0.0-2+deb12u1 | ||||
| libdb5.3 5.3.28+dfsg2-1 | ||||
| libdbus-1-3 1.14.10-1~deb12u1 | ||||
| libde265-0 1.0.11-1+deb12u2 | ||||
| libdebconfclient0 0.270 | ||||
| libdeflate0 1.14-1 | ||||
| libdrm-common 2.4.114-1 | ||||
| libdrm2 2.4.114-1+b1 | ||||
| libedit2 3.1-20221030-2 | ||||
| libexpat1 2.5.0-1+deb12u1 | ||||
| libext2fs2 1.47.0-2 | ||||
| libffi8 3.4.4-1 | ||||
| libfftw3-double3 3.3.10-1 | ||||
| libfontconfig1 2.14.1-4 | ||||
| libfontenc1 1:1.1.4-1 | ||||
| libfreetype6 2.12.1+dfsg-5+deb12u4 | ||||
| libfribidi0 1.0.8-2.1 | ||||
| libgcc-s1 12.2.0-14+deb12u1 | ||||
| libgcrypt20 1.10.1-3 | ||||
| libgdbm-compat4 1.23-3 | ||||
| libgdbm6 1.23-3 | ||||
| libgdk-pixbuf-2.0-0 2.42.10+dfsg-1+deb12u2 | ||||
| libgdk-pixbuf2.0-common 2.42.10+dfsg-1+deb12u2 | ||||
| libgif7 5.2.1-2.5 | ||||
| libglib2.0-0 2.74.6-2+deb12u6 | ||||
| libgme0 0.6.3-6 | ||||
| libgmp10 2:6.2.1+dfsg1-1.1 | ||||
| libgnutls30 3.7.9-2+deb12u5 | ||||
| libgomp1 12.2.0-14+deb12u1 | ||||
| libgpg-error0 1.46-1 | ||||
| libgraphite2-3 1.3.14-1 | ||||
| libgs-common 10.0.0~dfsg-11+deb12u7 | ||||
| libgs10 10.03.1~dfsg-1 | ||||
| libgs10-common 10.03.1~dfsg-1 | ||||
| libgsm1 1.0.22-1 | ||||
| libgssapi-krb5-2 1.20.1-2+deb12u3 | ||||
| libharfbuzz0b 6.0.0+dfsg-3 | ||||
| libheif1 1.15.1-1+deb12u1 | ||||
| libhogweed6 3.8.1-2 | ||||
| libhwy1 1.0.3-3+deb12u1 | ||||
| libice6 2:1.0.10-1 | ||||
| libicu72 72.1-3+deb12u1 | ||||
| libidn12 1.41-1 | ||||
| libidn2-0 2.3.3-1+b1 | ||||
| libijs-0.35 0.35-15 | ||||
| libimagequant0 2.17.0-1 | ||||
| libjbig0 2.1-6.1 | ||||
| libjbig2dec0 0.19-3 | ||||
| libjpeg62-turbo 1:2.1.5-2 | ||||
| libjxl0.7 0.7.0-10+deb12u1 | ||||
| libk5crypto3 1.20.1-2+deb12u3 | ||||
| libkeyutils1 1.6.3-2 | ||||
| libkrb5-3 1.20.1-2+deb12u3 | ||||
| libkrb5support0 1.20.1-2+deb12u3 | ||||
| libksba8 1.6.3-2 | ||||
| liblcms2-2 2.14-2 | ||||
| libldap-2.5-0 2.5.13+dfsg-5 | ||||
| liblept5 1.82.0-3+b3 | ||||
| liblerc4 4.0.0+ds-2 | ||||
| liblqr-1-0 0.4.2-2.1 | ||||
| libltdl7 2.4.7-7~deb12u1 | ||||
| liblz4-1 1.9.4-1 | ||||
| liblzma5 5.4.1-1 | ||||
| libmagic-mgc 1:5.44-3 | ||||
| libmagic1 1:5.44-3 | ||||
| libmagickcore-6.q16-6 8:6.9.11.60+dfsg-1.6+deb12u3 | ||||
| libmagickwand-6.q16-6 8:6.9.11.60+dfsg-1.6+deb12u3 | ||||
| libmariadb3 1:10.11.11-0+deb12u1 | ||||
| libmbedcrypto7 2.28.3-1 | ||||
| libmd0 1.0.4-2 | ||||
| libmfx1 22.5.4-1 | ||||
| libmount1 2.38.1-5+deb12u3 | ||||
| libmp3lame0 3.100-6 | ||||
| libmpg123-0 1.31.2-1+deb12u1 | ||||
| libncurses6 6.4-4 | ||||
| libncursesw6 6.4-4 | ||||
| libnettle8 3.8.1-2 | ||||
| libnghttp2-14 1.52.0-1+deb12u2 | ||||
| libnorm1 1.5.9+dfsg-2 | ||||
| libnpth0 1.6-3 | ||||
| libnsl2 1.3.0-2 | ||||
| libnspr4 2:4.35-1 | ||||
| libnss3 2:3.87.1-1+deb12u1 | ||||
| libnuma1 2.0.16-1 | ||||
| libogg0 1.3.5-3 | ||||
| libopenjp2-7 2.5.0-2+deb12u1 | ||||
| libopenmpt0 0.6.9-1 | ||||
| libopus0 1.3.1-3 | ||||
| libp11-kit0 0.24.1-2 | ||||
| libpam-modules 1.5.2-6+deb12u1 | ||||
| libpam-modules-bin 1.5.2-6+deb12u1 | ||||
| libpam-runtime 1.5.2-6+deb12u1 | ||||
| libpam0g 1.5.2-6+deb12u1 | ||||
| libpango-1.0-0 1.50.12+ds-1 | ||||
| libpangocairo-1.0-0 1.50.12+ds-1 | ||||
| libpangoft2-1.0-0 1.50.12+ds-1 | ||||
| libpaper1 1.1.29 | ||||
| libpcre2-8-0 10.42-1 | ||||
| libperl5.36 5.36.0-7+deb12u2 | ||||
| libpgm-5.3-0 5.3.128~dfsg-2 | ||||
| libpixman-1-0 0.42.2-1 | ||||
| libpng16-16 1.6.39-2 | ||||
| libpoppler126 22.12.0-2+deb12u1 | ||||
| libpq5 15.13-0+deb12u1 | ||||
| libpsl5 0.21.2-1 | ||||
| libqpdf29 11.9.0-1 | ||||
| librabbitmq4 0.11.0-1+deb12u1 | ||||
| librav1e0 0.5.1-6 | ||||
| libreadline8 8.2-1.3 | ||||
| librist4 0.2.7+dfsg-1 | ||||
| librsvg2-2 2.54.7+dfsg-1~deb12u1 | ||||
| librtmp1 2.4+20151223.gitfa8646d.1-2+b2 | ||||
| libsasl2-2 2.1.28+dfsg-10 | ||||
| libsasl2-modules-db 2.1.28+dfsg-10 | ||||
| libseccomp2 2.5.4-1+deb12u1 | ||||
| libselinux1 3.4-1+b6 | ||||
| libsemanage-common 3.4-1 | ||||
| libsemanage2 3.4-1+b5 | ||||
| libsepol2 3.4-2.1 | ||||
| libshine3 3.1.1-2 | ||||
| libsm6 2:1.2.3-1 | ||||
| libsmartcols1 2.38.1-5+deb12u3 | ||||
| libsnappy1v5 1.1.9-3 | ||||
| libsodium23 1.0.18-1 | ||||
| libsoxr0 0.1.3-4 | ||||
| libspeex1 1.2.1-2 | ||||
| libsqlite3-0 3.40.1-2+deb12u1 | ||||
| libsrt1.5-gnutls 1.5.1-1+deb12u1 | ||||
| libss2 1.47.0-2 | ||||
| libssh-gcrypt-4 0.10.6-0+deb12u1 | ||||
| libssh2-1 1.10.0-3+b1 | ||||
| libssl3 3.0.17-1~deb12u1 | ||||
| libstdc++6 12.2.0-14+deb12u1 | ||||
| libsvtav1enc1 1.4.1+dfsg-1 | ||||
| libswresample4 7:5.1.6-0+deb12u1 | ||||
| libsystemd0 252.38-1~deb12u1 | ||||
| libtasn1-6 4.19.0-2+deb12u1 | ||||
| libtesseract5 5.3.0-2 | ||||
| libthai-data 0.1.29-1 | ||||
| libthai0 0.1.29-1 | ||||
| libtheora0 1.1.1+dfsg.1-16.1+b1 | ||||
| libtiff6 4.5.0-6+deb12u2 | ||||
| libtinfo6 6.4-4 | ||||
| libtirpc-common 1.3.3+ds-1 | ||||
| libtirpc3 1.3.3+ds-1 | ||||
| libtwolame0 0.4.0-2 | ||||
| libudev1 252.38-1~deb12u1 | ||||
| libudfread0 1.1.2-1 | ||||
| libunistring2 1.0-2 | ||||
| libuuid1 2.38.1-5+deb12u3 | ||||
| libv4l-0 1.22.1-5+b2 | ||||
| libv4lconvert0 1.22.1-5+b2 | ||||
| libva-drm2 2.17.0-1 | ||||
| libva-x11-2 2.17.0-1 | ||||
| libva2 2.17.0-1 | ||||
| libvdpau1 1.5-2 | ||||
| libvorbis0a 1.3.7-1 | ||||
| libvorbisenc2 1.3.7-1 | ||||
| libvorbisfile3 1.3.7-1 | ||||
| libvpx7 1.12.0-1+deb12u4 | ||||
| libwebp7 1.2.4-0.2+deb12u1 | ||||
| libwebpdemux2 1.2.4-0.2+deb12u1 | ||||
| libwebpmux3 1.2.4-0.2+deb12u1 | ||||
| libx11-6 2:1.8.4-2+deb12u2 | ||||
| libx11-data 2:1.8.4-2+deb12u2 | ||||
| libx11-xcb1 2:1.8.4-2+deb12u2 | ||||
| libx264-164 2:0.164.3095+gitbaee400-3 | ||||
| libx265-199 3.5-2+b1 | ||||
| libxau6 1:1.0.9-1 | ||||
| libxcb-dri3-0 1.15-1 | ||||
| libxcb-render0 1.15-1 | ||||
| libxcb-shm0 1.15-1 | ||||
| libxcb1 1.15-1 | ||||
| libxdmcp6 1:1.1.2-3 | ||||
| libxext6 2:1.3.4-1+b1 | ||||
| libxfixes3 1:6.0.0-2 | ||||
| libxml2 2.9.14+dfsg-1.3~deb12u2 | ||||
| libxrender1 1:0.9.10-1.1 | ||||
| libxslt1.1 1.1.35-1+deb12u1 | ||||
| libxt6 1:1.2.1-1.1 | ||||
| libxvidcore4 2:1.3.7-1 | ||||
| libxxhash0 0.8.1-1 | ||||
| libzbar0 0.23.92-7+deb12u1 | ||||
| libzmq5 4.3.4-6 | ||||
| libzstd1 1.5.4+dfsg2-5 | ||||
| libzvbi-common 0.2.41-1 | ||||
| libzvbi0 0.2.41-1 | ||||
| login 1:4.13+dfsg1-1+deb12u1 | ||||
| logsave 1.47.0-2 | ||||
| mariadb-client 1:10.11.11-0+deb12u1 | ||||
| mariadb-client-core 1:10.11.11-0+deb12u1 | ||||
| mariadb-common 1:10.11.11-0+deb12u1 | ||||
| mawk 1.3.4.20200120-3.1 | ||||
| media-types 10.0.0 | ||||
| mount 2.38.1-5+deb12u3 | ||||
| mysql-common 5.8+1.1.0 | ||||
| ncurses-base 6.4-4 | ||||
| ncurses-bin 6.4-4 | ||||
| netbase 6.4 | ||||
| ocl-icd-libopencl1 2.3.1-1 | ||||
| openssl 3.0.17-1~deb12u1 | ||||
| passwd 1:4.13+dfsg1-1+deb12u1 | ||||
| perl 5.36.0-7+deb12u2 | ||||
| perl-base 5.36.0-7+deb12u2 | ||||
| perl-modules-5.36 5.36.0-7+deb12u2 | ||||
| pinentry-curses 1.2.1-1 | ||||
| pngquant 2.17.0-1 | ||||
| poppler-data 0.4.12-1 | ||||
| poppler-utils 22.12.0-2+deb12u1 | ||||
| postgresql-client 15+248 | ||||
| postgresql-client-15 15.13-0+deb12u1 | ||||
| postgresql-client-common 248 | ||||
| qpdf 11.9.0-1 | ||||
| readline-common 8.2-1.3 | ||||
| sed 4.9-1 | ||||
| sensible-utils 0.0.17+nmu1 | ||||
| shared-mime-info 2.2-1 | ||||
| sysvinit-utils 3.06-4 | ||||
| tar 1.34+dfsg-1.2+deb12u1 | ||||
| tesseract-ocr 5.3.0-2 | ||||
| tesseract-ocr-deu 1:4.1.0-2 | ||||
| tesseract-ocr-eng 1:4.1.0-2 | ||||
| tesseract-ocr-fra 1:4.1.0-2 | ||||
| tesseract-ocr-ita 1:4.1.0-2 | ||||
| tesseract-ocr-osd 1:4.1.0-2 | ||||
| tesseract-ocr-spa 1:4.1.0-2 | ||||
| tzdata 2025b-0+deb12u1 | ||||
| ucf 3.0043+nmu1+deb12u1 | ||||
| unpaper 7.0.0-0.1 | ||||
| usr-is-merged 37~deb12u1 | ||||
| util-linux 2.38.1-5+deb12u3 | ||||
| util-linux-extra 2.38.1-5+deb12u3 | ||||
| x11-common 1:7.7+23 | ||||
| xfonts-encodings 1:1.0.4-2.2 | ||||
| xfonts-utils 1:7.7+6 | ||||
| zlib1g 1:1.2.13.dfsg-1 | ||||
| @@ -170,11 +170,11 @@ Available options are `postgresql` and `mariadb`. | ||||
|  | ||||
|     !!! note | ||||
|  | ||||
|         A small pool is typically sufficient — for example, a size of 4. | ||||
|         Make sure your PostgreSQL server's max_connections setting is large enough to handle: | ||||
|         ```(Paperless workers + Celery workers) × pool size + safety margin``` | ||||
|         For example, with 4 Paperless workers and 2 Celery workers, and a pool size of 4: | ||||
|         (4 + 2) × 4 + 10 = 34 connections required. | ||||
|     A small pool is typically sufficient — for example, a size of 4. | ||||
|     Make sure your PostgreSQL server's max_connections setting is large enough to handle: | ||||
|     ```(Paperless workers + Celery workers) × pool size + safety margin``` | ||||
|     For example, with 4 Paperless workers and 2 Celery workers, and a pool size of 4: | ||||
|     (4 + 2) × 4 + 10 = 34 connections required. | ||||
|  | ||||
| #### [`PAPERLESS_DB_READ_CACHE_ENABLED=<bool>`](#PAPERLESS_DB_READ_CACHE_ENABLED) {#PAPERLESS_DB_READ_CACHE_ENABLED} | ||||
|  | ||||
| @@ -184,9 +184,9 @@ Available options are `postgresql` and `mariadb`. | ||||
|  | ||||
|     !!! danger | ||||
|  | ||||
|         **Do not modify the database outside the application while it is running.** | ||||
|         This includes actions such as restoring a backup, upgrading the database, or performing manual inserts. All external modifications must be done **only when the application is stopped**. | ||||
|         After making any such changes, you **must invalidate the DB read cache** using the `invalidate_cachalot` management command. | ||||
|     **Do not modify the database outside the application while it is running.** | ||||
|     This includes actions such as restoring a backup, upgrading the database, or performing manual inserts. All external modifications must be done **only when the application is stopped**. | ||||
|     After making any such changes, you **must invalidate the DB read cache** using the `invalidate_cachalot` management command. | ||||
|  | ||||
| #### [`PAPERLESS_READ_CACHE_TTL=<int>`](#PAPERLESS_READ_CACHE_TTL) {#PAPERLESS_READ_CACHE_TTL} | ||||
|  | ||||
| @@ -196,7 +196,7 @@ Available options are `postgresql` and `mariadb`. | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
|         A high TTL increases memory usage over time. Memory may be used until end of TTL, even if the cache is invalidated with the `invalidate_cachalot` command. | ||||
|     A high TTL increases memory usage over time. Memory may be used until end of TTL, even if the cache is invalidated with the `invalidate_cachalot` command. | ||||
|  | ||||
| In case of an out-of-memory (OOM) situation, Redis may stop accepting new data — including cache entries, scheduled tasks, and documents to consume. | ||||
| If your system has limited RAM, consider configuring a dedicated Redis instance for the read cache, with a memory limit and the eviction policy set to `allkeys-lru`. | ||||
| @@ -1805,3 +1805,23 @@ password. All of these options come from their similarly-named [Django settings] | ||||
| #### [`PAPERLESS_EMAIL_USE_SSL=<bool>`](#PAPERLESS_EMAIL_USE_SSL) {#PAPERLESS_EMAIL_USE_SSL} | ||||
|  | ||||
| : Defaults to false. | ||||
|  | ||||
| ## Remote OCR | ||||
|  | ||||
| #### [`PAPERLESS_REMOTE_OCR_ENGINE=<str>`](#PAPERLESS_REMOTE_OCR_ENGINE) {#PAPERLESS_REMOTE_OCR_ENGINE} | ||||
|  | ||||
| : The remote OCR engine to use. Currently only Azure AI is supported as "azureai". | ||||
|  | ||||
|     Defaults to None, which disables remote OCR. | ||||
|  | ||||
| #### [`PAPERLESS_REMOTE_OCR_API_KEY=<str>`](#PAPERLESS_REMOTE_OCR_API_KEY) {#PAPERLESS_REMOTE_OCR_API_KEY} | ||||
|  | ||||
| : The API key to use for the remote OCR engine. | ||||
|  | ||||
|     Defaults to None. | ||||
|  | ||||
| #### [`PAPERLESS_REMOTE_OCR_ENDPOINT=<str>`](#PAPERLESS_REMOTE_OCR_ENDPOINT) {#PAPERLESS_REMOTE_OCR_ENDPOINT} | ||||
|  | ||||
| : The endpoint to use for the remote OCR engine. This is required for Azure AI. | ||||
|  | ||||
|     Defaults to None. | ||||
|   | ||||
| @@ -25,9 +25,10 @@ physical documents into a searchable online archive so you can keep, well, _less | ||||
| ## Features | ||||
|  | ||||
| -   **Organize and index** your scanned documents with tags, correspondents, types, and more. | ||||
| -   _Your_ data is stored locally on _your_ server and is never transmitted or shared in any way. | ||||
| -   _Your_ data is stored locally on _your_ server and is never transmitted or shared in any way, unless you explicitly choose to do so. | ||||
| -   Performs **OCR** on your documents, adding searchable and selectable text, even to documents scanned with only images. | ||||
| -   Utilizes the open-source Tesseract engine to recognize more than 100 languages. | ||||
|     -   Utilizes the open-source Tesseract engine to recognize more than 100 languages. | ||||
|     -   _New!_ Supports remote OCR with Azure AI (opt-in). | ||||
| -   Documents are saved as PDF/A format which is designed for long term storage, alongside the unaltered originals. | ||||
| -   Uses machine-learning to automatically add tags, correspondents and document types to your documents. | ||||
| -   Supports PDF documents, images, plain text files, Office documents (Word, Excel, PowerPoint, and LibreOffice equivalents)[^1] and more. | ||||
|   | ||||
| @@ -414,7 +414,7 @@ fields and permissions, which will be merged. | ||||
|  | ||||
| #### Types {#workflow-trigger-types} | ||||
|  | ||||
| Currently, there are four events that correspond to workflow trigger 'types': | ||||
| Currently, there are three events that correspond to workflow trigger 'types': | ||||
|  | ||||
| 1. **Consumption Started**: _before_ a document is consumed, so events can include filters by source (mail, consumption | ||||
|    folder or API), file path, file name, mail rule | ||||
| @@ -427,7 +427,7 @@ Currently, there are four events that correspond to workflow trigger 'types': | ||||
|    added, created, updated date or you can specify a (date) custom field. You can also specify a day offset from the date (positive | ||||
|    offsets will trigger after the date, negative offsets will trigger before). | ||||
|  | ||||
| The following flow diagram illustrates the four document trigger types: | ||||
| The following flow diagram illustrates the three document trigger types: | ||||
|  | ||||
| ```mermaid | ||||
| flowchart TD | ||||
| @@ -882,6 +882,21 @@ how regularly you intend to scan documents and use paperless. | ||||
|     performed the task associated with the document, move it to the | ||||
|     inbox. | ||||
|  | ||||
| ## Remote OCR | ||||
|  | ||||
| !!! important | ||||
|  | ||||
|     This feature is disabled by default and will always remain strictly "opt-in". | ||||
|  | ||||
| Paperless-ngx supports performing OCR on documents using remote services. At the moment, this is limited to | ||||
| [Microsoft's Azure "Document Intelligence" service](https://azure.microsoft.com/en-us/products/ai-services/ai-document-intelligence). | ||||
| This is of course a paid service (with a free tier) which requires an Azure account and subscription. Azure AI is not affiliated with | ||||
| Paperless-ngx in any way. When enabled, Paperless-ngx will automatically send appropriate documents to Azure for OCR processing, bypassing | ||||
| the local OCR engine. See the [configuration](configuration.md#PAPERLESS_REMOTE_OCR_ENGINE) options for more details. | ||||
|  | ||||
| Additionally, when using a commercial service with this feature, consider both potential costs as well as any associated file size | ||||
| or page limitations (e.g. with a free tier). | ||||
|  | ||||
| ## Architecture | ||||
|  | ||||
| Paperless-ngx consists of the following components: | ||||
|   | ||||
| @@ -15,6 +15,7 @@ classifiers = [ | ||||
| # This will allow testing to not install a webserver, mysql, etc | ||||
|  | ||||
| dependencies = [ | ||||
|   "azure-ai-documentintelligence>=1.0.2", | ||||
|   "babel>=2.17", | ||||
|   "bleach~=6.2.0", | ||||
|   "celery[redis]~=5.5.1", | ||||
| @@ -232,6 +233,7 @@ testpaths = [ | ||||
|   "src/paperless_tesseract/tests/", | ||||
|   "src/paperless_tika/tests", | ||||
|   "src/paperless_text/tests/", | ||||
|   "src/paperless_remote/tests/", | ||||
| ] | ||||
| addopts = [ | ||||
|   "--pythonwarnings=all", | ||||
|   | ||||
| @@ -1636,7 +1636,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">45</context> | ||||
|           <context context-type="linenumber">44</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/trash/trash.component.html</context> | ||||
| @@ -1862,7 +1862,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">155</context> | ||||
|           <context context-type="linenumber">153</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2134950584701094962" datatype="html"> | ||||
| @@ -1918,77 +1918,63 @@ | ||||
|         <source>Result</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">46</context> | ||||
|           <context context-type="linenumber">45</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="5404910960991552159" datatype="html"> | ||||
|         <source>Dismiss selected</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">110</context> | ||||
|           <context context-type="linenumber">108</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8829078752502782653" datatype="html"> | ||||
|         <source>Dismiss all</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">111</context> | ||||
|           <context context-type="linenumber">109</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="1323591410517879795" datatype="html"> | ||||
|         <source>Confirm Dismiss All</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">152</context> | ||||
|           <context context-type="linenumber">150</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4157200209636243740" datatype="html"> | ||||
|         <source>Dismiss all <x id="PH" equiv-text="tasks.size"/> tasks?</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">153</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3597309129998924778" datatype="html"> | ||||
|         <source>Error dismissing tasks</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">161</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2132179171926568807" datatype="html"> | ||||
|         <source>Error dismissing task</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">170</context> | ||||
|           <context context-type="linenumber">151</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9011556615675272238" datatype="html"> | ||||
|         <source>queued</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">246</context> | ||||
|           <context context-type="linenumber">236</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6415892379431855826" datatype="html"> | ||||
|         <source>started</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">248</context> | ||||
|           <context context-type="linenumber">238</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="7510279840486540181" datatype="html"> | ||||
|         <source>completed</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">250</context> | ||||
|           <context context-type="linenumber">240</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4083337005045748464" datatype="html"> | ||||
|         <source>failed</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/admin/tasks/tasks.component.ts</context> | ||||
|           <context context-type="linenumber">252</context> | ||||
|           <context context-type="linenumber">242</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3418677553313974490" datatype="html"> | ||||
| @@ -2574,11 +2560,11 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1028</context> | ||||
|           <context context-type="linenumber">1025</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1393</context> | ||||
|           <context context-type="linenumber">1390</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -3190,7 +3176,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">981</context> | ||||
|           <context context-type="linenumber">978</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -6668,7 +6654,7 @@ | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1392</context> | ||||
|           <context context-type="linenumber">1389</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6490688569532630280" datatype="html"> | ||||
| @@ -7032,53 +7018,53 @@ | ||||
|         <source>Error retrieving metadata</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">669</context> | ||||
|           <context context-type="linenumber">666</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3456881259945295697" datatype="html"> | ||||
|         <source>Error retrieving suggestions.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">698</context> | ||||
|           <context context-type="linenumber">695</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="2194092841814123758" datatype="html"> | ||||
|         <source>Document "<x id="PH" equiv-text="newValues.title"/>" saved successfully.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">870</context> | ||||
|           <context context-type="linenumber">867</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">894</context> | ||||
|           <context context-type="linenumber">891</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6626387786259219838" datatype="html"> | ||||
|         <source>Error saving document "<x id="PH" equiv-text="this.document.title"/>"</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">900</context> | ||||
|           <context context-type="linenumber">897</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="448882439049417053" datatype="html"> | ||||
|         <source>Error saving document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">950</context> | ||||
|           <context context-type="linenumber">947</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8410796510716511826" datatype="html"> | ||||
|         <source>Do you really want to move the document "<x id="PH" equiv-text="this.document.title"/>" to the trash?</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">982</context> | ||||
|           <context context-type="linenumber">979</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="282586936710748252" datatype="html"> | ||||
|         <source>Documents can be restored prior to permanent deletion.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">983</context> | ||||
|           <context context-type="linenumber">980</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7089,7 +7075,7 @@ | ||||
|         <source>Move to trash</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">985</context> | ||||
|           <context context-type="linenumber">982</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7100,14 +7086,14 @@ | ||||
|         <source>Error deleting document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1004</context> | ||||
|           <context context-type="linenumber">1001</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="619486176823357521" datatype="html"> | ||||
|         <source>Reprocess confirm</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1024</context> | ||||
|           <context context-type="linenumber">1021</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-list/bulk-editor/bulk-editor.component.ts</context> | ||||
| @@ -7118,81 +7104,81 @@ | ||||
|         <source>This operation will permanently recreate the archive file for this document.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1025</context> | ||||
|           <context context-type="linenumber">1022</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="302054111564709516" datatype="html"> | ||||
|         <source>The archive file will be re-generated with the current settings.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1026</context> | ||||
|           <context context-type="linenumber">1023</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="8251197608401006898" datatype="html"> | ||||
|         <source>Reprocess operation for "<x id="PH" equiv-text="this.document.title"/>" will begin in the background. Close and re-open or reload this document after the operation has completed to see new content.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1036</context> | ||||
|           <context context-type="linenumber">1033</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4409560272830824468" datatype="html"> | ||||
|         <source>Error executing operation</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1047</context> | ||||
|           <context context-type="linenumber">1044</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6030453331794586802" datatype="html"> | ||||
|         <source>Error downloading document</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1096</context> | ||||
|           <context context-type="linenumber">1093</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4458954481601077369" datatype="html"> | ||||
|         <source>Page Fit</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1173</context> | ||||
|           <context context-type="linenumber">1170</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4663705961777238777" datatype="html"> | ||||
|         <source>PDF edit operation for "<x id="PH" equiv-text="this.document.title"/>" will begin in the background.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1411</context> | ||||
|           <context context-type="linenumber">1408</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="9043972994040261999" datatype="html"> | ||||
|         <source>Error executing PDF edit operation</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1423</context> | ||||
|           <context context-type="linenumber">1420</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="3740891324955700797" datatype="html"> | ||||
|         <source>Print failed.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1455</context> | ||||
|           <context context-type="linenumber">1452</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6457245677384603573" datatype="html"> | ||||
|         <source>Error loading document for printing.</source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1463</context> | ||||
|           <context context-type="linenumber">1460</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="6085793215710522488" datatype="html"> | ||||
|         <source>An error occurred loading tiff: <x id="PH" equiv-text="err.toString()"/></source> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1528</context> | ||||
|           <context context-type="linenumber">1525</context> | ||||
|         </context-group> | ||||
|         <context-group purpose="location"> | ||||
|           <context context-type="sourcefile">src/app/components/document-detail/document-detail.component.ts</context> | ||||
|           <context context-type="linenumber">1532</context> | ||||
|           <context context-type="linenumber">1529</context> | ||||
|         </context-group> | ||||
|       </trans-unit> | ||||
|       <trans-unit id="4958946940233632319" datatype="html"> | ||||
|   | ||||
| @@ -16,7 +16,6 @@ import { | ||||
|   NgbNavItem, | ||||
| } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { throwError } from 'rxjs' | ||||
| import { routes } from 'src/app/app-routing.module' | ||||
| import { | ||||
|   PaperlessTask, | ||||
| @@ -29,7 +28,6 @@ import { PermissionsGuard } from 'src/app/guards/permissions.guard' | ||||
| import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' | ||||
| import { PermissionsService } from 'src/app/services/permissions.service' | ||||
| import { TasksService } from 'src/app/services/tasks.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { environment } from 'src/environments/environment' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { PageHeaderComponent } from '../../common/page-header/page-header.component' | ||||
| @@ -125,7 +123,6 @@ describe('TasksComponent', () => { | ||||
|   let router: Router | ||||
|   let httpTestingController: HttpTestingController | ||||
|   let reloadSpy | ||||
|   let toastService: ToastService | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @@ -160,7 +157,6 @@ describe('TasksComponent', () => { | ||||
|     httpTestingController = TestBed.inject(HttpTestingController) | ||||
|     modalService = TestBed.inject(NgbModal) | ||||
|     router = TestBed.inject(Router) | ||||
|     toastService = TestBed.inject(ToastService) | ||||
|     fixture = TestBed.createComponent(TasksComponent) | ||||
|     component = fixture.componentInstance | ||||
|     jest.useFakeTimers() | ||||
| @@ -253,42 +249,6 @@ describe('TasksComponent', () => { | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(selected) | ||||
|   }) | ||||
|  | ||||
|   it('should show an error and re-enable modal buttons when dismissing multiple tasks fails', () => { | ||||
|     component.selectedTasks = new Set([tasks[0].id, tasks[1].id]) | ||||
|     const error = new Error('dismiss failed') | ||||
|     const toastSpy = jest.spyOn(toastService, 'showError') | ||||
|     const dismissSpy = jest | ||||
|       .spyOn(tasksService, 'dismissTasks') | ||||
|       .mockReturnValue(throwError(() => error)) | ||||
|  | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
|  | ||||
|     component.dismissTasks() | ||||
|     expect(modal).not.toBeUndefined() | ||||
|  | ||||
|     modal.componentInstance.confirmClicked.emit() | ||||
|  | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(new Set([tasks[0].id, tasks[1].id])) | ||||
|     expect(toastSpy).toHaveBeenCalledWith('Error dismissing tasks', error) | ||||
|     expect(modal.componentInstance.buttonsEnabled).toBe(true) | ||||
|     expect(component.selectedTasks.size).toBe(0) | ||||
|   }) | ||||
|  | ||||
|   it('should show an error when dismissing a single task fails', () => { | ||||
|     const error = new Error('dismiss failed') | ||||
|     const toastSpy = jest.spyOn(toastService, 'showError') | ||||
|     const dismissSpy = jest | ||||
|       .spyOn(tasksService, 'dismissTasks') | ||||
|       .mockReturnValue(throwError(() => error)) | ||||
|  | ||||
|     component.dismissTask(tasks[0]) | ||||
|  | ||||
|     expect(dismissSpy).toHaveBeenCalledWith(new Set([tasks[0].id])) | ||||
|     expect(toastSpy).toHaveBeenCalledWith('Error dismissing task', error) | ||||
|     expect(component.selectedTasks.size).toBe(0) | ||||
|   }) | ||||
|  | ||||
|   it('should support dismiss all tasks', () => { | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((m) => (modal = m[m.length - 1])) | ||||
|   | ||||
| @@ -24,7 +24,6 @@ import { PaperlessTask } from 'src/app/data/paperless-task' | ||||
| import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive' | ||||
| import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' | ||||
| import { TasksService } from 'src/app/services/tasks.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { PageHeaderComponent } from '../../common/page-header/page-header.component' | ||||
| import { LoadingComponentWithPermissions } from '../../loading-component/loading.component' | ||||
| @@ -73,7 +72,6 @@ export class TasksComponent | ||||
|   tasksService = inject(TasksService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private readonly router = inject(Router) | ||||
|   private readonly toastService = inject(ToastService) | ||||
|  | ||||
|   public activeTab: TaskTab | ||||
|   public selectedTasks: Set<number> = new Set() | ||||
| @@ -156,19 +154,11 @@ export class TasksComponent | ||||
|       modal.componentInstance.confirmClicked.pipe(first()).subscribe(() => { | ||||
|         modal.componentInstance.buttonsEnabled = false | ||||
|         modal.close() | ||||
|         this.tasksService.dismissTasks(tasks).subscribe({ | ||||
|           error: (e) => { | ||||
|             this.toastService.showError($localize`Error dismissing tasks`, e) | ||||
|             modal.componentInstance.buttonsEnabled = true | ||||
|           }, | ||||
|         }) | ||||
|         this.tasksService.dismissTasks(tasks) | ||||
|         this.clearSelection() | ||||
|       }) | ||||
|     } else { | ||||
|       this.tasksService.dismissTasks(tasks).subscribe({ | ||||
|         error: (e) => | ||||
|           this.toastService.showError($localize`Error dismissing task`, e), | ||||
|       }) | ||||
|       this.tasksService.dismissTasks(tasks) | ||||
|       this.clearSelection() | ||||
|     } | ||||
|   } | ||||
|   | ||||
| @@ -41,3 +41,9 @@ | ||||
|     min-width: 140px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .btn-group-xs { | ||||
|   > .btn { | ||||
|     border-radius: 0.15rem; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,18 +1,19 @@ | ||||
| <div class="mb-3"> | ||||
|   @if (title) { | ||||
|     <label class="form-label" [for]="inputId">{{title}}</label> | ||||
|     <label [for]="inputId">{{title}}</label> | ||||
|   } | ||||
|  | ||||
|   <div class="input-group" [class.is-invalid]="error"> | ||||
|     <button type="button" class="input-group-text" [style.background-color]="value" (click)="colorPicker.toggle()">   </button> | ||||
|     <span class="input-group-text" [style.background-color]="value">   </span> | ||||
|  | ||||
|     <ng-template #popContent> | ||||
|       <div style="min-width: 200px;" class="pb-3"> | ||||
|         <color-slider [color]="value" (onChangeComplete)="colorChanged($event.color.hex)"></color-slider> | ||||
|       </div> | ||||
|  | ||||
|     </ng-template> | ||||
|  | ||||
|     <input #inputField class="form-control" [class.is-invalid]="error" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)" [autoClose]="'outside'" [ngbPopover]="popContent" #colorPicker="ngbPopover" placement="bottom" popoverClass="shadow"> | ||||
|     <input #inputField class="form-control" [class.is-invalid]="error" [id]="inputId" [(ngModel)]="value" (change)="onChange(value)" [autoClose]="'outside'" [ngbPopover]="popContent" placement="bottom" popoverClass="shadow"> | ||||
|  | ||||
|     <button class="btn btn-outline-secondary" type="button" (click)="randomize()"> | ||||
|       <i-bs name="dice5"></i-bs> | ||||
|   | ||||
| @@ -42,8 +42,8 @@ describe('ColorComponent', () => { | ||||
|   }) | ||||
|  | ||||
|   it('should set swatch color', () => { | ||||
|     const swatch: HTMLButtonElement = fixture.nativeElement.querySelector( | ||||
|       'button.input-group-text' | ||||
|     const swatch: HTMLSpanElement = fixture.nativeElement.querySelector( | ||||
|       'span.input-group-text' | ||||
|     ) | ||||
|     expect(swatch.style.backgroundColor).toEqual('') | ||||
|     component.value = '#ff0000' | ||||
|   | ||||
| @@ -1212,7 +1212,7 @@ describe('DocumentDetailComponent', () => { | ||||
|   it('should support keyboard shortcuts', () => { | ||||
|     initNormally() | ||||
|  | ||||
|     const hasNextSpy = jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     const nextSpy = jest.spyOn(component, 'nextDoc') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 'arrowright', ctrlKey: true }) | ||||
| @@ -1226,32 +1226,21 @@ describe('DocumentDetailComponent', () => { | ||||
|     ) | ||||
|     expect(prevSpy).toHaveBeenCalled() | ||||
|  | ||||
|     const isDirtySpy = jest | ||||
|       .spyOn(openDocumentsService, 'isDirty') | ||||
|       .mockReturnValue(true) | ||||
|     jest.spyOn(openDocumentsService, 'isDirty').mockReturnValue(true) | ||||
|     const saveSpy = jest.spyOn(component, 'save') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true }) | ||||
|     ) | ||||
|     expect(saveSpy).toHaveBeenCalled() | ||||
|  | ||||
|     hasNextSpy.mockReturnValue(true) | ||||
|     jest.spyOn(openDocumentsService, 'isDirty').mockReturnValue(true) | ||||
|     jest.spyOn(component, 'hasNext').mockReturnValue(true) | ||||
|     const saveNextSpy = jest.spyOn(component, 'saveEditNext') | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true, shiftKey: true }) | ||||
|     ) | ||||
|     expect(saveNextSpy).toHaveBeenCalled() | ||||
|  | ||||
|     saveSpy.mockClear() | ||||
|     saveNextSpy.mockClear() | ||||
|     isDirtySpy.mockReturnValue(true) | ||||
|     hasNextSpy.mockReturnValue(false) | ||||
|     document.dispatchEvent( | ||||
|       new KeyboardEvent('keydown', { key: 's', ctrlKey: true, shiftKey: true }) | ||||
|     ) | ||||
|     expect(saveNextSpy).not.toHaveBeenCalled() | ||||
|     expect(saveSpy).toHaveBeenCalledWith(true) | ||||
|  | ||||
|     const closeSpy = jest.spyOn(component, 'close') | ||||
|     document.dispatchEvent(new KeyboardEvent('keydown', { key: 'escape' })) | ||||
|     expect(closeSpy).toHaveBeenCalled() | ||||
|   | ||||
| @@ -615,10 +615,7 @@ export class DocumentDetailComponent | ||||
|       }) | ||||
|       .pipe(takeUntil(this.unsubscribeNotifier)) | ||||
|       .subscribe(() => { | ||||
|         if (this.openDocumentService.isDirty(this.document)) { | ||||
|           if (this.hasNext()) this.saveEditNext() | ||||
|           else this.save(true) | ||||
|         } | ||||
|         if (this.openDocumentService.isDirty(this.document)) this.saveEditNext() | ||||
|       }) | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -51,7 +51,7 @@ describe('TasksService', () => { | ||||
|   }) | ||||
|  | ||||
|   it('calls acknowledge_tasks api endpoint on dismiss and reloads', () => { | ||||
|     tasksService.dismissTasks(new Set([1, 2, 3])).subscribe() | ||||
|     tasksService.dismissTasks(new Set([1, 2, 3])) | ||||
|     const req = httpTestingController.expectOne( | ||||
|       `${environment.apiBaseUrl}tasks/acknowledge/` | ||||
|     ) | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { HttpClient } from '@angular/common/http' | ||||
| import { Injectable, inject } from '@angular/core' | ||||
| import { Observable, Subject } from 'rxjs' | ||||
| import { first, takeUntil, tap } from 'rxjs/operators' | ||||
| import { first, takeUntil } from 'rxjs/operators' | ||||
| import { | ||||
|   PaperlessTask, | ||||
|   PaperlessTaskName, | ||||
| @@ -68,17 +68,14 @@ export class TasksService { | ||||
|   } | ||||
|  | ||||
|   public dismissTasks(task_ids: Set<number>) { | ||||
|     return this.http | ||||
|     this.http | ||||
|       .post(`${this.baseUrl}tasks/acknowledge/`, { | ||||
|         tasks: [...task_ids], | ||||
|       }) | ||||
|       .pipe( | ||||
|         first(), | ||||
|         takeUntil(this.unsubscribeNotifer), | ||||
|         tap(() => { | ||||
|           this.reload() | ||||
|         }) | ||||
|       ) | ||||
|       .pipe(first()) | ||||
|       .subscribe((r) => { | ||||
|         this.reload() | ||||
|       }) | ||||
|   } | ||||
|  | ||||
|   public cancelPending(): void { | ||||
|   | ||||
| @@ -161,21 +161,3 @@ class PaperlessNotePermissions(BasePermission): | ||||
|         perms = self.perms_map[request.method] | ||||
|  | ||||
|         return request.user.has_perms(perms) | ||||
|  | ||||
|  | ||||
| class AcknowledgeTasksPermissions(BasePermission): | ||||
|     """ | ||||
|     Permissions class that checks for model permissions for acknowledging tasks. | ||||
|     """ | ||||
|  | ||||
|     perms_map = { | ||||
|         "POST": ["documents.change_paperlesstask"], | ||||
|     } | ||||
|  | ||||
|     def has_permission(self, request, view): | ||||
|         if not request.user or not request.user.is_authenticated:  # pragma: no cover | ||||
|             return False | ||||
|  | ||||
|         perms = self.perms_map.get(request.method, []) | ||||
|  | ||||
|         return request.user.has_perms(perms) | ||||
|   | ||||
| @@ -76,9 +76,7 @@ def check_sanity(*, progress=False, scheduled=True) -> SanityCheckMessages: | ||||
|     messages = SanityCheckMessages() | ||||
|  | ||||
|     present_files = { | ||||
|         x.resolve() | ||||
|         for x in Path(settings.MEDIA_ROOT).glob("**/*") | ||||
|         if not x.is_dir() and x.name not in settings.IGNORABLE_FILES | ||||
|         x.resolve() for x in Path(settings.MEDIA_ROOT).glob("**/*") if not x.is_dir() | ||||
|     } | ||||
|  | ||||
|     lockfile = Path(settings.MEDIA_LOCK).resolve() | ||||
|   | ||||
| @@ -135,44 +135,6 @@ class TestTasks(DirectoriesMixin, APITestCase): | ||||
|         response = self.client.get(self.ENDPOINT + "?acknowledged=false") | ||||
|         self.assertEqual(len(response.data), 0) | ||||
|  | ||||
|     def test_acknowledge_tasks_requires_change_permission(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|             - A regular user initially without change permissions | ||||
|             - A regular user with change permissions | ||||
|         WHEN: | ||||
|             - API call is made to acknowledge tasks | ||||
|         THEN: | ||||
|             - The first user is forbidden from acknowledging tasks | ||||
|             - The second user is allowed to acknowledge tasks | ||||
|         """ | ||||
|         regular_user = User.objects.create_user(username="test") | ||||
|         self.client.force_authenticate(user=regular_user) | ||||
|  | ||||
|         task = PaperlessTask.objects.create( | ||||
|             task_id=str(uuid.uuid4()), | ||||
|             task_file_name="task_one.pdf", | ||||
|         ) | ||||
|  | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT + "acknowledge/", | ||||
|             {"tasks": [task.id]}, | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_403_FORBIDDEN) | ||||
|  | ||||
|         regular_user2 = User.objects.create_user(username="test2") | ||||
|         regular_user2.user_permissions.add( | ||||
|             Permission.objects.get(codename="change_paperlesstask"), | ||||
|         ) | ||||
|         regular_user2.save() | ||||
|         self.client.force_authenticate(user=regular_user2) | ||||
|  | ||||
|         response = self.client.post( | ||||
|             self.ENDPOINT + "acknowledge/", | ||||
|             {"tasks": [task.id]}, | ||||
|         ) | ||||
|         self.assertEqual(response.status_code, status.HTTP_200_OK) | ||||
|  | ||||
|     def test_tasks_owner_aware(self): | ||||
|         """ | ||||
|         GIVEN: | ||||
|   | ||||
| @@ -169,13 +169,6 @@ class TestSanityCheck(DirectoriesMixin, TestCase): | ||||
|         messages = check_sanity() | ||||
|         self.assertFalse(messages.has_warning) | ||||
|  | ||||
|     def test_ignore_ignorable_files(self): | ||||
|         self.make_test_data() | ||||
|         Path(self.dirs.media_dir, ".DS_Store").touch() | ||||
|         Path(self.dirs.media_dir, "desktop.ini").touch() | ||||
|         messages = check_sanity() | ||||
|         self.assertFalse(messages.has_warning) | ||||
|  | ||||
|     def test_archive_filename_no_checksum(self): | ||||
|         doc = self.make_test_data() | ||||
|         doc.archive_checksum = None | ||||
|   | ||||
| @@ -136,7 +136,6 @@ from documents.models import WorkflowAction | ||||
| from documents.models import WorkflowTrigger | ||||
| from documents.parsers import get_parser_class_for_mime_type | ||||
| from documents.parsers import parse_date_generator | ||||
| from documents.permissions import AcknowledgeTasksPermissions | ||||
| from documents.permissions import PaperlessAdminPermissions | ||||
| from documents.permissions import PaperlessNotePermissions | ||||
| from documents.permissions import PaperlessObjectPermissions | ||||
| @@ -2488,11 +2487,7 @@ class TasksViewSet(ReadOnlyModelViewSet): | ||||
|             queryset = PaperlessTask.objects.filter(task_id=task_id) | ||||
|         return queryset | ||||
|  | ||||
|     @action( | ||||
|         methods=["post"], | ||||
|         detail=False, | ||||
|         permission_classes=[IsAuthenticated, AcknowledgeTasksPermissions], | ||||
|     ) | ||||
|     @action(methods=["post"], detail=False) | ||||
|     def acknowledge(self, request): | ||||
|         serializer = AcknowledgeTasksViewSerializer(data=request.data) | ||||
|         serializer.is_valid(raise_exception=True) | ||||
|   | ||||
| @@ -322,6 +322,7 @@ INSTALLED_APPS = [ | ||||
|     "paperless_tesseract.apps.PaperlessTesseractConfig", | ||||
|     "paperless_text.apps.PaperlessTextConfig", | ||||
|     "paperless_mail.apps.PaperlessMailConfig", | ||||
|     "paperless_remote.apps.PaperlessRemoteParserConfig", | ||||
|     "django.contrib.admin", | ||||
|     "rest_framework", | ||||
|     "rest_framework.authtoken", | ||||
| @@ -1003,18 +1004,6 @@ THREADS_PER_WORKER = os.getenv( | ||||
| # Paperless Specific Settings                                                 # | ||||
| ############################################################################### | ||||
|  | ||||
| IGNORABLE_FILES: Final[list[str]] = [ | ||||
|     ".DS_Store", | ||||
|     ".DS_STORE", | ||||
|     "._*", | ||||
|     ".stfolder/*", | ||||
|     ".stversions/*", | ||||
|     ".localized/*", | ||||
|     "desktop.ini", | ||||
|     "@eaDir/*", | ||||
|     "Thumbs.db", | ||||
| ] | ||||
|  | ||||
| CONSUMER_POLLING = int(os.getenv("PAPERLESS_CONSUMER_POLLING", 0)) | ||||
|  | ||||
| CONSUMER_POLLING_DELAY = int(os.getenv("PAPERLESS_CONSUMER_POLLING_DELAY", 5)) | ||||
| @@ -1037,7 +1026,7 @@ CONSUMER_IGNORE_PATTERNS = list( | ||||
|     json.loads( | ||||
|         os.getenv( | ||||
|             "PAPERLESS_CONSUMER_IGNORE_PATTERNS", | ||||
|             json.dumps(IGNORABLE_FILES), | ||||
|             '[".DS_Store", ".DS_STORE", "._*", ".stfolder/*", ".stversions/*", ".localized/*", "desktop.ini", "@eaDir/*", "Thumbs.db"]', | ||||
|         ), | ||||
|     ), | ||||
| ) | ||||
| @@ -1401,3 +1390,10 @@ WEBHOOKS_ALLOW_INTERNAL_REQUESTS = __get_boolean( | ||||
|     "PAPERLESS_WEBHOOKS_ALLOW_INTERNAL_REQUESTS", | ||||
|     "true", | ||||
| ) | ||||
|  | ||||
| ############################################################################### | ||||
| # Remote Parser                                                               # | ||||
| ############################################################################### | ||||
| REMOTE_OCR_ENGINE = os.getenv("PAPERLESS_REMOTE_OCR_ENGINE") | ||||
| REMOTE_OCR_API_KEY = os.getenv("PAPERLESS_REMOTE_OCR_API_KEY") | ||||
| REMOTE_OCR_ENDPOINT = os.getenv("PAPERLESS_REMOTE_OCR_ENDPOINT") | ||||
|   | ||||
							
								
								
									
										4
									
								
								src/paperless_remote/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/paperless_remote/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| # this is here so that django finds the checks. | ||||
| from paperless_remote.checks import check_remote_parser_configured | ||||
|  | ||||
| __all__ = ["check_remote_parser_configured"] | ||||
							
								
								
									
										14
									
								
								src/paperless_remote/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/paperless_remote/apps.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| from django.apps import AppConfig | ||||
|  | ||||
| from paperless_remote.signals import remote_consumer_declaration | ||||
|  | ||||
|  | ||||
| class PaperlessRemoteParserConfig(AppConfig): | ||||
|     name = "paperless_remote" | ||||
|  | ||||
|     def ready(self): | ||||
|         from documents.signals import document_consumer_declaration | ||||
|  | ||||
|         document_consumer_declaration.connect(remote_consumer_declaration) | ||||
|  | ||||
|         AppConfig.ready(self) | ||||
							
								
								
									
										17
									
								
								src/paperless_remote/checks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								src/paperless_remote/checks.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| from django.conf import settings | ||||
| from django.core.checks import Error | ||||
| from django.core.checks import register | ||||
|  | ||||
|  | ||||
| @register() | ||||
| def check_remote_parser_configured(app_configs, **kwargs): | ||||
|     if settings.REMOTE_OCR_ENGINE == "azureai" and not ( | ||||
|         settings.REMOTE_OCR_ENDPOINT and settings.REMOTE_OCR_API_KEY | ||||
|     ): | ||||
|         return [ | ||||
|             Error( | ||||
|                 "Azure AI remote parser requires endpoint and API key to be configured.", | ||||
|             ), | ||||
|         ] | ||||
|  | ||||
|     return [] | ||||
							
								
								
									
										113
									
								
								src/paperless_remote/parsers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								src/paperless_remote/parsers.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| from pathlib import Path | ||||
|  | ||||
| from django.conf import settings | ||||
|  | ||||
| from paperless_tesseract.parsers import RasterisedDocumentParser | ||||
|  | ||||
|  | ||||
| class RemoteEngineConfig: | ||||
|     def __init__( | ||||
|         self, | ||||
|         engine: str, | ||||
|         api_key: str | None = None, | ||||
|         endpoint: str | None = None, | ||||
|     ): | ||||
|         self.engine = engine | ||||
|         self.api_key = api_key | ||||
|         self.endpoint = endpoint | ||||
|  | ||||
|     def engine_is_valid(self): | ||||
|         valid = self.engine in ["azureai"] and self.api_key is not None | ||||
|         if self.engine == "azureai": | ||||
|             valid = valid and self.endpoint is not None | ||||
|         return valid | ||||
|  | ||||
|  | ||||
| class RemoteDocumentParser(RasterisedDocumentParser): | ||||
|     """ | ||||
|     This parser uses a remote OCR engine to parse documents. Currently, it supports Azure AI Vision | ||||
|     as this is the only service that provides a remote OCR API with text-embedded PDF output. | ||||
|     """ | ||||
|  | ||||
|     logging_name = "paperless.parsing.remote" | ||||
|  | ||||
|     def get_settings(self) -> RemoteEngineConfig: | ||||
|         """ | ||||
|         Returns the configuration for the remote OCR engine, loaded from Django settings. | ||||
|         """ | ||||
|         return RemoteEngineConfig( | ||||
|             engine=settings.REMOTE_OCR_ENGINE, | ||||
|             api_key=settings.REMOTE_OCR_API_KEY, | ||||
|             endpoint=settings.REMOTE_OCR_ENDPOINT, | ||||
|         ) | ||||
|  | ||||
|     def supported_mime_types(self): | ||||
|         if self.settings.engine_is_valid(): | ||||
|             return { | ||||
|                 "application/pdf": ".pdf", | ||||
|                 "image/png": ".png", | ||||
|                 "image/jpeg": ".jpg", | ||||
|                 "image/tiff": ".tiff", | ||||
|                 "image/bmp": ".bmp", | ||||
|                 "image/gif": ".gif", | ||||
|                 "image/webp": ".webp", | ||||
|             } | ||||
|         else: | ||||
|             return {} | ||||
|  | ||||
|     def azure_ai_vision_parse( | ||||
|         self, | ||||
|         file: Path, | ||||
|     ) -> str | None: | ||||
|         """ | ||||
|         Uses Azure AI Vision to parse the document and return the text content. | ||||
|         It requests a searchable PDF output with embedded text. | ||||
|         The PDF is saved to the archive_path attribute. | ||||
|         Returns the text content extracted from the document. | ||||
|         If the parsing fails, it returns None. | ||||
|         """ | ||||
|         from azure.ai.documentintelligence import DocumentIntelligenceClient | ||||
|         from azure.ai.documentintelligence.models import AnalyzeDocumentRequest | ||||
|         from azure.ai.documentintelligence.models import AnalyzeOutputOption | ||||
|         from azure.ai.documentintelligence.models import DocumentContentFormat | ||||
|         from azure.core.credentials import AzureKeyCredential | ||||
|  | ||||
|         client = DocumentIntelligenceClient( | ||||
|             endpoint=self.settings.endpoint, | ||||
|             credential=AzureKeyCredential(self.settings.api_key), | ||||
|         ) | ||||
|  | ||||
|         with file.open("rb") as f: | ||||
|             analyze_request = AnalyzeDocumentRequest(bytes_source=f.read()) | ||||
|             poller = client.begin_analyze_document( | ||||
|                 model_id="prebuilt-read", | ||||
|                 body=analyze_request, | ||||
|                 output_content_format=DocumentContentFormat.TEXT, | ||||
|                 output=[AnalyzeOutputOption.PDF],  # request searchable PDF output | ||||
|                 content_type="application/json", | ||||
|             ) | ||||
|  | ||||
|         poller.wait() | ||||
|         result_id = poller.details["operation_id"] | ||||
|         result = poller.result() | ||||
|  | ||||
|         # Download the PDF with embedded text | ||||
|         self.archive_path = self.tempdir / "archive.pdf" | ||||
|         with self.archive_path.open("wb") as f: | ||||
|             for chunk in client.get_analyze_result_pdf( | ||||
|                 model_id="prebuilt-read", | ||||
|                 result_id=result_id, | ||||
|             ): | ||||
|                 f.write(chunk) | ||||
|  | ||||
|         client.close() | ||||
|         return result.content | ||||
|  | ||||
|     def parse(self, document_path: Path, mime_type, file_name=None): | ||||
|         if not self.settings.engine_is_valid(): | ||||
|             self.log.warning( | ||||
|                 "No valid remote parser engine is configured, content will be empty.", | ||||
|             ) | ||||
|             self.text = "" | ||||
|         elif self.settings.engine == "azureai": | ||||
|             self.text = self.azure_ai_vision_parse(document_path) | ||||
							
								
								
									
										18
									
								
								src/paperless_remote/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/paperless_remote/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| def get_parser(*args, **kwargs): | ||||
|     from paperless_remote.parsers import RemoteDocumentParser | ||||
|  | ||||
|     return RemoteDocumentParser(*args, **kwargs) | ||||
|  | ||||
|  | ||||
| def get_supported_mime_types(): | ||||
|     from paperless_remote.parsers import RemoteDocumentParser | ||||
|  | ||||
|     return RemoteDocumentParser(None).supported_mime_types() | ||||
|  | ||||
|  | ||||
| def remote_consumer_declaration(sender, **kwargs): | ||||
|     return { | ||||
|         "parser": get_parser, | ||||
|         "weight": 5, | ||||
|         "mime_types": get_supported_mime_types(), | ||||
|     } | ||||
							
								
								
									
										0
									
								
								src/paperless_remote/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								src/paperless_remote/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										
											BIN
										
									
								
								src/paperless_remote/tests/samples/simple-digital.pdf
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/paperless_remote/tests/samples/simple-digital.pdf
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										24
									
								
								src/paperless_remote/tests/test_checks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								src/paperless_remote/tests/test_checks.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| from unittest import TestCase | ||||
|  | ||||
| from django.test import override_settings | ||||
|  | ||||
| from paperless_remote import check_remote_parser_configured | ||||
|  | ||||
|  | ||||
| class TestChecks(TestCase): | ||||
|     @override_settings(REMOTE_OCR_ENGINE=None) | ||||
|     def test_no_engine(self): | ||||
|         msgs = check_remote_parser_configured(None) | ||||
|         self.assertEqual(len(msgs), 0) | ||||
|  | ||||
|     @override_settings(REMOTE_OCR_ENGINE="azureai") | ||||
|     @override_settings(REMOTE_OCR_API_KEY="somekey") | ||||
|     @override_settings(REMOTE_OCR_ENDPOINT=None) | ||||
|     def test_azure_no_endpoint(self): | ||||
|         msgs = check_remote_parser_configured(None) | ||||
|         self.assertEqual(len(msgs), 1) | ||||
|         self.assertTrue( | ||||
|             msgs[0].msg.startswith( | ||||
|                 "Azure AI remote parser requires endpoint and API key to be configured.", | ||||
|             ), | ||||
|         ) | ||||
							
								
								
									
										101
									
								
								src/paperless_remote/tests/test_parser.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								src/paperless_remote/tests/test_parser.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | ||||
| import uuid | ||||
| from pathlib import Path | ||||
| from unittest import mock | ||||
|  | ||||
| from django.test import TestCase | ||||
| from django.test import override_settings | ||||
|  | ||||
| from documents.tests.utils import DirectoriesMixin | ||||
| from documents.tests.utils import FileSystemAssertsMixin | ||||
| from paperless_remote.parsers import RemoteDocumentParser | ||||
| from paperless_remote.signals import get_parser | ||||
|  | ||||
|  | ||||
| class TestParser(DirectoriesMixin, FileSystemAssertsMixin, TestCase): | ||||
|     SAMPLE_FILES = Path(__file__).resolve().parent / "samples" | ||||
|  | ||||
|     def assertContainsStrings(self, content: str, strings: list[str]): | ||||
|         # Asserts that all strings appear in content, in the given order. | ||||
|         indices = [] | ||||
|         for s in strings: | ||||
|             if s in content: | ||||
|                 indices.append(content.index(s)) | ||||
|             else: | ||||
|                 self.fail(f"'{s}' is not in '{content}'") | ||||
|         self.assertListEqual(indices, sorted(indices)) | ||||
|  | ||||
|     @mock.patch("paperless_tesseract.parsers.run_subprocess") | ||||
|     @mock.patch("azure.ai.documentintelligence.DocumentIntelligenceClient") | ||||
|     def test_get_text_with_azure(self, mock_client_cls, mock_subprocess): | ||||
|         # Arrange mock Azure client | ||||
|         mock_client = mock.Mock() | ||||
|         mock_client_cls.return_value = mock_client | ||||
|  | ||||
|         # Simulate poller result and its `.details` | ||||
|         mock_poller = mock.Mock() | ||||
|         mock_poller.wait.return_value = None | ||||
|         mock_poller.details = {"operation_id": "fake-op-id"} | ||||
|         mock_client.begin_analyze_document.return_value = mock_poller | ||||
|         mock_poller.result.return_value.content = "This is a test document." | ||||
|  | ||||
|         # Return dummy PDF bytes | ||||
|         mock_client.get_analyze_result_pdf.return_value = [ | ||||
|             b"%PDF-", | ||||
|             b"1.7 ", | ||||
|             b"FAKEPDF", | ||||
|         ] | ||||
|  | ||||
|         # Simulate pdftotext by writing dummy text to sidecar file | ||||
|         def fake_run(cmd, *args, **kwargs): | ||||
|             with Path(cmd[-1]).open("w", encoding="utf-8") as f: | ||||
|                 f.write("This is a test document.") | ||||
|  | ||||
|         mock_subprocess.side_effect = fake_run | ||||
|  | ||||
|         with override_settings( | ||||
|             REMOTE_OCR_ENGINE="azureai", | ||||
|             REMOTE_OCR_API_KEY="somekey", | ||||
|             REMOTE_OCR_ENDPOINT="https://endpoint.cognitiveservices.azure.com", | ||||
|         ): | ||||
|             parser = get_parser(uuid.uuid4()) | ||||
|             parser.parse( | ||||
|                 self.SAMPLE_FILES / "simple-digital.pdf", | ||||
|                 "application/pdf", | ||||
|             ) | ||||
|  | ||||
|             self.assertContainsStrings( | ||||
|                 parser.text.strip(), | ||||
|                 ["This is a test document."], | ||||
|             ) | ||||
|  | ||||
|     @override_settings( | ||||
|         REMOTE_OCR_ENGINE="azureai", | ||||
|         REMOTE_OCR_API_KEY="key", | ||||
|         REMOTE_OCR_ENDPOINT="https://endpoint.cognitiveservices.azure.com", | ||||
|     ) | ||||
|     def test_supported_mime_types_valid_config(self): | ||||
|         parser = RemoteDocumentParser(uuid.uuid4()) | ||||
|         expected_types = { | ||||
|             "application/pdf": ".pdf", | ||||
|             "image/png": ".png", | ||||
|             "image/jpeg": ".jpg", | ||||
|             "image/tiff": ".tiff", | ||||
|             "image/bmp": ".bmp", | ||||
|             "image/gif": ".gif", | ||||
|             "image/webp": ".webp", | ||||
|         } | ||||
|         self.assertEqual(parser.supported_mime_types(), expected_types) | ||||
|  | ||||
|     def test_supported_mime_types_invalid_config(self): | ||||
|         parser = get_parser(uuid.uuid4()) | ||||
|         self.assertEqual(parser.supported_mime_types(), {}) | ||||
|  | ||||
|     @override_settings( | ||||
|         REMOTE_OCR_ENGINE=None, | ||||
|         REMOTE_OCR_API_KEY=None, | ||||
|         REMOTE_OCR_ENDPOINT=None, | ||||
|     ) | ||||
|     def test_parse_with_invalid_config(self): | ||||
|         parser = get_parser(uuid.uuid4()) | ||||
|         parser.parse(self.SAMPLE_FILES / "simple-digital.pdf", "application/pdf") | ||||
|         self.assertEqual(parser.text, "") | ||||
							
								
								
									
										39
									
								
								uv.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										39
									
								
								uv.lock
									
									
									
										generated
									
									
									
								
							| @@ -95,6 +95,34 @@ wheels = [ | ||||
|     { url = "https://files.pythonhosted.org/packages/02/ff/1175b0b7371e46244032d43a56862d0af455823b5280a50c63d99cc50f18/automat-25.4.16-py3-none-any.whl", hash = "sha256:04e9bce696a8d5671ee698005af6e5a9fa15354140a87f4870744604dcdd3ba1", size = 42842, upload-time = "2025-04-16T20:12:14.447Z" }, | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "azure-ai-documentintelligence" | ||||
| version = "1.0.2" | ||||
| source = { registry = "https://pypi.org/simple" } | ||||
| dependencies = [ | ||||
|     { name = "azure-core", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "isodate", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "typing-extensions", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
| ] | ||||
| sdist = { url = "https://files.pythonhosted.org/packages/44/7b/8115cd713e2caa5e44def85f2b7ebd02a74ae74d7113ba20bdd41fd6dd80/azure_ai_documentintelligence-1.0.2.tar.gz", hash = "sha256:4d75a2513f2839365ebabc0e0e1772f5601b3a8c9a71e75da12440da13b63484", size = 170940 } | ||||
| wheels = [ | ||||
|     { url = "https://files.pythonhosted.org/packages/d9/75/c9ec040f23082f54ffb1977ff8f364c2d21c79a640a13d1c1809e7fd6b1a/azure_ai_documentintelligence-1.0.2-py3-none-any.whl", hash = "sha256:e1fb446abbdeccc9759d897898a0fe13141ed29f9ad11fc705f951925822ed59", size = 106005 }, | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "azure-core" | ||||
| version = "1.33.0" | ||||
| source = { registry = "https://pypi.org/simple" } | ||||
| dependencies = [ | ||||
|     { name = "requests", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "six", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "typing-extensions", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
| ] | ||||
| sdist = { url = "https://files.pythonhosted.org/packages/75/aa/7c9db8edd626f1a7d99d09ef7926f6f4fb34d5f9fa00dc394afdfe8e2a80/azure_core-1.33.0.tar.gz", hash = "sha256:f367aa07b5e3005fec2c1e184b882b0b039910733907d001c20fb08ebb8c0eb9", size = 295633 } | ||||
| wheels = [ | ||||
|     { url = "https://files.pythonhosted.org/packages/07/b7/76b7e144aa53bd206bf1ce34fa75350472c3f69bf30e5c8c18bc9881035d/azure_core-1.33.0-py3-none-any.whl", hash = "sha256:9b5b6d0223a1d38c37500e6971118c1e0f13f54951e6893968b38910bc9cda8f", size = 207071 }, | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "babel" | ||||
| version = "2.17.0" | ||||
| @@ -1451,6 +1479,15 @@ wheels = [ | ||||
|     { url = "https://files.pythonhosted.org/packages/c7/fc/4e5a141c3f7c7bed550ac1f69e599e92b6be449dd4677ec09f325cad0955/inotifyrecursive-0.3.5-py3-none-any.whl", hash = "sha256:7e5f4a2e1dc2bef0efa3b5f6b339c41fb4599055a2b54909d020e9e932cc8d2f", size = 8009, upload-time = "2020-11-20T12:38:46.981Z" }, | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "isodate" | ||||
| version = "0.7.2" | ||||
| source = { registry = "https://pypi.org/simple" } | ||||
| sdist = { url = "https://files.pythonhosted.org/packages/54/4d/e940025e2ce31a8ce1202635910747e5a87cc3a6a6bb2d00973375014749/isodate-0.7.2.tar.gz", hash = "sha256:4cd1aa0f43ca76f4a6c6c0292a85f40b35ec2e43e315b59f06e6d32171a953e6", size = 29705 } | ||||
| wheels = [ | ||||
|     { url = "https://files.pythonhosted.org/packages/15/aa/0aca39a37d3c7eb941ba736ede56d689e7be91cab5d9ca846bde3999eba6/isodate-0.7.2-py3-none-any.whl", hash = "sha256:28009937d8031054830160fce6d409ed342816b543597cece116d966c6d99e15", size = 22320 }, | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "jinja2" | ||||
| version = "3.1.6" | ||||
| @@ -2119,6 +2156,7 @@ name = "paperless-ngx" | ||||
| version = "2.18.4" | ||||
| source = { virtual = "." } | ||||
| dependencies = [ | ||||
|     { name = "azure-ai-documentintelligence", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "babel", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "bleach", marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
|     { name = "celery", extra = ["redis"], marker = "sys_platform == 'darwin' or sys_platform == 'linux'" }, | ||||
| @@ -2255,6 +2293,7 @@ typing = [ | ||||
|  | ||||
| [package.metadata] | ||||
| requires-dist = [ | ||||
|     { name = "azure-ai-documentintelligence", specifier = ">=1.0.2" }, | ||||
|     { name = "babel", specifier = ">=2.17" }, | ||||
|     { name = "bleach", specifier = "~=6.2.0" }, | ||||
|     { name = "celery", extras = ["redis"], specifier = "~=5.5.1" }, | ||||
|   | ||||
		Reference in New Issue
	
	Block a user