Compare commits
	
		
			1129 Commits
		
	
	
		
			v1.16.2
			...
			feature-rt
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 292d7762c4 | ||
|   | 63e1f9f5d3 | ||
|   | bd4476d484 | ||
|   | 7a0334f353 | ||
|   | d03e48ea88 | ||
|   | 342e6d4679 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 584f1361ad | ||
|   | 05b1ff9738 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d65fcf70f3 | ||
|   | a5d3d51cc5 | ||
|   | f4489ca2e7 | ||
|   | e40893e74f | ||
|   | d002ae2e05 | ||
|   | bf430865b4 | ||
|   | a47d36f5e5 | ||
|   | 4392628bd7 | ||
|   | 6d25eb26a1 | ||
|   | 95fd1ae879 | ||
|   | 78f338484f | ||
|   | 40db1065dc | ||
|   | b720aa3cd1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e837f1e85b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ea2012bc81 | ||
|   | 8e39315586 | ||
|   | f009d9868e | ||
|   | 1bbcd0961b | ||
|   | 4fa2b54aed | ||
|   | 7281c110c6 | ||
|   | f812f2af4d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 47b4a602a7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca73c0d1f3 | ||
|   | 7f6a50be5b | ||
|   | 10e10f9ff4 | ||
|   | 95c24a50f7 | ||
|   | d06faa2fcb | ||
|   | bed66cced0 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ceaf60e6ad | ||
|   | 9885ca5103 | ||
|   | 2f22beaaee | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | fb2c6282a4 | ||
|   | 8c5b5d3948 | ||
|   | 4e5135fe70 | ||
|   | 579c35a3fe | ||
|   | 4aedcb856d | ||
|   | 0b34e70f6c | ||
|   | 7afc91e7b1 | ||
|   | 56b17ce6a2 | ||
|   | 954912cac3 | ||
|   | e46f6b1156 | ||
|   | 1d85caa8d0 | ||
|   | 622fcf96a0 | ||
|   | 654cc05f0e | ||
|   | 974dd24e69 | ||
|   | fe824e0faa | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 377d89ae06 | ||
|   | 629e24e031 | ||
|   | 38414025c8 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 1dc5b7a707 | ||
|   | f076418c50 | ||
|   | bbaad2cdfb | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ef01658335 | ||
|   | 9f4a6c3b42 | ||
|   | 5450bfb67b | ||
|   | ae2b302962 | ||
|   | 957691c454 | ||
|   | 6b17ba2934 | ||
|   | 4d3616cda9 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c4a9697e02 | ||
|   | c57b7520b9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 46bd09227f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 971f92a05c | ||
|   | 2c43b06910 | ||
|   | 0f8b2e69c9 | ||
|   | 00b04c2e86 | ||
|   | b3c66cae06 | ||
|   | 6a79d417b4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 98ef68f720 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ed3b7aa8f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e536600052 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bb820a2127 | ||
|   | 129933ff30 | ||
|   | 41fc11efff | ||
|   | a712bc72ca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fbe7acc6b0 | ||
|   | c4153b6fbf | ||
|   | 1f355a22e0 | ||
|   | 4af8070450 | ||
|   | d6d0071175 | ||
|   | ef51633b2c | ||
|   | d4963b9cbe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 01dabf7c05 | ||
|   | fc68f79cc8 | ||
|   | ebe1479503 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8c9fe4da06 | ||
|   | b2ef51af55 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 32b35d8e4b | ||
|   | c8bda18cf2 | ||
|   | 0a944975cc | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 48eaa31ecf | ||
|   | 1540e88a06 | ||
|   | b1aa57abcb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | df359730fe | ||
|   | 43ec154bc2 | ||
|   | 2c4a664df4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a196c14a58 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6f549506d6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8d463e05ae | ||
|   | 373c91911d | ||
|   | 1d3ac99c02 | ||
|   | cda4c8f87e | ||
|   | ef4f589094 | ||
|   | 3aeb45bf34 | ||
|   | b91da77a8a | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 33357a3fc2 | ||
|   | 025001499d | ||
|   | ccd6ad9936 | ||
|   | 979fcb0570 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 58afec98f1 | ||
|   | 91434a5c6f | ||
|   | 37c4545444 | ||
|   | 7b7b257725 | ||
|   | 1eff4b306f | ||
|   | 700af8caa2 | ||
|   | 90b43e154a | ||
|   | a3892302b0 | ||
|   | d2ee319684 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 5a6923a9aa | ||
|   | 0a634883a7 | ||
|   | 428f9cd761 | ||
|   | d828c1a2ff | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 25b49db7c0 | ||
|   | 55a40708a6 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | dae5bca883 | ||
|   | fc74da9b82 | ||
|   | 2b006907d5 | ||
|   | 1ba1afdce5 | ||
|   | a98317c52a | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ffddd0f323 | ||
|   | 4cb2f0acef | ||
|   | 98663e902f | ||
|   | 0f5e935214 | ||
|   | b9636a3def | ||
|   | 35574f3b86 | ||
|   | 00a8f0cd6e | ||
|   | 6779042242 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6379e7b54f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2fa742c94b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aa0da2f516 | ||
|   | f07441a408 | ||
|   | f6084acfc8 | ||
|   | 23ceb2a5ec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a698791059 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 41c1f38ab2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 83c85dc10e | ||
|   | a020d807d4 | ||
|   | 464ee51de8 | ||
|   | c57c1d5389 | ||
|   | af16bb3934 | ||
|   | fba416e8e1 | ||
|   | 3d8de50b5a | ||
|   | 86263a52ea | ||
|   | 754627681c | ||
|   | bf11dc8d1b | ||
|   | 84721b001f | ||
|   | f48a20c75f | ||
|   | 16f4552e0e | ||
|   | 86811d0733 | ||
|   | d2f9b5d5e5 | ||
|   | f5e1675107 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae016cae4b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c7f6d03508 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 955f2d0db9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba32684df6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f7adf40ac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a52031161b | ||
|   | 3e2c541b7b | ||
|   | 967fc98090 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22e95f45bd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 38c777ec0f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ccbc97399a | ||
|   | f43013a746 | ||
|   | 1335ab5f1b | ||
|   | 90b4691f16 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | f053ee3191 | ||
|   | 86748c1e96 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 22ded7d4c3 | ||
|   | ff5063849a | ||
|   | ec49284274 | ||
|   | 6bd5c34b54 | ||
|   | db0a2eb1a3 | ||
|   | 4948438378 | ||
|   | 76064178f5 | ||
|   | c772bd94b0 | ||
|   | 7f8f7fbb15 | ||
|   | 388d821f45 | ||
|   | 9c15623a89 | ||
|   | 966eb00de0 | ||
|   | 1c699278a3 | ||
|   | 4c6c976f63 | ||
|   | ebc9ce17b5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8039ce3c2b | ||
|   | 385d48f644 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f682fe25fc | ||
|   | 7924bf8611 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d75b909d28 | ||
|   | 47dfe85a7c | ||
|   | 4d0e8a338f | ||
|   | cfc64d37bb | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2db66280cc | ||
|   | 9f045f4494 | ||
|   | 4fdb28c8d6 | ||
|   | f1049cf889 | ||
|   | 8d664fad56 | ||
|   | f6ddcfa839 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ce59f2ad5e | ||
|   | 0de00a4ac1 | ||
|   | bec72dffeb | ||
|   | 463e95367c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 1739de2694 | ||
|   | fd8db27a88 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9ddf14bebe | ||
|   | 1cf8ea3aba | ||
|   | 134993fce6 | ||
|   | ff1955e014 | ||
|   | d83bbdc50b | ||
|   | 907b6d1294 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4e7bb1c8da | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 09ab694d05 | ||
|   | 03ced65d5f | ||
|   | 01d919cf31 | ||
|   | 8d5f331e63 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ed556ead6f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e10a904f33 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f2a05b61da | ||
|   | 21f96f0679 | ||
|   | 2ffabd54e5 | ||
|   | 1197437750 | ||
|   | d1339374d0 | ||
|   | 5ba4b9d6b2 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | b386ea9426 | ||
|   | 197174f400 | ||
|   | 97dceba783 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2ed4400827 | ||
|   | 58cbcbd6ef | ||
|   | 4aeb2e1a74 | ||
|   | 45c5f81b34 | ||
|   | 13201dbfff | ||
|   | 0b1523f4e5 | ||
|   | cd3b1a221e | ||
|   | 4855f4b8b1 | ||
|   | 6587470033 | ||
|   | 6487dab132 | ||
|   | b643a68fa3 | ||
|   | b60e16fe33 | ||
|   | c508be6ecd | ||
|   | 3b2d4fe876 | ||
|   | b47f301831 | ||
|   | a79b9de1a2 | ||
|   | e98da2e72c | ||
|   | 718171a125 | ||
|   | aaa130e20d | ||
|   | 4606caeaa8 | ||
|   | 4813a7bc70 | ||
|   | fb82aa0ee1 | ||
|   | c7e0c32226 | ||
|   | 607adf44f3 | ||
|   | 625780899d | ||
|   | 25542c56b9 | ||
|   | 45e2b7f814 | ||
|   | 6b34f592df | ||
|   | 6cf732e6ec | ||
|   | dfd959839f | ||
|   | d165e89ac3 | ||
|   | 421a87c94b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b55529b913 | ||
|   | c62d892969 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9e6aa55230 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6090305b77 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1b516a089 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 89aff63e52 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 11e9c4d8cc | ||
|   | 9d84e95771 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2aced1c305 | ||
|   | 454098630b | ||
|   | 8f3ab2791b | ||
|   | 61209b1057 | ||
|   | 38a817e887 | ||
|   | 5e3d1b26e7 | ||
|   | 4996b7e5f7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2c8fddb554 | ||
|   | ae05011062 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 50a6b7e154 | ||
|   | d0ce4113e0 | ||
|   | d55900b877 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2a73ab4693 | ||
|   | 2aea220c6d | ||
|   | b0c305e852 | ||
|   | 73a77d2a45 | ||
|   | 3a011e7c04 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d48b75d862 | ||
|   | f6e26d5953 | ||
|   | 7863780883 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c2ac9a26a2 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 58e8f796d1 | ||
|   | ba0f4718e5 | ||
|   | 6fb4daf03e | ||
|   | f6c34494a7 | ||
|   | 1e4d284b30 | ||
|   | 1f3406fd77 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 398faf36fc | ||
|   | ce841d4196 | ||
|   | c77f8acf41 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 212674f9df | ||
|   | 283ced56d1 | ||
|   | 530e57151d | ||
|   | 1141c3f361 | ||
|   | 49416d3372 | ||
|   | 88ae60a4a0 | ||
|   | 6651c80fb9 | ||
|   | 6d6650d5f6 | ||
|   | 6df252c99b | ||
|   | 5881f05dbc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 00eba3b223 | ||
|   | f7ab8d23a7 | ||
|   | 85b596d20d | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4d43f6b63d | ||
|   | ea1eb551a7 | ||
|   | 5842944d1e | ||
|   | 5781a0d51f | ||
|   | e5f48739a0 | ||
|   | d378c861f6 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9466bfdb00 | ||
|   | f02e8e0dc3 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c1ed87a44f | ||
|   | 16169ca331 | ||
|   | 26900e0766 | ||
|   | aa798604b3 | ||
|   | bb98fc5f65 | ||
|   | dc1918ad10 | ||
|   | ea632d0417 | ||
|   | 648dc709fd | ||
|   | 1a84f6a20e | ||
|   | 96af953e6f | ||
|   | 6db9e292ba | ||
|   | 2e2362e2df | ||
|   | 51dd95be3d | ||
|   | e16645b146 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0068f091bb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad6efd2898 | ||
|   | 86e380bb1c | ||
|   | 58aacd4814 | ||
|   | ad07791bac | ||
|   | 783090c2cd | ||
|   | 41a3c7c89b | ||
|   | 16cc7415c1 | ||
|   | 98c5cf89ef | ||
|   | 53e04e66cf | ||
|   | 2a6e79acc8 | ||
|   | 2da5e46386 | ||
|   | 4dbf8d7969 | ||
|   | 4a52fc27d4 | ||
|   | 05cd34c8af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8a622181fc | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 13f38bf3a1 | ||
|   | 16acc2d6ad | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c2c9a953d3 | ||
|   | 530f4a8b28 | ||
|   | 8eb1dc4f62 | ||
|   | a2b87fe012 | ||
|   | 3dcb973adb | ||
|   | 8e8810cbaa | ||
|   | b0aeec4c43 | ||
|   | 1ac298f6ff | ||
|   | 6d5f4e92cc | ||
|   | 416ad13aaf | ||
|   | a7e1299194 | ||
|   | a12e1fae72 | ||
|   | f525ac0af6 | ||
|   | 58bf9c552b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 22d257cd1f | ||
|   | f1bf1ddc54 | ||
|   | 6015cc0e4a | ||
|   | f9926d77d5 | ||
|   | 4f85dcecfc | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 30c31a3d4c | ||
|   | c64667d396 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9f6613fe05 | ||
|   | ea47af7034 | ||
|   | d46abeff01 | ||
|   | 2b39697ffb | ||
|   | 4b00a72ff5 | ||
|   | e590b2482e | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | eb7dd80410 | ||
|   | 86338465fb | ||
|   | a41dbdd12c | ||
|   | 1e10a438cd | ||
|   | ab34ea724d | ||
|   | fd8bfe1a80 | ||
|   | 9043f45350 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 5921e6d13e | ||
|   | ee2bfe2350 | ||
|   | 0957a7ca8e | ||
|   | f4e75c7fb7 | ||
|   | fae0e3b405 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ef335517ce | ||
|   | e2be166e67 | ||
|   | 37e34d92de | ||
|   | bd35030c59 | ||
|   | a82e3771ae | ||
|   | 3115106dc1 | ||
|   | d623af9c41 | ||
|   | 355a434a07 | ||
|   | 8da2535a65 | ||
|   | 5963dfe41b | ||
|   | c6dcaa0472 | ||
|   | 21063a5c22 | ||
|   | ba2f51bed1 | ||
|   | bbf64b7e93 | ||
|   | 3b6ce16f1c | ||
|   | 46e6be319f | ||
|   | e6d6f21d33 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 77b9b79a9e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f0016ad70c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 054468ffc2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 607c1282e3 | ||
|   | 3f4f4444f7 | ||
|   | 54372b5618 | ||
|   | 670a3f6c7f | ||
|   | 35a4d3fb54 | ||
|   | fb81612ed1 | ||
|   | c5d622279c | ||
|   | b93f655039 | ||
|   | 428ffb4729 | ||
|   | 061f33fb05 | ||
|   | da058b915b | ||
|   | cf869b1356 | ||
|   | 05e294fc81 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | bd904d9e6b | ||
|   | a7ac719711 | ||
|   | 370f6ddb3b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 3861e84f89 | ||
|   | 76001105b8 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d47cc9460b | ||
|   | fbeb03c377 | ||
|   | 2b13fa4712 | ||
|   | 98255f8e14 | ||
|   | eaeeb6447f | ||
|   | e6a9868e86 | ||
|   | f03592d17d | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e5db44bc2b | ||
|   | ffad42615f | ||
|   | 5576a073a5 | ||
|   | 151d337f6c | ||
|   | 74e89b0ee3 | ||
|   | 945fb675e9 | ||
|   | 7570945cf0 | ||
|   | d94d80a8c8 | ||
|   | 8a0a49dd57 | ||
|   | 66b2d90c50 | ||
|   | 5723bd8dd8 | ||
|   | 9b08ce1761 | ||
|   | a6248bec2d | ||
|   | b1f6f52486 | ||
|   | 638d9970fd | ||
|   | 5e8de4c1da | ||
|   | 088bad9030 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cfa908243b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cabeb00632 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2426c01978 | ||
|   | 80f2bee6e8 | ||
|   | 7ec51758eb | ||
|   | 9e93ae952a | ||
|   | 829836ddf6 | ||
|   | 1197a048bc | ||
|   | 7289c4ea56 | ||
|   | 55dadf0b00 | ||
|   | 341815cc03 | ||
|   | d22b27afe7 | ||
|   | 6ee2d023d5 | ||
|   | 5010cc6f15 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 383cced158 | ||
|   | a3d6967192 | ||
|   | c5881f75c9 | ||
|   | c4b7429e99 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | b1eced3612 | ||
|   | 9d5b07537d | ||
|   | 122e4141b0 | ||
|   | be2de4f15d | ||
|   | 92a920021d | ||
|   | 72000cac36 | ||
|   | 4510902677 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c2b9d2fa7b | ||
|   | cd38c39908 | ||
|   | 9016a1e6df | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 627254d5a7 | ||
|   | ff31558252 | ||
|   | 9454978264 | ||
|   | e2d25a7a09 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 85f824f032 | ||
|   | 1a48910e6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bffd5829d0 | ||
|   | 7e12bd1bef | ||
|   | af0817ab74 | ||
|   | fbf1a051a2 | ||
|   | 7ecf7f704a | ||
|   | 7b7a74d821 | ||
|   | e4acc33519 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2fd141d914 | ||
|   | aad814d342 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 7e21aaec17 | ||
|   | 6d5fdfe2e2 | ||
|   | 5942cd6cd2 | ||
|   | 5cd17e71e2 | ||
|   | 2f2ecaa61e | ||
|   | aa858a35e2 | ||
|   | d658150d42 | ||
|   | c312149b35 | ||
|   | 18a9a3df12 | ||
|   | 0cdd3581c9 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | cae79b811f | ||
|   | 6d953babcb | ||
|   | 975e5f3fd0 | ||
|   | 6cfe92bed1 | ||
|   | a51c0850c8 | ||
|   | 03415456bf | ||
|   | 0309a0fae1 | ||
|   | b48910bb94 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9a89786dd3 | ||
|   | 15a5261189 | ||
|   | f616da3b85 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 0e9cf016ec | ||
|   | 4481f12e32 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 66efaedcbb | ||
|   | 771c1fab92 | ||
|   | 8d6e7ed477 | ||
|   | 5d80511b9b | ||
|   | 90f90dc9b4 | ||
|   | a58e8498aa | ||
|   | ca355d5855 | ||
|   | 826322b610 | ||
|   | 80ff5677ea | ||
|   | 62c417cd51 | ||
|   | f27f25aa03 | ||
|   | 285a4b5aef | ||
|   | 47a2ded30d | ||
|   | 5b502b1e1a | ||
|   | aff56077a8 | ||
|   | 6e371ac5ac | ||
|   | 1b69b89d2d | ||
|   | 5a20c8e512 | ||
|   | 4ca1503beb | ||
|   | 567a7eb7f3 | ||
|   | 20f27fe32f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a50d6bb86 | ||
|   | 0b16c2db03 | ||
|   | 76ac888386 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 7cfa05d7f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e3496d0485 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d2c33c0074 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c5caecafa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 64651d5a84 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4493236879 | ||
|   | ce643942ea | ||
|   | 46d216b02f | ||
|   | 133d43ae30 | ||
|   | 27155cb7e3 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | b55c413774 | ||
|   | 69be86e16c | ||
|   | 65f6b0881e | ||
|   | c2bede40c7 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 33c2398de9 | ||
|   | 64cfc43891 | ||
|   | 6575c69409 | ||
|   | e3f4e0b775 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 5be89bfda5 | ||
|   | e1b573adeb | ||
|   | 0913c7aa9e | ||
|   | 5297626816 | ||
|   | c075642d78 | ||
|   | 253f1a44c1 | ||
|   | 26837c8871 | ||
|   | c732d31edd | ||
|   | c2595f28fb | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 58656a63cf | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f4ce178cfa | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 2c1cfc64d5 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4b84d93cee | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 0bfa347595 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c7a07b59fd | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c7c737d8c9 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 1126e668d1 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b71a94ebca | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 815137af19 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 3c4fbfec54 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b3a8eb7e06 | ||
|   | 1e5c64ff73 | ||
|   | a320bfa425 | ||
|   | 1c4dfc3c6e | ||
|   | 92ee4d33c3 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 01c2f01d3f | ||
|   | b83e6a5d5b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 16281b38e0 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ca11e116b6 | ||
|   | 1f32e4a642 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4c01089de0 | ||
|   | 47a416dd9b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 1ab1bbdd70 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 1e05cb168c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | d0383c1edf | ||
|   | c8ee35692c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 3e062a8021 | ||
|   | 0eb17e7102 | ||
|   | a14796cf90 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 84b1c1ce6c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 27e0b65c2d | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ef35576174 | ||
|   | 804faf726b | ||
|   | a94e5d2e47 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 54d817d656 | ||
|   | 00aa6c1f6a | ||
|   | ed34815393 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ceb15716c3 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | bc97f540de | ||
|   | 4ae67c79e0 | ||
|   | be857e989b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 232402197c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 12cbe84166 | ||
|   | b7b0be141c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 8f9f4d9b71 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b97b9b7a28 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | cf23732da3 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4e6b1d54e8 | ||
|   | fec8f90b78 | ||
|   | bedd4dafc4 | ||
|   | 0d92648d62 | ||
|   | 07989bc2fa | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 09ff2cb9a3 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 6aa2e60037 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 9f3ef532b8 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 654621b438 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f8cfa9e02d | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 7476aa5897 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 1d00a59706 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 82d4e5e456 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 07b6a36aab | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 0ceb6cc143 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 864a155ee8 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 15e2e575ee | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 253c524131 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 48c02b8554 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | d8fef22c9a | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | d82103e02d | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 9e5b22931e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 30cd361edb | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 48da0071bd | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 32037e3645 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4279038a62 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c8afe7503b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4dfae98250 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b44dca49f5 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 3d5892a841 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 365e06891a | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 0f436481b8 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | a5510be0bf | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 45a3c95a66 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 94cb13ce9d | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | fecc431b10 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c88a975bf0 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4ed26ab934 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 76a8da20bf | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 9f05747681 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f0bb9bb953 | ||
|   | 72c6ac09ac | ||
|   | 07c38d6353 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | dd1cafbc77 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 77e30d7844 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 1c4f772a14 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | bfc3148e4f | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 85db76c86e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f6c0d97172 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 67f846513b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ad4cf843dd | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | af79b1504f | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | caff72967e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 4abd21bbca | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | a2770a1bff | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 30e2d15321 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ba7e8b8627 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 58850e97ff | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b35acabc56 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | ef907f1bc4 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c25128917b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 9db6e64666 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f51a29a2e1 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b90025a2d0 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 03b5ae02e1 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 70627b992a | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | e830cd9baa | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 05fb10390e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | d2c802c9da | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 741c165a5b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 53d3ede184 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | a41d2f831d | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | e30b7d0fa4 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 7222741863 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 49cea717af | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 49a7b62155 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 3abc88985a | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 764b007d2c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | a5c0d3dfae | ||
|   | 386070923f | ||
|   | e79015d877 | ||
|   | b574396751 | ||
|   | 91400070a7 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 23d02085e8 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | a9459dca89 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | bd08cd1983 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 3ae91495c1 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 7f9fa46271 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f3b02d9922 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f2be582299 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | b2eb403f9b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 7e6a5927d7 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 2c4a3650c4 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 5b125d4513 | ||
|   | 66167aeb55 | ||
|   | 9804269cd1 | ||
|   | 3239c478a5 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 8f2ca5761b | ||
|   | ef9669cdb6 | ||
|   | caddcaf807 | ||
|   | 90707d661b | ||
|   | facb7226fe | ||
|   | b671f54cb7 | ||
|   | 55f26c9c4c | ||
|   | 18995b3561 | ||
|   | 8e3fbdddc7 | ||
|   | c19df84bef | ||
|   | e8527ba723 | ||
|   | 577b49df9d | ||
|   | 5b83bd03f5 | ||
|   | 73cbf6c33d | ||
|   | 95ab3b99f8 | ||
|   | 24dbf669a9 | ||
|   | 8716590cef | ||
|   | 10729f0362 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 800f54f263 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e63a543c29 | ||
|   | 0f08796e1b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d10c32bad6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 813ad6551c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5582b33d40 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 79afc236c9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f83a75d5f | ||
|   | 2a3baf5aec | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | f92ae3b232 | ||
|   | 03523244ef | ||
|   | 55e799b833 | ||
|   | 10929e9ac8 | ||
|   | 89757609c2 | ||
|   | 0f1dea67b7 | ||
|   | 38e035b95c | ||
|   | f695d4b9da | ||
|   | 4fbe7f16f3 | ||
|   | dcc2cc5001 | ||
|   | cb271deb8f | ||
|   | d480e91196 | ||
|   | 9880f9ebc7 | ||
|   | 999ae678c2 | ||
|   | 5f0eba694c | ||
|   | f893ba929a | ||
|   | d35b0423b6 | ||
|   | 651a22d056 | ||
|   | 8dee246854 | ||
|   | 90db397ec6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf973ff41e | ||
|   | fd8de5b1ea | ||
|   | 27772257a8 | ||
|   | 5665db844e | ||
|   | 21a9963a2b | ||
|   | 99f260225a | ||
|   | e4054d684c | ||
|   | 01af725d79 | ||
|   | db0f31adea | ||
|   | 226c771735 | ||
|   | b18b070622 | ||
|   | e52ae28426 | ||
|   | 6acd30eda1 | ||
|   | f170cc0354 | ||
|   | d9abae51b5 | ||
|   | ada67bd54e | ||
|   | f7f5d0efa6 | ||
|   | a2bdd64ad0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 48f4f21d28 | ||
|   | 57b8ee37ec | ||
|   | 71bf8eb332 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fb1e288580 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 720c6dd3b0 | ||
|   | be6506da08 | ||
|   | 4f6698e39f | ||
|   | cdcd22e6a6 | ||
|   | b1b3eb4406 | ||
|   | b0c6f5e56b | ||
|   | 322d8a61c2 | ||
|   | d41c4730cd | ||
|   | b38bb47491 | ||
|   | 6973691cce | ||
|   | 96176589ca | ||
|   | a74740877a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1de4072a48 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf2796b2af | ||
|   | 85315b768c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 76a147e58b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | de88c6de8c | ||
|   | 613b429540 | ||
|   | d9abb745a9 | ||
|   | 702225f535 | ||
|   | 612e6341a3 | ||
|   | 70b566f746 | ||
|   | 6dd69126ae | ||
|   | f3d6756fba | ||
|   | 8d60506884 | ||
|   | 9712ac109d | ||
|   | 379b4f8cd3 | ||
|   | 86d223fd93 | ||
|   | 205106b566 | ||
|   | 54099d8441 | ||
|   | 7c8b501c40 | ||
|   | 02bf0349ca | ||
|   | 9d72d1fc81 | ||
|   | 085e6da1f2 | ||
|   | 1ce0eae931 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ff5c25fc8c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ef863bec7c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 89117da57d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6170f4f56a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0b00f8f4f0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a45c128c38 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fe26655653 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2277fb5a58 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b6ab3095ab | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 591466d16d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 666c12f2e8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8180226eb7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0d5be4a730 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 49a2e2f3fa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 219563146d | ||
|   | 0d01295e79 | ||
|   | b65e195c27 | ||
|   | f5717cca1c | ||
|   | a2c9f6792e | ||
|   | 95c12c1840 | ||
|   | 4de1cb0a09 | ||
|   | ec9ebd3026 | ||
|   | 5ee9ad3e4f | ||
|   | 9e2135e2c7 | ||
|   | 116da276c7 | ||
|   | 7c9ab8c0b6 | ||
|   | 3a36d9b1ae | ||
|   | 5fce21e269 | ||
|   | a46bfc8b5f | ||
|   | a860c29304 | ||
|   | cc90b45022 | ||
|   | 78ae4c42f7 | ||
|   | d1292c59ea | ||
|   | 1a87c730bc | ||
|   | 7aa72f768f | ||
|   | 81b9f2d4e0 | ||
|   | a03a745295 | ||
|   | ce8bf90663 | ||
|   | e2ae919a84 | ||
|   | 9a0e44a731 | ||
|   | 650c816a7b | ||
|   | c8bfbb9315 | ||
|   | 80b32dd392 | ||
|   | e3352ea426 | ||
|   | faed7683be | ||
|   | 46a39190a4 | ||
|   | a2f738772c | ||
|   | a1697ff21c | ||
|   | b9fdf68be3 | ||
|   | ec971d0473 | ||
|   | d462464af9 | ||
|   | 423e0768f9 | ||
|   | 0ef3a141a8 | ||
|   | d532913d56 | ||
|   | 3205bb3bdf | ||
|   | b238ba054d | ||
|   | a8e13df249 | ||
|   | 9dc77d94ed | ||
|   | 714995877a | ||
|   | 729d7a11cd | ||
|   | 95cd86a541 | ||
|   | ca1b8344fa | ||
|   | e797295b82 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f216b322c2 | ||
|   | b97c19ca3c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d54c7ca27c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 077de8dcaa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6f50e5671a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bab9bdc832 | ||
|   | 8e0adbb0fb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6b175ae7e3 | ||
|   | 9494633da3 | ||
|   | c754a5f391 | ||
|   | 2b692c6fc8 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 35b6fb1a6d | ||
|   | 61566a34d1 | ||
|   | e14f4c94c2 | ||
|   | 52bdb1a80c | ||
|   | 88ee3bdb6d | ||
|   | 8f8a99a645 | ||
|   | 8d35f22daf | ||
|   | 6f8ef9bbdc | ||
|   | e7ddf6ba8f | ||
|   | 407a119b9a | ||
|   | 96aa2451bf | ||
|   | 7680aab04d | ||
|   | 3aef26b229 | ||
|   | 935141adc0 | ||
|   | 4300733d0c | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | c284a091c0 | ||
|   | 7e768bfe23 | ||
|   | 938f388a0b | ||
|   | d22a6fb8ea | ||
|   | 11142e2f05 | ||
|   | fe7fb488c0 | ||
|   | 0929c5086a | ||
|   | eb3ed7719b | ||
|   | 6c19a0f8c7 | ||
|   | a0ece589b0 | ||
|   | 8165071edf | ||
|   | a667974378 | ||
|   | fe1f88ce5d | ||
|   | 8f2715e437 | ||
|   | 57a3223c77 | ||
|   | df82ac8ac4 | ||
|   | 16adddc803 | ||
|   | d1ae82c5c2 | ||
|   | 0098936347 | ||
|   | 17b85f6400 | ||
|   | 48be9c0fd1 | ||
|   | ce36c2d0ea | ||
|   | 06c63ef4a4 | ||
|   | 03d93a7d6e | ||
|   | a0005c8b3e | ||
|   | 2220343c50 | ||
|   | 68fbed996f | ||
|   | 20a4d8949d | ||
|   | 1029ecfd49 | ||
|   | 4de2a3b09e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 651407c88e | ||
|   | 3e2e485f66 | ||
|   | 10efaad224 | ||
|   | 3dda02660c | ||
|   | 0b4e8141b0 | ||
|   | 16ab0efa59 | ||
|   | 20ca1ec547 | ||
|   | a1c9ab237f | ||
|   | a65239f7f1 | ||
|   | a0622675fd | ||
|   | 1e0b778097 | ||
|   | 3b666fef77 | ||
|   | 9291c98189 | ||
|   | f6dadd8c82 | ||
|   | 022bb272e6 | ||
|   | edad1a41e8 | ||
|   | 421e78a748 | ||
|   | b961df90a7 | ||
|   | 0b26b7098a | ||
|   | b09566a9a9 | ||
|   | b869deab66 | ||
|   | 3d552c3112 | ||
|   | 2a2bf3bf55 | ||
|   | 26863b8cdc | ||
|   | b0f7d07214 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 30c6557a32 | ||
|   | e18e173089 | ||
|   | 6a8bdbd4f6 | ||
|   | b5dec87a62 | ||
|   | e50d30876a | ||
|   | cbcd9ed67d | ||
|   | 6bcc26b487 | ||
|   | 56fcb3fee1 | ||
|   | 557e1790dd | ||
|   | 2e67697d36 | ||
|   | ca4500692f | ||
|   | 5aba6bff09 | ||
|   | 9ed74068b9 | ||
|   | be4685742c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 86b0a38811 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e528f0a9a | ||
|   | 170d7b6922 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f05249a9ad | ||
|   | 5957ed7af3 | ||
|   | 8ed63893eb | ||
|   | ea9dd926bc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f31d3b531f | ||
|   | d851448c32 | ||
|   | 65327d52a6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2ea5ae59b2 | ||
|   | a04d09028b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dd9255cb81 | ||
|   | 536d7ecd3e | ||
|   | f50aac08df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | db0f1d2159 | ||
|   | d97b565d6c | ||
|   | d3071f13d8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 768407c1d7 | ||
|   | c5d18b03cd | ||
|   | 5f05b44cde | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | beaa09e9b3 | ||
|   | d6960f537b | ||
|   | 0918eab004 | ||
|   | 9b16789a17 | ||
|   | 157240351f | ||
|   | 6ad3d45d60 | ||
|   | b715e4d426 | ||
|   | aad5e9e99f | ||
|   | 2a104ad33f | ||
|   | 851290ee89 | ||
|   | a8c6c55e3b | ||
|   | 992a647424 | ||
|   | c22461a1b6 | ||
|   | 23fefc3ab7 | ||
|   | 0beb9f0b5f | ||
|   | d376f9e7a3 | ||
|   | 07e7bcd30b | ||
|   | 95e86cb649 | ||
|   | 802e5591ce | ||
|   | 26d5730ad2 | ||
|   | 8c7554e081 | ||
|   | 9f5d47c320 | ||
|   | 4aa452ce63 | ||
|   | 7ef81ae10f | ||
|   | 3628292afa | ||
|   | 8aa5ecde62 | ||
|   | 2f149eac9d | ||
|   | fcfc705b87 | ||
|   | 7bd5c010a1 | ||
|   | 52168d8e61 | ||
|   | a3842d9228 | ||
|   | 452c51bd16 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22bedd9957 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cb318c723d | ||
|   | 9a81d3c28e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3ca59e3b7a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 07a12bdf15 | ||
|   | 62e81d8bf0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e295a41caa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c545a80aa3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 56f1a0cb51 | ||
|   | 7218b6da97 | ||
|   | 83f9f2d387 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 13a2e38385 | ||
|   | 996d942387 | ||
|   | cc42eb9fab | ||
|   | 44125be979 | ||
|   | c2e9cc9a51 | ||
|   | fcd10f2adc | ||
|   | 93009c1eed | ||
|   | d875be60d4 | ||
|   | db48d4c576 | ||
|   | 3293231ad2 | ||
|   | aa1f2d3b59 | ||
|   | f492b679e3 | ||
|   | 7ca84322bd | ||
|   | e3257b8fa3 | ||
|   | 0bcda5ded8 | ||
|   | 7ec82c0891 | ||
|   | 3241ac7dc2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e974605fc8 | ||
|   | ce13380533 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | e23e3acda3 | ||
|   | 63ab9972da | ||
|   | feb4901620 | ||
|   | 67788a1b1b | ||
|   | 001faf9ed7 | ||
|   | 7a464d8a6e | ||
|   | 5acd1c7c1b | ||
|   | de14540374 | ||
|   | 37e0c2667b | ||
|   | 252abb41c3 | ||
|   | fb2af341d8 | ||
|   | 931f5f9c27 | ||
|   | b5d04e575e | ||
|   | 3d395601fe | ||
|   | 6630ce646c | ||
|   | f5508eea1c | ||
|   | c8c460432f | ||
|   | 18299dafd2 | ||
|   | 817d09026e | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | d76b009390 | ||
|   | 9effed3ce1 | ||
|   | c1bbfc5dcf | ||
|   | b6c9cfb76f | ||
|   | 59ca7bbcf2 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 52c8d5e999 | ||
|   | 5851e7f1b7 | ||
|   | e05b3441de | ||
|   | 0d6e79cb93 | ||
|   | 76a102d901 | ||
|   | bbd4659fbf | ||
|   | 0880420ef6 | ||
|   | 2351c79282 | ||
|   | d6016fc798 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | bc17291006 | ||
|   | ed6cb14c4d | ||
|   | 08de8a04b8 | ||
|   | 5c67de8b47 | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 38b0408b1a | ||
|   | 9ccad7ea86 | ||
|   | 4a4e810a14 | ||
|   | 76d2df3bde | ||
|   | c02563d894 | ||
|   | 574ec6780b | ||
| ![Paperless-ngx Bot [bot]](/assets/img/avatar_default.png)  | 9e0f56982b | ||
|   | 1c66daf12b | ||
|   | 59d683849e | ||
|   | 9946acb1a0 | ||
|   | 83a760644d | ||
|   | 25ccff8640 | ||
|   | 5c4c5a7794 | ||
|   | cb6af97595 | ||
|   | c4407dccf6 | ||
|   | ecdea4c3c8 | ||
|   | 26d6f302cf | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ecf10622ef | ||
|   | 0fb553675b | 
							
								
								
									
										15
									
								
								.codecov.yml
									
									
									
									
									
								
							
							
						
						| @@ -1,3 +1,15 @@ | ||||
| codecov: | ||||
|   require_ci_to_pass: true | ||||
| # https://docs.codecov.com/docs/flags#recommended-automatic-flag-management | ||||
| # Require each flag to have 1 upload before notification | ||||
| flag_management: | ||||
|   individual_flags: | ||||
|     - name: backend | ||||
|       paths: | ||||
|         - src/ | ||||
|     - name: frontend | ||||
|       paths: | ||||
|         - src-ui/ | ||||
| # https://docs.codecov.com/docs/pull-request-comments | ||||
| # codecov will only comment if coverage changes | ||||
| comment: | ||||
| @@ -8,12 +20,9 @@ coverage: | ||||
|       default: | ||||
|         # https://docs.codecov.com/docs/commit-status#threshold | ||||
|         threshold: 1% | ||||
|         # https://docs.codecov.com/docs/commit-status#only_pulls | ||||
|         only_pulls: true | ||||
|     patch: | ||||
|       default: | ||||
|         # For the changed lines only, target 75% covered, but | ||||
|         # allow as low as 50% | ||||
|         target: 75% | ||||
|         threshold: 25% | ||||
|         only_pulls: true | ||||
|   | ||||
							
								
								
									
										3
									
								
								.codespellrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| [codespell] | ||||
| write-changes = True | ||||
| ignore-words-list = criterias,afterall,valeu,ureue,equest,ure | ||||
| @@ -1,21 +1,28 @@ | ||||
| # Tool caches | ||||
| **/__pycache__ | ||||
| /src-ui/.vscode | ||||
| /src-ui/node_modules | ||||
| /src-ui/dist | ||||
| **/.ruff_cache/ | ||||
| **/.mypy_cache/ | ||||
| # Virtual environment & similar | ||||
| .venv/ | ||||
| ./src-ui/node_modules | ||||
| ./src-ui/dist | ||||
| # IDE folders | ||||
| .idea/ | ||||
| .vscode/ | ||||
| ./src-ui/.vscode | ||||
| # VCS | ||||
| .git | ||||
| /export | ||||
| /consume | ||||
| /media | ||||
| /data | ||||
| /docs | ||||
| .pytest_cache | ||||
| /dist | ||||
| /scripts | ||||
| /resources | ||||
| # Test related | ||||
| **/.pytest_cache | ||||
| **/tests | ||||
| **/*.spec.ts | ||||
| **/htmlcov | ||||
| /src/.pytest_cache | ||||
| .idea | ||||
| .venv/ | ||||
| .vscode/ | ||||
| # Local folders | ||||
| ./export | ||||
| ./consume | ||||
| ./media | ||||
| ./data | ||||
| ./docs | ||||
| ./dist | ||||
| ./scripts | ||||
| ./resources | ||||
|   | ||||
							
								
								
									
										1
									
								
								.env
									
									
									
									
									
								
							
							
						
						| @@ -1,2 +1 @@ | ||||
| COMPOSE_PROJECT_NAME=paperless | ||||
| export PROMPT="(pipenv-projectname)$P$G" | ||||
|   | ||||
							
								
								
									
										24
									
								
								.github/ISSUE_TEMPLATE/bug-report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -6,14 +6,21 @@ body: | ||||
|   - type: markdown | ||||
|     attributes: | ||||
|       value: | | ||||
|         Have a question? 👉 [Start a new discussion](https://github.com/paperless-ngx/paperless-ngx/discussions/new) or [ask in chat](https://matrix.to/#/#paperlessngx:matrix.org). | ||||
|         ### ⚠️ Please remember: issues are for *bugs* | ||||
|         That is, something you believe affects every single user of Paperless-ngx, not just you. If you're not sure, start with one of the other options below. | ||||
|  | ||||
|         Before opening an issue, please double check: | ||||
|         Also, note that **Paperless-ngx does not perform OCR itself**, that is handled by other tools. Problems with OCR of specific files should likely be raised 'upstream', see https://github.com/ocrmypdf/OCRmyPDF/issues or https://github.com/tesseract-ocr/tesseract/issues | ||||
|   - type: markdown | ||||
|     attributes: | ||||
|       value: | | ||||
|         #### Have a question? 👉 [Start a new discussion](https://github.com/paperless-ngx/paperless-ngx/discussions/new) or [ask in chat](https://matrix.to/#/#paperlessngx:matrix.org). | ||||
|  | ||||
|         #### Before opening an issue, please double check: | ||||
|  | ||||
|         - [The troubleshooting documentation](https://docs.paperless-ngx.com/troubleshooting/). | ||||
|         - [The installation instructions](https://docs.paperless-ngx.com/setup/#installation). | ||||
|         - [Existing issues and discussions](https://github.com/paperless-ngx/paperless-ngx/search?q=&type=issues). | ||||
|         - Disable any customer container initialization scripts, if using any | ||||
|         - Disable any custom container initialization scripts, if using | ||||
|  | ||||
|         If you encounter issues while installing or configuring Paperless-ngx, please post in the ["Support" section of the discussions](https://github.com/paperless-ngx/paperless-ngx/discussions/new?category=support). | ||||
|   - type: textarea | ||||
| @@ -95,3 +102,14 @@ body: | ||||
|     attributes: | ||||
|       label: Other | ||||
|       description: Any other relevant details. | ||||
|   - type: checkboxes | ||||
|     id: required-checks | ||||
|     attributes: | ||||
|       label: Please confirm the following | ||||
|       options: | ||||
|         - label: I believe this issue is a bug that affects all users of Paperless-ngx, not something specific to my installation. | ||||
|           required: true | ||||
|         - label: I have already searched for relevant existing issues and discussions before opening this report. | ||||
|           required: true | ||||
|         - label: I have updated the title field above with a concise description. | ||||
|           required: true | ||||
|   | ||||
							
								
								
									
										20
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -8,7 +8,11 @@ Note: All PRs with code changes should be targeted to the `dev` branch, pure doc | ||||
| Please include a summary of the change and which issue is fixed (if any) and any relevant motivation / context. List any dependencies that are required for this change. If appropriate, please include an explanation of how your proposed change can be tested. Screenshots and / or videos can also be helpful if appropriate. | ||||
| --> | ||||
|  | ||||
| Fixes # (issue) | ||||
| <!-- | ||||
| ⚠️ Important: Pull requests that implement a new feature or enhancement *should almost always target an existing feature request* with evidence of community interest and discussion. This is in order to balance the work of implementing and maintaining new features / enhancements. If that is not currently the case, please open a feature request instead of this PR to gather feedback from both users and the project maintainers. | ||||
| --> | ||||
|  | ||||
| Closes #(issue or discussion) | ||||
|  | ||||
| ## Type of change | ||||
|  | ||||
| @@ -17,14 +21,20 @@ What type of change does your PR introduce to Paperless-ngx? | ||||
| NOTE: Please check only one box! | ||||
| --> | ||||
|  | ||||
| - [ ] Bug fix (non-breaking change which fixes an issue) | ||||
| - [ ] New feature (non-breaking change which adds functionality) | ||||
| - [ ] Breaking change (fix or feature that would cause existing functionality to not work as expected) | ||||
| - [ ] Other (please explain) | ||||
| - [ ] Bug fix: non-breaking change which fixes an issue. | ||||
| - [ ] New feature / Enhancement: non-breaking change which adds functionality. _Please read the important note above._ | ||||
| - [ ] Breaking change: fix or feature that would cause existing functionality to not work as expected. | ||||
| - [ ] Documentation only. | ||||
| - [ ] Other. Please explain: | ||||
|  | ||||
| ## Checklist: | ||||
|  | ||||
| <!-- | ||||
| NOTE: PRs that do not address the following will not be merged, please do not skip any relevant items. | ||||
| --> | ||||
|  | ||||
| - [ ] I have read & agree with the [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/main/CONTRIBUTING.md). | ||||
| - [ ] If applicable, I have included testing coverage for new code in this PR, for [backend](https://docs.paperless-ngx.com/development/#testing) and / or [front-end](https://docs.paperless-ngx.com/development/#testing-and-code-style) changes. | ||||
| - [ ] If applicable, I have tested my code for new features & regressions on both mobile & desktop devices, using the latest version of major browsers. | ||||
| - [ ] If applicable, I have checked that all tests pass, see [documentation](https://docs.paperless-ngx.com/development/#back-end-development). | ||||
| - [ ] I have run all `pre-commit` hooks, see [documentation](https://docs.paperless-ngx.com/development/#code-formatting-with-pre-commit-hooks). | ||||
|   | ||||
							
								
								
									
										44
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -8,7 +8,7 @@ updates: | ||||
|     target-branch: "dev" | ||||
|     # Look for `package.json` and `lock` files in the `/src-ui` directory | ||||
|     directory: "/src-ui" | ||||
|     # Check the npm registry for updates every month | ||||
|     open-pull-requests-limit: 10 | ||||
|     schedule: | ||||
|       interval: "monthly" | ||||
|     labels: | ||||
| @@ -17,6 +17,21 @@ updates: | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/frontend" | ||||
|     groups: | ||||
|       frontend-angular-dependencies: | ||||
|         patterns: | ||||
|           - "@angular*" | ||||
|           - "@ng-*" | ||||
|           - "ngx-*" | ||||
|           - "ng2-pdf-viewer" | ||||
|       frontend-jest-dependencies: | ||||
|         patterns: | ||||
|           - "@types/jest" | ||||
|           - "jest*" | ||||
|       frontend-eslint-dependencies: | ||||
|         patterns: | ||||
|           - "@typescript-eslint*" | ||||
|           - "eslint" | ||||
|  | ||||
|   # Enable version updates for Python | ||||
|   - package-ecosystem: "pip" | ||||
| @@ -32,8 +47,27 @@ updates: | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/backend" | ||||
|     ignore: | ||||
|       - dependency-name: "uvicorn" | ||||
|     groups: | ||||
|       development: | ||||
|         patterns: | ||||
|           - "*pytest*" | ||||
|           - "black" | ||||
|           - "ruff" | ||||
|           - "mkdocs-material" | ||||
|       django: | ||||
|         patterns: | ||||
|           - "*django*" | ||||
|       major-versions: | ||||
|         update-types: | ||||
|           - "major" | ||||
|       small-changes: | ||||
|         update-types: | ||||
|           - "minor" | ||||
|           - "patch" | ||||
|  | ||||
|   # Enable updates for Github Actions | ||||
|   # Enable updates for GitHub Actions | ||||
|   - package-ecosystem: "github-actions" | ||||
|     target-branch: "dev" | ||||
|     directory: "/" | ||||
| @@ -46,3 +80,9 @@ updates: | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/ci-cd" | ||||
|     groups: | ||||
|       actions: | ||||
|         update-types: | ||||
|           - "major" | ||||
|           - "minor" | ||||
|           - "patch" | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,23 +0,0 @@ | ||||
| # Number of days of inactivity before an issue becomes stale | ||||
| daysUntilStale: 30 | ||||
|  | ||||
| # Number of days of inactivity before a stale issue is closed | ||||
| daysUntilClose: 7 | ||||
|  | ||||
| # Only issues or pull requests with any of these labels are check if stale. Defaults to `[]` (disabled) | ||||
| any-of-labels: ['cant-reproduce','not a bug','unconfirmed'] | ||||
|  | ||||
| # Label to use when marking an issue as stale | ||||
| staleLabel: stale | ||||
|  | ||||
| # Comment to post when marking an issue as stale. Set to `false` to disable | ||||
| markComment: > | ||||
|   This issue has been automatically marked as stale because it has not had | ||||
|   recent activity. It will be closed if no further activity occurs. Thank you | ||||
|   for your contributions. | ||||
|  | ||||
| # Comment to post when closing a stale issue. Set to `false` to disable | ||||
| closeComment: false | ||||
|  | ||||
| # See https://github.com/marketplace/stale for more info on the app | ||||
| # and https://github.com/probot/stale for the configuration docs | ||||
							
								
								
									
										288
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,41 +16,47 @@ on: | ||||
| env: | ||||
|   # This is the version of pipenv all the steps will use | ||||
|   # If changing this, change Dockerfile | ||||
|   DEFAULT_PIP_ENV_VERSION: "2023.6.12" | ||||
|   # This is the default version of Python to use in most steps | ||||
|   # If changing this, change Dockerfile | ||||
|   DEFAULT_PYTHON_VERSION: "3.9" | ||||
|   DEFAULT_PIP_ENV_VERSION: "2023.12.1" | ||||
|   # This is the default version of Python to use in most steps which aren't specific | ||||
|   DEFAULT_PYTHON_VERSION: "3.10" | ||||
|  | ||||
| jobs: | ||||
|   pre-commit: | ||||
|     # We want to run on external PRs, but not on our own internal PRs as they'll be run | ||||
|     # by the push to the branch. Without this if check, checks are duplicated since | ||||
|     # internal PRs match both the push and pull_request events. | ||||
|     if: | ||||
|       github.event_name == 'push' || github.event.pull_request.head.repo.full_name != | ||||
|       github.repository | ||||
|  | ||||
|     name: Linting Checks | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout repository | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Install python | ||||
|         uses: actions/setup-python@v4 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Check files | ||||
|         uses: pre-commit/action@v3.0.0 | ||||
|         uses: pre-commit/action@v3.0.1 | ||||
|  | ||||
|   documentation: | ||||
|     name: "Build Documentation" | ||||
|     name: "Build & Deploy Documentation" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - pre-commit | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v4 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|           cache: "pipenv" | ||||
| @@ -58,7 +64,7 @@ jobs: | ||||
|       - | ||||
|         name: Install pipenv | ||||
|         run: | | ||||
|           pip install --user pipenv==${DEFAULT_PIP_ENV_VERSION} | ||||
|           pip install --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} | ||||
|       - | ||||
|         name: Install dependencies | ||||
|         run: | | ||||
| @@ -71,55 +77,44 @@ jobs: | ||||
|         name: Make documentation | ||||
|         run: | | ||||
|           pipenv --python ${{ steps.setup-python.outputs.python-version }} run mkdocs build --config-file ./mkdocs.yml | ||||
|       - | ||||
|         name: Deploy documentation | ||||
|         if: github.event_name == 'push' && github.ref == 'refs/heads/main' | ||||
|         run: | | ||||
|           echo "docs.paperless-ngx.com" > "${{ github.workspace }}/docs/CNAME" | ||||
|           git config --global user.name "${{ github.actor }}" | ||||
|           git config --global user.email "${{ github.actor }}@users.noreply.github.com" | ||||
|           pipenv --python ${{ steps.setup-python.outputs.python-version }} run mkdocs gh-deploy --force --no-history | ||||
|       - | ||||
|         name: Upload artifact | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: documentation | ||||
|           path: site/ | ||||
|           retention-days: 7 | ||||
|  | ||||
|   documentation-deploy: | ||||
|     name: "Deploy Documentation" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     if: github.event_name == 'push' && github.ref == 'refs/heads/main' | ||||
|     needs: | ||||
|       - documentation | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|       - | ||||
|         name: Deploy docs | ||||
|         uses: mhausenblas/mkdocs-deploy-gh-pages@master | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           CUSTOM_DOMAIN: docs.paperless-ngx.com | ||||
|           CONFIG_FILE: mkdocs.yml | ||||
|           EXTRA_PACKAGES: build-base | ||||
|  | ||||
|   tests-backend: | ||||
|     name: "Tests (${{ matrix.python-version }})" | ||||
|     name: "Backend Tests (Python ${{ matrix.python-version }})" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - pre-commit | ||||
|     strategy: | ||||
|       matrix: | ||||
|         python-version: ['3.8', '3.9', '3.10'] | ||||
|         python-version: ['3.9', '3.10', '3.11'] | ||||
|       fail-fast: false | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Start containers | ||||
|         run: | | ||||
|           docker compose --file ${GITHUB_WORKSPACE}/docker/compose/docker-compose.ci-test.yml pull --quiet | ||||
|           docker compose --file ${GITHUB_WORKSPACE}/docker/compose/docker-compose.ci-test.yml up --detach | ||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml pull --quiet | ||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml up --detach | ||||
|       - | ||||
|         name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v4 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: "${{ matrix.python-version }}" | ||||
|           cache: "pipenv" | ||||
| @@ -127,7 +122,7 @@ jobs: | ||||
|       - | ||||
|         name: Install pipenv | ||||
|         run: | | ||||
|           pip install --user pipenv==${DEFAULT_PIP_ENV_VERSION} | ||||
|           pip install --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} | ||||
|       - | ||||
|         name: Install system dependencies | ||||
|         run: | | ||||
| @@ -158,83 +153,164 @@ jobs: | ||||
|           cd src/ | ||||
|           pipenv --python ${{ steps.setup-python.outputs.python-version }} run pytest -ra | ||||
|       - | ||||
|         name: Upload coverage to Codecov | ||||
|         name: Upload coverage | ||||
|         if: ${{ matrix.python-version == env.DEFAULT_PYTHON_VERSION }} | ||||
|         uses: codecov/codecov-action@v3 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           # not required for public repos, but intermittently fails otherwise | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           # future expansion | ||||
|           flags: backend | ||||
|           name: backend-coverage-report | ||||
|           path: src/coverage.xml | ||||
|           retention-days: 7 | ||||
|           if-no-files-found: warn | ||||
|       - | ||||
|         name: Stop containers | ||||
|         if: always() | ||||
|         run: | | ||||
|           docker compose --file ${GITHUB_WORKSPACE}/docker/compose/docker-compose.ci-test.yml logs | ||||
|           docker compose --file ${GITHUB_WORKSPACE}/docker/compose/docker-compose.ci-test.yml down | ||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml logs | ||||
|           docker compose --file ${{ github.workspace }}/docker/compose/docker-compose.ci-test.yml down | ||||
|  | ||||
|   tests-frontend: | ||||
|     name: "Tests Frontend" | ||||
|   install-frontend-depedendencies: | ||||
|     name: "Install Frontend Dependencies" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - pre-commit | ||||
|     strategy: | ||||
|       matrix: | ||||
|         node-version: [16.x] | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|       - uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Use Node.js ${{ matrix.node-version }} | ||||
|         uses: actions/setup-node@v3 | ||||
|         name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: ${{ matrix.node-version }} | ||||
|           node-version: 20.x | ||||
|           cache: 'npm' | ||||
|           cache-dependency-path: 'src-ui/package-lock.json' | ||||
|       - name: Cache frontend dependencies | ||||
|         id: cache-frontend-deps | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/.npm | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||
|       - | ||||
|         name: Install dependencies | ||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||
|         run: cd src-ui && npm ci | ||||
|       - | ||||
|         name: Install Playwright | ||||
|         run: npx playwright install --with-deps | ||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||
|         run: cd src-ui && npx playwright install --with-deps | ||||
|  | ||||
|   tests-frontend: | ||||
|     name: "Frontend Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - install-frontend-depedendencies | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         node-version: [20.x] | ||||
|         shard-index: [1, 2, 3, 4] | ||||
|         shard-count: [4] | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
|           cache: 'npm' | ||||
|           cache-dependency-path: 'src-ui/package-lock.json' | ||||
|       - name: Cache frontend dependencies | ||||
|         id: cache-frontend-deps | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/.npm | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||
|       - name: Re-link Angular cli | ||||
|         run: cd src-ui && npm link @angular/cli | ||||
|       - | ||||
|         name: Linting checks | ||||
|         run: cd src-ui && npm run lint | ||||
|       - | ||||
|         name: Run Jest unit tests | ||||
|         run: cd src-ui && npm run test | ||||
|         run: cd src-ui && npm run test -- --max-workers=2 --shard=${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||
|       - | ||||
|         name: Upload Jest coverage | ||||
|         if: always() | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: jest-coverage-report | ||||
|           path: src-ui/coverage | ||||
|           name: jest-coverage-report-${{ matrix.shard-index }} | ||||
|           path: | | ||||
|             src-ui/coverage/coverage-final.json | ||||
|             src-ui/coverage/lcov.info | ||||
|             src-ui/coverage/clover.xml | ||||
|           retention-days: 7 | ||||
|           if-no-files-found: warn | ||||
|       - | ||||
|         name: Run Playwright e2e tests | ||||
|         run: cd src-ui && npx playwright test --shard ${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||
|       - | ||||
|         name: Upload Playwright test results | ||||
|         if: always() | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: playwright-report-${{ matrix.shard-index }} | ||||
|           path: src-ui/playwright-report | ||||
|           retention-days: 7 | ||||
|  | ||||
|   tests-coverage-upload: | ||||
|     name: "Upload Coverage" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - tests-backend | ||||
|       - tests-frontend | ||||
|     steps: | ||||
|       - | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Download frontend jest coverage | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           path: src-ui/coverage/ | ||||
|           pattern: jest-coverage-report-* | ||||
|       - | ||||
|         name: Download frontend playwright coverage | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           path: src-ui/coverage/ | ||||
|           pattern: playwright-report-* | ||||
|           merge-multiple: true | ||||
|       - | ||||
|         name: Upload frontend coverage to Codecov | ||||
|         if: always() | ||||
|         uses: codecov/codecov-action@v3 | ||||
|         uses: codecov/codecov-action@v4 | ||||
|         with: | ||||
|           # not required for public repos, but intermittently fails otherwise | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           flags: frontend | ||||
|           directory: src-ui/coverage/ | ||||
|           # dont include backend coverage files here | ||||
|           files: '!coverage.xml' | ||||
|       - | ||||
|         name: Download backend coverage | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: backend-coverage-report | ||||
|           path: src/ | ||||
|       - | ||||
|         name: Upload coverage to Codecov | ||||
|         uses: codecov/codecov-action@v4 | ||||
|         with: | ||||
|           # not required for public repos, but intermittently fails otherwise | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           # future expansion | ||||
|           flags: frontend | ||||
|       - | ||||
|         name: Run Playwright e2e tests | ||||
|         run: cd src-ui && npx playwright test | ||||
|       - | ||||
|         name: Upload Playwright test results | ||||
|         if: always() | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         with: | ||||
|           name: playwright-report | ||||
|           path: src-ui/playwright-report | ||||
|           retention-days: 7 | ||||
|           flags: backend | ||||
|           directory: src/ | ||||
|  | ||||
|   build-docker-image: | ||||
|     name: Build Docker image for ${{ github.ref_name }} | ||||
|     runs-on: ubuntu-22.04 | ||||
|     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v')) | ||||
|     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v')) | ||||
|     concurrency: | ||||
|       group: ${{ github.workflow }}-build-docker-image-${{ github.ref_name }} | ||||
|       cancel-in-progress: true | ||||
| @@ -252,7 +328,7 @@ jobs: | ||||
|         #  a tag | ||||
|         # Otherwise forks would require a Docker Hub account and secrets setup | ||||
|         run: | | ||||
|           if [[ ${{ github.repository_owner }} == "paperless-ngx" && ( ${{ github.ref_name }} == "main" || ${{ github.ref_name }} == "dev" || ${{ github.ref_name }} == "beta" || ${{ startsWith(github.ref, 'refs/tags/v') }} == "true" ) ]] ; then | ||||
|           if [[ ${{ github.repository_owner }} == "paperless-ngx" && ( ${{ github.ref_name }} == "dev" || ${{ github.ref_name }} == "beta" || ${{ startsWith(github.ref, 'refs/tags/v') }} == "true" ) ]] ; then | ||||
|             echo "Enabling DockerHub image push" | ||||
|             echo "enable=true" >> $GITHUB_OUTPUT | ||||
|           else | ||||
| @@ -269,7 +345,7 @@ jobs: | ||||
|       - | ||||
|         name: Gather Docker metadata | ||||
|         id: docker-meta | ||||
|         uses: docker/metadata-action@v4 | ||||
|         uses: docker/metadata-action@v5 | ||||
|         with: | ||||
|           images: | | ||||
|             ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }} | ||||
| @@ -284,26 +360,28 @@ jobs: | ||||
|             type=semver,pattern={{major}}.{{minor}} | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|       # If https://github.com/docker/buildx/issues/1044 is resolved, | ||||
|       # the append input with a native arm64 arch could be used to | ||||
|       # significantly speed up building | ||||
|       - | ||||
|         name: Set up Docker Buildx | ||||
|         uses: docker/setup-buildx-action@v2 | ||||
|         uses: docker/setup-buildx-action@v3 | ||||
|       - | ||||
|         name: Set up QEMU | ||||
|         uses: docker/setup-qemu-action@v2 | ||||
|         uses: docker/setup-qemu-action@v3 | ||||
|         with: | ||||
|           platforms: arm64 | ||||
|       - | ||||
|         name: Login to Github Container Registry | ||||
|         uses: docker/login-action@v2 | ||||
|         name: Login to GitHub Container Registry | ||||
|         uses: docker/login-action@v3 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.actor }} | ||||
|           password: ${{ secrets.GITHUB_TOKEN }} | ||||
|       - | ||||
|         name: Login to Docker Hub | ||||
|         uses: docker/login-action@v2 | ||||
|         uses: docker/login-action@v3 | ||||
|         # Don't attempt to login is not pushing to Docker Hub | ||||
|         if: steps.push-other-places.outputs.enable == 'true' | ||||
|         with: | ||||
| @@ -311,7 +389,7 @@ jobs: | ||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||
|       - | ||||
|         name: Login to Quay.io | ||||
|         uses: docker/login-action@v2 | ||||
|         uses: docker/login-action@v3 | ||||
|         # Don't attempt to login is not pushing to Quay.io | ||||
|         if: steps.push-other-places.outputs.enable == 'true' | ||||
|         with: | ||||
| @@ -320,11 +398,11 @@ jobs: | ||||
|           password: ${{ secrets.QUAY_ROBOT_TOKEN }} | ||||
|       - | ||||
|         name: Build and push | ||||
|         uses: docker/build-push-action@v4 | ||||
|         uses: docker/build-push-action@v5 | ||||
|         with: | ||||
|           context: . | ||||
|           file: ./Dockerfile | ||||
|           platforms: linux/amd64,linux/arm/v7,linux/arm64 | ||||
|           platforms: linux/amd64,linux/arm64 | ||||
|           push: ${{ github.event_name != 'pull_request' }} | ||||
|           tags: ${{ steps.docker-meta.outputs.tags }} | ||||
|           labels: ${{ steps.docker-meta.outputs.labels }} | ||||
| @@ -346,24 +424,26 @@ jobs: | ||||
|           docker cp frontend-extract:/usr/src/paperless/src/documents/static/frontend src/documents/static/frontend/ | ||||
|       - | ||||
|         name: Upload frontend artifact | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: frontend-compiled | ||||
|           path: src/documents/static/frontend/ | ||||
|           retention-days: 7 | ||||
|  | ||||
|   build-release: | ||||
|     name: "Build Release" | ||||
|     needs: | ||||
|       - build-docker-image | ||||
|       - documentation | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v4 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|           cache: "pipenv" | ||||
| @@ -371,11 +451,17 @@ jobs: | ||||
|       - | ||||
|         name: Install pipenv + tools | ||||
|         run: | | ||||
|           pip install --upgrade --user pipenv==${DEFAULT_PIP_ENV_VERSION} setuptools wheel | ||||
|           pip install --upgrade --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} setuptools wheel | ||||
|       - | ||||
|         name: Install Python dependencies | ||||
|         run: | | ||||
|           pipenv --python ${{ steps.setup-python.outputs.python-version }} sync --dev | ||||
|       - | ||||
|         name: Patch whitenoise | ||||
|         run: | | ||||
|           curl --fail --silent --show-error --location --output 484.patch https://github.com/evansd/whitenoise/pull/484.patch | ||||
|           patch -d $(pipenv --venv)/lib/python3.10/site-packages --verbose -p2 < 484.patch | ||||
|           rm 484.patch | ||||
|       - | ||||
|         name: Install system dependencies | ||||
|         run: | | ||||
| @@ -383,13 +469,13 @@ jobs: | ||||
|           sudo apt-get install -qq --no-install-recommends gettext liblept5 | ||||
|       - | ||||
|         name: Download frontend artifact | ||||
|         uses: actions/download-artifact@v3 | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: frontend-compiled | ||||
|           path: src/documents/static/frontend/ | ||||
|       - | ||||
|         name: Download documentation artifact | ||||
|         uses: actions/download-artifact@v3 | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: documentation | ||||
|           path: docs/_build/html/ | ||||
| @@ -455,13 +541,14 @@ jobs: | ||||
|           tar -cJf paperless-ngx.tar.xz paperless-ngx/ | ||||
|       - | ||||
|         name: Upload release artifact | ||||
|         uses: actions/upload-artifact@v3 | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: release | ||||
|           path: dist/paperless-ngx.tar.xz | ||||
|           retention-days: 7 | ||||
|  | ||||
|   publish-release: | ||||
|     name: "Publish Release" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     outputs: | ||||
|       prerelease: ${{ steps.get_version.outputs.prerelease }} | ||||
| @@ -473,7 +560,7 @@ jobs: | ||||
|     steps: | ||||
|       - | ||||
|         name: Download release artifact | ||||
|         uses: actions/download-artifact@v3 | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: release | ||||
|           path: ./ | ||||
| @@ -490,7 +577,7 @@ jobs: | ||||
|       - | ||||
|         name: Create Release and Changelog | ||||
|         id: create-release | ||||
|         uses: release-drafter/release-drafter@v5 | ||||
|         uses: release-drafter/release-drafter@v6 | ||||
|         with: | ||||
|           name: Paperless-ngx ${{ steps.get_version.outputs.version }} | ||||
|           tag: ${{ steps.get_version.outputs.version }} | ||||
| @@ -511,6 +598,7 @@ jobs: | ||||
|           asset_content_type: application/x-xz | ||||
|  | ||||
|   append-changelog: | ||||
|     name: "Append Changelog" | ||||
|     runs-on: ubuntu-22.04 | ||||
|     needs: | ||||
|       - publish-release | ||||
| @@ -518,12 +606,12 @@ jobs: | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           ref: main | ||||
|       - | ||||
|         name: Set up Python | ||||
|         uses: actions/setup-python@v4 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|           cache: "pipenv" | ||||
| @@ -531,7 +619,7 @@ jobs: | ||||
|       - | ||||
|         name: Install pipenv + tools | ||||
|         run: | | ||||
|           pip install --upgrade --user pipenv==${DEFAULT_PIP_ENV_VERSION} setuptools wheel | ||||
|           pip install --upgrade --user pipenv==${{ env.DEFAULT_PIP_ENV_VERSION }} setuptools wheel | ||||
|       - | ||||
|         name: Append Changelog to docs | ||||
|         id: append-Changelog | ||||
| @@ -552,12 +640,12 @@ jobs: | ||||
|           git push origin ${{ needs.publish-release.outputs.version }}-changelog | ||||
|       - | ||||
|         name: Create Pull Request | ||||
|         uses: actions/github-script@v6 | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             const { repo, owner } = context.repo; | ||||
|             const result = await github.rest.pulls.create({ | ||||
|               title: '[Documentation] Add ${{ needs.publish-release.outputs.version }} changelog', | ||||
|               title: 'Documentation: Add ${{ needs.publish-release.outputs.version }} changelog', | ||||
|               owner, | ||||
|               repo, | ||||
|               head: '${{ needs.publish-release.outputs.version }}-changelog', | ||||
|   | ||||
							
								
								
									
										27
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -19,9 +19,13 @@ concurrency: | ||||
|  | ||||
| jobs: | ||||
|   cleanup-images: | ||||
|     name: Cleanup Image Tags for paperless-ngx | ||||
|     name: Cleanup Image Tags for ${{ matrix.primary-name }} | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-22.04 | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         primary-name: ["paperless-ngx", "paperless-ngx/builder/cache/app"] | ||||
|     env: | ||||
|       # Requires a personal access token with the OAuth scope delete:packages | ||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||
| @@ -29,15 +33,15 @@ jobs: | ||||
|       - | ||||
|         name: Clean temporary images | ||||
|         if: "${{ env.TOKEN != '' }}" | ||||
|         uses: stumpylog/image-cleaner-action/ephemeral@v0.1.0 | ||||
|         uses: stumpylog/image-cleaner-action/ephemeral@v0.5.0 | ||||
|         with: | ||||
|           token: "${{ env.TOKEN }}" | ||||
|           owner: "${{ github.repository_owner }}" | ||||
|           is_org: "true" | ||||
|           package_name: "paperless-ngx" | ||||
|           package_name: "${{ matrix.primary-name }}" | ||||
|           scheme: "branch" | ||||
|           repo_name: "paperless-ngx" | ||||
|           match_regex: "feature-" | ||||
|           match_regex: "(feature|fix)" | ||||
|           do_delete: "true" | ||||
|  | ||||
|   cleanup-untagged-images: | ||||
| @@ -49,18 +53,7 @@ jobs: | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         include: | ||||
|           - primary-name: "paperless-ngx" | ||||
|           - primary-name: "paperless-ngx/builder/cache/app" | ||||
|           # TODO: Remove the above and replace with the below | ||||
|           # - primary-name: "builder/qpdf" | ||||
|           # - primary-name: "builder/cache/qpdf" | ||||
|           # - primary-name: "builder/pikepdf" | ||||
|           # - primary-name: "builder/cache/pikepdf" | ||||
|           # - primary-name: "builder/jbig2enc" | ||||
|           # - primary-name: "builder/cache/jbig2enc" | ||||
|           # - primary-name: "builder/psycopg2" | ||||
|           # - primary-name: "builder/cache/psycopg2" | ||||
|         primary-name: ["paperless-ngx", "paperless-ngx/builder/cache/app"] | ||||
|     env: | ||||
|       # Requires a personal access token with the OAuth scope delete:packages | ||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||
| @@ -68,7 +61,7 @@ jobs: | ||||
|       - | ||||
|         name: Clean untagged images | ||||
|         if: "${{ env.TOKEN != '' }}" | ||||
|         uses: stumpylog/image-cleaner-action/untagged@v0.1.0 | ||||
|         uses: stumpylog/image-cleaner-action/untagged@v0.5.0 | ||||
|         with: | ||||
|           token: "${{ env.TOKEN }}" | ||||
|           owner: "${{ github.repository_owner }}" | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -38,11 +38,11 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout repository | ||||
|       uses: actions/checkout@v3 | ||||
|       uses: actions/checkout@v4 | ||||
|  | ||||
|     # Initializes the CodeQL tools for scanning. | ||||
|     - name: Initialize CodeQL | ||||
|       uses: github/codeql-action/init@v2 | ||||
|       uses: github/codeql-action/init@v3 | ||||
|       with: | ||||
|         languages: ${{ matrix.language }} | ||||
|         # If you wish to specify custom queries, you can do so here or in a config file. | ||||
| @@ -51,4 +51,4 @@ jobs: | ||||
|         # queries: ./path/to/local/query, your-org/your-repo/queries@main | ||||
|  | ||||
|     - name: Perform CodeQL Analysis | ||||
|       uses: github/codeql-action/analyze@v2 | ||||
|       uses: github/codeql-action/analyze@v3 | ||||
|   | ||||
							
								
								
									
										34
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | ||||
| name: Crowdin Action | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: '2 */12 * * *' | ||||
|   push: | ||||
|     paths: [ | ||||
|       'src/locale/**', | ||||
|       'src-ui/messages.xlf', | ||||
|       'src-ui/src/locale/**' | ||||
|     ] | ||||
|     branches: [ dev ] | ||||
|  | ||||
| jobs: | ||||
|   synchronize-with-crowdin: | ||||
|     name: Crowdin Sync | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout | ||||
|       uses: actions/checkout@v4 | ||||
|     - name: crowdin action | ||||
|       uses: crowdin/github-action@v1 | ||||
|       with: | ||||
|         upload_translations: false | ||||
|         download_translations: true | ||||
|         crowdin_branch_name: 'dev' | ||||
|         localization_branch_name: l10n_dev | ||||
|         pull_request_labels: 'skip-changelog, translation' | ||||
|       env: | ||||
|         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|         CROWDIN_PROJECT_ID: ${{ secrets.CROWDIN_PROJECT_ID }} | ||||
|         CROWDIN_PERSONAL_TOKEN: ${{ secrets.CROWDIN_PERSONAL_TOKEN }} | ||||
							
								
								
									
										32
									
								
								.github/workflows/project-actions.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,10 +1,6 @@ | ||||
| name: Project Automations | ||||
|  | ||||
| on: | ||||
|   issues: | ||||
|     types: | ||||
|       - opened | ||||
|       - reopened | ||||
|   pull_request_target: #_target allows access to secrets | ||||
|     types: | ||||
|       - opened | ||||
| @@ -16,25 +12,7 @@ on: | ||||
| permissions: | ||||
|   contents: read | ||||
|  | ||||
| env: | ||||
|   todo: Todo | ||||
|   done: Done | ||||
|   in_progress: In Progress | ||||
|  | ||||
| jobs: | ||||
|   issue_opened_or_reopened: | ||||
|     name: issue_opened_or_reopened | ||||
|     runs-on: ubuntu-22.04 | ||||
|     if: github.event_name == 'issues' && (github.event.action == 'opened' || github.event.action == 'reopened') | ||||
|     steps: | ||||
|       - name: Add issue to project and set status to ${{ env.todo }} | ||||
|         uses: leonsteinhaeuser/project-beta-automations@v2.1.0 | ||||
|         with: | ||||
|           gh_token: ${{ secrets.GH_TOKEN }} | ||||
|           organization: paperless-ngx | ||||
|           project_id: 2 | ||||
|           resource_node_id: ${{ github.event.issue.node_id }} | ||||
|           status_value: ${{ env.todo }} # Target status | ||||
|   pr_opened_or_reopened: | ||||
|     name: pr_opened_or_reopened | ||||
|     runs-on: ubuntu-22.04 | ||||
| @@ -43,15 +21,7 @@ jobs: | ||||
|       pull-requests: write | ||||
|     if: github.event_name == 'pull_request_target' && (github.event.action == 'opened' || github.event.action == 'reopened') && github.event.pull_request.user.login != 'dependabot' | ||||
|     steps: | ||||
|       - name: Add PR to project and set status to "Needs Review" | ||||
|         uses: leonsteinhaeuser/project-beta-automations@v2.1.0 | ||||
|         with: | ||||
|           gh_token: ${{ secrets.GH_TOKEN }} | ||||
|           organization: paperless-ngx | ||||
|           project_id: 2 | ||||
|           resource_node_id: ${{ github.event.pull_request.node_id }} | ||||
|           status_value: "Needs Review" # Target status | ||||
|       - name: Label PR with release-drafter | ||||
|         uses: release-drafter/release-drafter@v5 | ||||
|         uses: release-drafter/release-drafter@v6 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										246
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -8,6 +8,7 @@ on: | ||||
| permissions: | ||||
|   issues: write | ||||
|   pull-requests: write | ||||
|   discussions: write | ||||
|  | ||||
| concurrency: | ||||
|   group: lock | ||||
| @@ -17,31 +18,264 @@ jobs: | ||||
|     name: 'Stale' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v8 | ||||
|       - uses: actions/stale@v9 | ||||
|         with: | ||||
|           days-before-stale: 30 | ||||
|           days-before-close: 7 | ||||
|           only-labels: 'cant-reproduce' | ||||
|           days-before-stale: 7 | ||||
|           days-before-close: 14 | ||||
|           any-of-labels: 'cant-reproduce,not a bug' | ||||
|           stale-issue-label: stale | ||||
|           stale-pr-label: stale | ||||
|           stale-issue-message: > | ||||
|             This issue has been automatically marked as stale because it has not had | ||||
|             recent activity. It will be closed if no further activity occurs. Thank you | ||||
|             for your contributions. | ||||
|             for your contributions. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|   lock-threads: | ||||
|     name: 'Lock Old Threads' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@v4 | ||||
|       - uses: dessant/lock-threads@v5 | ||||
|         with: | ||||
|           issue-inactive-days: '30' | ||||
|           pr-inactive-days: '30' | ||||
|           discussion-inactive-days: '30' | ||||
|           log-output: true | ||||
|           issue-comment: > | ||||
|             This issue has been automatically locked since there | ||||
|             has not been any recent activity after it was closed. | ||||
|             Please open a new discussion or issue for related concerns. | ||||
|             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|           pr-comment: > | ||||
|             This pull request has been automatically locked since there | ||||
|             has not been any recent activity after it was closed. | ||||
|             Please open a new discussion or issue for related concerns. | ||||
|             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|           discussion-comment: > | ||||
|             This discussion has been automatically locked since there | ||||
|             has not been any recent activity after it was closed. | ||||
|             Please open a new discussion for related concerns. | ||||
|             See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|   close-answered-discussions: | ||||
|     name: 'Close Answered Discussions' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             function sleep(ms) { | ||||
|               return new Promise(resolve => setTimeout(resolve, ms)); | ||||
|             } | ||||
|  | ||||
|             const query = `query($owner:String!, $name:String!) { | ||||
|               repository(owner:$owner, name:$name){ | ||||
|                 discussions(first:100, answered:true, states:[OPEN]) { | ||||
|                   nodes { | ||||
|                     id, | ||||
|                     number | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|             }`; | ||||
|             const variables = { | ||||
|               owner: context.repo.owner, | ||||
|               name: context.repo.repo, | ||||
|             } | ||||
|             const result = await github.graphql(query, variables) | ||||
|  | ||||
|             console.log(`Found ${result.repository.discussions.nodes.length} open answered discussions`) | ||||
|  | ||||
|             for (const discussion of result.repository.discussions.nodes) { | ||||
|               console.log(`Closing discussion #${discussion.number} (${discussion.id})`) | ||||
|  | ||||
|               const addCommentMutation = `mutation($discussion:ID!, $body:String!) { | ||||
|                 addDiscussionComment(input:{discussionId:$discussion, body:$body}) { | ||||
|                   clientMutationId | ||||
|                 } | ||||
|               }`; | ||||
|               const commentVariables = { | ||||
|                 discussion: discussion.id, | ||||
|                 body: 'This discussion has been automatically closed because it was marked as answered. Please see our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details.', | ||||
|               } | ||||
|               await github.graphql(addCommentMutation, commentVariables) | ||||
|  | ||||
|               const closeDiscussionMutation = `mutation($discussion:ID!, $reason:DiscussionCloseReason!) { | ||||
|                 closeDiscussion(input:{discussionId:$discussion, reason:$reason}) { | ||||
|                   clientMutationId | ||||
|                 } | ||||
|               }`; | ||||
|               const closeVariables = { | ||||
|                 discussion: discussion.id, | ||||
|                 reason: "RESOLVED", | ||||
|               } | ||||
|               await github.graphql(closeDiscussionMutation, closeVariables) | ||||
|  | ||||
|               await sleep(1000) | ||||
|             } | ||||
|   close-outdated-discussions: | ||||
|     name: 'Close Outdated Discussions' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             function sleep(ms) { | ||||
|               return new Promise(resolve => setTimeout(resolve, ms)); | ||||
|             } | ||||
|  | ||||
|             const CUTOFF_DAYS = 180; | ||||
|             const cutoff = new Date(); | ||||
|             cutoff.setDate(cutoff.getDate() - CUTOFF_DAYS); | ||||
|  | ||||
|             const query = `query( | ||||
|                 $owner:String!, | ||||
|                 $name:String!, | ||||
|                 $supportCategory:ID!, | ||||
|                 $generalCategory:ID!, | ||||
|               ) { | ||||
|               supportDiscussions: repository(owner:$owner, name:$name){ | ||||
|                 discussions( | ||||
|                   categoryId:$supportCategory, | ||||
|                   last:50, | ||||
|                   answered:false, | ||||
|                   states:[OPEN], | ||||
|                 ) { | ||||
|                   nodes { | ||||
|                     id, | ||||
|                     number, | ||||
|                     updatedAt | ||||
|                   } | ||||
|                 }, | ||||
|               }, | ||||
|               generalDiscussions: repository(owner:$owner, name:$name){ | ||||
|                 discussions( | ||||
|                   categoryId:$generalCategory, | ||||
|                   last:50, | ||||
|                   states:[OPEN], | ||||
|                 ) { | ||||
|                   nodes { | ||||
|                     id, | ||||
|                     number, | ||||
|                     updatedAt | ||||
|                   } | ||||
|                 } | ||||
|               } | ||||
|             }`; | ||||
|             const variables = { | ||||
|               owner: context.repo.owner, | ||||
|               name: context.repo.repo, | ||||
|               supportCategory: "DIC_kwDOG1Zs184CBKWK", | ||||
|               generalCategory: "DIC_kwDOG1Zs184CBKWJ" | ||||
|             } | ||||
|             const result = await github.graphql(query, variables); | ||||
|             const combinedDiscussions = [ | ||||
|               ...result.supportDiscussions.discussions.nodes, | ||||
|               ...result.generalDiscussions.discussions.nodes, | ||||
|             ] | ||||
|  | ||||
|             console.log(`Checking ${combinedDiscussions.length} open discussions`); | ||||
|  | ||||
|             for (const discussion of combinedDiscussions) { | ||||
|               if (new Date(discussion.updatedAt) < cutoff) { | ||||
|                 console.log(`Closing outdated discussion #${discussion.number} (${discussion.id}), last updated at ${discussion.updatedAt}`); | ||||
|                 const addCommentMutation = `mutation($discussion:ID!, $body:String!) { | ||||
|                   addDiscussionComment(input:{discussionId:$discussion, body:$body}) { | ||||
|                     clientMutationId | ||||
|                   } | ||||
|                 }`; | ||||
|                 const commentVariables = { | ||||
|                   discussion: discussion.id, | ||||
|                   body: 'This discussion has been automatically closed due to inactivity. Please see our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details.', | ||||
|                 } | ||||
|                 await github.graphql(addCommentMutation, commentVariables); | ||||
|  | ||||
|                 const closeDiscussionMutation = `mutation($discussion:ID!, $reason:DiscussionCloseReason!) { | ||||
|                   closeDiscussion(input:{discussionId:$discussion, reason:$reason}) { | ||||
|                     clientMutationId | ||||
|                   } | ||||
|                 }`; | ||||
|                 const closeVariables = { | ||||
|                   discussion: discussion.id, | ||||
|                   reason: "OUTDATED", | ||||
|                 } | ||||
|                 await github.graphql(closeDiscussionMutation, closeVariables); | ||||
|  | ||||
|                 await sleep(1000); | ||||
|               } | ||||
|             } | ||||
|   close-unsupported-feature-requests: | ||||
|     name: 'Close Unsupported Feature Requests' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             function sleep(ms) { | ||||
|               return new Promise(resolve => setTimeout(resolve, ms)); | ||||
|             } | ||||
|  | ||||
|             const CUTOFF_1_DAYS = 180; | ||||
|             const CUTOFF_1_COUNT = 5; | ||||
|             const CUTOFF_2_DAYS = 365; | ||||
|             const CUTOFF_2_COUNT = 10; | ||||
|  | ||||
|             const cutoff1Date = new Date(); | ||||
|             cutoff1Date.setDate(cutoff1Date.getDate() - CUTOFF_1_DAYS); | ||||
|             const cutoff2Date = new Date(); | ||||
|             cutoff2Date.setDate(cutoff2Date.getDate() - CUTOFF_2_DAYS); | ||||
|  | ||||
|             const query = `query( | ||||
|                 $owner:String!, | ||||
|                 $name:String!, | ||||
|                 $featureRequestsCategory:ID!, | ||||
|               ) { | ||||
|               repository(owner:$owner, name:$name){ | ||||
|                 discussions( | ||||
|                   categoryId:$featureRequestsCategory, | ||||
|                   last:100, | ||||
|                   states:[OPEN], | ||||
|                 ) { | ||||
|                   nodes { | ||||
|                     id, | ||||
|                     number, | ||||
|                     updatedAt, | ||||
|                     upvoteCount, | ||||
|                   } | ||||
|                 }, | ||||
|               } | ||||
|             }`; | ||||
|             const variables = { | ||||
|               owner: context.repo.owner, | ||||
|               name: context.repo.repo, | ||||
|               featureRequestsCategory: "DIC_kwDOG1Zs184CBNr4" | ||||
|             } | ||||
|             const result = await github.graphql(query, variables); | ||||
|  | ||||
|             for (const discussion of result.repository.discussions.nodes) { | ||||
|               const discussionDate = new Date(discussion.updatedAt); | ||||
|               if ((discussionDate < cutoff1Date && discussion.upvoteCount < CUTOFF_1_COUNT) || | ||||
|                   (discussionDate < cutoff2Date && discussion.upvoteCount < CUTOFF_2_COUNT)) { | ||||
|                 console.log(`Closing discussion #${discussion.number} (${discussion.id}), last updated at ${discussion.updatedAt} with votes ${discussion.upvoteCount}`); | ||||
|                 const addCommentMutation = `mutation($discussion:ID!, $body:String!) { | ||||
|                   addDiscussionComment(input:{discussionId:$discussion, body:$body}) { | ||||
|                     clientMutationId | ||||
|                   } | ||||
|                 }`; | ||||
|                 const commentVariables = { | ||||
|                   discussion: discussion.id, | ||||
|                   body: 'This discussion has been automatically closed due to lack of community support. Please see our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details.', | ||||
|                 } | ||||
|                 await github.graphql(addCommentMutation, commentVariables); | ||||
|  | ||||
|                 const closeDiscussionMutation = `mutation($discussion:ID!, $reason:DiscussionCloseReason!) { | ||||
|                   closeDiscussion(input:{discussionId:$discussion, reason:$reason}) { | ||||
|                     clientMutationId | ||||
|                   } | ||||
|                 }`; | ||||
|                 const closeVariables = { | ||||
|                   discussion: discussion.id, | ||||
|                   reason: "OUTDATED", | ||||
|                 } | ||||
|                 await github.graphql(closeDiscussionMutation, closeVariables); | ||||
|  | ||||
|                 await sleep(1000); | ||||
|               } | ||||
|             } | ||||
|   | ||||
| @@ -5,12 +5,14 @@ | ||||
| repos: | ||||
|   # General hooks | ||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|     rev: v4.4.0 | ||||
|     rev: v4.6.0 | ||||
|     hooks: | ||||
|       - id: check-docstring-first | ||||
|       - id: check-json | ||||
|         exclude: "tsconfig.*json" | ||||
|       - id: check-yaml | ||||
|         args: | ||||
|           - "--unsafe" | ||||
|       - id: check-toml | ||||
|       - id: check-executables-have-shebangs | ||||
|       - id: end-of-file-fixer | ||||
| @@ -26,8 +28,16 @@ repos: | ||||
|           - svg | ||||
|       - id: check-case-conflict | ||||
|       - id: detect-private-key | ||||
|   - repo: https://github.com/codespell-project/codespell | ||||
|     rev: v2.2.6 | ||||
|     hooks: | ||||
|       - id: codespell | ||||
|         exclude: "(^src-ui/src/locale/)|(^src-ui/e2e/)|(^src/paperless_mail/tests/samples/)" | ||||
|         exclude_types: | ||||
|           - pofile | ||||
|           - json | ||||
|   - repo: https://github.com/pre-commit/mirrors-prettier | ||||
|     rev: 'v2.7.1' | ||||
|     rev: 'v3.1.0' | ||||
|     hooks: | ||||
|       - id: prettier | ||||
|         types_or: | ||||
| @@ -36,17 +46,17 @@ repos: | ||||
|           - markdown | ||||
|         exclude: "(^Pipfile\\.lock$)" | ||||
|   # Python hooks | ||||
|   - repo: https://github.com/charliermarsh/ruff-pre-commit | ||||
|     rev: 'v0.0.272' | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: 'v0.4.1' | ||||
|     hooks: | ||||
|       - id: ruff | ||||
|   - repo: https://github.com/psf/black | ||||
|     rev: 23.3.0 | ||||
|   - repo: https://github.com/psf/black-pre-commit-mirror | ||||
|     rev: 24.4.0 | ||||
|     hooks: | ||||
|       - id: black | ||||
|   # Dockerfile hooks | ||||
|   - repo: https://github.com/AleksaC/hadolint-py | ||||
|     rev: v2.12.0.2 | ||||
|     rev: v2.12.0.3 | ||||
|     hooks: | ||||
|       - id: hadolint | ||||
|   # Shell script hooks | ||||
| @@ -57,6 +67,6 @@ repos: | ||||
|         args: | ||||
|           - "--tab" | ||||
|   - repo: https://github.com/shellcheck-py/shellcheck-py | ||||
|     rev: "v0.9.0.5" | ||||
|     rev: "v0.10.0.1" | ||||
|     hooks: | ||||
|       - id: shellcheck | ||||
|   | ||||
							
								
								
									
										20
									
								
								.prettierrc
									
									
									
									
									
								
							
							
						
						| @@ -1,4 +1,16 @@ | ||||
| # https://prettier.io/docs/en/options.html#semicolons | ||||
| semi: false | ||||
| # https://prettier.io/docs/en/options.html#quotes | ||||
| singleQuote: true | ||||
| { | ||||
|     # https://prettier.io/docs/en/options.html#semicolons | ||||
|     "semi": false, | ||||
|     # https://prettier.io/docs/en/options.html#quotes | ||||
|     "singleQuote": true, | ||||
|     # https://prettier.io/docs/en/options.html#trailing-commas | ||||
|     "trailingComma": "es5", | ||||
|     "overrides": [ | ||||
|         { | ||||
|             "files": ["index.md", "administration.md"], | ||||
|             "options": { | ||||
|                 "tabWidth": 4 | ||||
|             } | ||||
|         } | ||||
|     ] | ||||
| } | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| 3.8.16 | ||||
| 3.9.18 | ||||
|   | ||||
							
								
								
									
										46
									
								
								.ruff.toml
									
									
									
									
									
								
							
							
						
						| @@ -1,23 +1,47 @@ | ||||
| # https://beta.ruff.rs/docs/settings/ | ||||
| # https://beta.ruff.rs/docs/rules/ | ||||
| extend-select = ["I", "W", "UP", "COM", "DJ", "EXE", "ISC", "ICN", "G201", "INP", "PIE", "RSE", "SIM", "TID", "PLC", "PLE", "RUF"] | ||||
| # TODO PTH | ||||
| ignore = ["DJ001", "SIM105"] | ||||
| fix = true | ||||
| line-length = 88 | ||||
| respect-gitignore = true | ||||
| src = ["src"] | ||||
| target-version = "py38" | ||||
| format = "grouped" | ||||
| target-version = "py39" | ||||
| output-format = "grouped" | ||||
| show-fixes = true | ||||
|  | ||||
| [per-file-ignores] | ||||
| # https://docs.astral.sh/ruff/settings/ | ||||
| # https://docs.astral.sh/ruff/rules/ | ||||
| [lint] | ||||
| extend-select = [ | ||||
|   "W",     # https://docs.astral.sh/ruff/rules/#pycodestyle-e-w | ||||
|   "I",     # https://docs.astral.sh/ruff/rules/#isort-i | ||||
|   "UP",    # https://docs.astral.sh/ruff/rules/#pyupgrade-up | ||||
|   "COM",   # https://docs.astral.sh/ruff/rules/#flake8-commas-com | ||||
|   "DJ",    # https://docs.astral.sh/ruff/rules/#flake8-django-dj | ||||
|   "EXE",   # https://docs.astral.sh/ruff/rules/#flake8-executable-exe | ||||
|   "ISC",   # https://docs.astral.sh/ruff/rules/#flake8-implicit-str-concat-isc | ||||
|   "ICN",   # https://docs.astral.sh/ruff/rules/#flake8-import-conventions-icn | ||||
|   "G201",  # https://docs.astral.sh/ruff/rules/#flake8-logging-format-g | ||||
|   "INP",   # https://docs.astral.sh/ruff/rules/#flake8-no-pep420-inp | ||||
|   "PIE",   # https://docs.astral.sh/ruff/rules/#flake8-pie-pie | ||||
|   "Q",     # https://docs.astral.sh/ruff/rules/#flake8-quotes-q | ||||
|   "RSE",   # https://docs.astral.sh/ruff/rules/#flake8-raise-rse | ||||
|   "T20",   # https://docs.astral.sh/ruff/rules/#flake8-print-t20 | ||||
|   "SIM",   # https://docs.astral.sh/ruff/rules/#flake8-simplify-sim | ||||
|   "TID",   # https://docs.astral.sh/ruff/rules/#flake8-tidy-imports-tid | ||||
|   "TCH",   # https://docs.astral.sh/ruff/rules/#flake8-type-checking-tch | ||||
|   "PLC",   # https://docs.astral.sh/ruff/rules/#pylint-pl | ||||
|   "PLE",   # https://docs.astral.sh/ruff/rules/#pylint-pl | ||||
|   "RUF",   # https://docs.astral.sh/ruff/rules/#ruff-specific-rules-ruf | ||||
|   "FLY",   # https://docs.astral.sh/ruff/rules/#flynt-fly | ||||
| ] | ||||
| # TODO PTH https://docs.astral.sh/ruff/rules/#flake8-use-pathlib-pth | ||||
| ignore = ["DJ001", "SIM105", "RUF012"] | ||||
|  | ||||
| [lint.per-file-ignores] | ||||
| ".github/scripts/*.py" = ["E501", "INP001", "SIM117"] | ||||
| "docker/wait-for-redis.py" = ["INP001"] | ||||
| "docker/wait-for-redis.py" = ["INP001", "T201"] | ||||
| "*/tests/*.py" = ["E501", "SIM117"] | ||||
| "*/migrations/*.py" = ["E501", "SIM"] | ||||
| "*/migrations/*.py" = ["E501", "SIM", "T201"] | ||||
| "src/paperless_tesseract/tests/test_parser.py" = ["RUF001"] | ||||
| "src/documents/models.py" = ["SIM115"] | ||||
|  | ||||
| [isort] | ||||
| [lint.isort] | ||||
| force-single-line = true | ||||
|   | ||||
| @@ -11,7 +11,7 @@ If you want to implement something big: | ||||
|  | ||||
| ## Python | ||||
|  | ||||
| Paperless supports python 3.8 and 3.9. We format Python code with [Black](https://github.com/psf/black). | ||||
| Paperless supports python 3.9 - 3.11. We format Python code with [Black](https://github.com/psf/black). | ||||
|  | ||||
| ## Branches | ||||
|  | ||||
| @@ -45,7 +45,7 @@ Examples of `non-trivial` PRs might include: | ||||
|  | ||||
| - Additional features | ||||
| - Large changes to many distinct files | ||||
| - Breaking or depreciation of existing features | ||||
| - Breaking or deprecation of existing features | ||||
|  | ||||
| Our community review process for `non-trivial` PRs is the following: | ||||
|  | ||||
| @@ -58,6 +58,13 @@ Our community review process for `non-trivial` PRs is the following: | ||||
|  | ||||
| This process might be slow as community members have different schedules and time to dedicate to the Paperless project. However it ensures community code reviews are as brilliantly thorough as they once were with @jonaswinkler. | ||||
|  | ||||
| # AI-Generated Code | ||||
|  | ||||
| This project does not specifically prohibit the use of AI-generated code _during the process_ of creating a PR, however: | ||||
|  | ||||
| 1. Any code present in the final PR that was generated using AI sources should be clearly attributed as such and must not violate copyright protections. | ||||
| 2. We will not accept PRs that are entirely or mostly AI-derived. | ||||
|  | ||||
| # Translating Paperless-ngx | ||||
|  | ||||
| Some notes about translation: | ||||
| @@ -87,7 +94,7 @@ The following files need to be changed: | ||||
|  | ||||
| - src-ui/angular.json (under the _projects/paperless-ui/i18n/locales_ JSON key) | ||||
| - src/paperless/settings.py (in the _LANGUAGES_ array) | ||||
| - src-ui/src/app/services/settings.service.ts (inside the _getLanguageOptions_ method) | ||||
| - src-ui/src/app/services/settings.service.ts (inside the _LANGUAGE_OPTIONS_ array) | ||||
| - src-ui/src/app/app.module.ts (import locale from _angular/common/locales_ and call _registerLocaleData_) | ||||
|  | ||||
| Please add the language in the correct order, alphabetically by locale. | ||||
| @@ -130,3 +137,19 @@ All team members are notified when mentioned or assigned to a relevant issue or | ||||
| We are not overly strict with inviting people to the organization. If you have read the [team permissions](#permissions) and think having additional access would enhance your contributions, please reach out to an [admin](#structure) of the team. | ||||
|  | ||||
| The admins occasionally invite contributors directly if we believe having them on a team will accelerate their work. | ||||
|  | ||||
| # Automatic Repository Maintenance | ||||
|  | ||||
| The Paperless-ngx team appreciates all effort and interest from the community in filing bug reports, creating feature requests, sharing ideas and helping other | ||||
| community members. That said, in an effort to keep the repository organized and managebale the project uses automatic handling of certain areas: | ||||
|  | ||||
| - Issues that cannot be reproduced will be marked 'stale' after 7 days of inactivity and closed after 14 further days of inactivity. | ||||
| - Issues, pull requests and discussions that are closed will be locked after 30 days of inactivity. | ||||
| - Discussions with a marked answer will be automatically closed. | ||||
| - Discussions in the 'General' or 'Support' categories will be closed after 180 days of inactivity. | ||||
| - Feature requests that do not meet the following thresholds will be closed: 5 "up-votes" after 180 days of inactivity or 10 "up-votes" after 365 days. | ||||
|  | ||||
| In all cases, threads can be re-opened by project maintainers and, of course, users can always create a new discussion for related concerns. | ||||
| Finally, remember that all information remains searchable and 'closed' feature requests can still serve as inspiration for new features. | ||||
|  | ||||
| Thank you all for your contributions. | ||||
|   | ||||
							
								
								
									
										163
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						| @@ -5,14 +5,14 @@ | ||||
| # Purpose: Compiles the frontend | ||||
| # Notes: | ||||
| #  - Does NPM stuff with Typescript and such | ||||
| FROM --platform=$BUILDPLATFORM docker.io/node:16-bookworm-slim AS compile-frontend | ||||
| FROM --platform=$BUILDPLATFORM docker.io/node:20-bookworm-slim AS compile-frontend | ||||
|  | ||||
| COPY ./src-ui /src/src-ui | ||||
|  | ||||
| WORKDIR /src/src-ui | ||||
| RUN set -eux \ | ||||
|   && npm update npm -g \ | ||||
|   && npm ci --omit=optional | ||||
|   && npm ci | ||||
| RUN set -eux \ | ||||
|   && ./node_modules/.bin/ng build --configuration production | ||||
|  | ||||
| @@ -21,7 +21,7 @@ RUN set -eux \ | ||||
| # Comments: | ||||
| #  - pipenv dependencies are not left in the final image | ||||
| #  - pipenv can't touch the final image somehow | ||||
| FROM --platform=$BUILDPLATFORM docker.io/python:3.9-alpine as pipenv-base | ||||
| FROM --platform=$BUILDPLATFORM docker.io/python:3.11-alpine as pipenv-base | ||||
|  | ||||
| WORKDIR /usr/src/pipenv | ||||
|  | ||||
| @@ -29,7 +29,7 @@ COPY Pipfile* ./ | ||||
|  | ||||
| RUN set -eux \ | ||||
|   && echo "Installing pipenv" \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade pipenv==2023.6.12 \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade pipenv==2023.12.1 \ | ||||
|   && echo "Generating requirement.txt" \ | ||||
|     && pipenv requirements > requirements.txt | ||||
|  | ||||
| @@ -37,7 +37,7 @@ RUN set -eux \ | ||||
| # Purpose: The final image | ||||
| # Comments: | ||||
| #  - Don't leave anything extra in here | ||||
| FROM docker.io/python:3.9-slim-bookworm as main-app | ||||
| FROM docker.io/python:3.11-slim-bookworm as main-app | ||||
|  | ||||
| LABEL org.opencontainers.image.authors="paperless-ngx team <hello@paperless-ngx.com>" | ||||
| LABEL org.opencontainers.image.documentation="https://docs.paperless-ngx.com/" | ||||
| @@ -47,6 +47,21 @@ LABEL org.opencontainers.image.licenses="GPL-3.0-only" | ||||
|  | ||||
| ARG DEBIAN_FRONTEND=noninteractive | ||||
|  | ||||
| # Buildx provided, must be defined to use though | ||||
| ARG TARGETARCH | ||||
|  | ||||
| # Can be workflow provided, defaults set for manual building | ||||
| ARG JBIG2ENC_VERSION=0.29 | ||||
| ARG QPDF_VERSION=11.9.0 | ||||
| ARG GS_VERSION=10.02.1 | ||||
|  | ||||
| # Set Python environment variables | ||||
| ENV PYTHONDONTWRITEBYTECODE=1 \ | ||||
|     PYTHONUNBUFFERED=1 \ | ||||
|     # Ignore warning from Whitenoise | ||||
|     PYTHONWARNINGS="ignore:::django.http.response:517" \ | ||||
|     PNGX_CONTAINERIZED=1 | ||||
|  | ||||
| # | ||||
| # Begin installation and configuration | ||||
| # Order the steps below from least often changed to most | ||||
| @@ -67,23 +82,11 @@ ARG RUNTIME_PACKAGES="\ | ||||
|   gnupg \ | ||||
|   icc-profiles-free \ | ||||
|   imagemagick \ | ||||
|   # Image processing | ||||
|   liblept5 \ | ||||
|   liblcms2-2 \ | ||||
|   libtiff6 \ | ||||
|   libfreetype6 \ | ||||
|   libwebp7 \ | ||||
|   libopenjp2-7 \ | ||||
|   libimagequant0 \ | ||||
|   libraqm0 \ | ||||
|   libjpeg62-turbo \ | ||||
|   # PostgreSQL | ||||
|   libpq5 \ | ||||
|   postgresql-client \ | ||||
|   # MySQL / MariaDB | ||||
|   mariadb-client \ | ||||
|   # For Numpy | ||||
|   libatlas3-base \ | ||||
|   # OCRmyPDF dependencies | ||||
|   tesseract-ocr \ | ||||
|   tesseract-ocr-eng \ | ||||
| @@ -93,12 +96,11 @@ ARG RUNTIME_PACKAGES="\ | ||||
|   tesseract-ocr-spa \ | ||||
|   unpaper \ | ||||
|   pngquant \ | ||||
|   # pikepdf / qpdf | ||||
|   jbig2dec \ | ||||
|   # lxml | ||||
|   libxml2 \ | ||||
|   libxslt1.1 \ | ||||
|   libgnutls30 \ | ||||
|   libqpdf29 \ | ||||
|   # itself | ||||
|   qpdf \ | ||||
|   # Mime type detection | ||||
|   file \ | ||||
| @@ -107,9 +109,7 @@ ARG RUNTIME_PACKAGES="\ | ||||
|   zlib1g \ | ||||
|   # Barcode splitter | ||||
|   libzbar0 \ | ||||
|   poppler-utils \ | ||||
|   # RapidFuzz on armv7 | ||||
|   libatomic1" | ||||
|   poppler-utils" | ||||
|  | ||||
| # Install basic runtime packages. | ||||
| # These change very infrequently | ||||
| @@ -117,7 +117,37 @@ RUN set -eux \ | ||||
|   echo "Installing system packages" \ | ||||
|     && apt-get update \ | ||||
|     && apt-get install --yes --quiet --no-install-recommends ${RUNTIME_PACKAGES} \ | ||||
|     && rm -rf /var/lib/apt/lists/* \ | ||||
|     && echo "Installing pre-built updates" \ | ||||
|       && echo "Installing qpdf ${QPDF_VERSION}" \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/qpdf-${QPDF_VERSION}/qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./libqpdf29_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./qpdf_${QPDF_VERSION}-1_${TARGETARCH}.deb \ | ||||
|       && echo "Installing Ghostscript ${GS_VERSION}" \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output libgs10_${GS_VERSION}.dfsg-2_${TARGETARCH}.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output ghostscript_${GS_VERSION}.dfsg-2_${TARGETARCH}.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/ghostscript_${GS_VERSION}.dfsg-1_${TARGETARCH}.deb \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output libgs10-common_${GS_VERSION}.dfsg-2_all.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/ghostscript-${GS_VERSION}/libgs10-common_${GS_VERSION}.dfsg-1_all.deb \ | ||||
|         && dpkg --install ./libgs10-common_${GS_VERSION}.dfsg-2_all.deb \ | ||||
|         && dpkg --install ./libgs10_${GS_VERSION}.dfsg-2_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./ghostscript_${GS_VERSION}.dfsg-2_${TARGETARCH}.deb \ | ||||
|       && echo "Installing jbig2enc" \ | ||||
|         && curl --fail --silent --show-error --location \ | ||||
|           --output jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||
|           https://github.com/paperless-ngx/builder/releases/download/jbig2enc-${JBIG2ENC_VERSION}/jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||
|         && dpkg --install ./jbig2enc_${JBIG2ENC_VERSION}-1_${TARGETARCH}.deb \ | ||||
|       && echo "Cleaning up image layer" \ | ||||
|         && rm --force --verbose *.deb \ | ||||
|     && rm --recursive --force --verbose /var/lib/apt/lists/* \ | ||||
|   && echo "Installing supervisor" \ | ||||
|     && python3 -m pip install --default-timeout=1000 --upgrade --no-cache-dir supervisor==4.2.5 | ||||
|  | ||||
| @@ -164,45 +194,10 @@ RUN set -eux \ | ||||
|     && chmod 755 /usr/local/bin/paperless_cmd.sh \ | ||||
|     && mv flower-conditional.sh /usr/local/bin/flower-conditional.sh \ | ||||
|     && chmod 755 /usr/local/bin/flower-conditional.sh \ | ||||
|   && echo "Installing managment commands" \ | ||||
|   && echo "Installing management commands" \ | ||||
|     && chmod +x install_management_commands.sh \ | ||||
|     && ./install_management_commands.sh | ||||
|  | ||||
| # Buildx provided, must be defined to use though | ||||
| ARG TARGETARCH | ||||
| ARG TARGETVARIANT | ||||
|  | ||||
| # Can be workflow provided, defaults set for manual building | ||||
| ARG JBIG2ENC_VERSION=0.29 | ||||
| ARG QPDF_VERSION=11.3.0 | ||||
| ARG PIKEPDF_VERSION=7.2.0 | ||||
| ARG PSYCOPG2_VERSION=2.9.6 | ||||
|  | ||||
| # Install the built packages from the installer library images | ||||
| # These change sometimes | ||||
| RUN set -eux \ | ||||
|   && echo "Getting binaries" \ | ||||
|     && mkdir paperless-ngx \ | ||||
|     && curl --fail --silent --show-error --output paperless-ngx.tar.gz --location https://github.com/paperless-ngx/builder/archive/58bb061b9b3b63009852d6d875f9a305d9ae6ac9.tar.gz \ | ||||
|     && tar -xf paperless-ngx.tar.gz --directory paperless-ngx --strip-components=1 \ | ||||
|     && cd paperless-ngx \ | ||||
|     # Setting a specific revision ensures we know what this installed | ||||
|     # and ensures cache breaking on changes | ||||
|   && echo "Installing jbig2enc" \ | ||||
|     && cp ./jbig2enc/${JBIG2ENC_VERSION}/${TARGETARCH}${TARGETVARIANT}/jbig2 /usr/local/bin/ \ | ||||
|     && cp ./jbig2enc/${JBIG2ENC_VERSION}/${TARGETARCH}${TARGETVARIANT}/libjbig2enc* /usr/local/lib/ \ | ||||
|     && chmod a+x /usr/local/bin/jbig2 \ | ||||
|   && echo "Installing pikepdf and dependencies" \ | ||||
|     && python3 -m pip install --no-cache-dir ./pikepdf/${PIKEPDF_VERSION}/${TARGETARCH}${TARGETVARIANT}/*.whl \ | ||||
|     && python3 -m pip list \ | ||||
|   && echo "Installing psycopg2" \ | ||||
|     && python3 -m pip install --no-cache-dir ./psycopg2/${PSYCOPG2_VERSION}/${TARGETARCH}${TARGETVARIANT}/psycopg2*.whl \ | ||||
|     && python3 -m pip list \ | ||||
|   && echo "Cleaning up image layer" \ | ||||
|     && cd ../ \ | ||||
|     && rm -rf paperless-ngx \ | ||||
|     && rm paperless-ngx.tar.gz | ||||
|  | ||||
| WORKDIR /usr/src/paperless/src/ | ||||
|  | ||||
| # Python dependencies | ||||
| @@ -214,7 +209,11 @@ COPY --from=pipenv-base /usr/src/pipenv/requirements.txt ./ | ||||
| ARG BUILD_PACKAGES="\ | ||||
|   build-essential \ | ||||
|   git \ | ||||
|   default-libmysqlclient-dev" | ||||
|   # https://www.psycopg.org/docs/install.html#prerequisites | ||||
|   libpq-dev \ | ||||
|   # https://github.com/PyMySQL/mysqlclient#linux | ||||
|   default-libmysqlclient-dev \ | ||||
|   pkg-config" | ||||
|  | ||||
| # hadolint ignore=DL3042 | ||||
| RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \ | ||||
| @@ -225,34 +224,46 @@ RUN --mount=type=cache,target=/root/.cache/pip/,id=pip-cache \ | ||||
|     && python3 -m pip install --no-cache-dir --upgrade wheel \ | ||||
|   && echo "Installing Python requirements" \ | ||||
|     && python3 -m pip install --default-timeout=1000 --requirement requirements.txt \ | ||||
|   && echo "Patching whitenoise for compression speedup" \ | ||||
|     && curl --fail --silent --show-error --location --output 484.patch https://github.com/evansd/whitenoise/pull/484.patch \ | ||||
|     && patch -d /usr/local/lib/python3.11/site-packages --verbose -p2 < 484.patch \ | ||||
|     && rm 484.patch \ | ||||
|   && echo "Installing NLTK data" \ | ||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" snowball_data \ | ||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" stopwords \ | ||||
|     && python3 -W ignore::RuntimeWarning -m nltk.downloader -d "/usr/share/nltk_data" punkt \ | ||||
|   && echo "Cleaning up image" \ | ||||
|     && apt-get -y purge ${BUILD_PACKAGES} \ | ||||
|     && apt-get -y autoremove --purge \ | ||||
|     && apt-get --yes purge ${BUILD_PACKAGES} \ | ||||
|     && apt-get --yes autoremove --purge \ | ||||
|     && apt-get clean --yes \ | ||||
|     && rm -rf /var/lib/apt/lists/* \ | ||||
|     && rm -rf /tmp/* \ | ||||
|     && rm -rf /var/tmp/* \ | ||||
|     && rm -rf /var/cache/apt/archives/* \ | ||||
|     && truncate -s 0 /var/log/*log | ||||
|     && rm --recursive --force --verbose /var/lib/apt/lists/* \ | ||||
|     && rm --recursive --force --verbose /tmp/* \ | ||||
|     && rm --recursive --force --verbose /var/tmp/* \ | ||||
|     && rm --recursive --force --verbose /var/cache/apt/archives/* \ | ||||
|     && truncate --size 0 /var/log/*log | ||||
|  | ||||
| # copy backend | ||||
| COPY ./src ./ | ||||
| COPY --chown=1000:1000 ./src ./ | ||||
|  | ||||
| # copy frontend | ||||
| COPY --from=compile-frontend /src/src/documents/static/frontend/ ./documents/static/frontend/ | ||||
| COPY --from=compile-frontend --chown=1000:1000 /src/src/documents/static/frontend/ ./documents/static/frontend/ | ||||
|  | ||||
| # add users, setup scripts | ||||
| # Mount the compiled frontend to expected location | ||||
| RUN set -eux \ | ||||
|   && addgroup --gid 1000 paperless \ | ||||
|   && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ | ||||
|   && chown -R paperless:paperless /usr/src/paperless \ | ||||
|   && gosu paperless python3 manage.py collectstatic --clear --no-input --link \ | ||||
|   && gosu paperless python3 manage.py compilemessages | ||||
|   && echo "Setting up user/group" \ | ||||
|     && addgroup --gid 1000 paperless \ | ||||
|     && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ | ||||
|   && echo "Creating volume directories" \ | ||||
|     && mkdir --parents --verbose /usr/src/paperless/data \ | ||||
|     && mkdir --parents --verbose /usr/src/paperless/media \ | ||||
|     && mkdir --parents --verbose /usr/src/paperless/consume \ | ||||
|     && mkdir --parents --verbose /usr/src/paperless/export \ | ||||
|   && echo "Adjusting all permissions" \ | ||||
|     && chown --from root:root --changes --recursive paperless:paperless /usr/src/paperless \ | ||||
|   && echo "Collecting static files" \ | ||||
|     && gosu paperless python3 manage.py collectstatic --clear --no-input --link \ | ||||
|     && gosu paperless python3 manage.py compilemessages | ||||
|  | ||||
| VOLUME ["/usr/src/paperless/data", \ | ||||
|         "/usr/src/paperless/media", \ | ||||
| @@ -264,3 +275,5 @@ ENTRYPOINT ["/sbin/docker-entrypoint.sh"] | ||||
| EXPOSE 8000 | ||||
|  | ||||
| CMD ["/usr/local/bin/paperless_cmd.sh"] | ||||
|  | ||||
| HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000" ] | ||||
|   | ||||
							
								
								
									
										86
									
								
								Pipfile
									
									
									
									
									
								
							
							
						
						| @@ -3,69 +3,58 @@ url = "https://pypi.python.org/simple" | ||||
| verify_ssl = true | ||||
| name = "pypi" | ||||
|  | ||||
| [[source]] | ||||
| url = "https://www.piwheels.org/simple" | ||||
| verify_ssl = true | ||||
| name = "piwheels" | ||||
|  | ||||
| [packages] | ||||
| dateparser = "~=1.1" | ||||
| dateparser = "~=1.2" | ||||
| # WARNING: django does not use semver. | ||||
| #          Only patch versions are guaranteed to not introduce breaking changes. | ||||
| django = "~=4.1.9" | ||||
| django-cors-headers = "*" | ||||
| django = "~=4.2.11" | ||||
| django-allauth = "*" | ||||
| django-auditlog = "*" | ||||
| django-celery-results = "*" | ||||
| django-compression-middleware = "*" | ||||
| django-guardian = "*" | ||||
| django-cors-headers = "*" | ||||
| django-extensions = "*" | ||||
| django-filter = "~=22.1" | ||||
| djangorestframework = "~=3.14" | ||||
| django-filter = "~=24.2" | ||||
| django-guardian = "*" | ||||
| django-multiselectfield = "*" | ||||
| djangorestframework = "==3.14.0" | ||||
| djangorestframework-guardian = "*" | ||||
| drf-writable-nested = "*" | ||||
| bleach = "*" | ||||
| celery = {extras = ["redis"], version = "*"} | ||||
| channels = "~=4.1" | ||||
| channels-redis = "*" | ||||
| concurrent-log-handler = "*" | ||||
| filelock = "*" | ||||
| flower = "*" | ||||
| gotenberg-client = "*" | ||||
| gunicorn = "*" | ||||
| imap-tools = "*" | ||||
| inotifyrecursive = "~=0.3" | ||||
| langdetect = "*" | ||||
| mysqlclient = "*" | ||||
| nltk = "*" | ||||
| ocrmypdf = "~=15.4" | ||||
| pathvalidate = "*" | ||||
| pillow = "*" | ||||
| pikepdf = "*" | ||||
| python-gnupg = "*" | ||||
| python-dotenv = "*" | ||||
| python-dateutil = "*" | ||||
| python-magic = "*" | ||||
| python-ipware = "*" | ||||
| pdf2image = "*" | ||||
| psycopg2 = "*" | ||||
| python-dateutil = "*" | ||||
| python-dotenv = "*" | ||||
| python-gnupg = "*" | ||||
| python-ipware = "*" | ||||
| python-magic = "*" | ||||
| pyzbar = "*" | ||||
| rapidfuzz = "*" | ||||
| redis = {extras = ["hiredis"], version = "*"} | ||||
| scikit-learn = "~=1.2" | ||||
| whitenoise = "~=6.3" | ||||
| watchdog = "~=2.2" | ||||
| whoosh="~=2.7" | ||||
| inotifyrecursive = "~=0.3" | ||||
| ocrmypdf = "~=14.0" | ||||
| tqdm = "*" | ||||
| tika-client = "*" | ||||
| channels = "~=4.0" | ||||
| channels-redis = "*" | ||||
| uvicorn = {extras = ["standard"], version = "*"} | ||||
| concurrent-log-handler = "*" | ||||
| pyzbar = "*" | ||||
| mysqlclient = "*" | ||||
| celery = {extras = ["redis"], version = "*"} | ||||
| scikit-learn = "~=1.4" | ||||
| setproctitle = "*" | ||||
| nltk = "*" | ||||
| pdf2image = "*" | ||||
| flower = "*" | ||||
| bleach = "*" | ||||
| tika-client = "*" | ||||
| tqdm = "*" | ||||
| uvicorn = {extras = ["standard"], version = "==0.25.0"} | ||||
| watchdog = "~=4.0" | ||||
| whitenoise = "~=6.6" | ||||
| whoosh="~=2.7" | ||||
| zxing-cpp = {version = "*", platform_machine = "== 'x86_64'"} | ||||
| # | ||||
| # Packages locked due to issues (try to check if these are fixed in a release every so often) | ||||
| # | ||||
| # Pin this until piwheels is building 1.9 (see https://www.piwheels.org/project/scipy/) | ||||
| scipy = "==1.8.1" | ||||
| # v4 brings in extra dependencies for features not used here | ||||
| reportlab = "==3.6.12" | ||||
| # Pin this until piwheels is building a newer version (see https://www.piwheels.org/project/cryptography/) | ||||
| cryptography = "==40.0.1" | ||||
|  | ||||
| [dev-packages] | ||||
| # Linting | ||||
| @@ -81,11 +70,12 @@ pytest-httpx = "*" | ||||
| pytest-env = "*" | ||||
| pytest-sugar = "*" | ||||
| pytest-xdist = "*" | ||||
| "pdfminer.six" = "*" | ||||
| pytest-rerunfailures = "*" | ||||
| imagehash = "*" | ||||
| daphne = "*" | ||||
| # Documentation | ||||
| mkdocs-material = "*" | ||||
| mkdocs-glightbox = "*" | ||||
|  | ||||
| [typing-dev] | ||||
| mypy = "*" | ||||
| @@ -97,12 +87,10 @@ celery-types = "*" | ||||
| django-stubs = {extras= ["compatible-mypy"], version="*"} | ||||
| types-dateparser = "*" | ||||
| types-bleach = "*" | ||||
| types-humanfriendly = "*" | ||||
| types-redis = "*" | ||||
| types-tqdm = "*" | ||||
| types-Markdown = "*" | ||||
| types-Pygments = "*" | ||||
| types-backports = "*" | ||||
| types-colorama = "*" | ||||
| types-psycopg2 = "*" | ||||
| types-setuptools = "*" | ||||
|   | ||||
							
								
								
									
										5877
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										58
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -6,8 +6,11 @@ | ||||
| [](https://demo.paperless-ngx.com) | ||||
|  | ||||
| <p align="center"> | ||||
| <img src="https://github.com/paperless-ngx/paperless-ngx/raw/main/resources/logo/web/png/Black%20logo%20-%20no%20background.png#gh-light-mode-only" width="50%" /> | ||||
| <img src="https://github.com/paperless-ngx/paperless-ngx/raw/main/resources/logo/web/png/White%20logo%20-%20no%20background.png#gh-dark-mode-only" width="50%" /> | ||||
|   <picture> | ||||
|     <source media="(prefers-color-scheme: dark)" srcset="https://github.com/paperless-ngx/paperless-ngx/blob/main/resources/logo/web/png/White%20logo%20-%20no%20background.png" width="50%"> | ||||
|     <source media="(prefers-color-scheme: light)" srcset="https://github.com/paperless-ngx/paperless-ngx/raw/main/resources/logo/web/png/Black%20logo%20-%20no%20background.png" width="50%"> | ||||
|     <img src="https://github.com/paperless-ngx/paperless-ngx/raw/main/resources/logo/web/png/Black%20logo%20-%20no%20background.png" width="50%"> | ||||
|   </picture> | ||||
| </p> | ||||
|  | ||||
| <!-- omit in toc --> | ||||
| @@ -16,10 +19,9 @@ | ||||
|  | ||||
| Paperless-ngx is a document management system that transforms your physical documents into a searchable online archive so you can keep, well, _less paper_. | ||||
|  | ||||
| Paperless-ngx forked from [paperless-ng](https://github.com/jonaswinkler/paperless-ng) to continue the great work and distribute responsibility of supporting and advancing the project among a team of people. [Consider joining us!](#community-support) Discussion of this transition can be found in issues | ||||
| [#1599](https://github.com/jonaswinkler/paperless-ng/issues/1599) and [#1632](https://github.com/jonaswinkler/paperless-ng/issues/1632). | ||||
| Paperless-ngx is the official successor to the original [Paperless](https://github.com/the-paperless-project/paperless) & [Paperless-ng](https://github.com/jonaswinkler/paperless-ng) projects and is designed to distribute the responsibility of advancing and supporting the project among a team of people. [Consider joining us!](#community-support) | ||||
|  | ||||
| A demo is available at [demo.paperless-ngx.com](https://demo.paperless-ngx.com) using login `demo` / `demo`. _Note: demo content is reset frequently and confidential information should not be uploaded._ | ||||
| Thanks to the generous folks at [DigitalOcean](https://m.do.co/c/8d70b916d462), a demo is available at [demo.paperless-ngx.com](https://demo.paperless-ngx.com) using login `demo` / `demo`. _Note: demo content is reset frequently and confidential information should not be uploaded._ | ||||
|  | ||||
| - [Features](#features) | ||||
| - [Getting started](#getting-started) | ||||
| @@ -31,39 +33,31 @@ A demo is available at [demo.paperless-ngx.com](https://demo.paperless-ngx.com) | ||||
| - [Affiliated Projects](#affiliated-projects) | ||||
| - [Important Note](#important-note) | ||||
|  | ||||
| <p align="right">This project is supported by:<br/> | ||||
|   <a href="https://m.do.co/c/8d70b916d462" style="padding-top: 4px; display: block;"> | ||||
|     <picture> | ||||
|       <source media="(prefers-color-scheme: dark)" srcset="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/SVG/DO_Logo_horizontal_white.svg" width="140px"> | ||||
|       <source media="(prefers-color-scheme: light)" srcset="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/SVG/DO_Logo_horizontal_black_.svg" width="140px"> | ||||
|       <img src="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/SVG/DO_Logo_horizontal_black_.svg" width="140px"> | ||||
|     </picture> | ||||
|   </a> | ||||
| </p> | ||||
|  | ||||
| # Features | ||||
|  | ||||
|  | ||||
|  | ||||
| <picture> | ||||
|   <source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/docs/assets/screenshots/documents-smallcards-dark.png"> | ||||
|   <source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/docs/assets/screenshots/documents-smallcards.png"> | ||||
|   <img src="https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/docs/assets/screenshots/documents-smallcards.png"> | ||||
| </picture> | ||||
|  | ||||
| - Organize and index your scanned documents with tags, correspondents, types, and more. | ||||
| - Performs OCR on your documents, adds selectable text to image only documents and adds tags, correspondents and document types to your documents. | ||||
| - Supports PDF documents, images, plain text files, and Office documents (Word, Excel, Powerpoint, and LibreOffice equivalents). | ||||
|   - Office document support is optional and provided by Apache Tika (see [configuration](https://docs.paperless-ngx.com/configuration/#tika)) | ||||
| - Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and their format can be configured freely. | ||||
| - Single page application front end. | ||||
|   - Includes a dashboard that shows basic statistics and has document upload. | ||||
|   - Filtering by tags, correspondents, types, and more. | ||||
|   - Customizable views can be saved and displayed on the dashboard. | ||||
| - Full text search helps you find what you need. | ||||
|   - Auto completion suggests relevant words from your documents. | ||||
|   - Results are sorted by relevance to your search query. | ||||
|   - Highlighting shows you which parts of the document matched the query. | ||||
|   - Searching for similar documents ("More like this") | ||||
| - Email processing: Paperless adds documents from your email accounts. | ||||
|   - Configure multiple accounts and filters for each account. | ||||
|   - When adding documents from mail, paperless can move these mail to a new folder, mark them as read, flag them as important or delete them. | ||||
| - Machine learning powered document matching. | ||||
|   - Paperless-ngx learns from your documents and will be able to automatically assign tags, correspondents and types to documents once you've stored a few documents in paperless. | ||||
| - Optimized for multi core systems: Paperless-ngx consumes multiple documents in parallel. | ||||
| - The integrated sanity checker makes sure that your document archive is in good health. | ||||
| - [More screenshots are available in the documentation](https://docs.paperless-ngx.com/#screenshots). | ||||
| A full list of [features](https://docs.paperless-ngx.com/#features) and [screenshots](https://docs.paperless-ngx.com/#screenshots) are available in the [documentation](https://docs.paperless-ngx.com/). | ||||
|  | ||||
| # Getting started | ||||
|  | ||||
| The easiest way to deploy paperless is docker-compose. The files in the [`/docker/compose` directory](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) are configured to pull the image from Github Packages. | ||||
| The easiest way to deploy paperless is `docker compose`. The files in the [`/docker/compose` directory](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) are configured to pull the image from GitHub Packages. | ||||
|  | ||||
| If you'd like to jump right in, you can configure a docker-compose environment with our install script: | ||||
| If you'd like to jump right in, you can configure a `docker compose` environment with our install script: | ||||
|  | ||||
| ```bash | ||||
| bash -c "$(curl -L https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/install-paperless-ngx.sh)" | ||||
| @@ -85,7 +79,7 @@ If you feel like contributing to the project, please do! Bug fixes, enhancements | ||||
|  | ||||
| ## Community Support | ||||
|  | ||||
| People interested in continuing the work on paperless-ngx are encouraged to reach out here on github and in the [Matrix Room](https://matrix.to/#/#paperless:adnidor.de). If you would like to contribute to the project on an ongoing basis there are multiple [teams](https://github.com/orgs/paperless-ngx/people) (frontend, ci/cd, etc) that could use your help so please reach out! | ||||
| People interested in continuing the work on paperless-ngx are encouraged to reach out here on github and in the [Matrix Room](https://matrix.to/#/#paperless:matrix.org). If you would like to contribute to the project on an ongoing basis there are multiple [teams](https://github.com/orgs/paperless-ngx/people) (frontend, ci/cd, etc) that could use your help so please reach out! | ||||
|  | ||||
| ## Translation | ||||
|  | ||||
|   | ||||
							
								
								
									
										9
									
								
								SECURITY.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,9 @@ | ||||
| # Security Policy | ||||
|  | ||||
| ## Reporting a Vulnerability | ||||
|  | ||||
| The Paperless-ngx team and community take security bugs seriously. We appreciate your efforts to responsibly disclose your findings, and will make every effort to acknowledge your contributions. | ||||
|  | ||||
| To report a security issue, please use the GitHub Security Advisory ["Report a Vulnerability"](https://github.com/paperless-ngx/paperless-ngx/security/advisories/new) tab. | ||||
|  | ||||
| The team will send a response indicating the next steps in handling your report. After the initial reply to your report, the security team will keep you informed of the progress towards a fix and full announcement, and may ask for additional information or guidance. | ||||
| @@ -1,8 +1,6 @@ | ||||
| commit_message: '[ci skip]' | ||||
| pull_request_labels: [ | ||||
|   "skip-changelog", | ||||
|   "translation" | ||||
| ] | ||||
| project_id_env: CROWDIN_PROJECT_ID | ||||
| api_token_env: CROWDIN_PERSONAL_TOKEN | ||||
| preserve_hierarchy: true | ||||
| files: | ||||
|   - source: /src/locale/en_US/LC_MESSAGES/django.po | ||||
|     translation: /src/locale/%locale_with_underscore%/LC_MESSAGES/django.po | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| # docker-compose file for running paperless testing with actual gotenberg | ||||
| # Docker Compose file for running paperless testing with actual gotenberg | ||||
| # and Tika containers for a more end to end test of the Tika related functionality | ||||
| # Can be used locally or by the CI to start the nessecary containers with the | ||||
| # Can be used locally or by the CI to start the necessary containers with the | ||||
| # correct networking for the tests | ||||
|  | ||||
| version: "3.7" | ||||
| services: | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:7.8 | ||||
|     image: docker.io/gotenberg/gotenberg:7.10 | ||||
|     hostname: gotenberg | ||||
|     container_name: gotenberg | ||||
|     network_mode: host | ||||
| @@ -17,6 +17,8 @@ services: | ||||
|       - "gotenberg" | ||||
|       - "--chromium-disable-javascript=true" | ||||
|       - "--chromium-allow-list=file:///tmp/.*" | ||||
|       - "--log-level=warn" | ||||
|       - "--log-format=text" | ||||
|   tika: | ||||
|     image: ghcr.io/paperless-ngx/tika:latest | ||||
|     hostname: tika | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the Docker Hub. | ||||
| # docker compose file for running paperless from the Docker Hub. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -10,7 +10,7 @@ | ||||
| #   as this file and mounted to the correct folders inside the container. | ||||
| # - Paperless listens on port 8000. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Instead of SQLite (default), MariaDB is used as the database server. | ||||
| @@ -23,9 +23,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -60,11 +60,6 @@ services: | ||||
|       - tika | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-f", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -83,7 +78,7 @@ services: | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:7.8 | ||||
|     image: docker.io/gotenberg/gotenberg:7.10 | ||||
|     restart: unless-stopped | ||||
|     # The gotenberg chromium route is used to convert .eml files. We do not | ||||
|     # want to allow external content like tracking pixels or even javascript. | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the Docker Hub. | ||||
| # Docker Compose file for running paperless from the Docker Hub. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -10,7 +10,7 @@ | ||||
| #   as this file and mounted to the correct folders inside the container. | ||||
| # - Paperless listens on port 8000. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Instead of SQLite (default), MariaDB is used as the database server. | ||||
| @@ -19,9 +19,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -54,11 +54,6 @@ services: | ||||
|       - broker | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-f", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -73,7 +68,6 @@ services: | ||||
|       PAPERLESS_DBPASS: paperless # only needed if non-default password | ||||
|       PAPERLESS_DBPORT: 3306 | ||||
|  | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the Docker Hub. | ||||
| # Docker Compose file for running paperless from the Docker Hub. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -10,7 +10,7 @@ | ||||
| #   as this file and mounted to the correct folders inside the container. | ||||
| # - Paperless listens on port 8010. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Instead of SQLite (default), PostgreSQL is used as the database server. | ||||
| @@ -18,7 +18,7 @@ | ||||
| # To install and update paperless with this file, do the following: | ||||
| # | ||||
| # - Open portainer Stacks list and click 'Add stack' | ||||
| # - Paste the contents of this file and assign a name, e.g. 'Paperless' | ||||
| # - Paste the contents of this file and assign a name, e.g. 'paperless' | ||||
| # - Click 'Deploy the stack' and wait for it to be deployed | ||||
| # - Open the list of containers, select paperless_webserver_1 | ||||
| # - Click 'Console' and then 'Connect' to open the command line inside the container | ||||
| @@ -37,7 +37,7 @@ services: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:13 | ||||
|     image: docker.io/library/postgres:15 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - pgdata:/var/lib/postgresql/data | ||||
| @@ -54,11 +54,6 @@ services: | ||||
|       - broker | ||||
|     ports: | ||||
|       - "8010:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the docker container registry. | ||||
| # Docker Compose file for running paperless from the docker container registry. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -10,7 +10,7 @@ | ||||
| #   as this file and mounted to the correct folders inside the container. | ||||
| # - Paperless listens on port 8000. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Instead of SQLite (default), PostgreSQL is used as the database server. | ||||
| @@ -23,9 +23,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -39,7 +39,7 @@ services: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:13 | ||||
|     image: docker.io/library/postgres:15 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - pgdata:/var/lib/postgresql/data | ||||
| @@ -58,11 +58,6 @@ services: | ||||
|       - tika | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -77,7 +72,7 @@ services: | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:7.8 | ||||
|     image: docker.io/gotenberg/gotenberg:7.10 | ||||
|     restart: unless-stopped | ||||
|  | ||||
|     # The gotenberg chromium route is used to convert .eml files. We do not | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the Docker Hub. | ||||
| # Docker Compose file for running paperless from the Docker Hub. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -10,7 +10,7 @@ | ||||
| #   as this file and mounted to the correct folders inside the container. | ||||
| # - Paperless listens on port 8000. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Instead of SQLite (default), PostgreSQL is used as the database server. | ||||
| @@ -19,9 +19,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -35,7 +35,7 @@ services: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:13 | ||||
|     image: docker.io/library/postgres:15 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - pgdata:/var/lib/postgresql/data | ||||
| @@ -52,11 +52,6 @@ services: | ||||
|       - broker | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -67,7 +62,6 @@ services: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|       PAPERLESS_DBHOST: db | ||||
|  | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the docker container registry. | ||||
| # Docker Compose file for running paperless from the docker container registry. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # All compose files of paperless configure paperless in the following way: | ||||
| @@ -11,7 +11,7 @@ | ||||
| # | ||||
| # SQLite is used as the database. The SQLite file is stored in the data volume. | ||||
| # | ||||
| # In addition to that, this docker-compose file adds the following optional | ||||
| # In addition to that, this Docker Compose file adds the following optional | ||||
| # configurations: | ||||
| # | ||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||
| @@ -23,9 +23,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -47,11 +47,6 @@ services: | ||||
|       - tika | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -65,7 +60,7 @@ services: | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:7.8 | ||||
|     image: docker.io/gotenberg/gotenberg:7.10 | ||||
|     restart: unless-stopped | ||||
|  | ||||
|     # The gotenberg chromium route is used to convert .eml files. We do not | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| # docker-compose file for running paperless from the Docker Hub. | ||||
| # Docker Compose file for running paperless from the Docker Hub. | ||||
| # This file contains everything paperless needs to run. | ||||
| # Paperless supports amd64, arm and arm64 hardware. | ||||
| # | ||||
| @@ -16,9 +16,9 @@ | ||||
| # | ||||
| # - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' | ||||
| #   and '.env' into a folder. | ||||
| # - Run 'docker-compose pull'. | ||||
| # - Run 'docker-compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker-compose up -d'. | ||||
| # - Run 'docker compose pull'. | ||||
| # - Run 'docker compose run --rm webserver createsuperuser' to create a user. | ||||
| # - Run 'docker compose up -d'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
| @@ -38,11 +38,6 @@ services: | ||||
|       - broker | ||||
|     ports: | ||||
|       - "8000:8000" | ||||
|     healthcheck: | ||||
|       test: ["CMD", "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000"] | ||||
|       interval: 30s | ||||
|       timeout: 10s | ||||
|       retries: 5 | ||||
|     volumes: | ||||
|       - data:/usr/src/paperless/data | ||||
|       - media:/usr/src/paperless/media | ||||
| @@ -52,7 +47,6 @@ services: | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|  | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -86,17 +86,17 @@ initialize() { | ||||
| 		"${CONSUME_DIR}"; do | ||||
| 		if [[ ! -d "${dir}" ]]; then | ||||
| 			echo "Creating directory ${dir}" | ||||
| 			mkdir "${dir}" | ||||
| 			mkdir --parents "${dir}" | ||||
| 		fi | ||||
| 	done | ||||
|  | ||||
| 	local -r tmp_dir="/tmp/paperless" | ||||
| 	echo "Creating directory ${tmp_dir}" | ||||
| 	mkdir -p "${tmp_dir}" | ||||
| 	local -r tmp_dir="${PAPERLESS_SCRATCH_DIR:=/tmp/paperless}" | ||||
| 	echo "Creating directory scratch directory ${tmp_dir}" | ||||
| 	mkdir --parents "${tmp_dir}" | ||||
|  | ||||
| 	set +e | ||||
| 	echo "Adjusting permissions of paperless files. This may take a while." | ||||
| 	chown -R paperless:paperless ${tmp_dir} | ||||
| 	chown -R paperless:paperless "${tmp_dir}" | ||||
| 	for dir in \ | ||||
| 		"${export_dir}" \ | ||||
| 		"${DATA_DIR}" \ | ||||
|   | ||||
| @@ -80,7 +80,7 @@ django_checks() { | ||||
|  | ||||
| search_index() { | ||||
|  | ||||
| 	local -r index_version=6 | ||||
| 	local -r index_version=9 | ||||
| 	local -r index_version_file=${DATA_DIR}/.index_version | ||||
|  | ||||
| 	if [[ (! -f "${index_version_file}") || $(<"${index_version_file}") != "$index_version" ]]; then | ||||
|   | ||||
| @@ -13,6 +13,7 @@ for command in decrypt_documents \ | ||||
| 	document_retagger \ | ||||
| 	document_thumbnails \ | ||||
| 	document_sanity_checker \ | ||||
| 	document_fuzzy_match \ | ||||
| 	manage_superuser; | ||||
| do | ||||
| 	echo "installing $command..." | ||||
|   | ||||
| @@ -1,14 +1,15 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| SUPERVISORD_WORKING_DIR="${PAPERLESS_SUPERVISORD_WORKING_DIR:-$PWD}" | ||||
| rootless_args=() | ||||
| if [ "$(id -u)" == "$(id -u paperless)" ]; then | ||||
| 	rootless_args=( | ||||
| 		--user | ||||
| 		paperless | ||||
| 		--logfile | ||||
| 		supervisord.log | ||||
| 		"${SUPERVISORD_WORKING_DIR}/supervisord.log" | ||||
| 		--pidfile | ||||
| 		supervisord.pid | ||||
| 		"${SUPERVISORD_WORKING_DIR}/supervisord.pid" | ||||
| 	) | ||||
| fi | ||||
|  | ||||
|   | ||||
| @@ -15,6 +15,7 @@ stdout_logfile=/dev/stdout | ||||
| stdout_logfile_maxbytes=0 | ||||
| stderr_logfile=/dev/stderr | ||||
| stderr_logfile_maxbytes=0 | ||||
| environment = HOME="/usr/src/paperless",USER="paperless" | ||||
|  | ||||
| [program:consumer] | ||||
| command=python3 manage.py document_consumer | ||||
| @@ -25,6 +26,7 @@ stdout_logfile=/dev/stdout | ||||
| stdout_logfile_maxbytes=0 | ||||
| stderr_logfile=/dev/stderr | ||||
| stderr_logfile_maxbytes=0 | ||||
| environment = HOME="/usr/src/paperless",USER="paperless" | ||||
|  | ||||
| [program:celery] | ||||
|  | ||||
| @@ -37,6 +39,7 @@ stdout_logfile=/dev/stdout | ||||
| stdout_logfile_maxbytes=0 | ||||
| stderr_logfile=/dev/stderr | ||||
| stderr_logfile_maxbytes=0 | ||||
| environment = HOME="/usr/src/paperless",USER="paperless" | ||||
|  | ||||
| [program:celery-beat] | ||||
|  | ||||
| @@ -48,6 +51,7 @@ stdout_logfile=/dev/stdout | ||||
| stdout_logfile_maxbytes=0 | ||||
| stderr_logfile=/dev/stderr | ||||
| stderr_logfile_maxbytes=0 | ||||
| environment = HOME="/usr/src/paperless",USER="paperless" | ||||
|  | ||||
| [program:celery-flower] | ||||
| command = /usr/local/bin/flower-conditional.sh | ||||
| @@ -58,3 +62,4 @@ stdout_logfile=/dev/stdout | ||||
| stdout_logfile_maxbytes=0 | ||||
| stderr_logfile=/dev/stderr | ||||
| stderr_logfile_maxbytes=0 | ||||
| environment = HOME="/usr/src/paperless",USER="paperless" | ||||
|   | ||||
| @@ -5,17 +5,19 @@ | ||||
| Multiple options exist for making backups of your paperless instance, | ||||
| depending on how you installed paperless. | ||||
|  | ||||
| Before making backups, make sure that paperless is not running. | ||||
| Before making a backup, it's probably best to make sure that paperless is not actively | ||||
| consuming documents at that time. | ||||
|  | ||||
| Options available to any installation of paperless: | ||||
|  | ||||
| - Use the [document exporter](#exporter). The document exporter exports all your documents, | ||||
|   thumbnails and metadata to a specific folder. You may import your | ||||
|   documents into a fresh instance of paperless again or store your | ||||
|   documents in another DMS with this export. | ||||
| - The document exporter is also able to update an already existing | ||||
|   export. Therefore, incremental backups with `rsync` are entirely | ||||
|   possible. | ||||
| -   Use the [document exporter](#exporter). The document exporter exports all your documents, | ||||
|     thumbnails, metadata, and database contents to a specific folder. You may import your | ||||
|     documents and settings into a fresh instance of paperless again or store your | ||||
|     documents in another DMS with this export. | ||||
|  | ||||
|     The document exporter is also able to update an already existing | ||||
|     export. Therefore, incremental backups with `rsync` are entirely | ||||
|     possible. | ||||
|  | ||||
| !!! caution | ||||
|  | ||||
| @@ -25,31 +27,37 @@ Options available to any installation of paperless: | ||||
|  | ||||
| Options available to docker installations: | ||||
|  | ||||
| - Backup the docker volumes. These usually reside within | ||||
|   `/var/lib/docker/volumes` on the host and you need to be root in | ||||
|   order to access them. | ||||
| -   Backup the docker volumes. These usually reside within | ||||
|     `/var/lib/docker/volumes` on the host and you need to be root in | ||||
|     order to access them. | ||||
|  | ||||
|   Paperless uses 4 volumes: | ||||
|     Paperless uses 4 volumes: | ||||
|  | ||||
|   - `paperless_media`: This is where your documents are stored. | ||||
|   - `paperless_data`: This is where auxillary data is stored. This | ||||
|     folder also contains the SQLite database, if you use it. | ||||
|   - `paperless_pgdata`: Exists only if you use PostgreSQL and | ||||
|     contains the database. | ||||
|   - `paperless_dbdata`: Exists only if you use MariaDB and contains | ||||
|     the database. | ||||
|     -   `paperless_media`: This is where your documents are stored. | ||||
|     -   `paperless_data`: This is where auxiliary data is stored. This | ||||
|         folder also contains the SQLite database, if you use it. | ||||
|     -   `paperless_pgdata`: Exists only if you use PostgreSQL and | ||||
|         contains the database. | ||||
|     -   `paperless_dbdata`: Exists only if you use MariaDB and contains | ||||
|         the database. | ||||
|  | ||||
| Options available to bare-metal and non-docker installations: | ||||
|  | ||||
| - Backup the entire paperless folder. This ensures that if your | ||||
|   paperless instance crashes at some point or your disk fails, you can | ||||
|   simply copy the folder back into place and it works. | ||||
| -   Backup the entire paperless folder. This ensures that if your | ||||
|     paperless instance crashes at some point or your disk fails, you can | ||||
|     simply copy the folder back into place and it works. | ||||
|  | ||||
|   When using PostgreSQL or MariaDB, you'll also have to backup the | ||||
|   database. | ||||
|     When using PostgreSQL or MariaDB, you'll also have to backup the | ||||
|     database. | ||||
|  | ||||
| ### Restoring {#migrating-restoring} | ||||
|  | ||||
| If you've backed-up Paperless-ngx using the [document exporter](#exporter), | ||||
| restoring can simply be done with the [document importer](#importer). | ||||
|  | ||||
| Of course, other backup strategies require restoring any volumes, folders and database | ||||
| copies you created in the steps above. | ||||
|  | ||||
| ## Updating Paperless {#updating} | ||||
|  | ||||
| ### Docker Route {#docker-updating} | ||||
| @@ -59,34 +67,34 @@ you installed paperless-ngx in the first place. The releases are | ||||
| available at the [release | ||||
| page](https://github.com/paperless-ngx/paperless-ngx/releases). | ||||
|  | ||||
| First of all, ensure that paperless is stopped. | ||||
| First of all, make sure no active processes (like consumption) are running, then [make a backup](#backup). | ||||
|  | ||||
| After that, ensure that paperless is stopped: | ||||
|  | ||||
| ```shell-session | ||||
| $ cd /path/to/paperless | ||||
| $ docker-compose down | ||||
| $ docker compose down | ||||
| ``` | ||||
|  | ||||
| After that, [make a backup](#backup). | ||||
| 1.  If you pull the image from the docker hub, all you need to do is: | ||||
|  | ||||
| 1. If you pull the image from the docker hub, all you need to do is: | ||||
|     ```shell-session | ||||
|     $ docker compose pull | ||||
|     $ docker compose up | ||||
|     ``` | ||||
|  | ||||
|    ```shell-session | ||||
|    $ docker-compose pull | ||||
|    $ docker-compose up | ||||
|    ``` | ||||
|     The Docker Compose files refer to the `latest` version, which is | ||||
|     always the latest stable release. | ||||
|  | ||||
|    The docker-compose files refer to the `latest` version, which is | ||||
|    always the latest stable release. | ||||
| 1.  If you built the image yourself, do the following: | ||||
|  | ||||
| 2. If you built the image yourself, do the following: | ||||
|     ```shell-session | ||||
|     $ git pull | ||||
|     $ docker compose build | ||||
|     $ docker compose up | ||||
|     ``` | ||||
|  | ||||
|    ```shell-session | ||||
|    $ git pull | ||||
|    $ docker-compose build | ||||
|    $ docker-compose up | ||||
|    ``` | ||||
|  | ||||
| Running `docker-compose up` will also apply any new database migrations. | ||||
| Running `docker compose up` will also apply any new database migrations. | ||||
| If you see everything working, press CTRL+C once to gracefully stop | ||||
| paperless. Then you can start paperless-ngx with `-d` to have it run in | ||||
| the background. | ||||
| @@ -94,7 +102,7 @@ the background. | ||||
| !!! note | ||||
|  | ||||
|     In version 0.9.14, the update process was changed. In 0.9.13 and | ||||
|     earlier, the docker-compose files specified exact versions and pull | ||||
|     earlier, the Docker Compose files specified exact versions and pull | ||||
|     won't automatically update to newer versions. In order to enable | ||||
|     updates as described above, either get the new `docker-compose.yml` | ||||
|     file from | ||||
| @@ -139,7 +147,7 @@ following: | ||||
| 1.  Update dependencies. New paperless version may require additional | ||||
|     dependencies. The dependencies required are listed in the section | ||||
|     about | ||||
|     [bare metal installations](/setup#bare_metal). | ||||
|     [bare metal installations](setup.md#bare_metal). | ||||
|  | ||||
| 2.  Update python requirements. Keep in mind to activate your virtual | ||||
|     environment before that, if you use one. | ||||
| @@ -167,6 +175,16 @@ following: | ||||
|     This might not actually do anything. Not every new paperless version | ||||
|     comes with new database migrations. | ||||
|  | ||||
| ### Database Upgrades | ||||
|  | ||||
| In general, paperless does not require a specific version of PostgreSQL or MariaDB and it is | ||||
| safe to update them to newer versions. However, you should always take a backup and follow | ||||
| the instructions from your database's documentation for how to upgrade between major versions. | ||||
|  | ||||
| For PostgreSQL, refer to [Upgrading a PostgreSQL Cluster](https://www.postgresql.org/docs/current/upgrading.html). | ||||
|  | ||||
| For MariaDB, refer to [Upgrading MariaDB](https://mariadb.com/kb/en/upgrading/) | ||||
|  | ||||
| ## Downgrading Paperless {#downgrade-paperless} | ||||
|  | ||||
| Downgrades are possible. However, some updates also contain database | ||||
| @@ -202,11 +220,11 @@ Paperless comes with some management commands that perform various | ||||
| maintenance tasks on your paperless instance. You can invoke these | ||||
| commands in the following way: | ||||
|  | ||||
| With docker-compose, while paperless is running: | ||||
| With Docker Compose, while paperless is running: | ||||
|  | ||||
| ```shell-session | ||||
| $ cd /path/to/paperless | ||||
| $ docker-compose exec webserver <command> <arguments> | ||||
| $ docker compose exec webserver <command> <arguments> | ||||
| ``` | ||||
|  | ||||
| With docker, while paperless is running: | ||||
| @@ -229,26 +247,28 @@ with the argument `--help`. | ||||
|  | ||||
| ### Document exporter {#exporter} | ||||
|  | ||||
| The document exporter exports all your data from paperless into a folder | ||||
| for backup or migration to another DMS. | ||||
| The document exporter exports all your data (including your settings | ||||
| and database contents) from paperless into a folder for backup or | ||||
| migration to another DMS. | ||||
|  | ||||
| If you use the document exporter within a cronjob to backup your data | ||||
| you might use the `-T` flag behind exec to suppress "The input device | ||||
| is not a TTY" errors. For example: | ||||
| `docker-compose exec -T webserver document_exporter ../export` | ||||
| `docker compose exec -T webserver document_exporter ../export` | ||||
|  | ||||
| ``` | ||||
| document_exporter target [-c] [-d] [-f] [-na] [-nt] [-p] [-sm] [-z] | ||||
|  | ||||
| optional arguments: | ||||
| -c, --compare-checksums | ||||
| -d, --delete | ||||
| -f, --use-filename-format | ||||
| -c,  --compare-checksums | ||||
| -d,  --delete | ||||
| -f,  --use-filename-format | ||||
| -na, --no-archive | ||||
| -nt, --no-thumbnail | ||||
| -p, --use-folder-prefix | ||||
| -p,  --use-folder-prefix | ||||
| -sm, --split-manifest | ||||
| -z  --zip | ||||
| -z,  --zip | ||||
| -zn, --zip-name | ||||
| ``` | ||||
|  | ||||
| `target` is a folder to which the data gets written. This includes | ||||
| @@ -276,7 +296,7 @@ other files. | ||||
|  | ||||
| The filenames generated by this command follow the format | ||||
| `[date created] [correspondent] [title].[extension]`. If you want | ||||
| paperless to use `PAPERLESS_FILENAME_FORMAT` for exported filenames | ||||
| paperless to use [`PAPERLESS_FILENAME_FORMAT`](configuration.md#PAPERLESS_FILENAME_FORMAT) for exported filenames | ||||
| instead, specify `-f` or `--use-filename-format`. | ||||
|  | ||||
| If `-na` or `--no-archive` is provided, no archive files will be exported, | ||||
| @@ -304,7 +324,8 @@ manifest.json will still contain application wide information (e.g. tags, corres | ||||
| documenttype, etc) | ||||
|  | ||||
| If `-z` or `--zip` is provided, the export will be a zip file | ||||
| in the target directory, named according to the current date. | ||||
| in the target directory, named according to the current local date or the | ||||
| value set in `-zn` or `--zip-name`. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
| @@ -328,11 +349,17 @@ When you use the provided docker compose script, put the export inside | ||||
| the `export` folder in your paperless source directory. Specify | ||||
| `../export` as the `source`. | ||||
|  | ||||
| Note that .zip files (as can be generated from the exporter) are not supported. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Importing from a previous version of Paperless may work, but for best | ||||
|     results it is suggested to match the versions. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     The importer should be run against a completely empty installation (database and directories) of Paperless-ngx. | ||||
|  | ||||
| ### Document retagger {#retagger} | ||||
|  | ||||
| Say you've imported a few hundred documents and now want to introduce a | ||||
| @@ -341,7 +368,7 @@ currently-imported docs. This problem is common enough that there are | ||||
| tools for it. | ||||
|  | ||||
| ``` | ||||
| document_retagger [-h] [-c] [-T] [-t] [-i] [--use-first] [-f] | ||||
| document_retagger [-h] [-c] [-T] [-t] [-i] [--id-range] [--use-first] [-f] | ||||
|  | ||||
| optional arguments: | ||||
| -c, --correspondent | ||||
| @@ -349,6 +376,7 @@ optional arguments: | ||||
| -t, --document_type | ||||
| -s, --storage_path | ||||
| -i, --inbox-only | ||||
| --id-range | ||||
| --use-first | ||||
| -f, --overwrite | ||||
| ``` | ||||
| @@ -365,6 +393,11 @@ Specify `-i` to have the document retagger work on documents tagged with | ||||
| inbox tags only. This is useful when you don't want to mess with your | ||||
| already processed documents. | ||||
|  | ||||
| Specify `--id-range 1 100` to have the document retagger work only on a | ||||
| specific range of document id´s. This can be useful if you have a lot of | ||||
| documents and want to test the matching rules only on a subset of | ||||
| documents. | ||||
|  | ||||
| When multiple document types or correspondents match a single document, | ||||
| the retagger won't assign these to the document. Specify `--use-first` | ||||
| to override this behavior and just use the first correspondent or type | ||||
| @@ -381,7 +414,7 @@ that don't match a document anymore get removed as well. | ||||
| ### Managing the Automatic matching algorithm | ||||
|  | ||||
| The _Auto_ matching algorithm requires a trained neural network to work. | ||||
| This network needs to be updated whenever somethings in your data | ||||
| This network needs to be updated whenever something in your data | ||||
| changes. The docker image takes care of that automatically with the task | ||||
| scheduler. You can manually renew the classifier by invoking the | ||||
| following management command: | ||||
| @@ -396,6 +429,9 @@ This command takes no arguments. | ||||
|  | ||||
| Use this command to re-create document thumbnails. Optionally include the ` --document {id}` option to generate thumbnails for a specific document only. | ||||
|  | ||||
| You may also specify `--processes` to control the number of processes used to generate new thumbnails. The default is to utilize | ||||
| a quarter of the available processors. | ||||
|  | ||||
| ``` | ||||
| document_thumbnails | ||||
| ``` | ||||
| @@ -423,7 +459,7 @@ task scheduler. | ||||
| ### Managing filenames {#renamer} | ||||
|  | ||||
| If you use paperless' feature to | ||||
| [assign custom filenames to your documents](/advanced_usage#file-name-handling), you can use this command to move all your files after | ||||
| [assign custom filenames to your documents](advanced_usage.md#file-name-handling), you can use this command to move all your files after | ||||
| changing the naming scheme. | ||||
|  | ||||
| !!! warning | ||||
| @@ -448,19 +484,19 @@ collection for issues. | ||||
|  | ||||
| The issues detected by the sanity checker are as follows: | ||||
|  | ||||
| - Missing original files. | ||||
| - Missing archive files. | ||||
| - Inaccessible original files due to improper permissions. | ||||
| - Inaccessible archive files due to improper permissions. | ||||
| - Corrupted original documents by comparing their checksum against | ||||
|   what is stored in the database. | ||||
| - Corrupted archive documents by comparing their checksum against what | ||||
|   is stored in the database. | ||||
| - Missing thumbnails. | ||||
| - Inaccessible thumbnails due to improper permissions. | ||||
| - Documents without any content (warning). | ||||
| - Orphaned files in the media directory (warning). These are files | ||||
|   that are not referenced by any document im paperless. | ||||
| -   Missing original files. | ||||
| -   Missing archive files. | ||||
| -   Inaccessible original files due to improper permissions. | ||||
| -   Inaccessible archive files due to improper permissions. | ||||
| -   Corrupted original documents by comparing their checksum against | ||||
|     what is stored in the database. | ||||
| -   Corrupted archive documents by comparing their checksum against what | ||||
|     is stored in the database. | ||||
| -   Missing thumbnails. | ||||
| -   Inaccessible thumbnails due to improper permissions. | ||||
| -   Documents without any content (warning). | ||||
| -   Orphaned files in the media directory (warning). These are files | ||||
|     that are not referenced by any document in paperless. | ||||
|  | ||||
| ``` | ||||
| document_sanity_checker | ||||
| @@ -529,7 +565,7 @@ Documents can be stored in Paperless using GnuPG encryption. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     Encryption is deprecated since [paperless-ng 0.9](/changelog#paperless-ng-090) and doesn't really | ||||
|     Encryption is deprecated since [paperless-ng 0.9](changelog.md#paperless-ng-090) and doesn't really | ||||
|     provide any additional security, since you have to store the passphrase | ||||
|     in a configuration file on the same system as the encrypted documents | ||||
|     for paperless to work. Furthermore, the entire text content of the | ||||
| @@ -550,9 +586,37 @@ Enabling encryption is no longer supported. | ||||
|  | ||||
| Basic usage to disable encryption of your document store: | ||||
|  | ||||
| (Note: If `PAPERLESS_PASSPHRASE` isn't set already, you need to specify | ||||
| (Note: If [`PAPERLESS_PASSPHRASE`](configuration.md#PAPERLESS_PASSPHRASE) isn't set already, you need to specify | ||||
| it here) | ||||
|  | ||||
| ``` | ||||
| decrypt_documents [--passphrase SECR3TP4SSPHRA$E] | ||||
| ``` | ||||
|  | ||||
| ### Detecting duplicates {#fuzzy_duplicate} | ||||
|  | ||||
| Paperless already catches and prevents upload of exactly matching documents, | ||||
| however a new scan of an existing document may not produce an exact bit for bit | ||||
| duplicate. But the content should be exact or close, allowing detection. | ||||
|  | ||||
| This tool does a fuzzy match over document content, looking for | ||||
| those which look close according to a given ratio. | ||||
|  | ||||
| At this time, other metadata (such as correspondent or type) is not | ||||
| taken into account by the detection. | ||||
|  | ||||
| ``` | ||||
| document_fuzzy_match [--ratio] [--processes N] | ||||
| ``` | ||||
|  | ||||
| | Option      | Required | Default             | Description                                                                                                                    | | ||||
| | ----------- | -------- | ------------------- | ------------------------------------------------------------------------------------------------------------------------------ | | ||||
| | --ratio     | No       | 85.0                | a number between 0 and 100, setting how similar a document must be for it to be reported. Higher numbers mean more similarity. | | ||||
| | --processes | No       | 1/4 of system cores | Number of processes to use for matching. Setting 1 disables multiple processes                                                 | | ||||
| | --delete    | No       | False               | If provided, one document of a matched pair above the ratio will be deleted.                                                   | | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     If providing the `--delete` option, it is highly recommended to have a backup. | ||||
|     While every effort has been taken to ensure proper operation, there is always the | ||||
|     chance of deletion of a file you want to keep. | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| # Advanced Topics | ||||
|  | ||||
| Paperless offers a couple features that automate certain tasks and make | ||||
| Paperless offers a couple of features that automate certain tasks and make | ||||
| your life easier. | ||||
|  | ||||
| ## Matching tags, correspondents, document types, and storage paths {#matching} | ||||
| @@ -35,9 +35,10 @@ The following algorithms are available: | ||||
|   (i.e. preserve ordering) in the PDF. | ||||
| - **Regular expression:** Parses the match as a regular expression and | ||||
|   tries to find a match within the document. | ||||
| - **Fuzzy match:** I don't know. Look at the source. | ||||
| - **Fuzzy match:** Uses a partial matching based on locating the tag text | ||||
|   inside the document, using a [partial ratio](https://maxbachmann.github.io/RapidFuzz/Usage/fuzz.html#partial-ratio) | ||||
| - **Auto:** Tries to automatically match new documents. This does not | ||||
|   require you to set a match. See the notes below. | ||||
|   require you to set a match. See the [notes below](#automatic-matching). | ||||
|  | ||||
| When using the _any_ or _all_ matching algorithms, you can search for | ||||
| terms that consist of multiple words by enclosing them in double quotes. | ||||
| @@ -92,7 +93,7 @@ when using this feature: | ||||
|   decide when not to assign a certain tag, correspondent, document | ||||
|   type, or storage path. This will usually be the case as you start | ||||
|   filling up paperless with documents. Example: If all your documents | ||||
|   are either from "Webshop" and "Bank", paperless will assign one | ||||
|   are either from "Webshop" or "Bank", paperless will assign one | ||||
|   of these correspondents to ANY new document, if both are set to | ||||
|   automatic matching. | ||||
|  | ||||
| @@ -101,12 +102,12 @@ when using this feature: | ||||
| Sometimes you may want to do something arbitrary whenever a document is | ||||
| consumed. Rather than try to predict what you may want to do, Paperless | ||||
| lets you execute scripts of your own choosing just before or after a | ||||
| document is consumed using a couple simple hooks. | ||||
| document is consumed using a couple of simple hooks. | ||||
|  | ||||
| Just write a script, put it somewhere that Paperless can read & execute, | ||||
| and then put the path to that script in `paperless.conf` or | ||||
| `docker-compose.env` with the variable name of either | ||||
| `PAPERLESS_PRE_CONSUME_SCRIPT` or `PAPERLESS_POST_CONSUME_SCRIPT`. | ||||
| [`PAPERLESS_PRE_CONSUME_SCRIPT`](configuration.md#PAPERLESS_PRE_CONSUME_SCRIPT) or [`PAPERLESS_POST_CONSUME_SCRIPT`](configuration.md#PAPERLESS_POST_CONSUME_SCRIPT). | ||||
|  | ||||
| !!! info | ||||
|  | ||||
| @@ -126,6 +127,7 @@ script can access the following relevant environment variables set: | ||||
| | ----------------------- | ------------------------------------------------------------ | | ||||
| | `DOCUMENT_SOURCE_PATH`  | Original path of the consumed document                       | | ||||
| | `DOCUMENT_WORKING_PATH` | Path to a copy of the original that consumption will work on | | ||||
| | `TASK_ID`               | UUID of the task used to process the new document (if any)   | | ||||
|  | ||||
| !!! note | ||||
|  | ||||
| @@ -134,6 +136,11 @@ script can access the following relevant environment variables set: | ||||
|     be triggered, leading to failures as two tasks work on the | ||||
|     same document path | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     If your script modifies `DOCUMENT_WORKING_PATH` in a non-deterministic | ||||
|     way, this may allow duplicate documents to be stored | ||||
|  | ||||
| A simple but common example for this would be creating a simple script | ||||
| like this: | ||||
|  | ||||
| @@ -168,21 +175,22 @@ Executed after the consumer has successfully processed a document and | ||||
| has moved it into paperless. It receives the following environment | ||||
| variables: | ||||
|  | ||||
| | Environment Variable         | Description                                   | | ||||
| | ---------------------------- | --------------------------------------------- | | ||||
| | `DOCUMENT_ID`                | Database primary key of the document          | | ||||
| | `DOCUMENT_FILE_NAME`         | Formatted filename, not including paths       | | ||||
| | `DOCUMENT_CREATED`           | Date & time when document created             | | ||||
| | `DOCUMENT_MODIFIED`          | Date & time when document was last modified   | | ||||
| | `DOCUMENT_ADDED`             | Date & time when document was added           | | ||||
| | `DOCUMENT_SOURCE_PATH`       | Path to the original document file            | | ||||
| | `DOCUMENT_ARCHIVE_PATH`      | Path to the generate archive file (if any)    | | ||||
| | `DOCUMENT_THUMBNAIL_PATH`    | Path to the generated thumbnail               | | ||||
| | `DOCUMENT_DOWNLOAD_URL`      | URL for document download                     | | ||||
| | `DOCUMENT_THUMBNAIL_URL`     | URL for the document thumbnail                | | ||||
| | `DOCUMENT_CORRESPONDENT`     | Assigned correspondent (if any)               | | ||||
| | `DOCUMENT_TAGS`              | Comma separated list of tags applied (if any) | | ||||
| | `DOCUMENT_ORIGINAL_FILENAME` | Filename of original document                 | | ||||
| | Environment Variable         | Description                                    | | ||||
| | ---------------------------- | ---------------------------------------------- | | ||||
| | `DOCUMENT_ID`                | Database primary key of the document           | | ||||
| | `DOCUMENT_FILE_NAME`         | Formatted filename, not including paths        | | ||||
| | `DOCUMENT_CREATED`           | Date & time when document created              | | ||||
| | `DOCUMENT_MODIFIED`          | Date & time when document was last modified    | | ||||
| | `DOCUMENT_ADDED`             | Date & time when document was added            | | ||||
| | `DOCUMENT_SOURCE_PATH`       | Path to the original document file             | | ||||
| | `DOCUMENT_ARCHIVE_PATH`      | Path to the generate archive file (if any)     | | ||||
| | `DOCUMENT_THUMBNAIL_PATH`    | Path to the generated thumbnail                | | ||||
| | `DOCUMENT_DOWNLOAD_URL`      | URL for document download                      | | ||||
| | `DOCUMENT_THUMBNAIL_URL`     | URL for the document thumbnail                 | | ||||
| | `DOCUMENT_CORRESPONDENT`     | Assigned correspondent (if any)                | | ||||
| | `DOCUMENT_TAGS`              | Comma separated list of tags applied (if any)  | | ||||
| | `DOCUMENT_ORIGINAL_FILENAME` | Filename of original document                  | | ||||
| | `TASK_ID`                    | Task UUID used to import the document (if any) | | ||||
|  | ||||
| The script can be in any language, A simple shell script example: | ||||
|  | ||||
| @@ -197,7 +205,7 @@ The script can be in any language, A simple shell script example: | ||||
| !!! warning | ||||
|  | ||||
|     The post consumption script should not modify the document files | ||||
|     directly | ||||
|     directly. | ||||
|  | ||||
| The script's stdout and stderr will be logged line by line to the | ||||
| webserver log, along with the exit code of the script. | ||||
| @@ -233,8 +241,8 @@ webserver: | ||||
|  | ||||
| Troubleshooting: | ||||
|  | ||||
| - Monitor the docker-compose log | ||||
|   `cd ~/paperless-ngx; docker-compose logs -f` | ||||
| - Monitor the Docker Compose log | ||||
|   `cd ~/paperless-ngx; docker compose logs -f` | ||||
| - Check your script's permission e.g. in case of permission error | ||||
|   `sudo chmod 755 post-consumption-example.sh` | ||||
| - Pipe your scripts's output to a log file e.g. | ||||
| @@ -248,7 +256,8 @@ document. You will end up getting files like `0000123.pdf` in your media | ||||
| directory. This isn't necessarily a bad thing, because you normally | ||||
| don't have to access these files manually. However, if you wish to name | ||||
| your files differently, you can do that by adjusting the | ||||
| `PAPERLESS_FILENAME_FORMAT` configuration option. Paperless adds the | ||||
| [`PAPERLESS_FILENAME_FORMAT`](configuration.md#PAPERLESS_FILENAME_FORMAT) configuration option | ||||
| or using [storage paths (see below)](#storage-paths). Paperless adds the | ||||
| correct file extension e.g. `.pdf`, `.jpg` automatically. | ||||
|  | ||||
| This variable allows you to configure the filename (folders are allowed) | ||||
| @@ -281,6 +290,15 @@ will create a directory structure as follows: | ||||
|     paperless will report your files as missing and won't be able to find | ||||
|     them. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     Paperless checks the filename of a document whenever it is saved. Changing (or deleting) | ||||
|     a [storage path](#storage-paths) will automatically be reflected in the file system. However, | ||||
|     when changing `PAPERLESS_FILENAME_FORMAT` you will need to manually run the | ||||
|     [`document renamer`](administration.md#renamer) to move any existing documents. | ||||
|  | ||||
| #### Placeholders | ||||
|  | ||||
| Paperless provides the following placeholders within filenames: | ||||
|  | ||||
| - `{asn}`: The archive serial number of the document, or "none". | ||||
| @@ -311,6 +329,13 @@ Paperless provides the following placeholders within filenames: | ||||
| - `{added_day}`: Day added only (number 01-31). | ||||
| - `{owner_username}`: Username of document owner, if any, or "none" | ||||
| - `{original_name}`: Document original filename, minus the extension, if any, or "none" | ||||
| - `{doc_pk}`: The paperless identifier (primary key) for the document. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     When using file name placeholders, in particular when using `{tag_list}`, | ||||
|     you may run into the limits of your operating system's maximum path lengths. | ||||
|     In that case, files will retain the previous path instead and the issue logged. | ||||
|  | ||||
| Paperless will try to conserve the information from your database as | ||||
| much as possible. However, some characters that you can use in document | ||||
| @@ -322,34 +347,12 @@ paperless will automatically append `_01`, `_02`, etc to the filename. | ||||
| This happens if all the placeholders in a filename evaluate to the same | ||||
| value. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     You can affect how empty placeholders are treated by changing the | ||||
|     following setting to `true`. | ||||
|  | ||||
|     ``` | ||||
|     PAPERLESS_FILENAME_FORMAT_REMOVE_NONE=True | ||||
|     ``` | ||||
|  | ||||
|     Doing this results in all empty placeholders resolving to "" instead | ||||
|     of "none" as stated above. Spaces before empty placeholders are | ||||
|     removed as well, empty directories are omitted. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     Paperless checks the filename of a document whenever it is saved. | ||||
|     Therefore, you need to update the filenames of your documents and move | ||||
|     them after altering this setting by invoking the | ||||
|     [`document renamer`](/administration#renamer). | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     Make absolutely sure you get the spelling of the placeholders right, or | ||||
|     else paperless will use the default naming scheme instead. | ||||
| If there are any errors in the placeholders included in `PAPERLESS_FILENAME_FORMAT`, | ||||
| paperless will fall back to using the default naming scheme instead. | ||||
|  | ||||
| !!! caution | ||||
|  | ||||
|     As of now, you could totally tell paperless to store your files anywhere | ||||
|     As of now, you could potentially tell paperless to store your files anywhere | ||||
|     outside the media directory by setting | ||||
|  | ||||
|     ``` | ||||
| @@ -357,28 +360,25 @@ value. | ||||
|     ``` | ||||
|  | ||||
|     However, keep in mind that inside docker, if files get stored outside of | ||||
|     the predefined volumes, they will be lost after a restart of paperless. | ||||
|     the predefined volumes, they will be lost after a restart. | ||||
|  | ||||
| !!! warning | ||||
| ##### Empty placeholders | ||||
|  | ||||
|     When file naming handling, in particular when using `{tag_list}`, | ||||
|     you may run into the limits of your operating system's maximum | ||||
|     path lengths.  Files will retain the previous path instead and | ||||
|     the issue logged. | ||||
| You can affect how empty placeholders are treated by changing the | ||||
| [`PAPERLESS_FILENAME_FORMAT_REMOVE_NONE`](configuration.md#PAPERLESS_FILENAME_FORMAT_REMOVE_NONE) setting. | ||||
|  | ||||
| ## Storage paths | ||||
| Enabling this results in all empty placeholders resolving to "" instead of "none" as stated above. Spaces | ||||
| before empty placeholders are removed as well, empty directories are omitted. | ||||
|  | ||||
| One of the best things in Paperless is that you can not only access the | ||||
| documents via the web interface, but also via the file system. | ||||
| ### Storage paths | ||||
|  | ||||
| When a single storage layout is not sufficient for your use case, | ||||
| storage paths come to the rescue. Storage paths allow you to configure | ||||
| more precisely where each document is stored in the file system. | ||||
| When a single storage layout is not sufficient for your use case, storage paths allow for more complex | ||||
| structure to set precisely where each document is stored in the file system. | ||||
|  | ||||
| - Each storage path is a `PAPERLESS_FILENAME_FORMAT` and | ||||
| - Each storage path is a [`PAPERLESS_FILENAME_FORMAT`](configuration.md#PAPERLESS_FILENAME_FORMAT) and | ||||
|   follows the rules described above | ||||
| - Each document is assigned a storage path using the matching | ||||
|   algorithms described above, but can be overwritten at any time | ||||
| - Each document is assigned a storage path using the matching algorithms described above, but can be | ||||
|   overwritten at any time | ||||
|  | ||||
| For example, you could define the following two storage paths: | ||||
|  | ||||
| @@ -415,7 +415,7 @@ Insurances/                             # Insurances | ||||
| !!! tip | ||||
|  | ||||
|     Defining a storage path is optional. If no storage path is defined for a | ||||
|     document, the global `PAPERLESS_FILENAME_FORMAT` is applied. | ||||
|     document, the global [`PAPERLESS_FILENAME_FORMAT`](configuration.md#PAPERLESS_FILENAME_FORMAT) is applied. | ||||
|  | ||||
| ## Celery Monitoring {#celery-monitoring} | ||||
|  | ||||
| @@ -425,8 +425,10 @@ to view more detailed information about the health of the celery workers | ||||
| used for asynchronous tasks. This includes details on currently running, | ||||
| queued and completed tasks, timing and more. Flower can also be used | ||||
| with Prometheus, as it exports metrics. For details on its capabilities, | ||||
| refer to the Flower documentation. | ||||
| refer to the [Flower](https://flower.readthedocs.io/en/latest/index.html) | ||||
| documentation. | ||||
|  | ||||
| Flower can be enabled with the setting [PAPERLESS_ENABLE_FLOWER](configuration.md#PAPERLESS_ENABLE_FLOWER). | ||||
| To configure Flower further, create a `flowerconfig.py` and | ||||
| place it into the `src/paperless` directory. For a Docker | ||||
| installation, you can use volumes to accomplish this: | ||||
| @@ -435,6 +437,8 @@ installation, you can use volumes to accomplish this: | ||||
| services: | ||||
|   # ... | ||||
|   webserver: | ||||
|     environment: | ||||
|       - PAPERLESS_ENABLE_FLOWER | ||||
|     ports: | ||||
|       - 5555:5555 # (2)! | ||||
|     # ... | ||||
| @@ -443,7 +447,7 @@ services: | ||||
| ``` | ||||
|  | ||||
| 1. Note the `:ro` tag means the file will be mounted as read only. | ||||
| 2. `flower` runs by default on port 5555, but this can be configured | ||||
| 2. By default, Flower runs on port 5555, but this can be configured. | ||||
|  | ||||
| ## Custom Container Initialization | ||||
|  | ||||
| @@ -504,9 +508,21 @@ existing tables) with: | ||||
|     an older system may fix issues that can arise while setting up Paperless-ngx but | ||||
|     `utf8mb3` can cause issues with consumption (where `utf8mb4` does not). | ||||
|  | ||||
| ### Missing timezones | ||||
|  | ||||
| MySQL as well as MariaDB do not have any timezone information by default (though some | ||||
| docker images such as the official MariaDB image take care of this for you) which will | ||||
| cause unexpected behavior with date-based queries. | ||||
|  | ||||
| To fix this, execute one of the following commands: | ||||
|  | ||||
| MySQL: `mysql_tzinfo_to_sql /usr/share/zoneinfo | mysql -u root mysql -p` | ||||
|  | ||||
| MariaDB: `mariadb-tzinfo-to-sql /usr/share/zoneinfo | mariadb -u root mysql -p` | ||||
|  | ||||
| ## Barcodes {#barcodes} | ||||
|  | ||||
| Paperless is able to utilize barcodes for automatically preforming some tasks. | ||||
| Paperless is able to utilize barcodes for automatically performing some tasks. | ||||
|  | ||||
| At this time, the library utilized for detection of barcodes supports the following types: | ||||
|  | ||||
| @@ -524,11 +540,11 @@ At this time, the library utilized for detection of barcodes supports the follow | ||||
| You may check for updates on the [zbar library homepage](https://github.com/mchehab/zbar). | ||||
| For usage in Paperless, the type of barcode does not matter, only the contents of it. | ||||
|  | ||||
| For how to enable barcode usage, see [the configuration](/configuration#barcodes). | ||||
| For how to enable barcode usage, see [the configuration](configuration.md#barcodes). | ||||
| The two settings may be enabled independently, but do have interactions as explained | ||||
| below. | ||||
|  | ||||
| ### Document Splitting | ||||
| ### Document Splitting {#document-splitting} | ||||
|  | ||||
| When enabled, Paperless will look for a barcode with the configured value and create a new document | ||||
| starting from the next page. The page with the barcode on it will _not_ be retained. It | ||||
| @@ -543,3 +559,132 @@ If document splitting via barcode is also enabled, documents will be split when | ||||
| barcode is located. However, differing from the splitting, the page with the | ||||
| barcode _will_ be retained. This allows application of a barcode to any page, including | ||||
| one which holds data to keep in the document. | ||||
|  | ||||
| ### Tag Assignment | ||||
|  | ||||
| When enabled, Paperless will parse barcodes and attempt to interpret and assign tags. | ||||
|  | ||||
| See the relevant settings [`PAPERLESS_CONSUMER_ENABLE_TAG_BARCODE`](configuration.md#PAPERLESS_CONSUMER_ENABLE_TAG_BARCODE) | ||||
| and [`PAPERLESS_CONSUMER_TAG_BARCODE_MAPPING`](configuration.md#PAPERLESS_CONSUMER_TAG_BARCODE_MAPPING) | ||||
| for more information. | ||||
|  | ||||
| ## Automatic collation of double-sided documents {#collate} | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     If your scanner supports double-sided scanning natively, you do not need this feature. | ||||
|  | ||||
| This feature is turned off by default, see [configuration](configuration.md#collate) on how to turn it on. | ||||
|  | ||||
| ### Summary | ||||
|  | ||||
| If you have a scanner with an automatic document feeder (ADF) that only scans a single side, | ||||
| this feature makes scanning double-sided documents much more convenient by automatically | ||||
| collating two separate scans into one document, reordering the pages as necessary. | ||||
|  | ||||
| ### Usage example | ||||
|  | ||||
| Suppose you have a double-sided document with 6 pages (3 sheets of paper). First, | ||||
| put the stack into your ADF as normal, ensuring that page 1 is scanned first. Your ADF | ||||
| will now scan pages 1, 3, and 5. Then you (or your scanner, if it supports it) upload | ||||
| the scan into the correct sub-directory of the consume folder (`double-sided` by default; | ||||
| keep in mind that Paperless will _not_ automatically create the directory for you.) | ||||
| Paperless will then process the scan and move it into an internal staging area. | ||||
|  | ||||
| The next step is to turn your stack upside down (without reordering the sheets of paper), | ||||
| and scan it once again, your ADF will now scan pages 6, 4, and 2, in that order. Once this | ||||
| scan is copied into the sub-directory, Paperless will collate the previous scan with the | ||||
| new one, reversing the order of the pages on the second, "even numbered" scan. The | ||||
| resulting document will have the pages 1-6 in the correct order, and this new file will | ||||
| then be processed as normal. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     When scanning the even numbered pages, you can omit the last empty pages, if there are | ||||
|     any. For example, if page 6 is empty, you only need to scan pages 2 and 4. _Do not_ omit | ||||
|     empty pages in the middle of the document. | ||||
|  | ||||
| ### Things that could go wrong | ||||
|  | ||||
| Paperless will notice when the first, "odd numbered" scan has less pages than the second | ||||
| scan (this can happen when e.g. the ADF skipped a few pages in the first pass). In that | ||||
| case, Paperless will remove the staging copy as well as the scan, and give you an error | ||||
| message asking you to restart the process from scratch, by scanning the odd pages again, | ||||
| followed by the even pages. | ||||
|  | ||||
| It's important that the scan files get consumed in the correct order, and one at a time. | ||||
| You therefore need to make sure that Paperless is running while you upload the files into | ||||
| the directory; and if you're using [polling](configuration.md#polling), make sure that | ||||
| `CONSUMER_POLLING` is set to a value lower than it takes for the second scan to appear, | ||||
| like 5-10 or even lower. | ||||
|  | ||||
| Another thing that might happen is that you start a double sided scan, but then forget | ||||
| to upload the second file. To avoid collating the wrong documents if you then come back | ||||
| a day later to scan a new double-sided document, Paperless will only keep an "odd numbered | ||||
| pages" file for up to 30 minutes. If more time passes, it will consider the next incoming | ||||
| scan a completely new "odd numbered pages" one. The old staging file will get discarded. | ||||
|  | ||||
| ### Interaction with "subdirs as tags" | ||||
|  | ||||
| The collation feature can be used together with the [subdirs as tags](configuration.md#consume_config) | ||||
| feature (but this is not a requirement). Just create a correctly named double-sided subdir | ||||
| in the hierarchy and upload your scans there. For example, both `double-sided/foo/bar` as | ||||
| well as `foo/bar/double-sided` will cause the collated document to be treated as if it | ||||
| were uploaded into `foo/bar` and receive both `foo` and `bar` tags, but not `double-sided`. | ||||
|  | ||||
| ### Interaction with document splitting | ||||
|  | ||||
| You can use the [document splitting](#document-splitting) feature, but if you use a normal | ||||
| single-sided split marker page, the split document(s) will have an empty page at the front (or | ||||
| whatever else was on the backside of the split marker page.) You can work around that by having | ||||
| a split marker page that has the split barcode on _both_ sides. This way, the extra page will | ||||
| get automatically removed. | ||||
|  | ||||
| ## SSO and third party authentication with Paperless-ngx | ||||
|  | ||||
| Paperless-ngx has a built-in authentication system from Django but you can easily integrate an | ||||
| external authentication solution using one of the following methods: | ||||
|  | ||||
| ### Remote User authentication | ||||
|  | ||||
| This is a simple option that uses remote user authentication made available by certain SSO | ||||
| applications. See the relevant configuration options for more information: | ||||
| [PAPERLESS_ENABLE_HTTP_REMOTE_USER](configuration.md#PAPERLESS_ENABLE_HTTP_REMOTE_USER), | ||||
| [PAPERLESS_HTTP_REMOTE_USER_HEADER_NAME](configuration.md#PAPERLESS_HTTP_REMOTE_USER_HEADER_NAME) | ||||
| and [PAPERLESS_LOGOUT_REDIRECT_URL](configuration.md#PAPERLESS_LOGOUT_REDIRECT_URL) | ||||
|  | ||||
| ### OpenID Connect and social authentication | ||||
|  | ||||
| Version 2.5.0 of Paperless-ngx added support for integrating other authentication systems via | ||||
| the [django-allauth](https://github.com/pennersr/django-allauth) package. Once set up, users | ||||
| can either log in or (optionally) sign up using any third party systems you integrate. See the | ||||
| relevant [configuration settings](configuration.md#PAPERLESS_SOCIALACCOUNT_PROVIDERS) and | ||||
| [django-allauth docs](https://docs.allauth.org/en/latest/socialaccount/configuration.html) | ||||
| for more information. | ||||
|  | ||||
| To associate an existing Paperless-ngx account with a social account, first login with your | ||||
| regular credentials and then choose "My Profile" from the user dropdown in the app and you | ||||
| will see options to connect social account(s). If enabled, signup options will be available | ||||
| on the login page. | ||||
|  | ||||
| As an example, to set up login via Github, the following environment variables would need to be | ||||
| set: | ||||
|  | ||||
| ```conf | ||||
| PAPERLESS_APPS="allauth.socialaccount.providers.github" | ||||
| PAPERLESS_SOCIALACCOUNT_PROVIDERS='{"github": {"APPS": [{"provider_id": "github","name": "Github","client_id": "<CLIENT_ID>","secret": "<CLIENT_SECRET>"}]}}' | ||||
| ``` | ||||
|  | ||||
| Or, to use OpenID Connect ("OIDC"), via Keycloak in this example: | ||||
|  | ||||
| ```conf | ||||
| PAPERLESS_APPS="allauth.socialaccount.providers.openid_connect" | ||||
| PAPERLESS_SOCIALACCOUNT_PROVIDERS=' | ||||
| {"openid_connect": {"APPS": [{"provider_id": "keycloak","name": "Keycloak","client_id": "paperless","secret": "<CLIENT_SECRET>","settings": { "server_url": "https://<KEYCLOAK_SERVER>/realms/<REALM>/.well-known/openid-configuration"}}]}}' | ||||
| ``` | ||||
|  | ||||
| More details about configuration option for various providers can be found in the [allauth documentation](https://docs.allauth.org/en/latest/socialaccount/providers/index.html#provider-specifics). | ||||
|  | ||||
| ### Disabling Regular Login | ||||
|  | ||||
| Once external auth is set up, 'regular' login can be disabled with the [PAPERLESS_DISABLE_REGULAR_LOGIN](configuration.md#PAPERLESS_DISABLE_REGULAR_LOGIN) setting. | ||||
|   | ||||
							
								
								
									
										168
									
								
								docs/api.md
									
									
									
									
									
								
							
							
						
						| @@ -6,19 +6,24 @@ provides a browsable API for most of its endpoints, which you can | ||||
| inspect at `http://<paperless-host>:<port>/api/`. This also documents | ||||
| most of the available filters and ordering fields. | ||||
|  | ||||
| The API provides 7 main endpoints: | ||||
| The API provides the following main endpoints: | ||||
|  | ||||
| - `/api/correspondents/`: Full CRUD support. | ||||
| - `/api/custom_fields/`: Full CRUD support. | ||||
| - `/api/documents/`: Full CRUD support, except POSTing new documents. | ||||
|   See below. | ||||
| - `/api/correspondents/`: Full CRUD support. | ||||
| - `/api/document_types/`: Full CRUD support. | ||||
| - `/api/groups/`: Full CRUD support. | ||||
| - `/api/logs/`: Read-Only. | ||||
| - `/api/tags/`: Full CRUD support. | ||||
| - `/api/tasks/`: Read-only. | ||||
| - `/api/mail_accounts/`: Full CRUD support. | ||||
| - `/api/mail_rules/`: Full CRUD support. | ||||
| - `/api/profile/`: GET, PATCH | ||||
| - `/api/share_links/`: Full CRUD support. | ||||
| - `/api/storage_paths/`: Full CRUD support. | ||||
| - `/api/tags/`: Full CRUD support. | ||||
| - `/api/tasks/`: Read-only. | ||||
| - `/api/users/`: Full CRUD support. | ||||
| - `/api/groups/`: Full CRUD support. | ||||
| - `/api/workflows/`: Full CRUD support. | ||||
|  | ||||
| All of these endpoints except for the logging endpoint allow you to | ||||
| fetch (and edit and delete where appropriate) individual objects by | ||||
| @@ -47,8 +52,15 @@ fields: | ||||
|   Read-only. | ||||
| - `archived_file_name`: Verbose filename of the archived document. | ||||
|   Read-only. Null if no archived document is available. | ||||
| - `notes`: Array of notes associated with the document. | ||||
| - `set_permissions`: Allows setting document permissions. Optional, | ||||
|   write-only. See [below](#permissions). | ||||
| - `custom_fields`: Array of custom fields & values, specified as | ||||
|   `{ field: CUSTOM_FIELD_ID, value: VALUE }` | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Note that all endpoint URLs must end with a `/`slash. | ||||
|  | ||||
| ## Downloading documents | ||||
|  | ||||
| @@ -124,9 +136,15 @@ File metadata is reported as a list of objects in the following form: | ||||
| depends on the file type and the metadata available in that specific | ||||
| document. Paperless only reports PDF metadata at this point. | ||||
|  | ||||
| ## Documents additional endpoints | ||||
|  | ||||
| - `/api/documents/<id>/notes/`: Retrieve notes for a document. | ||||
| - `/api/documents/<id>/share_links/`: Retrieve share links for a document. | ||||
| - `/api/documents/<id>/history/`: Retrieve history of changes for a document. | ||||
|  | ||||
| ## Authorization | ||||
|  | ||||
| The REST api provides three different forms of authentication. | ||||
| The REST api provides four different forms of authentication. | ||||
|  | ||||
| 1.  Basic authentication | ||||
|  | ||||
| @@ -147,6 +165,10 @@ The REST api provides three different forms of authentication. | ||||
|  | ||||
| 3.  Token authentication | ||||
|  | ||||
|     You can create (or re-create) an API token by opening the "My Profile" | ||||
|     link in the user dropdown found in the web UI and clicking the circular | ||||
|     arrow button. | ||||
|  | ||||
|     Paperless also offers an endpoint to acquire authentication tokens. | ||||
|  | ||||
|     POST a username and password as a form or json string to | ||||
| @@ -158,7 +180,13 @@ The REST api provides three different forms of authentication. | ||||
|     Authorization: Token <token> | ||||
|     ``` | ||||
|  | ||||
|     Tokens can be managed and revoked in the paperless admin. | ||||
|     Tokens can also be managed in the Django admin. | ||||
|  | ||||
| 4.  Remote User authentication | ||||
|  | ||||
|     If enabled (see | ||||
|     [configuration](configuration.md#PAPERLESS_ENABLE_HTTP_REMOTE_USER_API)), | ||||
|     you can authenticate against the API using Remote User auth. | ||||
|  | ||||
| ## Searching for documents | ||||
|  | ||||
| @@ -167,8 +195,8 @@ specific query parameters cause the API to return full text search | ||||
| results: | ||||
|  | ||||
| - `/api/documents/?query=your%20search%20query`: Search for a document | ||||
|   using a full text query. For details on the syntax, see [Basic Usage - Searching](/usage#basic-usage_searching). | ||||
| - `/api/documents/?more_like=1234`: Search for documents similar to | ||||
|   using a full text query. For details on the syntax, see [Basic Usage - Searching](usage.md#basic-usage_searching). | ||||
| - `/api/documents/?more_like_id=1234`: Search for documents similar to | ||||
|   the document with id 1234. | ||||
|  | ||||
| Pagination works exactly the same as it does for normal requests on this | ||||
| @@ -257,9 +285,12 @@ The endpoint supports the following optional form fields: | ||||
| - `correspondent`: Specify the ID of a correspondent that the consumer | ||||
|   should use for the document. | ||||
| - `document_type`: Similar to correspondent. | ||||
| - `storage_path`: Similar to correspondent. | ||||
| - `tags`: Similar to correspondent. Specify this multiple times to | ||||
|   have multiple tags added to the document. | ||||
| - `archive_serial_number`: An optional archive serial number to set. | ||||
| - `custom_fields`: An array of custom field ids to assign (with an empty | ||||
|   value) to the document. | ||||
|  | ||||
| The endpoint will immediately return HTTP 200 if the document consumption | ||||
| process was started successfully, with the UUID of the consumption task | ||||
| @@ -272,26 +303,111 @@ consumption including the ID of a created document if consumption succeeded. | ||||
| ## Permissions | ||||
|  | ||||
| All objects (documents, tags, etc.) allow setting object-level permissions | ||||
| with an optional `set_permissions` parameter which is of the form: | ||||
| with optional `owner` and / or a `set_permissions` parameters which are of | ||||
| the form: | ||||
|  | ||||
| ``` | ||||
| { | ||||
|   "owner": user_id, | ||||
|   "view": { | ||||
|       "users": [...], | ||||
|       "groups": [...], | ||||
|   }, | ||||
|   "change": { | ||||
|       "users": [...], | ||||
|       "groups": [...], | ||||
|   }, | ||||
| "owner": ..., | ||||
| "set_permissions": { | ||||
|     "view": { | ||||
|         "users": [...], | ||||
|         "groups": [...], | ||||
|     }, | ||||
|     "change": { | ||||
|         "users": [...], | ||||
|         "groups": [...], | ||||
|     }, | ||||
| } | ||||
| ``` | ||||
|  | ||||
| If this parameter is supplied the object's permissions will be overwritten, | ||||
| !!! note | ||||
|  | ||||
|     Arrays should contain user or group ID numbers. | ||||
|  | ||||
| If these parameters are supplied the object's permissions will be overwritten, | ||||
| assuming the authenticated user has permission to do so (the user must be | ||||
| the object owner or a superuser). | ||||
|  | ||||
| ### Retrieving full permissions | ||||
|  | ||||
| By default, the API will return a truncated version of object-level | ||||
| permissions, returning `user_can_change` indicating whether the current user | ||||
| can edit the object (either because they are the object owner or have permissions | ||||
| granted). You can pass the parameter `full_perms=true` to API calls to view the | ||||
| full permissions of objects in a format that mirrors the `set_permissions` | ||||
| parameter above. | ||||
|  | ||||
| ## Bulk Editing | ||||
|  | ||||
| The API supports various bulk-editing operations which are executed asynchronously. | ||||
|  | ||||
| ### Documents | ||||
|  | ||||
| For bulk operations on documents, use the endpoint `/api/documents/bulk_edit/` which accepts | ||||
| a json payload of the format: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "documents": [LIST_OF_DOCUMENT_IDS], | ||||
|   "method": METHOD, // see below | ||||
|   "parameters": args // see below | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The following methods are supported: | ||||
|  | ||||
| - `set_correspondent` | ||||
|   - Requires `parameters`: `{ "correspondent": CORRESPONDENT_ID }` | ||||
| - `set_document_type` | ||||
|   - Requires `parameters`: `{ "document_type": DOCUMENT_TYPE_ID }` | ||||
| - `set_storage_path` | ||||
|   - Requires `parameters`: `{ "storage_path": STORAGE_PATH_ID }` | ||||
| - `add_tag` | ||||
|   - Requires `parameters`: `{ "tag": TAG_ID }` | ||||
| - `remove_tag` | ||||
|   - Requires `parameters`: `{ "tag": TAG_ID }` | ||||
| - `modify_tags` | ||||
|   - Requires `parameters`: `{ "add_tags": [LIST_OF_TAG_IDS] }` and / or `{ "remove_tags": [LIST_OF_TAG_IDS] }` | ||||
| - `delete` | ||||
|   - No `parameters` required | ||||
| - `redo_ocr` | ||||
|   - No `parameters` required | ||||
| - `set_permissions` | ||||
|   - Requires `parameters`: | ||||
|     - `"permissions": PERMISSIONS_OBJ` (see format [above](#permissions)) and / or | ||||
|     - `"owner": OWNER_ID or null` | ||||
|     - `"merge": true or false` (defaults to false) | ||||
|   - The `merge` flag determines if the supplied permissions will overwrite all existing permissions (including | ||||
|     removing them) or be merged with existing permissions. | ||||
| - `merge` | ||||
|   - No additional `parameters` required. | ||||
|   - The ordering of the merged document is determined by the list of IDs. | ||||
|   - Optional `parameters`: | ||||
|     - `"metadata_document_id": DOC_ID` apply metadata (tags, correspondent, etc.) from this document to the merged document. | ||||
| - `split` | ||||
|   - Requires `parameters`: | ||||
|     - `"pages": [..]` The list should be a list of pages and/or a ranges, separated by commas e.g. `"[1,2-3,4,5-7]"` | ||||
|   - The split operation only accepts a single document. | ||||
| - `rotate` | ||||
|   - Requires `parameters`: | ||||
|     - `"degrees": DEGREES`. Must be an integer i.e. 90, 180, 270 | ||||
|  | ||||
| ### Objects | ||||
|  | ||||
| Bulk editing for objects (tags, document types etc.) currently supports set permissions or delete | ||||
| operations, using the endpoint: `/api/bulk_edit_objects/`, which requires a json payload of the format: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "objects": [LIST_OF_OBJECT_IDS], | ||||
|   "object_type": "tags", "correspondents", "document_types" or "storage_paths", | ||||
|   "operation": "set_permissions" or "delete", | ||||
|   "owner": OWNER_ID, // optional | ||||
|   "permissions": { "view": { "users": [] ... }, "change": { ... } }, // (see 'set_permissions' format above) | ||||
|   "merge": true / false // defaults to false, see above | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## API Versioning | ||||
|  | ||||
| The REST API is versioned since Paperless-ngx 1.3.0. | ||||
| @@ -348,3 +464,13 @@ Initial API version. | ||||
|   color to use for a specific tag, which is either black or white | ||||
|   depending on the brightness of `Tag.color`. | ||||
| - Removed field `Tag.colour`. | ||||
|  | ||||
| #### Version 3 | ||||
|  | ||||
| - Permissions endpoints have been added. | ||||
| - The format of the `/api/ui_settings/` has changed. | ||||
|  | ||||
| #### Version 4 | ||||
|  | ||||
| - Consumption templates were refactored to workflows and API endpoints | ||||
|   changed as such. | ||||
|   | ||||
| @@ -20,6 +20,28 @@ | ||||
|         margin-left: 4%; | ||||
|         float: left; | ||||
|     } | ||||
|  | ||||
|     .grid-flipped-left { | ||||
|         width: 66%; | ||||
|         float: left; | ||||
|     } | ||||
|  | ||||
|     .grid-flipped-right { | ||||
|         width: 29%; | ||||
|         margin-left: 4%; | ||||
|         float: left; | ||||
|     } | ||||
|  | ||||
|     .grid-half-left { | ||||
|         width: 48%; | ||||
|         float: left; | ||||
|     } | ||||
|  | ||||
|     .grid-half-right { | ||||
|         width: 48%; | ||||
|         margin-left: 4%; | ||||
|         float: left; | ||||
|     } | ||||
| } | ||||
|  | ||||
| .grid-left > p { | ||||
| @@ -31,6 +53,48 @@ | ||||
|     margin: 0; | ||||
| } | ||||
|  | ||||
| .clear { | ||||
|     clear: both; | ||||
|     margin-bottom: 20px; | ||||
|     display: block; | ||||
| } | ||||
|  | ||||
| .index-callout { | ||||
|     margin-right: .5rem; | ||||
| } | ||||
|  | ||||
| /* make code in headers not bold */ | ||||
| h4 code { | ||||
|     font-weight: normal; | ||||
| } | ||||
|  | ||||
| /* Hide config vars from sidebar, toc and move the border on mobile case their hidden */ | ||||
| .md-nav.md-nav--secondary .md-nav__item .md-nav__link[href*="PAPERLESS_"], | ||||
| .md-nav.md-nav--secondary .md-nav__item .md-nav__link[href*="USERMAP_"] { | ||||
|     display: none; | ||||
| } | ||||
|  | ||||
| @media screen and (max-width: 76.1875em) { | ||||
|     .md-nav--primary .md-nav__item { | ||||
|         border-top: none; | ||||
|     } | ||||
|  | ||||
|     .md-nav--primary .md-nav__link { | ||||
|         border-top: .05rem solid var(--md-default-fg-color--lightest); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /* Show search shortcut key */ | ||||
| [data-md-toggle="search"]:not(:checked) ~ .md-header .md-search__form::after { | ||||
|     position: absolute; | ||||
|     top: .3rem; | ||||
|     right: .3rem; | ||||
|     display: block; | ||||
|     padding: .1rem .4rem; | ||||
|     color: var(--md-default-fg-color--lighter); | ||||
|     font-weight: bold; | ||||
|     font-size: .8rem; | ||||
|     border: .05rem solid var(--md-default-fg-color--lighter); | ||||
|     border-radius: .1rem; | ||||
|     content: "/"; | ||||
|   } | ||||
|   | ||||
| Before Width: | Height: | Size: 740 KiB After Width: | Height: | Size: 1.8 MiB | 
| Before Width: | Height: | Size: 383 KiB After Width: | Height: | Size: 501 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/custom_field1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 21 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/custom_field2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.2 MiB | 
| Before Width: | Height: | Size: 704 KiB After Width: | Height: | Size: 644 KiB | 
| Before Width: | Height: | Size: 474 KiB After Width: | Height: | Size: 667 KiB | 
| Before Width: | Height: | Size: 616 KiB After Width: | Height: | Size: 1003 KiB | 
| Before Width: | Height: | Size: 708 KiB After Width: | Height: | Size: 1.7 MiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/documents-smallcards-slimsidebar.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 MiB | 
| Before Width: | Height: | Size: 705 KiB After Width: | Height: | Size: 1.8 MiB | 
| Before Width: | Height: | Size: 480 KiB After Width: | Height: | Size: 925 KiB | 
| Before Width: | Height: | Size: 689 KiB After Width: | Height: | Size: 1.7 MiB | 
| Before Width: | Height: | Size: 685 KiB After Width: | Height: | Size: 1.8 MiB | 
| Before Width: | Height: | Size: 859 KiB After Width: | Height: | Size: 2.3 MiB | 
| Before Width: | Height: | Size: 706 KiB After Width: | Height: | Size: 726 KiB | 
| Before Width: | Height: | Size: 76 KiB After Width: | Height: | Size: 169 KiB | 
| Before Width: | Height: | Size: 393 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/mobile1.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 432 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/mobile2.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 280 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/mobile3.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 246 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/new-correspondent.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 27 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/new-document_type.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 29 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/new-storage_path.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 48 KiB | 
| Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 45 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/permissions_document.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 559 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/permissions_global.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 116 KiB | 
| Before Width: | Height: | Size: 54 KiB After Width: | Height: | Size: 87 KiB | 
| Before Width: | Height: | Size: 516 KiB After Width: | Height: | Size: 792 KiB | 
							
								
								
									
										
											BIN
										
									
								
								docs/assets/screenshots/workflow.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 137 KiB | 
							
								
								
									
										1607
									
								
								docs/changelog.md
									
									
									
									
									
								
							
							
						
						| @@ -9,7 +9,7 @@ following way: | ||||
| - `main` always represents the latest release and will only see | ||||
|   changes when a new release is made. | ||||
| - `dev` contains the code that will be in the next release. | ||||
| - `feature-X` contain bigger changes that will be in some release, but | ||||
| - `feature-X` contains bigger changes that will be in some release, but | ||||
|   not necessarily the next one. | ||||
|  | ||||
| When making functional changes to Paperless-ngx, _always_ make your changes | ||||
| @@ -58,10 +58,10 @@ first-time setup. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Every command is executed directly from the root folder of the project unless specified otherwise. | ||||
|       Every command is executed directly from the root folder of the project unless specified otherwise. | ||||
|  | ||||
| 1.  Install prerequisites + pipenv as mentioned in | ||||
|     [Bare metal route](/setup#bare_metal). | ||||
|     [Bare metal route](setup.md#bare_metal). | ||||
|  | ||||
| 2.  Copy `paperless.conf.example` to `paperless.conf` and enable debug | ||||
|     mode within the file via `PAPERLESS_DEBUG=true`. | ||||
| @@ -177,68 +177,69 @@ The front end is built using AngularJS. In order to get started, you need Node.j | ||||
|  | ||||
|     The following commands are all performed in the `src-ui`-directory. You will need a running back end (including an active session) to connect to the back end API. To spin it up refer to the commands under the section [above](#back-end-development). | ||||
|  | ||||
| 1. Install the Angular CLI. You might need sudo privileges | ||||
|    to perform this command: | ||||
| 1.  Install the Angular CLI. You might need sudo privileges to perform this command: | ||||
|  | ||||
|    ```bash | ||||
|    $ npm install -g @angular/cli | ||||
|    ``` | ||||
|     ```bash | ||||
|     $ npm install -g @angular/cli | ||||
|     ``` | ||||
|  | ||||
| 2. Make sure that it's on your path. | ||||
| 2.  Make sure that it's on your path. | ||||
|  | ||||
| 3. Install all necessary modules: | ||||
| 3.  Install all necessary modules: | ||||
|  | ||||
|    ```bash | ||||
|    $ npm install | ||||
|    ``` | ||||
|     ```bash | ||||
|     $ npm install | ||||
|     ``` | ||||
|  | ||||
| 4. You can launch a development server by running: | ||||
| 4.  You can launch a development server by running: | ||||
|  | ||||
|    ```bash | ||||
|    $ ng serve | ||||
|    ``` | ||||
|     ```bash | ||||
|     $ ng serve | ||||
|     ``` | ||||
|  | ||||
|    This will automatically update whenever you save. However, in-place | ||||
|    compilation might fail on syntax errors, in which case you need to | ||||
|    restart it. | ||||
|     This will automatically update whenever you save. However, in-place | ||||
|     compilation might fail on syntax errors, in which case you need to | ||||
|     restart it. | ||||
|  | ||||
|    By default, the development server is available on `http://localhost:4200/` and is configured to access the API at | ||||
|    `http://localhost:8000/api/`, which is the default of the backend. If you enabled `DEBUG` on the back end, several security overrides for allowed hosts, CORS and X-Frame-Options are in place so that the front end behaves exactly as in production. | ||||
|     By default, the development server is available on `http://localhost:4200/` and is configured to access the API at | ||||
|     `http://localhost:8000/api/`, which is the default of the backend. If you enabled `DEBUG` on the back end, several security overrides for allowed hosts, CORS and X-Frame-Options are in place so that the front end behaves exactly as in production. | ||||
|  | ||||
| ### Testing and code style | ||||
|  | ||||
| - The front end code (.ts, .html, .scss) use `prettier` for code | ||||
|   formatting via the Git `pre-commit` hooks which run automatically on | ||||
|   commit. See [above](#code-formatting-with-pre-commit-hooks) for installation instructions. You can also run this via the CLI with a | ||||
|   command such as | ||||
| The front end code (.ts, .html, .scss) use `prettier` for code | ||||
| formatting via the Git `pre-commit` hooks which run automatically on | ||||
| commit. See [above](#code-formatting-with-pre-commit-hooks) for installation instructions. You can also run this via the CLI with a | ||||
| command such as | ||||
|  | ||||
|   ```bash | ||||
|   $ git ls-files -- '*.ts' | xargs pre-commit run prettier --files | ||||
|   ``` | ||||
| ```bash | ||||
| $ git ls-files -- '*.ts' | xargs pre-commit run prettier --files | ||||
| ``` | ||||
|  | ||||
| - Front end testing uses Jest and Playwright. Unit tests and e2e tests, | ||||
|   respectively, can be run non-interactively with: | ||||
| Front end testing uses Jest and Playwright. Unit tests and e2e tests, | ||||
| respectively, can be run non-interactively with: | ||||
|  | ||||
|   ```bash | ||||
|   $ ng test | ||||
|   $ npx playwright test | ||||
|   ``` | ||||
| ```bash | ||||
| $ ng test | ||||
| $ npx playwright test | ||||
| ``` | ||||
|  | ||||
|   - Playwright also includes a UI which can be run with: | ||||
| Playwright also includes a UI which can be run with: | ||||
|  | ||||
|     ```bash | ||||
|     $ npx playwright test --ui | ||||
|     ``` | ||||
| ```bash | ||||
| $ npx playwright test --ui | ||||
| ``` | ||||
|  | ||||
| - In order to build the front end and serve it as part of Django, execute: | ||||
| ### Building the frontend | ||||
|  | ||||
|   ```bash | ||||
|   $ ng build --configuration production | ||||
|   ``` | ||||
| In order to build the front end and serve it as part of Django, execute: | ||||
|  | ||||
|   This will build the front end and put it in a location from which the | ||||
|   Django server will serve it as static content. This way, you can verify | ||||
|   that authentication is working. | ||||
| ```bash | ||||
| $ ng build --configuration production | ||||
| ``` | ||||
|  | ||||
| This will build the front end and put it in a location from which the | ||||
| Django server will serve it as static content. This way, you can verify | ||||
| that authentication is working. | ||||
|  | ||||
| ## Localization | ||||
|  | ||||
| @@ -276,27 +277,17 @@ Adding new languages requires adding the translated files in the | ||||
|     } | ||||
|     ``` | ||||
|  | ||||
| 2.  Add the language to the available options in | ||||
| 2.  Add the language to the `LANGUAGE_OPTIONS` array in | ||||
|     `src-ui/src/app/services/settings.service.ts`: | ||||
|  | ||||
|     ```typescript | ||||
|     getLanguageOptions(): LanguageOption[] { | ||||
|         return [ | ||||
|             {code: "en-us", name: $localize`English (US)`, englishName: "English (US)", dateInputFormat: "mm/dd/yyyy"}, | ||||
|             {code: "en-gb", name: $localize`English (GB)`, englishName: "English (GB)", dateInputFormat: "dd/mm/yyyy"}, | ||||
|             {code: "de", name: $localize`German`, englishName: "German", dateInputFormat: "dd.mm.yyyy"}, | ||||
|             {code: "nl", name: $localize`Dutch`, englishName: "Dutch", dateInputFormat: "dd-mm-yyyy"}, | ||||
|             {code: "fr", name: $localize`French`, englishName: "French", dateInputFormat: "dd/mm/yyyy"}, | ||||
|             {code: "pt-br", name: $localize`Portuguese (Brazil)`, englishName: "Portuguese (Brazil)", dateInputFormat: "dd/mm/yyyy"} | ||||
|             // Add your new language here | ||||
|         ] | ||||
|     } | ||||
|     ``` | ||||
|  | ||||
|     `dateInputFormat` is a special string that defines the behavior of | ||||
|     the date input fields and absolutely needs to contain "dd", "mm" | ||||
|     and "yyyy". | ||||
|  | ||||
|     ``` | ||||
|  | ||||
| 3.  Import and register the Angular data for this locale in | ||||
|     `src-ui/src/app/app.module.ts`: | ||||
|  | ||||
|   | ||||
							
								
								
									
										37
									
								
								docs/faq.md
									
									
									
									
									
								
							
							
						
						| @@ -3,15 +3,16 @@ | ||||
| ## _What's the general plan for Paperless-ngx?_ | ||||
|  | ||||
| **A:** While Paperless-ngx is already considered largely | ||||
| "feature-complete" it is a community-driven project and development | ||||
| will be guided in this way. New features can be submitted via GitHub | ||||
| discussions and "up-voted" by the community but this is not a | ||||
| guarantee the feature will be implemented. This project will always be | ||||
| "feature-complete", it is a community-driven project and development | ||||
| will be guided in this way. New features can be submitted via | ||||
| [GitHub discussions](https://github.com/paperless-ngx/paperless-ngx/discussions) | ||||
| and "up-voted" by the community, but this is not a | ||||
| guarantee that the feature will be implemented. This project will always be | ||||
| open to collaboration in the form of PRs, ideas etc. | ||||
|  | ||||
| ## _I'm using docker. Where are my documents?_ | ||||
|  | ||||
| **A:** Your documents are stored inside the docker volume | ||||
| **A:** By default, your documents are stored inside the docker volume | ||||
| `paperless_media`. Docker manages this volume automatically for you. It | ||||
| is a persistent storage and will persist as long as you don't | ||||
| explicitly delete it. The actual location depends on your host operating | ||||
| @@ -45,8 +46,8 @@ elsewhere. Here are a couple notes about that. | ||||
| - By default, paperless uses the internal ID of each document as its | ||||
|   filename. This might not be very convenient for export. However, you | ||||
|   can adjust the way files are stored in paperless by | ||||
|   [configuring the filename format](/advanced_usage#file-name-handling). | ||||
| - [The exporter](/administration#exporter) is | ||||
|   [configuring the filename format](advanced_usage.md#file-name-handling). | ||||
| - [The exporter](administration.md#exporter) is | ||||
|   another easy way to get your files out of paperless with reasonable | ||||
|   file names. | ||||
|  | ||||
| @@ -58,7 +59,7 @@ elsewhere. Here are a couple notes about that. | ||||
|   WebP images are processed with OCR and converted into PDF documents. | ||||
| - Plain text documents are supported as well and are added verbatim to | ||||
|   paperless. | ||||
| - With the optional Tika integration enabled (see [Tika configuration](/configuration#tika), | ||||
| - With the optional Tika integration enabled (see [Tika configuration](https://docs.paperless-ngx.com/configuration#tika)), | ||||
|   Paperless also supports various Office documents (.docx, .doc, odt, | ||||
|   .ppt, .pptx, .odp, .xls, .xlsx, .ods). | ||||
|  | ||||
| @@ -77,20 +78,27 @@ has to do much less work to serve the data. | ||||
| !!! note | ||||
|  | ||||
|     You can adjust some of the settings so that paperless uses less | ||||
|     processing power. See [setup](/setup#less-powerful-devices) for details. | ||||
|     processing power. See [setup](setup.md#less-powerful-devices) for details. | ||||
|  | ||||
| ## _How do I install paperless-ngx on Raspberry Pi?_ | ||||
|  | ||||
| **A:** Docker images are available for armv7 and arm64 hardware, so just | ||||
| follow the docker-compose instructions. Apart from more required disk | ||||
| **A:** Docker images are available for arm64 hardware, so just | ||||
| follow the [Docker Compose instructions](https://docs.paperless-ngx.com/setup/#installation). Apart from more required disk | ||||
| space compared to a bare metal installation, docker comes with close to | ||||
| zero overhead, even on Raspberry Pi. | ||||
|  | ||||
| If you decide to got with the bare metal route, be aware that some of | ||||
| If you decide to go with the bare metal route, be aware that some of | ||||
| the python requirements do not have precompiled packages for ARM / | ||||
| ARM64. Installation of these will require additional development | ||||
| libraries and compilation will take a long time. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     For ARMv7 (32-bit) systems, paperless may still function, but it could require | ||||
|     modifications to the Dockerfile (if using Docker) or additional | ||||
|     tools for installing bare metal.  It is suggested to upgrade to arm64 | ||||
|     instead. | ||||
|  | ||||
| ## _How do I run this on Unraid?_ | ||||
|  | ||||
| **A:** Paperless-ngx is available as [community | ||||
| @@ -102,10 +110,7 @@ Fahrer](https://github.com/Tooa) created a container template for that. | ||||
| **A:** I honestly don't know! As for all other devices that might be | ||||
| able to run paperless, you're a bit on your own. If you can't run the | ||||
| docker image, the documentation has instructions for bare metal | ||||
| installs. I'm running paperless on an i3 processor from 2015 or so. | ||||
| This is also what I use to test new releases with. Apart from that, I | ||||
| also have a Raspberry Pi, which I occasionally build the image on and | ||||
| see if it works. | ||||
| installs. | ||||
|  | ||||
| ## _How do I proxy this with NGINX?_ | ||||
|  | ||||
|   | ||||
							
								
								
									
										198
									
								
								docs/index.md
									
									
									
									
									
								
							
							
						
						| @@ -5,9 +5,16 @@ | ||||
| **Paperless-ngx** is a _community-supported_ open-source document management system that transforms your | ||||
| physical documents into a searchable online archive so you can keep, well, _less paper_. | ||||
|  | ||||
| [Get started](/setup){ .md-button .md-button--primary .index-callout } | ||||
| [Get started](setup.md){ .md-button .md-button--primary .index-callout } | ||||
| [Demo](https://demo.paperless-ngx.com){ .md-button .md-button--secondary target=\_blank } | ||||
|  | ||||
| <div style="display: flex; justify-content: end; margin-top: -1.5rem;"> | ||||
|   <a href="https://m.do.co/c/8d70b916d462" target="_blank"> | ||||
|     <img src="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/PoweredByDO/DO_Powered_by_Badge_white.svg#only-dark" class="no-lightbox" width="150px"> | ||||
|     <img src="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/PoweredByDO/DO_Powered_by_Badge_black.svg#only-light" class="no-lightbox" width="150px"> | ||||
|   </a> | ||||
| </div> | ||||
|  | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
| {.index-screenshot} | ||||
| @@ -15,103 +22,162 @@ physical documents into a searchable online archive so you can keep, well, _less | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| ## Why This Exists | ||||
| ## Features | ||||
|  | ||||
| Paper is a nightmare. Environmental issues aside, there's no excuse for | ||||
| it in the 21st century. It takes up space, collects dust, doesn't | ||||
| support any form of a search feature, indexing is tedious, it's heavy | ||||
| and prone to damage & loss. | ||||
| -   **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. | ||||
| -   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. | ||||
| -   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. | ||||
| -   Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and their format can be configured freely with different configurations assigned to different documents. | ||||
| -   **Beautiful, modern web application** that features: | ||||
|     -   Customizable dashboard with statistics. | ||||
|     -   Filtering by tags, correspondents, types, and more. | ||||
|     -   Bulk editing of tags, correspondents, types and more. | ||||
|     -   Drag-and-drop uploading of documents throughout the app. | ||||
|     -   Customizable views can be saved and displayed on the dashboard and / or sidebar. | ||||
|     -   Support for custom fields of various data types. | ||||
|     -   Shareable public links with optional expiration. | ||||
| -   **Full text search** helps you find what you need: | ||||
|     -   Auto completion suggests relevant words from your documents. | ||||
|     -   Results are sorted by relevance to your search query. | ||||
|     -   Highlighting shows you which parts of the document matched the query. | ||||
|     -   Searching for similar documents ("More like this") | ||||
| -   **Email processing**[^1]: import documents from your email accounts: | ||||
|     -   Configure multiple accounts and rules for each account. | ||||
|     -   After processing, paperless can perform actions on the messages such as marking as read, deleting and more. | ||||
| -   A built-in robust **multi-user permissions** system that supports 'global' permissions as well as per document or object. | ||||
| -   A powerful workflow system that gives you even more control. | ||||
| -   **Optimized** for multi core systems: Paperless-ngx consumes multiple documents in parallel. | ||||
| -   The integrated sanity checker makes sure that your document archive is in good health. | ||||
|  | ||||
| This software is designed to make "going paperless" easier. No more worrying | ||||
| about finding stuff again, feed documents right from the post box into | ||||
| the scanner and then shred them. Perhaps you might find it useful too. | ||||
| [^1]: Office document and email consumption support is optional and provided by Apache Tika (see [configuration](https://docs.paperless-ngx.com/configuration/#tika)) | ||||
|  | ||||
| ## Paperless, a history | ||||
|  | ||||
| Paperless is a simple Django application running in two parts: a | ||||
| _Consumer_ (the thing that does the indexing) and the _Web server_ (the | ||||
| part that lets you search & download already-indexed documents). If you | ||||
| want to learn more about its functions keep on reading after the | ||||
| installation section. | ||||
| Paperless-ngx is the official successor to the original [Paperless](https://github.com/the-paperless-project/paperless) & [Paperless-ng](https://github.com/jonaswinkler/paperless-ng) projects and is designed to distribute the responsibility of advancing and supporting the project among a team of people. [Consider joining us!](https://github.com/paperless-ngx/paperless-ngx#community-support) | ||||
|  | ||||
| Paperless-ngx is a document management system that transforms your | ||||
| physical documents into a searchable online archive so you can keep, | ||||
| well, _less paper_. | ||||
|  | ||||
| Paperless-ngx forked from paperless-ng to continue the great work and | ||||
| distribute responsibility of supporting and advancing the project among | ||||
| a team of people. | ||||
|  | ||||
| NG stands for both Angular (the framework used for the Frontend) and | ||||
| next-gen. Publishing this project under a different name also avoids | ||||
| confusion between paperless and paperless-ngx. | ||||
|  | ||||
| If you want to learn about what's different in paperless-ngx from | ||||
| Paperless, check out these resources in the documentation: | ||||
|  | ||||
| - [Some screenshots](#screenshots) of the new UI are available. | ||||
| - Read [this section](/advanced_usage#automatic-matching) if you want to learn about how paperless automates all | ||||
|   tagging using machine learning. | ||||
| - Paperless now comes with a [proper email consumer](/usage#usage-email) that's fully tested and production ready. | ||||
| - Paperless creates searchable PDF/A documents from whatever you put into the consumption directory. This means | ||||
|   that you can select text in image-only documents coming from your scanner. | ||||
| - See [this note](/administration#encryption) about GnuPG encryption in paperless-ngx. | ||||
| - Paperless is now integrated with a | ||||
|   [task processing queue](/setup#task_processor) that tells you at a glance when and why something is not working. | ||||
| - The [changelog](/changelog) contains a detailed list of all changes in paperless-ngx. | ||||
| Further discussion of the transition between these projects can be found at | ||||
| [ng#1599](https://github.com/jonaswinkler/paperless-ng/issues/1599) and [ng#1632](https://github.com/jonaswinkler/paperless-ng/issues/1632). | ||||
|  | ||||
| ## Screenshots | ||||
|  | ||||
| This is what Paperless-ngx looks like. | ||||
| Paperless-ngx aims to be as nice to use as it is useful. Check out some screenshots below. | ||||
|  | ||||
| The dashboard shows customizable views on your document and allows | ||||
| document uploads: | ||||
| <div class="grid-flipped-left" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="grid-flipped-right" markdown> | ||||
|   The dashboard shows saved views which can be sorted. Documents can be uploaded with the button or dropped anywhere in the application. | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/dashboard.png) | ||||
| The document list provides three different styles to browse your documents. | ||||
|  | ||||
| The document list provides three different styles to scroll through your | ||||
| documents: | ||||
| {: style="width:32%"} | ||||
| {: style="width:32%"} | ||||
| {: style="width:32%"} | ||||
|  | ||||
| [](assets/screenshots/documents-table.png) | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/documents-smallcards.png) | ||||
| <div class="grid-left" markdown> | ||||
|   Use the 'slim' sidebar to focus on your docs and minimize the UI. | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/documents-largecards.png) | ||||
| Of course, Paperless-ngx also supports dark mode: | ||||
|  | ||||
| Paperless-ngx also supports dark mode: | ||||
|  | ||||
|  | ||||
| [](assets/screenshots/documents-smallcards-dark.png) | ||||
| <div class="clear"></div> | ||||
|  | ||||
| Extensive filtering mechanisms: | ||||
| <div class="grid-left" markdown> | ||||
|   Quickly find documents with extensive filtering mechanisms. | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
| <div class="grid-left" markdown> | ||||
|   And perform bulk edit operations to set tags, correspondents, etc. as well as permissions. | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/documents-filter.png) | ||||
| Side-by-side editing of documents. | ||||
|  | ||||
| Bulk editing of document tags, correspondents, etc.: | ||||
|  | ||||
|  | ||||
| [](assets/screenshots/bulk-edit.png) | ||||
| <div class="grid-left" markdown> | ||||
|   Support for custom fields. | ||||
|  | ||||
| Side-by-side editing of documents: | ||||
|  | ||||
|  | ||||
| [](assets/screenshots/editing.png) | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| Tag editing. This looks about the same for correspondents and document | ||||
| types. | ||||
| <div class="grid-left" markdown> | ||||
|   A robust permissions system with support for 'global' and document / object permissions. | ||||
|  | ||||
| [](assets/screenshots/new-tag.png) | ||||
|  | ||||
|  | ||||
| Searching provides auto complete and highlights the results. | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/search-preview.png) | ||||
| <div class="grid-left" markdown> | ||||
|   Searching provides auto complete and highlights the results. | ||||
|  | ||||
| [](assets/screenshots/search-results.png) | ||||
|  | ||||
|  | ||||
| Fancy mail filters! | ||||
| </div> | ||||
| <div class="grid-right" markdown> | ||||
|    | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| [](assets/screenshots/mail-rules-edited.png) | ||||
| Tag, correspondent, document type and storage path editing. | ||||
|  | ||||
| {: style="width:21%; float: left"} | ||||
| {: style="width:21%; margin-left: 4%; float: left"} | ||||
| {: style="width:21%; margin-left: 4%; float: left"} | ||||
| {: style="width:21%; margin-left: 4%; float: left"} | ||||
|  | ||||
| <div class="clear"></div> | ||||
|  | ||||
| <div class="grid-half-left" markdown> | ||||
|   Mail rules support various filters and actions for incoming e-mails. | ||||
|  | ||||
|  | ||||
|  | ||||
| </div> | ||||
| <div class="grid-half-right" markdown> | ||||
|   Workflows provide finer control over the document pipeline and trigger actions. | ||||
|  | ||||
|  | ||||
|  | ||||
| </div> | ||||
| <div class="clear"></div> | ||||
|  | ||||
| <div class="clear"></div> | ||||
|  | ||||
| Mobile devices are supported. | ||||
|  | ||||
| [](assets/screenshots/mobile.png) | ||||
| {: style="width:32%"} | ||||
| {: style="width:32%"} | ||||
| {: style="width:32%"} | ||||
|  | ||||
| ## Support | ||||
|  | ||||
| @@ -131,7 +197,7 @@ People interested in continuing the work on paperless-ngx are encouraged to reac | ||||
|  | ||||
| ### Translation | ||||
|  | ||||
| Paperless-ngx is available in many languages that are coordinated on [Crowdin](https://crwd.in/paperless-ngx). If you want to help out by translating paperless-ngx into your language, please head over to https://crwd.in/paperless-ngx, and thank you! | ||||
| Paperless-ngx is available in many languages that are coordinated on [Crowdin](https://crwd.in/paperless-ngx). If you want to help out by translating paperless-ngx into your language, please head over to the [Paperless-ngx project at Crowdin](https://crwd.in/paperless-ngx), and thank you! | ||||
|  | ||||
| ## Scanners & Software | ||||
|  | ||||
|   | ||||
							
								
								
									
										147
									
								
								docs/setup.md
									
									
									
									
									
								
							
							
						
						| @@ -25,12 +25,16 @@ necessary configuration files, pull the docker image, start paperless | ||||
| and create your user account. This script essentially performs all the | ||||
| steps described in [Docker setup](#docker_hub) automatically. | ||||
|  | ||||
| 1.  Make sure that docker and docker-compose are installed. | ||||
| 1.  Make sure that Docker and Docker Compose are installed. | ||||
|  | ||||
|     !!! tip | ||||
|  | ||||
|         See the Docker installation instructions at https://docs.docker.com/engine/install/ | ||||
|  | ||||
| 2.  Download and run the installation script: | ||||
|  | ||||
|     ```shell-session | ||||
|     $ bash -c "$(curl -L https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/install-paperless-ngx.sh)" | ||||
|     $ bash -c "$(curl --location --silent --show-error https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/install-paperless-ngx.sh)" | ||||
|     ``` | ||||
|  | ||||
|     !!! note | ||||
| @@ -62,19 +66,19 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|         For new installations, it is recommended to use PostgreSQL as the | ||||
|         database backend. | ||||
|  | ||||
| 3.  Install [Docker](https://www.docker.com/) and | ||||
|     [docker-compose](https://docs.docker.com/compose/install/). | ||||
| 3.  Install [Docker](https://docs.docker.com/engine/install/) and | ||||
|     [Docker Compose](https://docs.docker.com/compose/install/). | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
|         If you want to use the included `docker-compose.*.yml` file, you | ||||
|         need to have at least Docker version **17.09.0** and docker-compose | ||||
|         version **1.17.0**. To check do: `docker-compose -v` or `docker -v` | ||||
|         need to have at least Docker version **17.09.0** and Docker Compose | ||||
|         version **v2**. To check do: `docker compose version` or `docker -v` | ||||
|  | ||||
|         See the [Docker installation guide](https://docs.docker.com/engine/install/) on how to install the current | ||||
|         version of Docker for your operating system or Linux distribution of | ||||
|         choice. To get the latest version of docker-compose, follow the | ||||
|         [docker-compose installation guide](https://docs.docker.com/compose/install/linux/) if your package repository | ||||
|         choice. To get the latest version of Docker Compose, follow the | ||||
|         [Docker Compose installation guide](https://docs.docker.com/compose/install/linux/) if your package repository | ||||
|         doesn't include it. | ||||
|  | ||||
| 4.  Modify `docker-compose.yml` to your preferences. You may want to | ||||
| @@ -92,7 +96,7 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|     - /home/jonaswinkler/paperless-inbox:/usr/src/paperless/consume | ||||
|     ``` | ||||
|  | ||||
|     Don't change the part after the colon or paperless wont find your | ||||
|     Don't change the part after the colon or paperless won't find your | ||||
|     documents. | ||||
|  | ||||
|     You may also need to change the default port that the webserver will | ||||
| @@ -117,6 +121,10 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|  | ||||
|     **Rootless** | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
|         It is currently not possible to run the container rootless if additional languages are specified via `PAPERLESS_OCR_LANGUAGES`. | ||||
|  | ||||
|     If you want to run Paperless as a rootless container, you will need | ||||
|     to do the following in your `docker-compose.yml`: | ||||
|  | ||||
| @@ -124,7 +132,7 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|       user in the container. This value (`user_id` below), should be | ||||
|       the same id that `USERMAP_UID` and `USERMAP_GID` are set to in | ||||
|       the next step. See `USERMAP_UID` and `USERMAP_GID` | ||||
|       [here](/configuration#docker). | ||||
|       [here](configuration.md#docker). | ||||
|  | ||||
|     Your entry for Paperless should contain something like: | ||||
|  | ||||
| @@ -148,12 +156,12 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|     !!! note | ||||
|  | ||||
|         You can copy any setting from the file `paperless.conf.example` and | ||||
|         paste it here. Have a look at [configuration](/configuration) to see what's available. | ||||
|         paste it here. Have a look at [configuration](configuration.md) to see what's available. | ||||
|  | ||||
|     !!! note | ||||
|  | ||||
|         You can utilize Docker secrets for configuration settings by | ||||
|         appending `_FILE` to configuration values. For example `PAPERLESS_DBUSER` | ||||
|         appending `_FILE` to configuration values. For example [`PAPERLESS_DBUSER`](configuration.md#PAPERLESS_DBUSER) | ||||
|         can be set using `PAPERLESS_DBUSER_FILE=/var/run/secrets/password.txt`. | ||||
|  | ||||
|     !!! warning | ||||
| @@ -162,16 +170,16 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|         system notifications with `inotify`. When storing the consumption | ||||
|         directory on such a file system, paperless will not pick up new | ||||
|         files with the default configuration. You will need to use | ||||
|         `PAPERLESS_CONSUMER_POLLING`, which will disable inotify. See | ||||
|         [here](/configuration#polling). | ||||
|         [`PAPERLESS_CONSUMER_POLLING`](configuration.md#PAPERLESS_CONSUMER_POLLING), which will disable inotify. See | ||||
|         [here](configuration.md#polling). | ||||
|  | ||||
| 6.  Run `docker-compose pull`. This will pull the image. | ||||
| 6.  Run `docker compose pull`. This will pull the image. | ||||
|  | ||||
| 7.  To be able to login, you will need a super user. To create it, | ||||
|     execute the following command: | ||||
|  | ||||
|     ```shell-session | ||||
|     $ docker-compose run --rm webserver createsuperuser | ||||
|     $ docker compose run --rm webserver createsuperuser | ||||
|     ``` | ||||
|  | ||||
|     or using docker exec from within the container: | ||||
| @@ -183,7 +191,7 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|     This will prompt you to set a username, an optional e-mail address | ||||
|     and finally a password (at least 8 characters). | ||||
|  | ||||
| 8.  Run `docker-compose up -d`. This will create and start the necessary containers. | ||||
| 8.  Run `docker compose up -d`. This will create and start the necessary containers. | ||||
|  | ||||
| 9.  The default `docker-compose.yml` exports the webserver on your local | ||||
|     port | ||||
| @@ -209,39 +217,27 @@ steps described in [Docker setup](#docker_hub) automatically. | ||||
|     root as well. | ||||
|  | ||||
| 3.  In the `docker-compose.yml` file, find the line that instructs | ||||
|     docker-compose to pull the paperless image from Docker Hub: | ||||
|     Docker Compose to pull the paperless image from Docker Hub: | ||||
|  | ||||
|     ```yaml | ||||
|     webserver: | ||||
|       image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     ``` | ||||
|  | ||||
|     and replace it with a line that instructs docker-compose to build | ||||
|     and replace it with a line that instructs Docker Compose to build | ||||
|     the image from the current working directory instead: | ||||
|  | ||||
|     ```yaml | ||||
|     webserver: | ||||
|       build: | ||||
|         context: . | ||||
|         args: | ||||
|           QPDF_VERSION: x.y.x | ||||
|           PIKEPDF_VERSION: x.y.z | ||||
|           PSYCOPG2_VERSION: x.y.z | ||||
|           JBIG2ENC_VERSION: 0.29 | ||||
|     ``` | ||||
|  | ||||
|     !!! note | ||||
|  | ||||
|         You should match the build argument versions to the version for the | ||||
|         release you have checked out. These are pre-built images with | ||||
|         certain, more updated software. If you want to build these images | ||||
|         your self, that is possible, but beyond the scope of these steps. | ||||
|  | ||||
| 4.  Follow steps 3 to 8 of [Docker Setup](#docker_hub). When asked to run | ||||
|     `docker-compose pull` to pull the image, do | ||||
|     `docker compose pull` to pull the image, do | ||||
|  | ||||
|     ```shell-session | ||||
|     $ docker-compose build | ||||
|     $ docker compose build | ||||
|     ``` | ||||
|  | ||||
|     instead to build the image. | ||||
| @@ -255,10 +251,11 @@ supported. | ||||
|  | ||||
| 1.  Install dependencies. Paperless requires the following packages. | ||||
|  | ||||
|     - `python3` 3.8, 3.9 | ||||
|     - `python3` - 3.9 - 3.11 are supported | ||||
|     - `python3-pip` | ||||
|     - `python3-dev` | ||||
|     - `default-libmysqlclient-dev` for MariaDB | ||||
|     - `pkg-config` for mysqlclient (python dependency) | ||||
|     - `fonts-liberation` for generating thumbnails for plain text | ||||
|       files | ||||
|     - `imagemagick` >= 6 for PDF conversion | ||||
| @@ -273,7 +270,7 @@ supported. | ||||
|     Use this list for your preferred package management: | ||||
|  | ||||
|     ``` | ||||
|     python3 python3-pip python3-dev imagemagick fonts-liberation gnupg libpq-dev default-libmysqlclient-dev libmagic-dev mime-support libzbar0 poppler-utils | ||||
|     python3 python3-pip python3-dev imagemagick fonts-liberation gnupg libpq-dev default-libmysqlclient-dev pkg-config libmagic-dev mime-support libzbar0 poppler-utils | ||||
|     ``` | ||||
|  | ||||
|     These dependencies are required for OCRmyPDF, which is used for text | ||||
| @@ -341,41 +338,41 @@ supported. | ||||
|     home folder of the user you created before (`/opt/paperless`). | ||||
|  | ||||
|     Optional: If you cloned the git repo, you will have to | ||||
|     compile the frontend yourself, see [here](/development#front-end-development) | ||||
|     compile the frontend yourself, see [here](development.md#front-end-development) | ||||
|     and use the `build` step, not `serve`. | ||||
|  | ||||
| 6.  Configure paperless. See [configuration](/configuration) for details. | ||||
| 6.  Configure paperless. See [configuration](configuration.md) for details. | ||||
|     Edit the included `paperless.conf` and adjust the settings to your | ||||
|     needs. Required settings for getting | ||||
|     paperless running are: | ||||
|  | ||||
|     - `PAPERLESS_REDIS` should point to your redis server, such as | ||||
|     - [`PAPERLESS_REDIS`](configuration.md#PAPERLESS_REDIS) should point to your redis server, such as | ||||
|       <redis://localhost:6379>. | ||||
|     - `PAPERLESS_DBENGINE` optional, and should be one of `postgres`, | ||||
|     - [`PAPERLESS_DBENGINE`](configuration.md#PAPERLESS_DBENGINE) optional, and should be one of `postgres`, | ||||
|       `mariadb`, or `sqlite` | ||||
|     - `PAPERLESS_DBHOST` should be the hostname on which your | ||||
|     - [`PAPERLESS_DBHOST`](configuration.md#PAPERLESS_DBHOST) should be the hostname on which your | ||||
|       PostgreSQL server is running. Do not configure this to use | ||||
|       SQLite instead. Also configure port, database name, user and | ||||
|       password as necessary. | ||||
|     - `PAPERLESS_CONSUMPTION_DIR` should point to a folder which | ||||
|     - [`PAPERLESS_CONSUMPTION_DIR`](configuration.md#PAPERLESS_CONSUMPTION_DIR) should point to a folder which | ||||
|       paperless should watch for documents. You might want to have | ||||
|       this somewhere else. Likewise, `PAPERLESS_DATA_DIR` and | ||||
|       `PAPERLESS_MEDIA_ROOT` define where paperless stores its data. | ||||
|       this somewhere else. Likewise, [`PAPERLESS_DATA_DIR`](configuration.md#PAPERLESS_DATA_DIR) and | ||||
|       [`PAPERLESS_MEDIA_ROOT`](configuration.md#PAPERLESS_MEDIA_ROOT) define where paperless stores its data. | ||||
|       If you like, you can point both to the same directory. | ||||
|     - `PAPERLESS_SECRET_KEY` should be a random sequence of | ||||
|     - [`PAPERLESS_SECRET_KEY`](configuration.md#PAPERLESS_SECRET_KEY) should be a random sequence of | ||||
|       characters. It's used for authentication. Failure to do so | ||||
|       allows third parties to forge authentication credentials. | ||||
|     - `PAPERLESS_URL` if you are behind a reverse proxy. This should | ||||
|     - [`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) if you are behind a reverse proxy. This should | ||||
|       point to your domain. Please see | ||||
|       [configuration](/configuration) for more | ||||
|       [configuration](configuration.md) for more | ||||
|       information. | ||||
|  | ||||
|     Many more adjustments can be made to paperless, especially the OCR | ||||
|     part. The following options are recommended for everyone: | ||||
|  | ||||
|     - Set `PAPERLESS_OCR_LANGUAGE` to the language most of your | ||||
|     - Set [`PAPERLESS_OCR_LANGUAGE`](configuration.md#PAPERLESS_OCR_LANGUAGE) to the language most of your | ||||
|       documents are written in. | ||||
|     - Set `PAPERLESS_TIME_ZONE` to your local time zone. | ||||
|     - Set [`PAPERLESS_TIME_ZONE`](configuration.md#PAPERLESS_TIME_ZONE) to your local time zone. | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
| @@ -521,7 +518,7 @@ supported. | ||||
|     not available for most distributions. | ||||
|  | ||||
| 15. Optional: If using the NLTK machine learning processing (see | ||||
|     `PAPERLESS_ENABLE_NLTK` in [configuration](/configuration#software_tweaks) for details), | ||||
|     [`PAPERLESS_ENABLE_NLTK`](configuration.md#PAPERLESS_ENABLE_NLTK) for details), | ||||
|     download the NLTK data for the Snowball | ||||
|     Stemmer, Stopwords and Punkt tokenizer to your | ||||
|     `PAPERLESS_DATA_DIR/nltk`. Refer to the [NLTK | ||||
| @@ -552,7 +549,7 @@ to | ||||
| image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
| ``` | ||||
|  | ||||
| and then run `docker-compose up -d` which will pull the new image | ||||
| and then run `docker compose up -d` which will pull the new image | ||||
| recreate the container. That's it! | ||||
|  | ||||
| Users who installed with the bare-metal route should also update their | ||||
| @@ -570,7 +567,7 @@ your setup depending on how you installed paperless. | ||||
| This setup describes how to update an existing paperless Docker | ||||
| installation. The important things to keep in mind are as follows: | ||||
|  | ||||
| - Read the [changelog](/changelog) and | ||||
| - Read the [changelog](changelog.md) and | ||||
|   take note of breaking changes. | ||||
| - You should decide if you want to stick with SQLite or want to | ||||
|   migrate your database to PostgreSQL. See [documentation](#sqlite_to_psql) | ||||
| @@ -581,7 +578,7 @@ installation. The important things to keep in mind are as follows: | ||||
| - The task scheduler of paperless, which is used to execute periodic | ||||
|   tasks such as email checking and maintenance, requires a | ||||
|   [redis](https://redis.io/) message broker instance. The | ||||
|   docker-compose route takes care of that. | ||||
|   Docker Compose route takes care of that. | ||||
| - The layout of the folder structure for your documents and data | ||||
|   remains the same, so you can just plug your old docker volumes into | ||||
|   paperless-ngx and expect it to find everything where it should be. | ||||
| @@ -592,7 +589,7 @@ Migration to paperless-ngx is then performed in a few simple steps: | ||||
|  | ||||
|     ```bash | ||||
|     $ cd /path/to/current/paperless | ||||
|     $ docker-compose down | ||||
|     $ docker compose down | ||||
|     ``` | ||||
|  | ||||
| 2.  Do a backup for two purposes: If something goes wrong, you still | ||||
| @@ -600,7 +597,7 @@ Migration to paperless-ngx is then performed in a few simple steps: | ||||
|     switch back to paperless. | ||||
|  | ||||
| 3.  Download the latest release of paperless-ngx. You can either go with | ||||
|     the docker-compose files from | ||||
|     the Docker Compose files from | ||||
|     [here](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) | ||||
|     or clone the repository to build the image yourself (see | ||||
|     [above](#docker_build)). You can | ||||
| @@ -630,14 +627,14 @@ Migration to paperless-ngx is then performed in a few simple steps: | ||||
|     See [Docker setup](#docker_hub) details on | ||||
|     which edits are advised. | ||||
|  | ||||
| 6.  [Update paperless.](/administration#updating) | ||||
| 6.  [Update paperless.](administration.md#updating) | ||||
|  | ||||
| 7.  In order to find your existing documents with the new search | ||||
|     feature, you need to invoke a one-time operation that will create | ||||
|     the search index: | ||||
|  | ||||
|     ```shell-session | ||||
|     $ docker-compose run --rm webserver document_index reindex | ||||
|     $ docker compose run --rm webserver document_index reindex | ||||
|     ``` | ||||
|  | ||||
|     This will migrate your database and create the search index. After | ||||
| @@ -646,7 +643,7 @@ Migration to paperless-ngx is then performed in a few simple steps: | ||||
| 8.  Start paperless-ngx. | ||||
|  | ||||
|     ```bash | ||||
|     $ docker-compose up -d | ||||
|     $ docker compose up -d | ||||
|     ``` | ||||
|  | ||||
|     This will run paperless in the background and automatically start it | ||||
| @@ -669,28 +666,28 @@ commands as well. | ||||
| 1.  Stop and remove the paperless container | ||||
| 2.  If using an external database, stop the container | ||||
| 3.  Update Redis configuration | ||||
|     a) If `REDIS_URL` is already set, change it to `PAPERLESS_REDIS` | ||||
|     a) If `REDIS_URL` is already set, change it to [`PAPERLESS_REDIS`](configuration.md#PAPERLESS_REDIS) | ||||
|     and continue to step 4. | ||||
|     b) Otherwise, in the `docker-compose.yml` add a new service for | ||||
|     Redis, following [the example compose | ||||
|     files](https://github.com/paperless-ngx/paperless-ngx/tree/main/docker/compose) | ||||
|     c) Set the environment variable `PAPERLESS_REDIS` so it points to | ||||
|     c) Set the environment variable [`PAPERLESS_REDIS`](configuration.md#PAPERLESS_REDIS) so it points to | ||||
|     the new Redis container | ||||
| 4.  Update user mapping | ||||
|     a) If set, change the environment variable `PUID` to `USERMAP_UID` | ||||
|     b) If set, change the environment variable `PGID` to `USERMAP_GID` | ||||
| 5.  Update configuration paths | ||||
|     a) Set the environment variable `PAPERLESS_DATA_DIR` to `/config` | ||||
|     a) Set the environment variable [`PAPERLESS_DATA_DIR`](configuration.md#PAPERLESS_DATA_DIR) to `/config` | ||||
| 6.  Update media paths | ||||
|     a) Set the environment variable `PAPERLESS_MEDIA_ROOT` to | ||||
|     a) Set the environment variable [`PAPERLESS_MEDIA_ROOT`](configuration.md#PAPERLESS_MEDIA_ROOT) to | ||||
|     `/data/media` | ||||
| 7.  Update timezone | ||||
|     a) Set the environment variable `PAPERLESS_TIME_ZONE` to the same | ||||
|     a) Set the environment variable [`PAPERLESS_TIME_ZONE`](configuration.md#PAPERLESS_TIME_ZONE) to the same | ||||
|     value as `TZ` | ||||
| 8.  Modify the `image:` to point to | ||||
|     `ghcr.io/paperless-ngx/paperless-ngx:latest` or a specific version | ||||
|     if preferred. | ||||
| 9.  Start the containers as before, using `docker-compose`. | ||||
| 9.  Start the containers as before, using `docker compose`. | ||||
|  | ||||
| ## Moving data from SQLite to PostgreSQL or MySQL/MariaDB {#sqlite_to_psql} | ||||
|  | ||||
| @@ -717,7 +714,7 @@ below use PostgreSQL, but are applicable to MySQL/MariaDB with the | ||||
| !!! warning | ||||
|  | ||||
|     MySQL is case insensitive by default, treating values like "Name" and | ||||
|     "NAME" as identical. See [MySQL caveats](/advanced_usage#mysql-caveats) for details. | ||||
|     "NAME" as identical. See [MySQL caveats](advanced_usage.md#mysql-caveats) for details. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
| @@ -738,7 +735,7 @@ below use PostgreSQL, but are applicable to MySQL/MariaDB with the | ||||
|     file to `docker-compose.yml`. Remember to adjust the consumption | ||||
|     directory, if necessary. | ||||
|     b) Without docker, configure the database in your `paperless.conf` | ||||
|     file. See [configuration](/configuration) for | ||||
|     file. See [configuration](configuration.md) for | ||||
|     details. | ||||
|  | ||||
| 3.  Open a shell and initialize the database: | ||||
| @@ -748,7 +745,7 @@ below use PostgreSQL, but are applicable to MySQL/MariaDB with the | ||||
|  | ||||
|         ``` shell-session | ||||
|         $ cd /path/to/paperless | ||||
|         $ docker-compose run --rm webserver /bin/bash | ||||
|         $ docker compose run --rm webserver /bin/bash | ||||
|         ``` | ||||
|  | ||||
|         This will launch the container and initialize the PostgreSQL | ||||
| @@ -801,7 +798,7 @@ Execute this: | ||||
|  | ||||
| ```shell-session | ||||
| $ cd /path/to/paperless | ||||
| $ docker-compose run --rm webserver migrate documents 0023 | ||||
| $ docker compose run --rm webserver migrate documents 0023 | ||||
| ``` | ||||
|  | ||||
| Or without docker: | ||||
| @@ -822,36 +819,36 @@ the Pi and configuring some options in paperless can help improve | ||||
| performance immensely: | ||||
|  | ||||
| - Stick with SQLite to save some resources. | ||||
| - Consider setting `PAPERLESS_OCR_PAGES` to 1, so that paperless will | ||||
| - Consider setting [`PAPERLESS_OCR_PAGES`](configuration.md#PAPERLESS_OCR_PAGES) to 1, so that paperless will | ||||
|   only OCR the first page of your documents. In most cases, this page | ||||
|   contains enough information to be able to find it. | ||||
| - `PAPERLESS_TASK_WORKERS` and `PAPERLESS_THREADS_PER_WORKER` are | ||||
| - [`PAPERLESS_TASK_WORKERS`](configuration.md#PAPERLESS_TASK_WORKERS) and [`PAPERLESS_THREADS_PER_WORKER`](configuration.md#PAPERLESS_THREADS_PER_WORKER) are | ||||
|   configured to use all cores. The Raspberry Pi models 3 and up have 4 | ||||
|   cores, meaning that paperless will use 2 workers and 2 threads per | ||||
|   worker. This may result in sluggish response times during | ||||
|   consumption, so you might want to lower these settings (example: 2 | ||||
|   workers and 1 thread to always have some computing power left for | ||||
|   other tasks). | ||||
| - Keep `PAPERLESS_OCR_MODE` at its default value `skip` and consider | ||||
| - Keep [`PAPERLESS_OCR_MODE`](configuration.md#PAPERLESS_OCR_MODE) at its default value `skip` and consider | ||||
|   OCR'ing your documents before feeding them into paperless. Some | ||||
|   scanners are able to do this! | ||||
| - Set `PAPERLESS_OCR_SKIP_ARCHIVE_FILE` to `with_text` to skip archive | ||||
| - Set [`PAPERLESS_OCR_SKIP_ARCHIVE_FILE`](configuration.md#PAPERLESS_OCR_SKIP_ARCHIVE_FILE) to `with_text` to skip archive | ||||
|   file generation for already ocr'ed documents, or `always` to skip it | ||||
|   for all documents. | ||||
| - If you want to perform OCR on the device, consider using | ||||
|   `PAPERLESS_OCR_CLEAN=none`. This will speed up OCR times and use | ||||
|   less memory at the expense of slightly worse OCR results. | ||||
| - If using docker, consider setting `PAPERLESS_WEBSERVER_WORKERS` to 1. This will save some memory. | ||||
| - Consider setting `PAPERLESS_ENABLE_NLTK` to false, to disable the | ||||
| - If using docker, consider setting [`PAPERLESS_WEBSERVER_WORKERS`](configuration.md#PAPERLESS_WEBSERVER_WORKERS) to 1. This will save some memory. | ||||
| - Consider setting [`PAPERLESS_ENABLE_NLTK`](configuration.md#PAPERLESS_ENABLE_NLTK) to false, to disable the | ||||
|   more advanced language processing, which can take more memory and | ||||
|   processing time. | ||||
|  | ||||
| For details, refer to [configuration](/configuration). | ||||
| For details, refer to [configuration](configuration.md). | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Updating the | ||||
|     [automatic matching algorithm](/advanced_usage#automatic-matching) takes quite a bit of time. However, the update mechanism | ||||
|     [automatic matching algorithm](advanced_usage.md#automatic-matching) takes quite a bit of time. However, the update mechanism | ||||
|     checks if your data has changed before doing the heavy lifting. If you | ||||
|     experience the algorithm taking too much cpu time, consider changing the | ||||
|     schedule in the admin interface to daily. You can also manually invoke | ||||
|   | ||||
| @@ -6,7 +6,7 @@ Check for the following issues: | ||||
|  | ||||
| - Ensure that the directory you're putting your documents in is the | ||||
|   folder paperless is watching. With docker, this setting is performed | ||||
|   in the `docker-compose.yml` file. Without docker, look at the | ||||
|   in the `docker-compose.yml` file. Without Docker, look at the | ||||
|   `CONSUMPTION_DIR` setting. Don't adjust this setting if you're | ||||
|   using docker. | ||||
|  | ||||
| @@ -46,8 +46,7 @@ run: | ||||
| If you notice that the consumer will only pickup files in the | ||||
| consumption directory at startup, but won't find any other files added | ||||
| later, you will need to enable filesystem polling with the configuration | ||||
| option `PAPERLESS_CONSUMER_POLLING`, see | ||||
| `[here](/configuration#polling). | ||||
| option [`PAPERLESS_CONSUMER_POLLING`](configuration.md#PAPERLESS_CONSUMER_POLLING). | ||||
|  | ||||
| This will disable listening to filesystem changes with inotify and | ||||
| paperless will manually check the consumption directory for changes | ||||
| @@ -121,7 +120,7 @@ Gotenberg raises this error. | ||||
|  | ||||
| You can increase the timeout by configuring a command flag for Gotenberg | ||||
| (see also [here](https://gotenberg.dev/docs/modules/api#properties)). If | ||||
| using docker-compose, this is achieved by the following configuration | ||||
| using Docker Compose, this is achieved by the following configuration | ||||
| change in the `docker-compose.yml` file: | ||||
|  | ||||
| ```yaml | ||||
| @@ -139,13 +138,13 @@ command: | ||||
| You might encounter errors such as: | ||||
|  | ||||
| ```shell-session | ||||
| The following error occured while consuming document.pdf: [Errno 13] Permission denied: '/usr/src/paperless/src/../consume/document.pdf' | ||||
| The following error occurred while consuming document.pdf: [Errno 13] Permission denied: '/usr/src/paperless/src/../consume/document.pdf' | ||||
| ``` | ||||
|  | ||||
| This happens when paperless does not have permission to delete files | ||||
| inside the consumption directory. Ensure that `USERMAP_UID` and | ||||
| `USERMAP_GID` are set to the user id and group id you use on the host | ||||
| operating system, if these are different from `1000`. See [Docker setup](/setup#docker_hub). | ||||
| operating system, if these are different from `1000`. See [Docker setup](setup.md#docker_hub). | ||||
|  | ||||
| Also ensure that you are able to read and write to the consumption | ||||
| directory on the host. | ||||
| @@ -265,8 +264,8 @@ This probably indicates paperless tried to consume the same file twice. | ||||
| This can happen for a number of reasons, depending on how documents are | ||||
| placed into the consume folder. If paperless is using inotify (the | ||||
| default) to check for documents, try adjusting the | ||||
| [inotify configuration](/configuration#inotify). If polling is enabled, try adjusting the | ||||
| [polling configuration](/configuration#polling). | ||||
| [inotify configuration](configuration.md#inotify). If polling is enabled, try adjusting the | ||||
| [polling configuration](configuration.md#polling). | ||||
|  | ||||
| ## Consumer fails waiting for file to remain unmodified. | ||||
|  | ||||
| @@ -278,7 +277,7 @@ You might find messages like these in your log files: | ||||
|  | ||||
| This indicates paperless timed out while waiting for the file to be | ||||
| completely written to the consume folder. Adjusting | ||||
| [polling configuration](/configuration#polling) values should resolve the issue. | ||||
| [polling configuration](configuration.md#polling) values should resolve the issue. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
| @@ -297,8 +296,8 @@ This indicates paperless was unable to open the file, as the OS reported | ||||
| the file as still being in use. To prevent a crash, paperless did not | ||||
| try to consume the file. If paperless is using inotify (the default) to | ||||
| check for documents, try adjusting the | ||||
| [inotify configuration](/configuration#inotify). If polling is enabled, try adjusting the | ||||
| [polling configuration](/configuration#polling). | ||||
| [inotify configuration](configuration.md#inotify). If polling is enabled, try adjusting the | ||||
| [polling configuration](configuration.md#polling). | ||||
|  | ||||
| !!! note | ||||
|  | ||||
| @@ -320,7 +319,7 @@ many workers attempting to access the database simultaneously. | ||||
|  | ||||
| Consider changing to the PostgreSQL database if you will be processing | ||||
| many documents at once often. Otherwise, try tweaking the | ||||
| `PAPERLESS_DB_TIMEOUT` setting to allow more time for the database to | ||||
| [`PAPERLESS_DB_TIMEOUT`](configuration.md#PAPERLESS_DB_TIMEOUT) setting to allow more time for the database to | ||||
| unlock. This may have minor performance implications. | ||||
|  | ||||
| ## gunicorn fails to start with "is not a valid port number" | ||||
| @@ -330,7 +329,7 @@ environment variable named `${serviceName}_PORT`. This is | ||||
| the same environment variable which is used by Paperless to optionally | ||||
| change the port gunicorn listens on. | ||||
|  | ||||
| To fix this, set `PAPERLESS_PORT` again to your desired port, or the | ||||
| To fix this, set [`PAPERLESS_PORT`](configuration.md#PAPERLESS_PORT) again to your desired port, or the | ||||
| default of 8000. | ||||
|  | ||||
| ## Database Warns about unique constraint "documents_tag_name_uniq | ||||
| @@ -345,3 +344,15 @@ STATEMENT:  INSERT INTO "documents_tag" ("owner_id", "name", "match", "matching_ | ||||
|  | ||||
| This can happen during heavy consumption when using polling. Paperless will handle it correctly and the file | ||||
| will still be consumed | ||||
|  | ||||
| ## Consumption fails with "Ghostscript PDF/A rendering failed" | ||||
|  | ||||
| Newer versions of OCRmyPDF will fail if it encounters errors during processing. | ||||
| This is intentional as the output archive file may differ in unexpected or undesired | ||||
| ways from the original. As the logs indicate, if you encounter this error you can set | ||||
| `PAPERLESS_OCR_USER_ARGS: '{"continue_on_soft_render_error": true}'` to try to 'force' | ||||
| processing documents with this issue. | ||||
|  | ||||
| ## Platform-Specific Deployment Troubleshooting | ||||
|  | ||||
| A user-maintained wiki page is available to help troubleshoot issues that may arise when trying to deploy Paperless-ngx on specific platforms, for example SELinux. Please see [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Platform%E2%80%90Specific-Troubleshooting). | ||||
|   | ||||
							
								
								
									
										322
									
								
								docs/usage.md
									
									
									
									
									
								
							
							
						
						| @@ -62,7 +62,7 @@ following operations on your documents: | ||||
|     paperless to create archived versions for digital documents, you can | ||||
|     configure that by configuring | ||||
|     `PAPERLESS_OCR_SKIP_ARCHIVE_FILE=with_text`. Please read the | ||||
|     [relevant section in the documentation](/configuration#ocr). | ||||
|     [relevant section in the documentation](configuration.md#ocr). | ||||
|  | ||||
| !!! note | ||||
|  | ||||
| @@ -103,25 +103,14 @@ Typically, you're looking at an FTP server like | ||||
|  | ||||
| ### Web UI Upload | ||||
|  | ||||
| The dashboard has a file drop field to upload documents to paperless. | ||||
| Simply drag a file onto this field or select a file with the file | ||||
| dialog. Multiple files are supported. | ||||
|  | ||||
| You can also upload documents on any other page of the web UI by | ||||
| dragging-and-dropping files into your browser window. | ||||
| The dashboard has a button to upload documents to paperless or you | ||||
| can simply drag a file anywhere into the app to initiate the consumption | ||||
| process. | ||||
|  | ||||
| ### Mobile upload {#usage-mobile_upload} | ||||
|  | ||||
| The mobile app over at [https://github.com/qcasey/paperless_share](https://github.com/qcasey/paperless_share) | ||||
| allows Android users to share any documents with paperless. This can be | ||||
| combined with any of the mobile scanning apps out there, such as Office | ||||
| Lens. | ||||
|  | ||||
| Furthermore, there is the [Paperless | ||||
| App](https://github.com/bauerj/paperless_app) as well, which not only | ||||
| has document upload, but also document browsing and download features. | ||||
|  | ||||
| Another option is [Paperless Mobile](https://github.com/astubenbord/paperless-mobile), an Android app that supports document upload, scanning, management of labels and more. | ||||
| Please see [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Affiliated-Projects) for a user-maintained list of affiliated projects and | ||||
| software (e.g. for mobile devices) that is compatible with Paperless-ngx. | ||||
|  | ||||
| ### IMAP (Email) {#usage-email} | ||||
|  | ||||
| @@ -145,9 +134,9 @@ These rules perform the following: | ||||
| 5.  If documents were consumed from a mail, the rule action is performed | ||||
|     on that mail. | ||||
|  | ||||
| Paperless will completely ignore mails that do not match your filters. | ||||
| It will also only perform the action on mails that it has consumed | ||||
| documents from. | ||||
| Paperless will check all emails only once and completely ignore messages | ||||
| that do not match your filters. It will also only perform the rule action | ||||
| on e-mails that it has consumed documents from. | ||||
|  | ||||
| The actions all ensure that the same mail is not consumed twice by | ||||
| different means. These are as follows: | ||||
| @@ -160,7 +149,7 @@ different means. These are as follows: | ||||
| - **Flag:** Sets the 'important' flag on mails with consumed | ||||
|   documents. Paperless will not consume flagged mails. | ||||
| - **Move to folder:** Moves consumed mails out of the way so that | ||||
|   paperless wont consume them again. | ||||
|   paperless won't consume them again. | ||||
| - **Add custom Tag:** Adds a custom tag to mails with consumed | ||||
|   documents (the IMAP standard calls these "keywords"). Paperless | ||||
|   will not consume mails already tagged. Not all mail servers support | ||||
| @@ -208,48 +197,287 @@ different means. These are as follows: | ||||
|     them further. | ||||
|  | ||||
| Paperless is set up to check your mails every 10 minutes. This can be | ||||
| configured via `PAPERLESS_EMAIL_TASK_CRON` (see [software tweaks](/configuration#software_tweaks)) | ||||
| configured via [`PAPERLESS_EMAIL_TASK_CRON`](configuration.md#PAPERLESS_EMAIL_TASK_CRON) | ||||
|  | ||||
| ### REST API | ||||
|  | ||||
| You can also submit a document using the REST API, see [POSTing documents](/api#file-uploads) | ||||
| You can also submit a document using the REST API, see [POSTing documents](api.md#file-uploads) | ||||
| for details. | ||||
|  | ||||
| ## Permissions | ||||
|  | ||||
| As of version 1.14.0 Paperless-ngx added core support for user / group permissions. Permissions is | ||||
| based around 'global' permissions as well as 'object-level' permissions. Global permissions designate | ||||
| which parts of the application a user can access (e.g. Documents, Tags, Settings) and object-level | ||||
| determine which objects are visible or editable. All objects have an 'owner' and 'view' and 'edit' | ||||
| permissions which can be granted to other users or groups. | ||||
| Permissions in Paperless-ngx are based around ['global' permissions](#global-permissions) as well as | ||||
| ['object-level' permissions](#object-permissions). Global permissions determine which parts of the | ||||
| application a user can access (e.g. Documents, Tags, Settings) and object-level determine which | ||||
| objects are visible or editable. All objects have an 'owner' and 'view' and 'edit' permissions which | ||||
| can be granted to other users or groups. The paperless-ngx permissions system uses the built-in user | ||||
| model of the backend framework, Django. | ||||
|  | ||||
| Permissions uses the built-in user model of the backend framework, Django. | ||||
| !!! tip | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     After migration to version 1.14.0 all existing documents, tags etc. will have no explicit owner | ||||
|     set which means they will be visible / editable by all users. Once an object has an owner set, | ||||
|     only the owner can explicitly grant / revoke permissions. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     When first migrating to permissions it is recommended to use a 'superuser' account (which | ||||
|     would usually have been setup during installation) to ensure you have full permissions. | ||||
|  | ||||
|     Note that superusers have access to all objects. | ||||
|     Object-level permissions only apply to the object itself. In other words, setting permissions | ||||
|     for a Tag will _not_ affect the permissions of documents that have the Tag. | ||||
|  | ||||
| Permissions can be set using the new "Permissions" tab when editing documents, or bulk-applied | ||||
| in the UI by selecting documents and choosing the "Permissions" button. Owner can also optionally | ||||
| be set for documents uploaded via the API. Documents consumed via the consumption dir currently | ||||
| do not have an owner set. | ||||
| in the UI by selecting documents and choosing the "Permissions" button. | ||||
|  | ||||
| ### Default permissions | ||||
|  | ||||
| [Workflows](#workflows) provide advanced ways to control permissions. | ||||
|  | ||||
| For objects created via the web UI (tags, doc types, etc.) the default is to set the current user | ||||
| as owner and no extra permissions, but you can explicitly set these under Settings > Permissions. | ||||
|  | ||||
| Documents consumed via the consumption directory do not have an owner or additional permissions set by default, but again, can be controlled with [Workflows](#workflows). | ||||
|  | ||||
| ### Users and Groups | ||||
|  | ||||
| Paperless-ngx versions after 1.14.0 allow creating and editing users and groups via the 'frontend' UI. | ||||
| These can be found under Settings > Users & Groups, assuming the user has access. If a user is designated | ||||
| Paperless-ngx supports editing users and groups via the 'frontend' UI, which can be found under | ||||
| Settings > Users & Groups, assuming the user has access. If a user is designated | ||||
| as a member of a group those permissions will be inherited and this is reflected in the UI. Explicit | ||||
| permissions can be granted to limit access to certain parts of the UI (and corresponding API endpoints). | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Superusers can access all parts of the front and backend application as well as any and all objects. | ||||
|  | ||||
| #### Admin Status | ||||
|  | ||||
| Admin status (Django 'staff status') grants access to viewing the paperless logs and the system status dialog | ||||
| as well as accessing the Django backend. | ||||
|  | ||||
| #### Detailed Explanation of Global Permissions {#global-permissions} | ||||
|  | ||||
| Global permissions define what areas of the app and API endpoints the user can access. For example, they | ||||
| determine if a user can create, edit, delete or view _any_ documents, but individual documents themselves | ||||
| still have "object-level" permissions. | ||||
|  | ||||
| | Type          | Details                                                                                                                                                                                             | | ||||
| | ------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | AppConfig     | _Change_ or higher permissions grants access to the "Application Configuration" area.                                                                                                               | | ||||
| | Correspondent | Grants global permissions to add, edit, delete or view Correspondents.                                                                                                                              | | ||||
| | CustomField   | Grants global permissions to add, edit, delete or view Custom Fields.                                                                                                                               | | ||||
| | Document      | Grants global permissions to add, edit, delete or view Documents.                                                                                                                                   | | ||||
| | DocumentType  | Grants global permissions to add, edit, delete or view Document Types.                                                                                                                              | | ||||
| | Group         | Grants global permissions to add, edit, delete or view Groups.                                                                                                                                      | | ||||
| | MailAccount   | Grants global permissions to add, edit, delete or view Mail Accounts.                                                                                                                               | | ||||
| | MailRule      | Grants global permissions to add, edit, delete or view Mail Rules.                                                                                                                                  | | ||||
| | Note          | Grants global permissions to add, edit, delete or view Notes.                                                                                                                                       | | ||||
| | PaperlessTask | Grants global permissions to view or dismiss (_Change_) File Tasks.                                                                                                                                 | | ||||
| | SavedView     | Grants global permissions to add, edit, delete or view Saved Views.                                                                                                                                 | | ||||
| | ShareLink     | Grants global permissions to add, delete or view Share Links.                                                                                                                                       | | ||||
| | StoragePath   | Grants global permissions to add, edit, delete or view Storage Paths.                                                                                                                               | | ||||
| | Tag           | Grants global permissions to add, edit, delete or view Tags.                                                                                                                                        | | ||||
| | UISettings    | Grants global permissions to add, edit, delete or view the UI settings that are used by the web app.<br/>Users expected to access the web UI should usually be granted at least _View_ permissions. | | ||||
| | User          | Grants global permissions to add, edit, delete or view Users.                                                                                                                                       | | ||||
| | Workflow      | Grants global permissions to add, edit, delete or view Workflows.<br/>Note that Workflows are global, in other words all users who can access workflows have access to the same set of them.        | | ||||
|  | ||||
| #### Detailed Explanation of Object Permissions {#object-permissions} | ||||
|  | ||||
| | Type  | Details                                                                                                                                                                                                                                                                                                                                  | | ||||
| | ----- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | Owner | By default objects are only visible and editable by their owner.<br/>Only the object owner can grant permissions to other users or groups.<br/>Additionally, only document owners can create share links and add / remove custom fields.<br/>For backwards compatibility objects can have no owner which makes them visible to any user. | | ||||
| | View  | Confers the ability to view (not edit) a document, tag, etc.<br/>Users without 'view' (or higher) permissions will be shown _'Private'_ in place of the object name for example when viewing a document with a tag for which the user doesn't have permissions.                                                                          | | ||||
| | Edit  | Confers the ability to edit (and view) a document, tag, etc.                                                                                                                                                                                                                                                                             | | ||||
|  | ||||
| ### Password reset | ||||
|  | ||||
| In order to enable the password reset feature you will need to setup an SMTP backend, see | ||||
| [`PAPERLESS_EMAIL_HOST`](configuration.md#PAPERLESS_EMAIL_HOST). If your installation does not have | ||||
| [`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) set, the reset link included in emails will use the server host. | ||||
|  | ||||
| ## Workflows | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     v2.3 added "Workflows" and existing "Consumption Templates" were converted automatically to the new more powerful format. | ||||
|  | ||||
| Workflows allow hooking into the Paperless-ngx document pipeline, for example to alter what metadata (tags, doc types) and | ||||
| permissions (owner, privileges) are assigned to documents. Workflows can have multiple 'triggers' and 'actions'. Triggers | ||||
| are events (with optional filtering rules) that will cause the workflow to be run and actions are the set of sequential | ||||
| actions to apply. | ||||
|  | ||||
| In general, workflows and any actions they contain are applied sequentially by sort order. For "assignment" actions, subsequent | ||||
| workflow actions will override previous assignments, except for assignments that accept multiple items e.g. tags, custom | ||||
| fields and permissions, which will be merged. | ||||
|  | ||||
| ### Workflow Triggers | ||||
|  | ||||
| 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 | ||||
| 2. **Document Added**: _after_ a document is added. At this time, file path and source information is no longer available, | ||||
|    but the document content has been extracted and metadata such as document type, tags, etc. have been set, so these can now | ||||
|    be used for filtering. | ||||
| 3. **Document Updated**: when a document is updated. Similar to 'added' events, triggers can include filtering by content matching, | ||||
|    tags, doc type, or correspondent. | ||||
|  | ||||
| The following flow diagram illustrates the three trigger types: | ||||
|  | ||||
| ```mermaid | ||||
| flowchart TD | ||||
|     consumption{"Matching | ||||
|     'Consumption' | ||||
|     trigger(s)"} | ||||
|  | ||||
|     added{"Matching | ||||
|     'Added' | ||||
|     trigger(s)"} | ||||
|  | ||||
|     updated{"Matching | ||||
|     'Updated' | ||||
|     trigger(s)"} | ||||
|  | ||||
|     A[New Document] --> consumption | ||||
|     consumption --> |Yes| C[Workflow Actions Run] | ||||
|     consumption --> |No| D | ||||
|     C --> D[Document Added] | ||||
|     D -- Paperless-ngx 'matching' of tags, etc. --> added | ||||
|     added --> |Yes| F[Workflow Actions Run] | ||||
|     added --> |No| G | ||||
|     F --> G[Document Finalized] | ||||
|     H[Existing Document Changed] --> updated | ||||
|     updated --> |Yes| J[Workflow Actions Run] | ||||
|     updated --> |No| K | ||||
|     J --> K[Document Saved] | ||||
| ``` | ||||
|  | ||||
| #### Filters {#workflow-trigger-filters} | ||||
|  | ||||
| Workflows allow you to filter by: | ||||
|  | ||||
| - Source, e.g. documents uploaded via consume folder, API (& the web UI) and mail fetch | ||||
| - File name, including wildcards e.g. \*.pdf will apply to all pdfs | ||||
| - File path, including wildcards. Note that enabling `PAPERLESS_CONSUMER_RECURSIVE` would allow, for | ||||
|   example, automatically assigning documents to different owners based on the upload directory. | ||||
| - Mail rule. Choosing this option will force 'mail fetch' to be the workflow source. | ||||
| - Content matching (`Added` and `Updated` triggers only). Filter document content using the matching settings. | ||||
| - Tags (`Added` and `Updated` triggers only). Filter for documents with any of the specified tags | ||||
| - Document type (`Added` and `Updated` triggers only). Filter documents with this doc type | ||||
| - Correspondent (`Added` and `Updated` triggers only). Filter documents with this correspondent | ||||
|  | ||||
| ### Workflow Actions | ||||
|  | ||||
| There are currently two types of workflow actions, "Assignment", which can assign: | ||||
|  | ||||
| - Title, see [title placeholders](usage.md#title-placeholders) below | ||||
| - Tags, correspondent, document type and storage path | ||||
| - Document owner | ||||
| - View and / or edit permissions to users or groups | ||||
| - Custom fields. Note that no value for the field will be set | ||||
|  | ||||
| and "Removal" actions, which can remove either all of or specific sets of the following: | ||||
|  | ||||
| - Tags, correspondents, document types or storage paths | ||||
| - Document owner | ||||
| - View and / or edit permissions | ||||
| - Custom fields | ||||
|  | ||||
| #### Title placeholders | ||||
|  | ||||
| Workflow titles can include placeholders but the available options differ depending on the type of | ||||
| workflow trigger. This is because at the time of consumption (when the title is to be set), no automatic tags etc. have been | ||||
| applied. You can use the following placeholders with any trigger type: | ||||
|  | ||||
| - `{correspondent}`: assigned correspondent name | ||||
| - `{document_type}`: assigned document type name | ||||
| - `{owner_username}`: assigned owner username | ||||
| - `{added}`: added datetime | ||||
| - `{added_year}`: added year | ||||
| - `{added_year_short}`: added year | ||||
| - `{added_month}`: added month | ||||
| - `{added_month_name}`: added month name | ||||
| - `{added_month_name_short}`: added month short name | ||||
| - `{added_day}`: added day | ||||
| - `{added_time}`: added time in HH:MM format | ||||
| - `{original_filename}`: original file name without extension | ||||
|  | ||||
| The following placeholders are only available for "added" or "updated" triggers | ||||
|  | ||||
| - `{created}`: created datetime | ||||
| - `{created_year}`: created year | ||||
| - `{created_year_short}`: created year | ||||
| - `{created_month}`: created month | ||||
| - `{created_month_name}`: created month name | ||||
| - `{created_month_name_short}`: created month short name | ||||
| - `{created_day}`: created day | ||||
| - `{created_time}`: created time in HH:MM format | ||||
|  | ||||
| ### Workflow permissions | ||||
|  | ||||
| All users who have application permissions for editing workflows can see the same set | ||||
| of workflows. In other words, workflows themselves intentionally do not have an owner or permissions. | ||||
|  | ||||
| Given their potentially far-reaching capabilities, you may want to restrict access to workflows. | ||||
|  | ||||
| Upon migration, existing installs will grant access to workflows to users who can add | ||||
| documents (and superusers who can always access all parts of the app). | ||||
|  | ||||
| ## Custom Fields {#custom-fields} | ||||
|  | ||||
| Paperless-ngx supports the use of custom fields for documents as of v2.0, allowing a user | ||||
| to optionally attach data to documents which does not fit in the existing set of fields | ||||
| Paperless-ngx provides. | ||||
|  | ||||
| 1. First, create a custom field (under "Manage"), with a given name and data type. This could be something like "Invoice Number" or "Date Paid", with a data type of "Number", "Date", "String", etc. | ||||
| 2. Once created, a field can be used with documents and data stored. To do so, use the "Custom Fields" menu on the document detail page, choose your existing field and click "Add". Once the field is visible in the form you can enter the appropriate | ||||
|    data which will be validated according to the custom field "data type". | ||||
| 3. Fields can be removed by hovering over the field name revealing a "Remove" button. | ||||
|  | ||||
| !!! important | ||||
|  | ||||
|     Added / removed fields, as well as any data is not saved to the document until you | ||||
|     actually hit the "Save" button, similar to other changes on the document details page. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Once the data type for a field is set, it cannot be changed. | ||||
|  | ||||
| Multiple fields may be attached to a document but the same field name cannot be assigned multiple times to the a single document. | ||||
|  | ||||
| The following custom field types are supported: | ||||
|  | ||||
| - `Text`: any text | ||||
| - `Boolean`: true / false (check / unchecked) field | ||||
| - `Date`: date | ||||
| - `URL`: a valid url | ||||
| - `Integer`: integer number e.g. 12 | ||||
| - `Number`: float number e.g. 12.3456 | ||||
| - `Monetary`: [ISO 4217 currency code](https://en.wikipedia.org/wiki/ISO_4217#List_of_ISO_4217_currency_codes) and a number with exactly two decimals, e.g. USD12.30 | ||||
| - `Document Link`: reference(s) to other document(s) displayed as links, automatically creates a symmetrical link in reverse | ||||
|  | ||||
| ## Share Links | ||||
|  | ||||
| Paperless-ngx added the ability to create shareable links to files in version 2.0. You can find the button for this on the document detail screen. | ||||
|  | ||||
| - Share links do not require a user to login and thus link directly to a file. | ||||
| - Links are unique and are of the form `{paperless-url}/share/{randomly-generated-slug}`. | ||||
| - Links can optionally have an expiration time set. | ||||
| - After a link expires or is deleted users will be redirected to the regular paperless-ngx login. | ||||
|  | ||||
| !!! tip | ||||
|  | ||||
|     If your paperless-ngx instance is behind a reverse-proxy you may want to create an exception to bypass any authentication layers that are part of your setup in order to make links truly publicly-accessible. Of course, do so with caution. | ||||
|  | ||||
| ## PDF Actions | ||||
|  | ||||
| Paperless-ngx supports 3 basic editing operations for PDFs (these operations cannot be performed on non-PDF files): | ||||
|  | ||||
| - Merging documents: available when selecting multiple documents for 'bulk editing' | ||||
| - Rotating documents: available when selecting multiple documents for 'bulk editing' and from an individual document's details page. | ||||
| - Splitting documents: available from an individual document's details page | ||||
|  | ||||
| !!! important | ||||
|  | ||||
|     Note that rotation alters the Paperless-ngx _original_ file, which would, for example, invalidate a digital signature. | ||||
|  | ||||
| ## Document History | ||||
|  | ||||
| As of version 2.7, Paperless-ngx automatically records all changes to a document and records this in an audit log. The feature requires [`PAPERLESS_AUDIT_LOG_ENABLED`](configuration.md#PAPERLESS_AUDIT_LOG_ENABLED) be enabled, which it is by default as of version 2.7. | ||||
| Changes to documents are visible under the "History" tab. Note that certain changes such as those made by workflows, record the 'actor' | ||||
| as "System". | ||||
|  | ||||
| ## Best practices {#basic-searching} | ||||
|  | ||||
| Paperless offers a couple tools that help you organize your document | ||||
| @@ -458,7 +686,7 @@ Once you have scanned in a document, proceed in paperless as follows. | ||||
|     paperless will assign them automatically. After consuming a couple | ||||
|     documents, you can even ask paperless to *learn* when to assign tags and | ||||
|     correspondents by itself. For details on this feature, see | ||||
|     [advanced matching](/advanced_usage#matching). | ||||
|     [advanced matching](advanced_usage.md#matching). | ||||
|  | ||||
| ### Task management | ||||
|  | ||||
|   | ||||
| @@ -38,7 +38,6 @@ ask_docker_folder() { | ||||
| 			echo "Invalid folder: $result" | ||||
| 		fi | ||||
|  | ||||
|  | ||||
| 	done | ||||
| } | ||||
|  | ||||
| @@ -57,14 +56,9 @@ if ! command -v docker &> /dev/null ; then | ||||
| 	exit 1 | ||||
| fi | ||||
|  | ||||
| DOCKER_COMPOSE_CMD="docker-compose" | ||||
| if ! command -v ${DOCKER_COMPOSE_CMD} ; then | ||||
| 	if docker compose version &> /dev/null ; then | ||||
| 		DOCKER_COMPOSE_CMD="docker compose" | ||||
| 	else | ||||
| 		echo "docker-compose executable not found. Is docker-compose installed?" | ||||
| 		exit 1 | ||||
| 	fi | ||||
| if ! docker compose &> /dev/null ; then | ||||
| 	echo "docker compose plugin not found. Is docker compose installed?" | ||||
| 	exit 1 | ||||
| fi | ||||
|  | ||||
| # Check if user has permissions to run Docker by trying to get the status of Docker (docker status). | ||||
| @@ -72,12 +66,22 @@ fi | ||||
| if ! docker stats --no-stream &> /dev/null ; then | ||||
| 	echo "" | ||||
| 	echo "WARN: It look like the current user does not have Docker permissions." | ||||
| 	echo "WARN: Use 'sudo usermod -aG docker $USER' to assign Docker permissions to the user." | ||||
| 	echo "WARN: Use 'sudo usermod -aG docker $USER' to assign Docker permissions to the user (may require restarting shell)." | ||||
| 	echo "" | ||||
| 	sleep 3 | ||||
| fi | ||||
|  | ||||
| default_time_zone=$(timedatectl show -p Timezone --value) | ||||
| # Added handling for timezone for busybox based linux, not having timedatectl available (i.e. QNAP QTS) | ||||
| # if neither timedatectl nor /etc/TZ is succeeding, defaulting to GMT. | ||||
| if  command -v timedatectl &> /dev/null ; then | ||||
| 	default_time_zone=$(timedatectl show -p Timezone --value) | ||||
| elif [ -f /etc/TZ ] && [ -f /etc/tzlist ] ; then | ||||
| 	TZ=$(cat /etc/TZ) | ||||
| 	default_time_zone=$(grep -B 1 -m 1 "$TZ" /etc/tzlist | head -1 | cut -f 2 -d =) | ||||
| else | ||||
| 	echo "WARN: unable to detect timezone, defaulting to Etc/UTC" | ||||
| 	default_time_zone="Etc/UTC" | ||||
| fi | ||||
|  | ||||
| set -e | ||||
|  | ||||
| @@ -321,7 +325,8 @@ fi | ||||
| wget "https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/docker/compose/docker-compose.$DOCKER_COMPOSE_VERSION.yml" -O docker-compose.yml | ||||
| wget "https://raw.githubusercontent.com/paperless-ngx/paperless-ngx/main/docker/compose/.env" -O .env | ||||
|  | ||||
| SECRET_KEY=$(tr --delete --complement 'a-zA-Z0-9' < /dev/urandom 2>/dev/null | head --bytes 64) | ||||
| SECRET_KEY=$(LC_ALL=C tr -dc 'a-zA-Z0-9!#$%&()*+,-./:;<=>?@[\]^_`{|}~' < /dev/urandom | dd bs=1 count=64 2>/dev/null) | ||||
|  | ||||
|  | ||||
| DEFAULT_LANGUAGES=("deu eng fra ita spa") | ||||
|  | ||||
| @@ -340,7 +345,7 @@ read -r -a OCR_LANGUAGES_ARRAY <<< "${_split_langs}" | ||||
| 	fi | ||||
| 	echo "PAPERLESS_TIME_ZONE=$TIME_ZONE" | ||||
| 	echo "PAPERLESS_OCR_LANGUAGE=$OCR_LANGUAGE" | ||||
| 	echo "PAPERLESS_SECRET_KEY=$SECRET_KEY" | ||||
| 	echo "PAPERLESS_SECRET_KEY='$SECRET_KEY'" | ||||
| 	if [[ ! ${DEFAULT_LANGUAGES[*]} =~ ${OCR_LANGUAGES_ARRAY[*]} ]] ; then | ||||
| 		echo "PAPERLESS_OCR_LANGUAGES=${OCR_LANGUAGES_ARRAY[*]}" | ||||
| 	fi | ||||
| @@ -382,8 +387,16 @@ if [ "$l1" -eq "$l2" ] ; then | ||||
| fi | ||||
|  | ||||
|  | ||||
| ${DOCKER_COMPOSE_CMD} pull | ||||
| docker compose pull | ||||
|  | ||||
| ${DOCKER_COMPOSE_CMD} run --rm -e DJANGO_SUPERUSER_PASSWORD="$PASSWORD" webserver createsuperuser --noinput --username "$USERNAME" --email "$EMAIL" | ||||
| if [ "$DATABASE_BACKEND" == "postgres" ] || [ "$DATABASE_BACKEND" == "mariadb" ] ; then | ||||
| 	echo "Starting DB first for initialization" | ||||
| 	docker compose up --detach db | ||||
| 	# hopefully enough time for even the slower systems | ||||
| 	sleep 15 | ||||
| 	docker compose stop | ||||
| fi | ||||
|  | ||||
| ${DOCKER_COMPOSE_CMD} up --detach | ||||
| docker compose run --rm -e DJANGO_SUPERUSER_PASSWORD="$PASSWORD" webserver createsuperuser --noinput --username "$USERNAME" --email "$EMAIL" | ||||
|  | ||||
| docker compose up --detach | ||||
|   | ||||
							
								
								
									
										12
									
								
								mkdocs.yml
									
									
									
									
									
								
							
							
						
						| @@ -28,6 +28,7 @@ theme: | ||||
|     repo: fontawesome/brands/github | ||||
|   favicon: assets/favicon.png | ||||
| repo_url: https://github.com/paperless-ngx/paperless-ngx | ||||
| repo_name: paperless-ngx/paperless-ngx | ||||
| edit_uri: blob/main/docs/ | ||||
| extra_css: | ||||
|   - assets/extra.css | ||||
| @@ -42,6 +43,12 @@ markdown_extensions: | ||||
|   - pymdownx.superfences | ||||
|   - pymdownx.inlinehilite | ||||
|   - pymdownx.snippets | ||||
|   - footnotes | ||||
|   - pymdownx.superfences: | ||||
|       custom_fences: | ||||
|         - name: mermaid | ||||
|           class: mermaid | ||||
|           format: !!python/name:pymdownx.superfences.fence_code_format | ||||
| strict: true | ||||
| nav: | ||||
|     - index.md | ||||
| @@ -64,3 +71,8 @@ extra: | ||||
|       link: https://hub.docker.com/r/paperlessngx/paperless-ngx | ||||
|     - icon: material/chat | ||||
|       link: https://matrix.to/#/#paperless:matrix.org | ||||
| plugins: | ||||
|   - search | ||||
|   - glightbox: | ||||
|        skip_classes: | ||||
|          - no-lightbox | ||||
|   | ||||
							
								
								
									
										36
									
								
								paperless-ngx.code-workspace
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,36 @@ | ||||
| { | ||||
| 	"folders": [ | ||||
| 		{ | ||||
| 			"path": "." | ||||
| 		}, | ||||
| 		{ | ||||
| 			"path": "./src", | ||||
| 			"name": "Backend" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"path": "./src-ui", | ||||
| 			"name": "Frontend" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"path": "./.github", | ||||
| 			"name": "CI/CD" | ||||
| 		}, | ||||
| 		{ | ||||
| 			"path": "./docs", | ||||
| 			"name": "Documentation" | ||||
| 		} | ||||
|  | ||||
| 	], | ||||
| 	"settings": { | ||||
| 		"files.exclude": { | ||||
| 			"**/__pycache__": true, | ||||
| 			"**/.mypy_cache": true, | ||||
| 			"**/.ruff_cache": true, | ||||
| 			"**/.pytest_cache": true, | ||||
| 			"**/.idea": true, | ||||
| 			"**/.venv": true, | ||||
| 			"**/.coverage": true, | ||||
| 			"**/coverage.json": true | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -66,6 +66,13 @@ | ||||
| #PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS=false | ||||
| #PAPERLESS_CONSUMER_ENABLE_BARCODES=false | ||||
| #PAPERLESS_CONSUMER_BARCODE_STRING=PATCHT | ||||
| #PAPERLESS_CONSUMER_BARCODE_UPSCALE=0.0 | ||||
| #PAPERLESS_CONSUMER_BARCODE_DPI=300 | ||||
| #PAPERLESS_CONSUMER_ENABLE_TAG_BARCODE=false | ||||
| #PAPERLESS_CONSUMER_TAG_BARCODE_MAPPING={"TAG:(.*)": "\\g<1>"} | ||||
| #PAPERLESS_CONSUMER_ENABLE_COLLATE_DOUBLE_SIDED=false | ||||
| #PAPERLESS_CONSUMER_COLLATE_DOUBLE_SIDED_SUBDIR_NAME=double-sided | ||||
| #PAPERLESS_CONSUMER_COLLATE_DOUBLE_SIDED_TIFF_SUPPORT=false | ||||
| #PAPERLESS_PRE_CONSUME_SCRIPT=/path/to/an/arbitrary/script.sh | ||||
| #PAPERLESS_POST_CONSUME_SCRIPT=/path/to/an/arbitrary/script.sh | ||||
| #PAPERLESS_FILENAME_DATE_ORDER=YMD | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| docker run -p 5432:5432 -e POSTGRES_PASSWORD=password -v paperless_pgdata:/var/lib/postgresql/data -d postgres:13 | ||||
| docker run -p 5432:5432 -e POSTGRES_PASSWORD=password -v paperless_pgdata:/var/lib/postgresql/data -d postgres:15 | ||||
| docker run -d -p 6379:6379 redis:latest | ||||
| docker run -p 3000:3000 -d gotenberg/gotenberg:7.8 gotenberg --chromium-disable-javascript=true --chromium-allow-list="file:///tmp/.*" | ||||
| docker run -p 9998:9998 -d ghcr.io/paperless-ngx/tika:latest | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| { | ||||
|   "root": true, | ||||
|   "ignorePatterns": [ | ||||
|     "projects/**/*" | ||||
|     "projects/**/*", | ||||
|     "/src/app/components/common/pdf-viewer/**" | ||||
|   ], | ||||
|   "overrides": [ | ||||
|     { | ||||
| @@ -24,7 +25,7 @@ | ||||
|           "error", | ||||
|           { | ||||
|             "type": "attribute", | ||||
|             "prefix": "app", | ||||
|             "prefix": "pngx", | ||||
|             "style": "camelCase" | ||||
|           } | ||||
|         ], | ||||
| @@ -32,7 +33,7 @@ | ||||
|           "error", | ||||
|           { | ||||
|             "type": "element", | ||||
|             "prefix": "app", | ||||
|             "prefix": "pngx", | ||||
|             "style": "kebab-case" | ||||
|           } | ||||
|         ] | ||||
|   | ||||
| @@ -12,32 +12,40 @@ | ||||
|       }, | ||||
|       "root": "", | ||||
|       "sourceRoot": "src", | ||||
|       "prefix": "app", | ||||
|       "prefix": "pngx", | ||||
|       "i18n": { | ||||
|         "sourceLocale": "en-US", | ||||
|         "locales": { | ||||
|           "ar-AR": "src/locale/messages.ar_AR.xlf", | ||||
|           "af-ZA": "src/locale/messages.af_ZA.xlf", | ||||
|           "bg-BG": "src/locale/messages.bg_BG.xlf", | ||||
|           "be-BY": "src/locale/messages.be_BY.xlf", | ||||
|           "ca-ES": "src/locale/messages.ca_ES.xlf", | ||||
|           "cs-CZ": "src/locale/messages.cs_CZ.xlf", | ||||
|           "da-DK": "src/locale/messages.da_DK.xlf", | ||||
|           "de-DE": "src/locale/messages.de_DE.xlf", | ||||
|           "el-GR": "src/locale/messages.el_GR.xlf", | ||||
|           "en-GB": "src/locale/messages.en_GB.xlf", | ||||
|           "es-ES": "src/locale/messages.es_ES.xlf", | ||||
|           "fi-FI": "src/locale/messages.fi_FI.xlf", | ||||
|           "fr-FR": "src/locale/messages.fr_FR.xlf", | ||||
|           "hu-HU": "src/locale/messages.hu_HU.xlf", | ||||
|           "it-IT": "src/locale/messages.it_IT.xlf", | ||||
|           "ja-JP": "src/locale/messages.ja_JP.xlf", | ||||
|           "lb-LU": "src/locale/messages.lb_LU.xlf", | ||||
|           "nl-NL": "src/locale/messages.nl_NL.xlf", | ||||
|           "no-NO": "src/locale/messages.no_NO.xlf", | ||||
|           "pl-PL": "src/locale/messages.pl_PL.xlf", | ||||
|           "pt-BR": "src/locale/messages.pt_BR.xlf", | ||||
|           "pt-PT": "src/locale/messages.pt_PT.xlf", | ||||
|           "ro-RO": "src/locale/messages.ro_RO.xlf", | ||||
|           "ru-RU": "src/locale/messages.ru_RU.xlf", | ||||
|           "sk-SK": "src/locale/messages.sk_SK.xlf", | ||||
|           "sl-SI": "src/locale/messages.sl_SI.xlf", | ||||
|           "sr-CS": "src/locale/messages.sr_CS.xlf", | ||||
|           "sv-SE": "src/locale/messages.sv_SE.xlf", | ||||
|           "tr-TR": "src/locale/messages.tr_TR.xlf", | ||||
|           "uk-UA": "src/locale/messages.uk_UA.xlf", | ||||
|           "zh-CN": "src/locale/messages.zh_CN.xlf" | ||||
|         } | ||||
|       }, | ||||
| @@ -58,7 +66,7 @@ | ||||
|               "src/assets", | ||||
|               "src/manifest.webmanifest", | ||||
|               { | ||||
|                 "glob": "pdf.worker.min.js", | ||||
|                 "glob": "{pdf.worker.min.js,pdf.min.js}", | ||||
|                 "input": "node_modules/pdfjs-dist/build/", | ||||
|                 "output": "/assets/js/" | ||||
|               } | ||||
| @@ -68,7 +76,10 @@ | ||||
|             ], | ||||
|             "scripts": [], | ||||
|             "allowedCommonJsDependencies": [ | ||||
|               "ng2-pdf-viewer" | ||||
|               "pdfjs-dist", | ||||
|               "pdfjs-dist/web/pdf_viewer", | ||||
|               "filesize", | ||||
|               "file-saver" | ||||
|             ], | ||||
|             "vendorChunk": true, | ||||
|             "extractLicenses": false, | ||||
| @@ -102,7 +113,7 @@ | ||||
|                 { | ||||
|                   "type": "anyComponentStyle", | ||||
|                   "maximumWarning": "6kb", | ||||
|                   "maximumError": "10kb" | ||||
|                   "maximumError": "30kb" | ||||
|                 } | ||||
|               ] | ||||
|             }, | ||||
| @@ -117,18 +128,18 @@ | ||||
|         "serve": { | ||||
|           "builder": "@angular-devkit/build-angular:dev-server", | ||||
|           "options": { | ||||
|             "browserTarget": "paperless-ui:build:en-US" | ||||
|             "buildTarget": "paperless-ui:build:en-US" | ||||
|           }, | ||||
|           "configurations": { | ||||
|             "production": { | ||||
|               "browserTarget": "paperless-ui:build:production" | ||||
|               "buildTarget": "paperless-ui:build:production" | ||||
|             } | ||||
|           } | ||||
|         }, | ||||
|         "extract-i18n": { | ||||
|           "builder": "@angular-devkit/build-angular:extract-i18n", | ||||
|           "options": { | ||||
|             "browserTarget": "paperless-ui:build" | ||||
|             "buildTarget": "paperless-ui:build" | ||||
|           } | ||||
|         }, | ||||
|         "test": { | ||||
| @@ -159,7 +170,6 @@ | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "defaultProject": "paperless-ui", | ||||
|   "cli": { | ||||
|     "schematicCollections": [ | ||||
|       "@angular-eslint/schematics" | ||||
|   | ||||
							
								
								
									
										56
									
								
								src-ui/e2e/admin/settings.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,56 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
|  | ||||
| const REQUESTS_HAR = 'e2e/admin/requests/api-settings.har' | ||||
|  | ||||
| test('should activate / deactivate save button when settings change', async ({ | ||||
|   page, | ||||
| }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/settings') | ||||
|   await expect(page.getByRole('button', { name: 'Save' })).toBeDisabled() | ||||
|   await page.getByLabel('Use system setting').click() | ||||
|   await page.getByRole('button', { name: 'Save' }).scrollIntoViewIfNeeded() | ||||
|   await expect(page.getByRole('button', { name: 'Save' })).toBeEnabled() | ||||
| }) | ||||
|  | ||||
| test('should warn on unsaved changes', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/settings') | ||||
|   await page.getByLabel('Use system setting').click() | ||||
|   await page.getByRole('link', { name: 'Dashboard' }).click() | ||||
|   await expect(page.getByRole('dialog')).toHaveText(/unsaved changes/) | ||||
|   await page.getByRole('button', { name: 'Cancel' }).click() | ||||
|   await page.getByLabel('Use system setting').click() | ||||
|   await page.getByRole('link', { name: 'Dashboard' }).click() | ||||
|   await expect(page.getByRole('dialog')).toHaveCount(0) | ||||
| }) | ||||
|  | ||||
| test('should apply appearance changes when set', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/settings') | ||||
|   await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /auto/) | ||||
|   await page.getByLabel('Use system setting').click() | ||||
|   await page.getByLabel('Enable dark mode').click() | ||||
|   await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /dark/) | ||||
| }) | ||||
|  | ||||
| test('should toggle saved view options when set & saved', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/settings/savedviews') | ||||
|   await page.getByLabel('Show on dashboard').first().click() | ||||
|   await page.getByLabel('Show in sidebar').first().click() | ||||
|   const updatePromise = page.waitForRequest((request) => { | ||||
|     if (!request.url().includes('8')) return true // skip other saved views | ||||
|     const data = request.postDataJSON() | ||||
|     const isValid = | ||||
|       data['show_on_dashboard'] === true && data['show_in_sidebar'] === true | ||||
|     return ( | ||||
|       isValid && | ||||
|       request.method() === 'PATCH' && | ||||
|       request.url().includes('/api/saved_views/') | ||||
|     ) | ||||
|   }) | ||||
|   await page.getByRole('button', { name: 'Save' }).scrollIntoViewIfNeeded() | ||||
|   await page.getByRole('button', { name: 'Save' }).click() | ||||
|   await updatePromise | ||||
| }) | ||||
| @@ -10,7 +10,7 @@ test('dashboard inbox link', async ({ page }) => { | ||||
|   await page.goto('/dashboard') | ||||
|   await page.getByRole('link', { name: 'Documents in inbox' }).click() | ||||
|   await expect(page).toHaveURL(/tags__id__all=9/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/8 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/8 documents/) | ||||
| }) | ||||
|  | ||||
| test('dashboard total documents link', async ({ page }) => { | ||||
| @@ -18,7 +18,7 @@ test('dashboard total documents link', async ({ page }) => { | ||||
|   await page.goto('/dashboard') | ||||
|   await page.getByRole('link').filter({ hasText: 'Total documents' }).click() | ||||
|   await expect(page).toHaveURL(/documents/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/61 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/61 documents/) | ||||
|   await page.getByRole('button', { name: 'Reset filters' }) | ||||
| }) | ||||
|  | ||||
| @@ -26,19 +26,20 @@ test('dashboard saved view show all', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR3, { notFound: 'fallback' }) | ||||
|   await page.goto('/dashboard') | ||||
|   await page | ||||
|     .locator('app-widget-frame') | ||||
|     .locator('pngx-widget-frame') | ||||
|     .filter({ hasText: 'Inbox' }) | ||||
|     .getByRole('link', { name: 'Show all' }) | ||||
|     .first() | ||||
|     .click() | ||||
|   await expect(page).toHaveURL(/view\/7/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/8 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/8 documents/) | ||||
| }) | ||||
|  | ||||
| test('dashboard saved view document links', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR4, { notFound: 'fallback' }) | ||||
|   await page.goto('/dashboard') | ||||
|   await page | ||||
|     .locator('app-widget-frame') | ||||
|     .locator('pngx-widget-frame') | ||||
|     .filter({ hasText: 'Inbox' }) | ||||
|     .locator('table') | ||||
|     .getByRole('link', { name: /test/ }) | ||||
|   | ||||
| @@ -8,17 +8,13 @@ test('should activate / deactivate save button when changes are saved', async ({ | ||||
| }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents/175/') | ||||
|   await page.waitForSelector('app-document-detail app-input-text:first-child') | ||||
|   await page.waitForSelector('pngx-document-detail pngx-input-text:first-child') | ||||
|   await expect(page.getByTitle('Storage path', { exact: true })).toHaveText( | ||||
|     /\w+/ | ||||
|   ) | ||||
|   await expect( | ||||
|     page.getByRole('button', { name: 'Save', exact: true }) | ||||
|   ).toBeDisabled() | ||||
|   await expect(page.getByRole('button', { name: 'Save' }).nth(1)).toBeDisabled() | ||||
|   await page.getByTitle('Storage path').getByTitle('Clear all').click() | ||||
|   await expect( | ||||
|     page.getByRole('button', { name: 'Save', exact: true }) | ||||
|   ).toBeEnabled() | ||||
|   await expect(page.getByRole('button', { name: 'Save' }).nth(1)).toBeEnabled() | ||||
| }) | ||||
|  | ||||
| test('should warn on unsaved changes', async ({ page }) => { | ||||
| @@ -27,16 +23,12 @@ test('should warn on unsaved changes', async ({ page }) => { | ||||
|   await expect(page.getByTitle('Correspondent', { exact: true })).toHaveText( | ||||
|     /\w+/ | ||||
|   ) | ||||
|   await expect( | ||||
|     page.getByRole('button', { name: 'Save', exact: true }) | ||||
|   ).toBeDisabled() | ||||
|   await expect(page.getByRole('button', { name: 'Save' }).nth(1)).toBeDisabled() | ||||
|   await page | ||||
|     .getByTitle('Storage path', { exact: true }) | ||||
|     .getByTitle('Clear all') | ||||
|     .click() | ||||
|   await expect( | ||||
|     page.getByRole('button', { name: 'Save', exact: true }) | ||||
|   ).toBeEnabled() | ||||
|   await expect(page.getByRole('button', { name: 'Save' }).nth(1)).toBeEnabled() | ||||
|   await page.getByRole('button', { name: 'Close', exact: true }).click() | ||||
|   await expect(page.getByRole('dialog')).toHaveText(/unsaved changes/) | ||||
|   await page.getByRole('button', { name: 'Cancel' }).click() | ||||
| @@ -79,13 +71,13 @@ test('should show a mobile preview', async ({ page }) => { | ||||
|   await page.setViewportSize({ width: 400, height: 1000 }) | ||||
|   await expect(page.getByRole('tab', { name: 'Preview' })).toBeVisible() | ||||
|   await page.getByRole('tab', { name: 'Preview' }).click() | ||||
|   await page.waitForSelector('pdf-viewer') | ||||
|   await page.waitForSelector('pngx-pdf-viewer') | ||||
| }) | ||||
|  | ||||
| test('should show a list of notes', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents/175/notes') | ||||
|   await expect(page.locator('app-document-notes')).toBeVisible() | ||||
|   await expect(page.locator('pngx-document-notes')).toBeVisible() | ||||
|   await expect( | ||||
|     await page.getByRole('button', { | ||||
|       name: /delete note/i, | ||||
| @@ -94,51 +86,6 @@ test('should show a list of notes', async ({ page }) => { | ||||
|   ).toHaveCount(4) | ||||
| }) | ||||
|  | ||||
| test('should support note deletion', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents/175/notes') | ||||
|   await expect(page.locator('app-document-notes')).toBeVisible() | ||||
|   const deletePromise = page.waitForRequest( | ||||
|     (request) => | ||||
|       request.method() === 'DELETE' && | ||||
|       request.url().includes('/api/documents/175/notes/') | ||||
|   ) | ||||
|   await page | ||||
|     .getByRole('button', { name: /delete note/i, includeHidden: true }) | ||||
|     .first() | ||||
|     .click() | ||||
|   await deletePromise | ||||
| }) | ||||
|  | ||||
| test('should support note insertion', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents/175/notes') | ||||
|   await expect(page.locator('app-document-notes')).toBeVisible() | ||||
|   await expect( | ||||
|     await page.getByRole('button', { | ||||
|       name: /delete note/i, | ||||
|       includeHidden: true, | ||||
|     }) | ||||
|   ).toHaveCount(4) | ||||
|   await page.getByPlaceholder('Enter note').fill('This is a new note') | ||||
|   const addPromise = page.waitForRequest((request) => { | ||||
|     if (!request.url().includes('/notes/')) { | ||||
|       // ignore other requests | ||||
|       return true | ||||
|     } else { | ||||
|       const data = request.postDataJSON() | ||||
|       const isValid = data['note'] === 'This is a new note' | ||||
|       return ( | ||||
|         isValid && | ||||
|         request.method() === 'POST' && | ||||
|         request.url().includes('/notes/') | ||||
|       ) | ||||
|     } | ||||
|   }) | ||||
|   await page.getByRole('button', { name: 'Add note' }).click() | ||||
|   await addPromise | ||||
| }) | ||||
|  | ||||
| test('should support quick filters', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR2, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents/175/details') | ||||
|   | ||||
| @@ -13,33 +13,33 @@ test('basic filtering', async ({ page }) => { | ||||
|   await page.getByRole('button', { name: 'Tags' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Inbox' }).click() | ||||
|   await expect(page).toHaveURL(/tags__id__all=9/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/8 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/8 documents/) | ||||
|   await page.getByRole('button', { name: 'Document type' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Invoice Test 3' }).click() | ||||
|   await expect(page).toHaveURL(/document_type__id__in=1/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/3 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/3 documents/) | ||||
|   await page.getByRole('button', { name: 'Reset filters' }).first().click() | ||||
|   await page.getByRole('button', { name: 'Correspondent' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Test Correspondent 1' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Correspondent 9' }).click() | ||||
|   await expect(page).toHaveURL(/correspondent__id__in=12,1/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/7 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/7 documents/) | ||||
|   await page | ||||
|     .locator('app-filter-editor') | ||||
|     .locator('pngx-filter-editor') | ||||
|     .getByTitle('Correspondent') | ||||
|     .getByText('Exclude') | ||||
|     .click() | ||||
|   await expect(page).toHaveURL(/correspondent__id__none=12,1/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/54 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/54 documents/) | ||||
|   // clear button | ||||
|   await page.getByRole('button', { name: '2 selected', exact: true }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/61 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/61 documents/) | ||||
|   await page.getByRole('button', { name: 'Storage path' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Testing 12' }).click() | ||||
|   await expect(page).toHaveURL(/storage_path__id__in=5/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/8 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/8 documents/) | ||||
|   await page.getByRole('button', { name: 'Reset filters' }).first().click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/61 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/61 documents/) | ||||
| }) | ||||
|  | ||||
| test('text filtering', async ({ page }) => { | ||||
| @@ -47,54 +47,55 @@ test('text filtering', async ({ page }) => { | ||||
|   await page.goto('/documents') | ||||
|   await page.getByRole('textbox').click() | ||||
|   await page.getByRole('textbox').fill('test') | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/32 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/32 documents/) | ||||
|   await expect(page).toHaveURL(/title_content=test/) | ||||
|   await page.getByRole('button', { name: 'Title & content' }).click() | ||||
|   await page.getByRole('button', { name: 'Title', exact: true }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/9 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/9 documents/) | ||||
|   await expect(page).toHaveURL(/title__icontains=test/) | ||||
|   await page.getByRole('button', { name: 'Title', exact: true }).click() | ||||
|   await page.getByRole('button', { name: 'Advanced search' }).click() | ||||
|   await expect(page).toHaveURL(/query=test/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/26 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/26 documents/) | ||||
|   await page.getByRole('button', { name: 'Advanced search' }).click() | ||||
|   await page.getByRole('button', { name: 'ASN' }).click() | ||||
|   await page.getByRole('textbox').fill('1123') | ||||
|   await expect(page).toHaveURL(/archive_serial_number=1123/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/one document/i) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/one document/i) | ||||
|   await page.locator('select').selectOption('greater') | ||||
|   await page.getByRole('textbox').click() | ||||
|   await page.getByRole('textbox').fill('1123') | ||||
|   await expect(page).toHaveURL(/archive_serial_number__gt=1123/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/5 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/5 documents/) | ||||
|   await page.locator('select').selectOption('less') | ||||
|   await expect(page).toHaveURL(/archive_serial_number__lt=1123/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/0 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/0 documents/) | ||||
|   await page.locator('select').selectOption('is null') | ||||
|   await expect(page).toHaveURL(/archive_serial_number__isnull=1/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/55 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/55 documents/) | ||||
|   await page.locator('select').selectOption('not null') | ||||
|   await expect(page).toHaveURL(/archive_serial_number__isnull=0/) | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/6 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/6 documents/) | ||||
| }) | ||||
|  | ||||
| test('date filtering', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR3, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents') | ||||
|   await page.getByRole('button', { name: 'Created' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Last 3 months' }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/one document/i) | ||||
|   await page.getByRole('button', { name: 'Created Clear selected' }).click() | ||||
|   await page.getByRole('button', { name: 'Created' }).click() | ||||
|   await page.getByRole('button', { name: 'Dates' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'Last 3 months' }).first().click() | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/one document/i) | ||||
|   await page.getByRole('button', { name: 'Dates Clear selected' }).click() | ||||
|   await page.getByRole('button', { name: 'Dates' }).click() | ||||
|   await page | ||||
|     .getByRole('menuitem', { name: 'After mm/dd/yyyy' }) | ||||
|     .getByRole('button') | ||||
|     .first() | ||||
|     .click() | ||||
|   await page.getByRole('combobox', { name: 'Select month' }).selectOption('12') | ||||
|   await page.getByRole('combobox', { name: 'Select year' }).selectOption('2022') | ||||
|   await page.getByText('11', { exact: true }).click() | ||||
|   await page.getByRole('button', { name: 'Title & content' }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText(/2 documents/) | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText(/2 documents/) | ||||
| }) | ||||
|  | ||||
| test('sorting', async ({ page }) => { | ||||
| @@ -105,7 +106,7 @@ test('sorting', async ({ page }) => { | ||||
|   await expect(page).toHaveURL(/sort=archive_serial_number/) | ||||
|   await page.getByRole('button', { name: 'Sort' }).click() | ||||
|   await page | ||||
|     .locator('app-page-header') | ||||
|     .locator('pngx-page-header') | ||||
|     .getByRole('button', { name: 'Correspondent' }) | ||||
|     .click() | ||||
|   await expect(page).toHaveURL(/sort=correspondent__name/) | ||||
| @@ -114,7 +115,7 @@ test('sorting', async ({ page }) => { | ||||
|   await expect(page).toHaveURL(/sort=title/) | ||||
|   await page.getByRole('button', { name: 'Sort' }).click() | ||||
|   await page | ||||
|     .locator('app-page-header') | ||||
|     .locator('pngx-page-header') | ||||
|     .getByRole('button', { name: 'Document type' }) | ||||
|     .click() | ||||
|   await expect(page).toHaveURL(/sort=document_type__name/) | ||||
| @@ -131,49 +132,49 @@ test('sorting', async ({ page }) => { | ||||
|   await page.getByRole('button', { name: 'Notes' }).click() | ||||
|   await expect(page).toHaveURL(/sort=num_notes/) | ||||
|   await page.getByRole('button', { name: 'Sort' }).click() | ||||
|   await page.locator('.w-100 > label > .toolbaricon').first().click() | ||||
|   await page.locator('.w-100 > label > i-bs').first().click() | ||||
|   await expect(page).not.toHaveURL(/reverse=1/) | ||||
| }) | ||||
|  | ||||
| test('change views', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR5, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents') | ||||
|   await page.locator('app-page-header label').first().click() | ||||
|   await expect(page.locator('app-document-list table')).toBeVisible() | ||||
|   await page.locator('app-page-header label').nth(1).click() | ||||
|   await expect(page.locator('app-document-card-small').first()).toBeAttached() | ||||
|   await page.locator('app-page-header label').nth(2).click() | ||||
|   await expect(page.locator('app-document-card-large').first()).toBeAttached() | ||||
|   await page.locator('.btn-group label').first().click() | ||||
|   await expect(page.locator('pngx-document-list table')).toBeVisible() | ||||
|   await page.locator('.btn-group label').nth(1).click() | ||||
|   await expect(page.locator('pngx-document-card-small').first()).toBeAttached() | ||||
|   await page.locator('.btn-group label').nth(2).click() | ||||
|   await expect(page.locator('pngx-document-card-large').first()).toBeAttached() | ||||
| }) | ||||
|  | ||||
| test('bulk edit', async ({ page }) => { | ||||
|   await page.routeFromHAR(REQUESTS_HAR6, { notFound: 'fallback' }) | ||||
|   await page.goto('/documents') | ||||
|  | ||||
|   await page.locator('app-document-card-small').nth(0).click() | ||||
|   await page.locator('pngx-document-card-small').nth(0).click() | ||||
|   await page | ||||
|     .locator('app-document-card-small') | ||||
|     .locator('pngx-document-card-small') | ||||
|     .nth(3) | ||||
|     .click({ | ||||
|       modifiers: ['Shift'], | ||||
|     }) | ||||
|  | ||||
|   await expect(page.locator('app-document-list')).toHaveText( | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText( | ||||
|     /Selected 4 of 61 documents/i | ||||
|   ) | ||||
|  | ||||
|   await page.getByRole('button', { name: 'Page' }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText( | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText( | ||||
|     /Selected 50 of 61 documents/i | ||||
|   ) | ||||
|   await page.getByRole('button', { name: 'All' }).click() | ||||
|   await expect(page.locator('app-document-list')).toHaveText( | ||||
|   await expect(page.locator('pngx-document-list')).toHaveText( | ||||
|     /Selected 61 of 61 documents/i | ||||
|   ) | ||||
|   await page.getByRole('button', { name: 'Cancel' }).click() | ||||
|  | ||||
|   await page.locator('app-document-card-small').nth(1).click() | ||||
|   await page.locator('app-document-card-small').nth(2).click() | ||||
|   await page.locator('pngx-document-card-small').nth(1).click() | ||||
|   await page.locator('pngx-document-card-small').nth(2).click() | ||||
|  | ||||
|   await page.getByRole('button', { name: 'Tags' }).click() | ||||
|   await page.getByRole('menuitem', { name: 'TagWithPartial' }).click() | ||||
|   | ||||